Mercurial > mplayer.hg
annotate stream/realrtsp/real.c @ 21783:4511c04bc4a0
Fix potential buffer overflow in asm rules matching code
author | rtogni |
---|---|
date | Sun, 31 Dec 2006 13:27:53 +0000 |
parents | fa99b3d31d13 |
children | 04281be4fbdd |
rev | line source |
---|---|
9922 | 1 /* |
2 * This file was ported to MPlayer from xine CVS real.c,v 1.8 2003/03/30 17:11:50 | |
3 */ | |
4 | |
5 /* | |
6 * Copyright (C) 2002 the xine project | |
7 * | |
8 * This file is part of xine, a free video player. | |
9 * | |
10 * xine is free software; you can redistribute it and/or modify | |
11 * it under the terms of the GNU General Public License as published by | |
12 * the Free Software Foundation; either version 2 of the License, or | |
13 * (at your option) any later version. | |
14 * | |
15 * xine is distributed in the hope that it will be useful, | |
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
18 * GNU General Public License for more details. | |
19 * | |
20 * You should have received a copy of the GNU General Public License | |
21 * along with this program; if not, write to the Free Software | |
22 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA | |
23 * | |
24 * | |
25 * special functions for real streams. | |
26 * adopted from joschkas real tools. | |
27 * | |
28 */ | |
29 | |
30 #include <stdio.h> | |
31 #include <string.h> | |
32 | |
10115 | 33 #include "../config.h" |
21372 | 34 #include "libavutil/common.h" |
21507
fa99b3d31d13
Hack around libavutil/bswap.h compilation problems due to always_inline undefined.
reimar
parents:
21372
diff
changeset
|
35 #include "mpbswap.h" |
9922 | 36 #include "real.h" |
37 #include "asmrp.h" | |
38 #include "sdpplin.h" | |
12266 | 39 #include "xbuffer.h" |
19256
ed6ca050bba5
The real hash function is just a md5, replace with lavu code
rtogni
parents:
19074
diff
changeset
|
40 #if USE_LIBAVUTIL_SO |
ed6ca050bba5
The real hash function is just a md5, replace with lavu code
rtogni
parents:
19074
diff
changeset
|
41 #include "ffmpeg/md5.h" |
ed6ca050bba5
The real hash function is just a md5, replace with lavu code
rtogni
parents:
19074
diff
changeset
|
42 #else |
ed6ca050bba5
The real hash function is just a md5, replace with lavu code
rtogni
parents:
19074
diff
changeset
|
43 #include "libavutil/md5.h" |
ed6ca050bba5
The real hash function is just a md5, replace with lavu code
rtogni
parents:
19074
diff
changeset
|
44 #endif |
20527 | 45 #include "../http.h" |
46 #include "mp_msg.h" | |
9922 | 47 |
48 /* | |
49 #define LOG | |
50 */ | |
51 | |
19258 | 52 static const unsigned char xor_table[] = { |
9922 | 53 0x05, 0x18, 0x74, 0xd0, 0x0d, 0x09, 0x02, 0x53, |
54 0xc0, 0x01, 0x05, 0x05, 0x67, 0x03, 0x19, 0x70, | |
55 0x08, 0x27, 0x66, 0x10, 0x10, 0x72, 0x08, 0x09, | |
56 0x63, 0x11, 0x03, 0x71, 0x08, 0x08, 0x70, 0x02, | |
57 0x10, 0x57, 0x05, 0x18, 0x54, 0x00, 0x00, 0x00 }; | |
58 | |
59 | |
10115 | 60 #define BE_32C(x,y) (*((uint32_t*)(x))=be2me_32(y)) |
9922 | 61 |
10115 | 62 #define BE_16(x) be2me_16(*(uint16_t*)(x)) |
9922 | 63 |
10115 | 64 #define BE_32(x) be2me_32(*(uint32_t*)(x)) |
9922 | 65 |
12845 | 66 #ifndef MAX |
9922 | 67 #define MAX(x,y) ((x>y) ? x : y) |
12845 | 68 #endif |
9922 | 69 |
18792
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
70 #define BUF_SIZE 4096 |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
71 |
9922 | 72 #ifdef LOG |
73 static void hexdump (const char *buf, int length) { | |
74 | |
75 int i; | |
76 | |
77 printf (" hexdump> "); | |
78 for (i = 0; i < length; i++) { | |
79 unsigned char c = buf[i]; | |
80 | |
81 printf ("%02x", c); | |
82 | |
83 if ((i % 16) == 15) | |
84 printf ("\n "); | |
85 | |
86 if ((i % 2) == 1) | |
87 printf (" "); | |
88 | |
89 } | |
90 printf ("\n"); | |
91 } | |
92 #endif | |
93 | |
94 | |
95 static void calc_response_string (char *result, char *challenge) { | |
96 | |
19263 | 97 char zres[16]; |
9922 | 98 int i; |
99 | |
19256
ed6ca050bba5
The real hash function is just a md5, replace with lavu code
rtogni
parents:
19074
diff
changeset
|
100 av_md5_sum(zres, challenge, 64); |
9922 | 101 |
102 /* convert zres to ascii string */ | |
103 for (i=0; i<16; i++ ) { | |
104 char a, b; | |
105 | |
106 a = (zres[i] >> 4) & 15; | |
107 b = zres[i] & 15; | |
108 | |
109 result[i*2] = ((a<10) ? (a+48) : (a+87)) & 255; | |
110 result[i*2+1] = ((b<10) ? (b+48) : (b+87)) & 255; | |
111 } | |
112 } | |
113 | |
19266 | 114 static void real_calc_response_and_checksum (char *response, char *chksum, char *challenge) { |
9922 | 115 |
116 int ch_len, table_len, resp_len; | |
117 int i; | |
118 char *ptr; | |
119 char buf[128]; | |
120 | |
121 /* initialize return values */ | |
122 memset(response, 0, 64); | |
123 memset(chksum, 0, 34); | |
124 | |
125 /* initialize buffer */ | |
126 memset(buf, 0, 128); | |
127 ptr=buf; | |
128 BE_32C(ptr, 0xa1e9149d); | |
129 ptr+=4; | |
130 BE_32C(ptr, 0x0e6b3b59); | |
131 ptr+=4; | |
132 | |
133 /* some (length) checks */ | |
134 if (challenge != NULL) | |
135 { | |
136 ch_len = strlen (challenge); | |
137 | |
138 if (ch_len == 40) /* what a hack... */ | |
139 { | |
140 challenge[32]=0; | |
141 ch_len=32; | |
142 } | |
143 if ( ch_len > 56 ) ch_len=56; | |
144 | |
145 /* copy challenge to buf */ | |
146 memcpy(ptr, challenge, ch_len); | |
147 } | |
148 | |
149 table_len = strlen(xor_table); | |
150 | |
151 if (table_len > 56) table_len=56; | |
152 | |
153 /* xor challenge bytewise with xor_table */ | |
154 for (i=0; i<table_len; i++) | |
155 ptr[i] = ptr[i] ^ xor_table[i]; | |
156 | |
157 calc_response_string (response, buf); | |
158 | |
159 /* add tail */ | |
160 resp_len = strlen (response); | |
161 strcpy (&response[resp_len], "01d0a8e3"); | |
162 | |
163 /* calculate checksum */ | |
164 for (i=0; i<resp_len/4; i++) | |
165 chksum[i] = response[i*4]; | |
166 } | |
167 | |
168 | |
169 /* | |
170 * takes a MLTI-Chunk and a rule number got from match_asm_rule, | |
171 * returns a pointer to selected data and number of bytes in that. | |
172 */ | |
173 | |
12266 | 174 static int select_mlti_data(const char *mlti_chunk, int mlti_size, int selection, char **out) { |
9922 | 175 |
176 int numrules, codec, size; | |
177 int i; | |
178 | |
179 /* MLTI chunk should begin with MLTI */ | |
180 | |
181 if ((mlti_chunk[0] != 'M') | |
182 ||(mlti_chunk[1] != 'L') | |
183 ||(mlti_chunk[2] != 'T') | |
184 ||(mlti_chunk[3] != 'I')) | |
185 { | |
186 #ifdef LOG | |
187 printf("libreal: MLTI tag not detected, copying data\n"); | |
188 #endif | |
12266 | 189 *out = xbuffer_copyin(*out, 0, mlti_chunk, mlti_size); |
9922 | 190 return mlti_size; |
191 } | |
192 | |
193 mlti_chunk+=4; | |
194 | |
195 /* next 16 bits are the number of rules */ | |
196 numrules=BE_16(mlti_chunk); | |
197 if (selection >= numrules) return 0; | |
198 | |
199 /* now <numrules> indices of codecs follows */ | |
200 /* we skip to selection */ | |
201 mlti_chunk+=(selection+1)*2; | |
202 | |
203 /* get our index */ | |
204 codec=BE_16(mlti_chunk); | |
205 | |
206 /* skip to number of codecs */ | |
207 mlti_chunk+=(numrules-selection)*2; | |
208 | |
209 /* get number of codecs */ | |
210 numrules=BE_16(mlti_chunk); | |
211 | |
212 if (codec >= numrules) { | |
20530 | 213 mp_msg(MSGT_STREAM, MSGL_WARN, "realrtsp: codec index >= number of codecs. %i %i\n", |
214 codec, numrules); | |
9922 | 215 return 0; |
216 } | |
217 | |
218 mlti_chunk+=2; | |
219 | |
220 /* now seek to selected codec */ | |
221 for (i=0; i<codec; i++) { | |
222 size=BE_32(mlti_chunk); | |
223 mlti_chunk+=size+4; | |
224 } | |
225 | |
226 size=BE_32(mlti_chunk); | |
227 | |
228 #ifdef LOG | |
229 hexdump(mlti_chunk+4, size); | |
230 #endif | |
12266 | 231 *out = xbuffer_copyin(*out, 0, mlti_chunk+4, size); |
9922 | 232 return size; |
233 } | |
234 | |
235 /* | |
236 * looking at stream description. | |
237 */ | |
238 | |
19266 | 239 static rmff_header_t *real_parse_sdp(char *data, char **stream_rules, uint32_t bandwidth) { |
9922 | 240 |
241 sdpplin_t *desc; | |
242 rmff_header_t *header; | |
12266 | 243 char *buf; |
9922 | 244 int len, i; |
245 int max_bit_rate=0; | |
246 int avg_bit_rate=0; | |
247 int max_packet_size=0; | |
248 int avg_packet_size=0; | |
249 int duration=0; | |
250 | |
251 | |
252 if (!data) return NULL; | |
253 | |
254 desc=sdpplin_parse(data); | |
255 | |
256 if (!desc) return NULL; | |
257 | |
12266 | 258 buf = xbuffer_init(2048); |
9922 | 259 header=calloc(1,sizeof(rmff_header_t)); |
260 | |
261 header->fileheader=rmff_new_fileheader(4+desc->stream_count); | |
262 header->cont=rmff_new_cont( | |
263 desc->title, | |
264 desc->author, | |
265 desc->copyright, | |
266 desc->abstract); | |
267 header->data=rmff_new_dataheader(0,0); | |
268 header->streams=calloc(1,sizeof(rmff_mdpr_t*)*(desc->stream_count+1)); | |
269 #ifdef LOG | |
270 printf("number of streams: %u\n", desc->stream_count); | |
271 #endif | |
272 | |
273 for (i=0; i<desc->stream_count; i++) { | |
274 | |
275 int j=0; | |
276 int n; | |
277 char b[64]; | |
21783
4511c04bc4a0
Fix potential buffer overflow in asm rules matching code
rtogni
parents:
21507
diff
changeset
|
278 int rulematches[MAX_RULEMATCHES]; |
9922 | 279 |
280 #ifdef LOG | |
281 printf("calling asmrp_match with:\n%s\n%u\n", desc->stream[i]->asm_rule_book, bandwidth); | |
282 #endif | |
283 n=asmrp_match(desc->stream[i]->asm_rule_book, bandwidth, rulematches); | |
284 for (j=0; j<n; j++) { | |
285 #ifdef LOG | |
286 printf("asmrp rule match: %u for stream %u\n", rulematches[j], desc->stream[i]->stream_id); | |
287 #endif | |
288 sprintf(b,"stream=%u;rule=%u,", desc->stream[i]->stream_id, rulematches[j]); | |
12266 | 289 *stream_rules = xbuffer_strcat(*stream_rules, b); |
9922 | 290 } |
291 | |
14144
1fcb18e39ef9
Fix streaming if not mlti_data (for some non-multirate streams)
rtognimp
parents:
13676
diff
changeset
|
292 if (!desc->stream[i]->mlti_data) { |
1fcb18e39ef9
Fix streaming if not mlti_data (for some non-multirate streams)
rtognimp
parents:
13676
diff
changeset
|
293 len = 0; |
1fcb18e39ef9
Fix streaming if not mlti_data (for some non-multirate streams)
rtognimp
parents:
13676
diff
changeset
|
294 buf = NULL; |
1fcb18e39ef9
Fix streaming if not mlti_data (for some non-multirate streams)
rtognimp
parents:
13676
diff
changeset
|
295 } else |
12266 | 296 len=select_mlti_data(desc->stream[i]->mlti_data, desc->stream[i]->mlti_data_size, rulematches[0], &buf); |
9922 | 297 |
298 header->streams[i]=rmff_new_mdpr( | |
299 desc->stream[i]->stream_id, | |
300 desc->stream[i]->max_bit_rate, | |
301 desc->stream[i]->avg_bit_rate, | |
302 desc->stream[i]->max_packet_size, | |
303 desc->stream[i]->avg_packet_size, | |
304 desc->stream[i]->start_time, | |
305 desc->stream[i]->preroll, | |
306 desc->stream[i]->duration, | |
307 desc->stream[i]->stream_name, | |
308 desc->stream[i]->mime_type, | |
309 len, | |
310 buf); | |
311 | |
312 duration=MAX(duration,desc->stream[i]->duration); | |
313 max_bit_rate+=desc->stream[i]->max_bit_rate; | |
314 avg_bit_rate+=desc->stream[i]->avg_bit_rate; | |
315 max_packet_size=MAX(max_packet_size, desc->stream[i]->max_packet_size); | |
316 if (avg_packet_size) | |
317 avg_packet_size=(avg_packet_size + desc->stream[i]->avg_packet_size) / 2; | |
318 else | |
319 avg_packet_size=desc->stream[i]->avg_packet_size; | |
320 } | |
321 | |
12266 | 322 if (*stream_rules && strlen(*stream_rules) && (*stream_rules)[strlen(*stream_rules)-1] == ',') |
323 (*stream_rules)[strlen(*stream_rules)-1]=0; /* delete last ',' in stream_rules */ | |
9922 | 324 |
325 header->prop=rmff_new_prop( | |
326 max_bit_rate, | |
327 avg_bit_rate, | |
328 max_packet_size, | |
329 avg_packet_size, | |
330 0, | |
331 duration, | |
332 0, | |
333 0, | |
334 0, | |
335 desc->stream_count, | |
336 desc->flags); | |
337 | |
338 rmff_fix_header(header); | |
12266 | 339 buf = xbuffer_free(buf); |
9922 | 340 |
341 return header; | |
342 } | |
343 | |
12266 | 344 int real_get_rdt_chunk(rtsp_t *rtsp_session, char **buffer) { |
9922 | 345 |
346 int n=1; | |
347 uint8_t header[8]; | |
348 rmff_pheader_t ph; | |
349 int size; | |
11595
95413c6d76a0
keyframe detection support by rgselk <rgselknospam@yahoo.com>
alex
parents:
11506
diff
changeset
|
350 int flags1, flags2; |
9922 | 351 int unknown1; |
352 uint32_t ts; | |
11595
95413c6d76a0
keyframe detection support by rgselk <rgselknospam@yahoo.com>
alex
parents:
11506
diff
changeset
|
353 static uint32_t prev_ts = -1; |
95413c6d76a0
keyframe detection support by rgselk <rgselknospam@yahoo.com>
alex
parents:
11506
diff
changeset
|
354 static int prev_stream_number = -1; |
9922 | 355 |
356 n=rtsp_read_data(rtsp_session, header, 8); | |
357 if (n<8) return 0; | |
358 if (header[0] != 0x24) | |
359 { | |
20530 | 360 mp_msg(MSGT_STREAM, MSGL_WARN, "realrtsp: rdt chunk not recognized: got 0x%02x\n", |
361 header[0]); | |
9922 | 362 return 0; |
363 } | |
13289
2cb80075204c
chunk size fix from Ross Finlayson, ported from xine
diego
parents:
12845
diff
changeset
|
364 size=(header[1]<<16)+(header[2]<<8)+(header[3]); |
9922 | 365 flags1=header[4]; |
366 if ((flags1!=0x40)&&(flags1!=0x42)) | |
367 { | |
368 #ifdef LOG | |
369 printf("got flags1: 0x%02x\n",flags1); | |
370 #endif | |
13676 | 371 if(header[6] == 0x06) { |
20530 | 372 mp_msg(MSGT_STREAM, MSGL_INFO, "realrtsp: Stream EOF detected\n"); |
13676 | 373 return -1; |
374 } | |
9922 | 375 header[0]=header[5]; |
376 header[1]=header[6]; | |
377 header[2]=header[7]; | |
378 n=rtsp_read_data(rtsp_session, header+3, 5); | |
379 if (n<5) return 0; | |
380 #ifdef LOG | |
381 printf("ignoring bytes:\n"); | |
382 hexdump(header, 8); | |
383 #endif | |
384 n=rtsp_read_data(rtsp_session, header+4, 4); | |
385 if (n<4) return 0; | |
386 flags1=header[4]; | |
387 size-=9; | |
388 } | |
11595
95413c6d76a0
keyframe detection support by rgselk <rgselknospam@yahoo.com>
alex
parents:
11506
diff
changeset
|
389 flags2=header[7]; |
95413c6d76a0
keyframe detection support by rgselk <rgselknospam@yahoo.com>
alex
parents:
11506
diff
changeset
|
390 // header[5..6] == frame number in stream |
13289
2cb80075204c
chunk size fix from Ross Finlayson, ported from xine
diego
parents:
12845
diff
changeset
|
391 unknown1=(header[5]<<16)+(header[6]<<8)+(header[7]); |
9922 | 392 n=rtsp_read_data(rtsp_session, header, 6); |
393 if (n<6) return 0; | |
394 ts=BE_32(header); | |
395 | |
396 #ifdef LOG | |
11595
95413c6d76a0
keyframe detection support by rgselk <rgselknospam@yahoo.com>
alex
parents:
11506
diff
changeset
|
397 printf("ts: %u, size: %u, flags: 0x%02x, unknown values: 0x%06x 0x%02x 0x%02x\n", |
9922 | 398 ts, size, flags1, unknown1, header[4], header[5]); |
399 #endif | |
400 size+=2; | |
401 | |
402 ph.object_version=0; | |
403 ph.length=size; | |
404 ph.stream_number=(flags1>>1)&1; | |
405 ph.timestamp=ts; | |
406 ph.reserved=0; | |
11595
95413c6d76a0
keyframe detection support by rgselk <rgselknospam@yahoo.com>
alex
parents:
11506
diff
changeset
|
407 if ((flags2&1) == 0 && (prev_ts != ts || prev_stream_number != ph.stream_number)) |
95413c6d76a0
keyframe detection support by rgselk <rgselknospam@yahoo.com>
alex
parents:
11506
diff
changeset
|
408 { |
95413c6d76a0
keyframe detection support by rgselk <rgselknospam@yahoo.com>
alex
parents:
11506
diff
changeset
|
409 prev_ts = ts; |
95413c6d76a0
keyframe detection support by rgselk <rgselknospam@yahoo.com>
alex
parents:
11506
diff
changeset
|
410 prev_stream_number = ph.stream_number; |
95413c6d76a0
keyframe detection support by rgselk <rgselknospam@yahoo.com>
alex
parents:
11506
diff
changeset
|
411 ph.flags=2; |
95413c6d76a0
keyframe detection support by rgselk <rgselknospam@yahoo.com>
alex
parents:
11506
diff
changeset
|
412 } |
95413c6d76a0
keyframe detection support by rgselk <rgselknospam@yahoo.com>
alex
parents:
11506
diff
changeset
|
413 else |
95413c6d76a0
keyframe detection support by rgselk <rgselknospam@yahoo.com>
alex
parents:
11506
diff
changeset
|
414 ph.flags=0; |
12266 | 415 *buffer = xbuffer_ensure_size(*buffer, 12+size); |
416 rmff_dump_pheader(&ph, *buffer); | |
9922 | 417 size-=12; |
12266 | 418 n=rtsp_read_data(rtsp_session, (*buffer)+12, size); |
9922 | 419 |
12271
4adb4a3b52a2
More bounds checking fixes (thnaks to Miguel Freitas)
rtognimp
parents:
12266
diff
changeset
|
420 return (n <= 0) ? 0 : n+12; |
9922 | 421 } |
422 | |
19266 | 423 static int convert_timestamp(char *str, int *sec, int *msec) { |
11506
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
424 int hh, mm, ss, ms = 0; |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
425 if (sscanf(str, "%d:%d:%d.%d", &hh, &mm, &ss, &ms) < 3) { |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
426 hh = 0; |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
427 if (sscanf(str, "%d:%d.%d", &mm, &ss, &ms) < 2) { |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
428 mm = 0; |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
429 if (sscanf(str, "%d.%d", &ss, &ms) < 1) { |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
430 ss = 0; |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
431 ms = 0; |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
432 } |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
433 } |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
434 } |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
435 if (sec) |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
436 *sec = hh * 3600 + mm * 60 + ss; |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
437 if (msec) |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
438 *msec = ms; |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
439 return 1; |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
440 } |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
441 |
14159 | 442 //! maximum size of the rtsp description, must be < INT_MAX |
443 #define MAX_DESC_BUF (20 * 1024 * 1024) | |
20527 | 444 rmff_header_t *real_setup_and_get_header(rtsp_t *rtsp_session, uint32_t bandwidth, |
445 char *username, char *password) { | |
9922 | 446 |
447 char *description=NULL; | |
448 char *session_id=NULL; | |
449 rmff_header_t *h; | |
450 char *challenge1; | |
451 char challenge2[64]; | |
452 char checksum[34]; | |
12266 | 453 char *subscribe; |
454 char *buf = xbuffer_init(256); | |
9922 | 455 char *mrl=rtsp_get_mrl(rtsp_session); |
456 unsigned int size; | |
457 int status; | |
17332
88adbc28f60b
This patch makes real rtsp tell the server to deliver data at specified
rtognimp
parents:
14159
diff
changeset
|
458 uint32_t maxbandwidth = bandwidth; |
20527 | 459 char* authfield = NULL; |
9922 | 460 |
461 /* get challenge */ | |
462 challenge1=strdup(rtsp_search_answers(rtsp_session,"RealChallenge1")); | |
463 #ifdef LOG | |
464 printf("real: Challenge1: %s\n", challenge1); | |
465 #endif | |
17332
88adbc28f60b
This patch makes real rtsp tell the server to deliver data at specified
rtognimp
parents:
14159
diff
changeset
|
466 |
88adbc28f60b
This patch makes real rtsp tell the server to deliver data at specified
rtognimp
parents:
14159
diff
changeset
|
467 /* set a reasonable default to get the best stream, unless bandwidth given */ |
88adbc28f60b
This patch makes real rtsp tell the server to deliver data at specified
rtognimp
parents:
14159
diff
changeset
|
468 if (!bandwidth) |
88adbc28f60b
This patch makes real rtsp tell the server to deliver data at specified
rtognimp
parents:
14159
diff
changeset
|
469 bandwidth = 10485800; |
9922 | 470 |
471 /* request stream description */ | |
20527 | 472 rtsp_send_describe: |
9922 | 473 rtsp_schedule_field(rtsp_session, "Accept: application/sdp"); |
474 sprintf(buf, "Bandwidth: %u", bandwidth); | |
475 rtsp_schedule_field(rtsp_session, buf); | |
476 rtsp_schedule_field(rtsp_session, "GUID: 00000000-0000-0000-0000-000000000000"); | |
477 rtsp_schedule_field(rtsp_session, "RegionData: 0"); | |
478 rtsp_schedule_field(rtsp_session, "ClientID: Linux_2.4_6.0.9.1235_play32_RN01_EN_586"); | |
479 rtsp_schedule_field(rtsp_session, "SupportsMaximumASMBandwidth: 1"); | |
480 rtsp_schedule_field(rtsp_session, "Language: en-US"); | |
481 rtsp_schedule_field(rtsp_session, "Require: com.real.retain-entity-for-setup"); | |
20527 | 482 if(authfield) |
483 rtsp_schedule_field(rtsp_session, authfield); | |
9922 | 484 status=rtsp_request_describe(rtsp_session,NULL); |
485 | |
20527 | 486 if (status == 401) { |
487 int authlen, b64_authlen; | |
488 char *authreq; | |
489 char* authstr = NULL; | |
490 | |
491 if (authfield) { | |
492 mp_msg(MSGT_STREAM, MSGL_ERR, "realrtsp: authorization failed, check your credentials\n"); | |
493 goto autherr; | |
494 } | |
495 if (!(authreq = rtsp_search_answers(rtsp_session,"WWW-Authenticate"))) { | |
496 mp_msg(MSGT_STREAM, MSGL_ERR, "realrtsp: 401 but no auth request, aborting\n"); | |
497 goto autherr; | |
498 } | |
499 if (!username) { | |
500 mp_msg(MSGT_STREAM, MSGL_ERR, "realrtsp: auth required but no username supplied\n"); | |
501 goto autherr; | |
502 } | |
503 if (!strstr(authreq, "Basic")) { | |
504 mp_msg(MSGT_STREAM, MSGL_ERR, "realrtsp: authenticator not supported (%s)\n", authreq); | |
505 goto autherr; | |
506 } | |
507 authlen = strlen(username) + (password ? strlen(password) : 0) + 2; | |
508 authstr = malloc(authlen); | |
509 sprintf(authstr, "%s:%s", username, password ? password : ""); | |
510 authfield = malloc(authlen*2+22); | |
511 strcpy(authfield, "Authorization: Basic "); | |
512 b64_authlen = base64_encode(authstr, authlen, authfield+21, authlen*2); | |
513 free(authstr); | |
514 if (b64_authlen < 0) { | |
515 mp_msg(MSGT_STREAM, MSGL_ERR, "realrtsp: base64 output overflow, this should never happen\n"); | |
516 goto autherr; | |
517 } | |
518 authfield[b64_authlen+21] = 0; | |
519 goto rtsp_send_describe; | |
520 } | |
521 autherr: | |
522 | |
523 if (authfield) | |
524 free(authfield); | |
525 | |
9922 | 526 if ( status<200 || status>299 ) |
527 { | |
528 char *alert=rtsp_search_answers(rtsp_session,"Alert"); | |
529 if (alert) { | |
20530 | 530 mp_msg(MSGT_STREAM, MSGL_WARN, "realrtsp: got message from server:\n%s\n", |
531 alert); | |
9922 | 532 } |
533 rtsp_send_ok(rtsp_session); | |
12266 | 534 buf = xbuffer_free(buf); |
9922 | 535 return NULL; |
536 } | |
537 | |
538 /* receive description */ | |
539 size=0; | |
540 if (!rtsp_search_answers(rtsp_session,"Content-length")) | |
20530 | 541 mp_msg(MSGT_STREAM, MSGL_WARN, "real: got no Content-length!\n"); |
9922 | 542 else |
543 size=atoi(rtsp_search_answers(rtsp_session,"Content-length")); | |
544 | |
14159 | 545 // as size is unsigned this also catches the case (size < 0) |
546 if (size > MAX_DESC_BUF) { | |
20530 | 547 mp_msg(MSGT_STREAM, MSGL_ERR, "realrtsp: Content-length for description too big (> %uMB)!\n", |
14159 | 548 MAX_DESC_BUF/(1024*1024) ); |
549 xbuffer_free(buf); | |
550 return NULL; | |
551 } | |
552 | |
9922 | 553 if (!rtsp_search_answers(rtsp_session,"ETag")) |
20530 | 554 mp_msg(MSGT_STREAM, MSGL_WARN, "realrtsp: got no ETag!\n"); |
9922 | 555 else |
556 session_id=strdup(rtsp_search_answers(rtsp_session,"ETag")); | |
557 | |
558 #ifdef LOG | |
14159 | 559 printf("real: Stream description size: %u\n", size); |
9922 | 560 #endif |
561 | |
19074
d385666efa27
removes unused parentheses lefted behind in the r19075 sizeof(char) cleanups, noticed by dalias
reynaldo
parents:
19070
diff
changeset
|
562 description=malloc(size+1); |
9922 | 563 |
12271
4adb4a3b52a2
More bounds checking fixes (thnaks to Miguel Freitas)
rtognimp
parents:
12266
diff
changeset
|
564 if( rtsp_read_data(rtsp_session, description, size) <= 0) { |
4adb4a3b52a2
More bounds checking fixes (thnaks to Miguel Freitas)
rtognimp
parents:
12266
diff
changeset
|
565 buf = xbuffer_free(buf); |
4adb4a3b52a2
More bounds checking fixes (thnaks to Miguel Freitas)
rtognimp
parents:
12266
diff
changeset
|
566 return NULL; |
4adb4a3b52a2
More bounds checking fixes (thnaks to Miguel Freitas)
rtognimp
parents:
12266
diff
changeset
|
567 } |
9922 | 568 description[size]=0; |
569 | |
570 /* parse sdp (sdpplin) and create a header and a subscribe string */ | |
12266 | 571 subscribe = xbuffer_init(256); |
9922 | 572 strcpy(subscribe, "Subscribe: "); |
12266 | 573 h=real_parse_sdp(description, &subscribe, bandwidth); |
574 if (!h) { | |
575 subscribe = xbuffer_free(subscribe); | |
576 buf = xbuffer_free(buf); | |
577 return NULL; | |
578 } | |
9922 | 579 rmff_fix_header(h); |
580 | |
581 #ifdef LOG | |
582 printf("Title: %s\nCopyright: %s\nAuthor: %s\nStreams: %i\n", | |
583 h->cont->title, h->cont->copyright, h->cont->author, h->prop->num_streams); | |
584 #endif | |
585 | |
586 /* setup our streams */ | |
587 real_calc_response_and_checksum (challenge2, checksum, challenge1); | |
12266 | 588 buf = xbuffer_ensure_size(buf, strlen(challenge2) + strlen(checksum) + 32); |
9922 | 589 sprintf(buf, "RealChallenge2: %s, sd=%s", challenge2, checksum); |
590 rtsp_schedule_field(rtsp_session, buf); | |
12266 | 591 buf = xbuffer_ensure_size(buf, strlen(session_id) + 32); |
9922 | 592 sprintf(buf, "If-Match: %s", session_id); |
593 rtsp_schedule_field(rtsp_session, buf); | |
594 rtsp_schedule_field(rtsp_session, "Transport: x-pn-tng/tcp;mode=play,rtp/avp/tcp;unicast;mode=play"); | |
12266 | 595 buf = xbuffer_ensure_size(buf, strlen(mrl) + 32); |
9922 | 596 sprintf(buf, "%s/streamid=0", mrl); |
18772
eb60c209a117
better RTSP RFC compliance (fixes in CSeq and TEARDOWN handling)
ben
parents:
17332
diff
changeset
|
597 rtsp_request_setup(rtsp_session,buf,NULL); |
9922 | 598 |
599 if (h->prop->num_streams > 1) { | |
600 rtsp_schedule_field(rtsp_session, "Transport: x-pn-tng/tcp;mode=play,rtp/avp/tcp;unicast;mode=play"); | |
12266 | 601 buf = xbuffer_ensure_size(buf, strlen(session_id) + 32); |
9922 | 602 sprintf(buf, "If-Match: %s", session_id); |
603 rtsp_schedule_field(rtsp_session, buf); | |
604 | |
12266 | 605 buf = xbuffer_ensure_size(buf, strlen(mrl) + 32); |
9922 | 606 sprintf(buf, "%s/streamid=1", mrl); |
18772
eb60c209a117
better RTSP RFC compliance (fixes in CSeq and TEARDOWN handling)
ben
parents:
17332
diff
changeset
|
607 rtsp_request_setup(rtsp_session,buf,NULL); |
9922 | 608 } |
609 /* set stream parameter (bandwidth) with our subscribe string */ | |
610 rtsp_schedule_field(rtsp_session, subscribe); | |
611 rtsp_request_setparameter(rtsp_session,NULL); | |
612 | |
17332
88adbc28f60b
This patch makes real rtsp tell the server to deliver data at specified
rtognimp
parents:
14159
diff
changeset
|
613 /* set delivery bandwidth */ |
88adbc28f60b
This patch makes real rtsp tell the server to deliver data at specified
rtognimp
parents:
14159
diff
changeset
|
614 if (maxbandwidth) { |
88adbc28f60b
This patch makes real rtsp tell the server to deliver data at specified
rtognimp
parents:
14159
diff
changeset
|
615 sprintf(buf, "SetDeliveryBandwidth: Bandwidth=%u;BackOff=0", maxbandwidth); |
88adbc28f60b
This patch makes real rtsp tell the server to deliver data at specified
rtognimp
parents:
14159
diff
changeset
|
616 rtsp_schedule_field(rtsp_session, buf); |
88adbc28f60b
This patch makes real rtsp tell the server to deliver data at specified
rtognimp
parents:
14159
diff
changeset
|
617 rtsp_request_setparameter(rtsp_session,NULL); |
88adbc28f60b
This patch makes real rtsp tell the server to deliver data at specified
rtognimp
parents:
14159
diff
changeset
|
618 } |
88adbc28f60b
This patch makes real rtsp tell the server to deliver data at specified
rtognimp
parents:
14159
diff
changeset
|
619 |
11506
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
620 { |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
621 int s_ss = 0, s_ms = 0, e_ss = 0, e_ms = 0; |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
622 char *str; |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
623 if ((str = rtsp_get_param(rtsp_session, "start"))) { |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
624 convert_timestamp(str, &s_ss, &s_ms); |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
625 free(str); |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
626 } |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
627 if ((str = rtsp_get_param(rtsp_session, "end"))) { |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
628 convert_timestamp(str, &e_ss, &e_ms); |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
629 free(str); |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
630 } |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
631 str = buf + sprintf(buf, s_ms ? "%s%d.%d-" : "%s%d-", "Range: npt=", s_ss, s_ms); |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
632 if (e_ss || e_ms) |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
633 sprintf(str, e_ms ? "%d.%d" : "%d", e_ss, e_ms); |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
634 } |
fff1b6f1a9cc
Real rtsp Range parameter (Start and End) support.
rtognimp
parents:
10115
diff
changeset
|
635 rtsp_schedule_field(rtsp_session, buf); |
9922 | 636 /* and finally send a play request */ |
637 rtsp_request_play(rtsp_session,NULL); | |
638 | |
12266 | 639 subscribe = xbuffer_free(subscribe); |
640 buf = xbuffer_free(buf); | |
9922 | 641 return h; |
642 } | |
18792
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
643 |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
644 struct real_rtsp_session_t * |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
645 init_real_rtsp_session (void) |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
646 { |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
647 struct real_rtsp_session_t *real_rtsp_session = NULL; |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
648 |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
649 real_rtsp_session = malloc (sizeof (struct real_rtsp_session_t)); |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
650 real_rtsp_session->recv = xbuffer_init (BUF_SIZE); |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
651 |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
652 return real_rtsp_session; |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
653 } |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
654 |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
655 void |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
656 free_real_rtsp_session (struct real_rtsp_session_t* real_session) |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
657 { |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
658 if (!real_session) |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
659 return; |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
660 |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
661 xbuffer_free (real_session->recv); |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
662 free (real_session); |
6a015ba5bf45
move real_rtsp init/uninit code to its dedicated place to simplify a bit rtsp session demuxer
ben
parents:
18772
diff
changeset
|
663 } |