2 * RTMP network protocol
3 * Copyright (c) 2009 Kostya Shishkov
5 * This file is part of FFmpeg.
7 * FFmpeg is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
12 * FFmpeg is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with FFmpeg; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
23 * @file libavformat/rtmpproto.c
27 #include "libavcodec/bytestream.h"
28 #include "libavutil/avstring.h"
29 #include "libavutil/lfg.h"
30 #include "libavutil/sha.h"
39 /* we can't use av_log() with URLContext yet... */
40 #if LIBAVFORMAT_VERSION_MAJOR < 53
41 #define LOG_CONTEXT NULL
46 /** RTMP protocol handler state */
48 STATE_START
, ///< client has not done anything yet
49 STATE_HANDSHAKED
, ///< client has performed handshake
50 STATE_CONNECTING
, ///< client connected to server successfully
51 STATE_READY
, ///< client has sent all needed commands and waits for server reply
52 STATE_PLAYING
, ///< client has started receiving multimedia data from server
55 /** protocol handler context */
56 typedef struct RTMPContext
{
57 URLContext
* stream
; ///< TCP stream used in interactions with RTMP server
58 RTMPPacket prev_pkt
[2][RTMP_CHANNELS
]; ///< packet history used when reading and sending packets
59 int chunk_size
; ///< size of the chunks RTMP packets are divided into
60 char playpath
[256]; ///< path to filename to play (with possible "mp4:" prefix)
61 ClientState state
; ///< current state
62 int main_channel_id
; ///< an additional channel ID which is used for some invocations
63 uint8_t* flv_data
; ///< buffer with data for demuxer
64 int flv_size
; ///< current buffer size
65 int flv_off
; ///< number of bytes read from current buffer
66 uint32_t video_ts
; ///< current video timestamp in milliseconds
67 uint32_t audio_ts
; ///< current audio timestamp in milliseconds
70 #define PLAYER_KEY_OPEN_PART_LEN 30 ///< length of partial key used for first client digest signing
71 /** Client key used for digest signing */
72 static const uint8_t rtmp_player_key
[] = {
73 'G', 'e', 'n', 'u', 'i', 'n', 'e', ' ', 'A', 'd', 'o', 'b', 'e', ' ',
74 'F', 'l', 'a', 's', 'h', ' ', 'P', 'l', 'a', 'y', 'e', 'r', ' ', '0', '0', '1',
76 0xF0, 0xEE, 0xC2, 0x4A, 0x80, 0x68, 0xBE, 0xE8, 0x2E, 0x00, 0xD0, 0xD1, 0x02,
77 0x9E, 0x7E, 0x57, 0x6E, 0xEC, 0x5D, 0x2D, 0x29, 0x80, 0x6F, 0xAB, 0x93, 0xB8,
78 0xE6, 0x36, 0xCF, 0xEB, 0x31, 0xAE
81 #define SERVER_KEY_OPEN_PART_LEN 36 ///< length of partial key used for first server digest signing
82 /** Key used for RTMP server digest signing */
83 static const uint8_t rtmp_server_key
[] = {
84 'G', 'e', 'n', 'u', 'i', 'n', 'e', ' ', 'A', 'd', 'o', 'b', 'e', ' ',
85 'F', 'l', 'a', 's', 'h', ' ', 'M', 'e', 'd', 'i', 'a', ' ',
86 'S', 'e', 'r', 'v', 'e', 'r', ' ', '0', '0', '1',
88 0xF0, 0xEE, 0xC2, 0x4A, 0x80, 0x68, 0xBE, 0xE8, 0x2E, 0x00, 0xD0, 0xD1, 0x02,
89 0x9E, 0x7E, 0x57, 0x6E, 0xEC, 0x5D, 0x2D, 0x29, 0x80, 0x6F, 0xAB, 0x93, 0xB8,
90 0xE6, 0x36, 0xCF, 0xEB, 0x31, 0xAE
94 * Generates 'connect' call and sends it to the server.
96 static void gen_connect(URLContext
*s
, RTMPContext
*rt
, const char *proto
,
97 const char *host
, int port
, const char *app
)
103 ff_rtmp_packet_create(&pkt
, RTMP_VIDEO_CHANNEL
, RTMP_PT_INVOKE
, 0, 4096);
106 snprintf(tcurl
, sizeof(tcurl
), "%s://%s:%d/%s", proto
, host
, port
, app
);
107 ff_amf_write_string(&p
, "connect");
108 ff_amf_write_number(&p
, 1.0);
109 ff_amf_write_object_start(&p
);
110 ff_amf_write_field_name(&p
, "app");
111 ff_amf_write_string(&p
, app
);
113 snprintf(ver
, sizeof(ver
), "%s %d,%d,%d,%d", RTMP_CLIENT_PLATFORM
, RTMP_CLIENT_VER1
,
114 RTMP_CLIENT_VER2
, RTMP_CLIENT_VER3
, RTMP_CLIENT_VER4
);
115 ff_amf_write_field_name(&p
, "flashVer");
116 ff_amf_write_string(&p
, ver
);
117 ff_amf_write_field_name(&p
, "tcUrl");
118 ff_amf_write_string(&p
, tcurl
);
119 ff_amf_write_field_name(&p
, "fpad");
120 ff_amf_write_bool(&p
, 0);
121 ff_amf_write_field_name(&p
, "capabilities");
122 ff_amf_write_number(&p
, 15.0);
123 ff_amf_write_field_name(&p
, "audioCodecs");
124 ff_amf_write_number(&p
, 1639.0);
125 ff_amf_write_field_name(&p
, "videoCodecs");
126 ff_amf_write_number(&p
, 252.0);
127 ff_amf_write_field_name(&p
, "videoFunction");
128 ff_amf_write_number(&p
, 1.0);
129 ff_amf_write_object_end(&p
);
131 pkt
.data_size
= p
- pkt
.data
;
133 ff_rtmp_packet_write(rt
->stream
, &pkt
, rt
->chunk_size
, rt
->prev_pkt
[1]);
137 * Generates 'createStream' call and sends it to the server. It should make
138 * the server allocate some channel for media streams.
140 static void gen_create_stream(URLContext
*s
, RTMPContext
*rt
)
145 av_log(LOG_CONTEXT
, AV_LOG_DEBUG
, "Creating stream...\n");
146 ff_rtmp_packet_create(&pkt
, RTMP_VIDEO_CHANNEL
, RTMP_PT_INVOKE
, 0, 25);
149 ff_amf_write_string(&p
, "createStream");
150 ff_amf_write_number(&p
, 3.0);
151 ff_amf_write_null(&p
);
153 ff_rtmp_packet_write(rt
->stream
, &pkt
, rt
->chunk_size
, rt
->prev_pkt
[1]);
154 ff_rtmp_packet_destroy(&pkt
);
158 * Generates 'play' call and sends it to the server, then pings the server
159 * to start actual playing.
161 static void gen_play(URLContext
*s
, RTMPContext
*rt
)
166 av_log(LOG_CONTEXT
, AV_LOG_DEBUG
, "Sending play command for '%s'\n", rt
->playpath
);
167 ff_rtmp_packet_create(&pkt
, RTMP_VIDEO_CHANNEL
, RTMP_PT_INVOKE
, 0,
168 29 + strlen(rt
->playpath
));
169 pkt
.extra
= rt
->main_channel_id
;
172 ff_amf_write_string(&p
, "play");
173 ff_amf_write_number(&p
, 0.0);
174 ff_amf_write_null(&p
);
175 ff_amf_write_string(&p
, rt
->playpath
);
176 ff_amf_write_number(&p
, 0.0);
178 ff_rtmp_packet_write(rt
->stream
, &pkt
, rt
->chunk_size
, rt
->prev_pkt
[1]);
179 ff_rtmp_packet_destroy(&pkt
);
181 // set client buffer time disguised in ping packet
182 ff_rtmp_packet_create(&pkt
, RTMP_NETWORK_CHANNEL
, RTMP_PT_PING
, 1, 10);
185 bytestream_put_be16(&p
, 3);
186 bytestream_put_be32(&p
, 1);
187 bytestream_put_be32(&p
, 256); //TODO: what is a good value here?
189 ff_rtmp_packet_write(rt
->stream
, &pkt
, rt
->chunk_size
, rt
->prev_pkt
[1]);
190 ff_rtmp_packet_destroy(&pkt
);
194 * Generates ping reply and sends it to the server.
196 static void gen_pong(URLContext
*s
, RTMPContext
*rt
, RTMPPacket
*ppkt
)
201 ff_rtmp_packet_create(&pkt
, RTMP_NETWORK_CHANNEL
, RTMP_PT_PING
, ppkt
->timestamp
+ 1, 6);
203 bytestream_put_be16(&p
, 7);
204 bytestream_put_be32(&p
, AV_RB32(ppkt
->data
+2) + 1);
205 ff_rtmp_packet_write(rt
->stream
, &pkt
, rt
->chunk_size
, rt
->prev_pkt
[1]);
206 ff_rtmp_packet_destroy(&pkt
);
209 //TODO: Move HMAC code somewhere. Eventually.
210 #define HMAC_IPAD_VAL 0x36
211 #define HMAC_OPAD_VAL 0x5C
214 * Calculates HMAC-SHA2 digest for RTMP handshake packets.
216 * @param src input buffer
217 * @param len input buffer length (should be 1536)
218 * @param gap offset in buffer where 32 bytes should not be taken into account
219 * when calculating digest (since it will be used to store that digest)
220 * @param key digest key
221 * @param keylen digest key length
222 * @param dst buffer where calculated digest will be stored (32 bytes)
224 static void rtmp_calc_digest(const uint8_t *src
, int len
, int gap
,
225 const uint8_t *key
, int keylen
, uint8_t *dst
)
228 uint8_t hmac_buf
[64+32] = {0};
231 sha
= av_mallocz(av_sha_size
);
234 memcpy(hmac_buf
, key
, keylen
);
236 av_sha_init(sha
, 256);
237 av_sha_update(sha
,key
, keylen
);
238 av_sha_final(sha
, hmac_buf
);
240 for (i
= 0; i
< 64; i
++)
241 hmac_buf
[i
] ^= HMAC_IPAD_VAL
;
243 av_sha_init(sha
, 256);
244 av_sha_update(sha
, hmac_buf
, 64);
246 av_sha_update(sha
, src
, len
);
247 } else { //skip 32 bytes used for storing digest
248 av_sha_update(sha
, src
, gap
);
249 av_sha_update(sha
, src
+ gap
+ 32, len
- gap
- 32);
251 av_sha_final(sha
, hmac_buf
+ 64);
253 for (i
= 0; i
< 64; i
++)
254 hmac_buf
[i
] ^= HMAC_IPAD_VAL
^ HMAC_OPAD_VAL
; //reuse XORed key for opad
255 av_sha_init(sha
, 256);
256 av_sha_update(sha
, hmac_buf
, 64+32);
257 av_sha_final(sha
, dst
);
263 * Puts HMAC-SHA2 digest of packet data (except for the bytes where this digest
264 * will be stored) into that packet.
266 * @param buf handshake data (1536 bytes)
267 * @return offset to the digest inside input data
269 static int rtmp_handshake_imprint_with_digest(uint8_t *buf
)
271 int i
, digest_pos
= 0;
273 for (i
= 8; i
< 12; i
++)
274 digest_pos
+= buf
[i
];
275 digest_pos
= (digest_pos
% 728) + 12;
277 rtmp_calc_digest(buf
, RTMP_HANDSHAKE_PACKET_SIZE
, digest_pos
,
278 rtmp_player_key
, PLAYER_KEY_OPEN_PART_LEN
,
284 * Verifies that the received server response has the expected digest value.
286 * @param buf handshake data received from the server (1536 bytes)
287 * @param off position to search digest offset from
288 * @return 0 if digest is valid, digest position otherwise
290 static int rtmp_validate_digest(uint8_t *buf
, int off
)
292 int i
, digest_pos
= 0;
295 for (i
= 0; i
< 4; i
++)
296 digest_pos
+= buf
[i
+ off
];
297 digest_pos
= (digest_pos
% 728) + off
+ 4;
299 rtmp_calc_digest(buf
, RTMP_HANDSHAKE_PACKET_SIZE
, digest_pos
,
300 rtmp_server_key
, SERVER_KEY_OPEN_PART_LEN
,
302 if (!memcmp(digest
, buf
+ digest_pos
, 32))
308 * Performs handshake with the server by means of exchanging pseudorandom data
309 * signed with HMAC-SHA2 digest.
311 * @return 0 if handshake succeeds, negative value otherwise
313 static int rtmp_handshake(URLContext
*s
, RTMPContext
*rt
)
316 uint8_t tosend
[RTMP_HANDSHAKE_PACKET_SIZE
+1] = {
317 3, // unencrypted data
318 0, 0, 0, 0, // client uptime
324 uint8_t clientdata
[RTMP_HANDSHAKE_PACKET_SIZE
];
325 uint8_t serverdata
[RTMP_HANDSHAKE_PACKET_SIZE
+1];
327 int server_pos
, client_pos
;
330 av_log(LOG_CONTEXT
, AV_LOG_DEBUG
, "Handshaking...\n");
332 av_lfg_init(&rnd
, 0xDEADC0DE);
333 // generate handshake packet - 1536 bytes of pseudorandom data
334 for (i
= 9; i
<= RTMP_HANDSHAKE_PACKET_SIZE
; i
++)
335 tosend
[i
] = av_lfg_get(&rnd
) >> 24;
336 client_pos
= rtmp_handshake_imprint_with_digest(tosend
+ 1);
338 url_write(rt
->stream
, tosend
, RTMP_HANDSHAKE_PACKET_SIZE
+ 1);
339 i
= url_read_complete(rt
->stream
, serverdata
, RTMP_HANDSHAKE_PACKET_SIZE
+ 1);
340 if (i
!= RTMP_HANDSHAKE_PACKET_SIZE
+ 1) {
341 av_log(LOG_CONTEXT
, AV_LOG_ERROR
, "Cannot read RTMP handshake response\n");
344 i
= url_read_complete(rt
->stream
, clientdata
, RTMP_HANDSHAKE_PACKET_SIZE
);
345 if (i
!= RTMP_HANDSHAKE_PACKET_SIZE
) {
346 av_log(LOG_CONTEXT
, AV_LOG_ERROR
, "Cannot read RTMP handshake response\n");
350 av_log(LOG_CONTEXT
, AV_LOG_DEBUG
, "Server version %d.%d.%d.%d\n",
351 serverdata
[5], serverdata
[6], serverdata
[7], serverdata
[8]);
353 server_pos
= rtmp_validate_digest(serverdata
+ 1, 772);
355 server_pos
= rtmp_validate_digest(serverdata
+ 1, 8);
357 av_log(LOG_CONTEXT
, AV_LOG_ERROR
, "Server response validating failed\n");
362 rtmp_calc_digest(tosend
+ 1 + client_pos
, 32, 0,
363 rtmp_server_key
, sizeof(rtmp_server_key
),
365 rtmp_calc_digest(clientdata
, RTMP_HANDSHAKE_PACKET_SIZE
-32, 0,
368 if (memcmp(digest
, clientdata
+ RTMP_HANDSHAKE_PACKET_SIZE
- 32, 32)) {
369 av_log(LOG_CONTEXT
, AV_LOG_ERROR
, "Signature mismatch\n");
373 for (i
= 0; i
< RTMP_HANDSHAKE_PACKET_SIZE
; i
++)
374 tosend
[i
] = av_lfg_get(&rnd
) >> 24;
375 rtmp_calc_digest(serverdata
+ 1 + server_pos
, 32, 0,
376 rtmp_player_key
, sizeof(rtmp_player_key
),
378 rtmp_calc_digest(tosend
, RTMP_HANDSHAKE_PACKET_SIZE
- 32, 0,
380 tosend
+ RTMP_HANDSHAKE_PACKET_SIZE
- 32);
382 // write reply back to the server
383 url_write(rt
->stream
, tosend
, RTMP_HANDSHAKE_PACKET_SIZE
);
388 * Parses received packet and may perform some action depending on
389 * the packet contents.
390 * @return 0 for no errors, negative values for serious errors which prevent
391 * further communications, positive values for uncritical errors
393 static int rtmp_parse_result(URLContext
*s
, RTMPContext
*rt
, RTMPPacket
*pkt
)
396 const uint8_t *data_end
= pkt
->data
+ pkt
->data_size
;
399 case RTMP_PT_CHUNK_SIZE
:
400 if (pkt
->data_size
!= 4) {
401 av_log(LOG_CONTEXT
, AV_LOG_ERROR
,
402 "Chunk size change packet is not 4 bytes long (%d)\n", pkt
->data_size
);
405 rt
->chunk_size
= AV_RB32(pkt
->data
);
406 if (rt
->chunk_size
<= 0) {
407 av_log(LOG_CONTEXT
, AV_LOG_ERROR
, "Incorrect chunk size %d\n", rt
->chunk_size
);
410 av_log(LOG_CONTEXT
, AV_LOG_DEBUG
, "New chunk size = %d\n", rt
->chunk_size
);
413 t
= AV_RB16(pkt
->data
);
415 gen_pong(s
, rt
, pkt
);
418 //TODO: check for the messages sent for wrong state?
419 if (!memcmp(pkt
->data
, "\002\000\006_error", 9)) {
422 if (!ff_amf_get_field_value(pkt
->data
+ 9, data_end
,
423 "description", tmpstr
, sizeof(tmpstr
)))
424 av_log(LOG_CONTEXT
, AV_LOG_ERROR
, "Server error: %s\n",tmpstr
);
426 } else if (!memcmp(pkt
->data
, "\002\000\007_result", 10)) {
428 case STATE_HANDSHAKED
:
429 gen_create_stream(s
, rt
);
430 rt
->state
= STATE_CONNECTING
;
432 case STATE_CONNECTING
:
433 //extract a number from the result
434 if (pkt
->data
[10] || pkt
->data
[19] != 5 || pkt
->data
[20]) {
435 av_log(LOG_CONTEXT
, AV_LOG_WARNING
, "Unexpected reply on connect()\n");
437 rt
->main_channel_id
= (int) av_int2dbl(AV_RB64(pkt
->data
+ 21));
440 rt
->state
= STATE_READY
;
443 } else if (!memcmp(pkt
->data
, "\002\000\010onStatus", 11)) {
444 const uint8_t* ptr
= pkt
->data
+ 11;
448 for (i
= 0; i
< 2; i
++) {
449 t
= ff_amf_tag_size(ptr
, data_end
);
454 t
= ff_amf_get_field_value(ptr
, data_end
,
455 "level", tmpstr
, sizeof(tmpstr
));
456 if (!t
&& !strcmp(tmpstr
, "error")) {
457 if (!ff_amf_get_field_value(ptr
, data_end
,
458 "description", tmpstr
, sizeof(tmpstr
)))
459 av_log(LOG_CONTEXT
, AV_LOG_ERROR
, "Server error: %s\n",tmpstr
);
462 t
= ff_amf_get_field_value(ptr
, data_end
,
463 "code", tmpstr
, sizeof(tmpstr
));
464 if (!t
&& !strcmp(tmpstr
, "NetStream.Play.Start")) {
465 rt
->state
= STATE_PLAYING
;
475 * Interacts with the server by receiving and sending RTMP packets until
476 * there is some significant data (media data or expected status notification).
478 * @param s reading context
479 * @param for_header non-zero value tells function to work until it gets notification from the server that playing has been started, otherwise function will work until some media data is received (or an error happens)
480 * @return 0 for successful operation, negative value in case of error
482 static int get_packet(URLContext
*s
, int for_header
)
484 RTMPContext
*rt
= s
->priv_data
;
489 if ((ret
= ff_rtmp_packet_read(rt
->stream
, &rpkt
,
490 rt
->chunk_size
, rt
->prev_pkt
[0])) != 0) {
492 return AVERROR(EAGAIN
);
498 ret
= rtmp_parse_result(s
, rt
, &rpkt
);
499 if (ret
< 0) {//serious error in current packet
500 ff_rtmp_packet_destroy(&rpkt
);
503 if (for_header
&& rt
->state
== STATE_PLAYING
) {
504 ff_rtmp_packet_destroy(&rpkt
);
507 if (!rpkt
.data_size
) {
508 ff_rtmp_packet_destroy(&rpkt
);
511 if (rpkt
.type
== RTMP_PT_VIDEO
|| rpkt
.type
== RTMP_PT_AUDIO
||
512 rpkt
.type
== RTMP_PT_NOTIFY
) {
514 uint32_t ts
= rpkt
.timestamp
;
516 if (rpkt
.type
== RTMP_PT_VIDEO
) {
517 rt
->video_ts
+= rpkt
.timestamp
;
519 } else if (rpkt
.type
== RTMP_PT_AUDIO
) {
520 rt
->audio_ts
+= rpkt
.timestamp
;
523 // generate packet header and put data into buffer for FLV demuxer
525 rt
->flv_size
= rpkt
.data_size
+ 15;
526 rt
->flv_data
= p
= av_realloc(rt
->flv_data
, rt
->flv_size
);
527 bytestream_put_byte(&p
, rpkt
.type
);
528 bytestream_put_be24(&p
, rpkt
.data_size
);
529 bytestream_put_be24(&p
, ts
);
530 bytestream_put_byte(&p
, ts
>> 24);
531 bytestream_put_be24(&p
, 0);
532 bytestream_put_buffer(&p
, rpkt
.data
, rpkt
.data_size
);
533 bytestream_put_be32(&p
, 0);
534 ff_rtmp_packet_destroy(&rpkt
);
536 } else if (rpkt
.type
== RTMP_PT_METADATA
) {
537 // we got raw FLV data, make it available for FLV demuxer
539 rt
->flv_size
= rpkt
.data_size
;
540 rt
->flv_data
= av_realloc(rt
->flv_data
, rt
->flv_size
);
541 memcpy(rt
->flv_data
, rpkt
.data
, rpkt
.data_size
);
542 ff_rtmp_packet_destroy(&rpkt
);
545 ff_rtmp_packet_destroy(&rpkt
);
550 static int rtmp_close(URLContext
*h
)
552 RTMPContext
*rt
= h
->priv_data
;
554 av_freep(&rt
->flv_data
);
555 url_close(rt
->stream
);
561 * Opens RTMP connection and verifies that the stream can be played.
563 * URL syntax: rtmp://server[:port][/app][/playpath]
564 * where 'app' is first one or two directories in the path
565 * (e.g. /ondemand/, /flash/live/, etc.)
566 * and 'playpath' is a file name (the rest of the path,
567 * may be prefixed with "mp4:")
569 static int rtmp_open(URLContext
*s
, const char *uri
, int flags
)
572 char proto
[8], hostname
[256], path
[1024], app
[128], *fname
;
577 is_input
= !(flags
& URL_WRONLY
);
579 rt
= av_mallocz(sizeof(RTMPContext
));
581 return AVERROR(ENOMEM
);
584 url_split(proto
, sizeof(proto
), NULL
, 0, hostname
, sizeof(hostname
), &port
,
585 path
, sizeof(path
), s
->filename
);
588 port
= RTMP_DEFAULT_PORT
;
589 snprintf(buf
, sizeof(buf
), "tcp://%s:%d", hostname
, port
);
591 if (url_open(&rt
->stream
, buf
, URL_RDWR
) < 0)
595 av_log(LOG_CONTEXT
, AV_LOG_ERROR
, "RTMP output is not supported yet.\n");
598 rt
->state
= STATE_START
;
599 if (rtmp_handshake(s
, rt
))
602 rt
->chunk_size
= 128;
603 rt
->state
= STATE_HANDSHAKED
;
604 //extract "app" part from path
605 if (!strncmp(path
, "/ondemand/", 10)) {
607 memcpy(app
, "ondemand", 9);
609 char *p
= strchr(path
+ 1, '/');
614 fname
= strchr(p
+ 1, '/');
617 av_strlcpy(app
, path
+ 1, p
- path
);
620 av_strlcpy(app
, path
+ 1, fname
- path
- 1);
624 if (!strcmp(fname
+ strlen(fname
) - 4, ".f4v") ||
625 !strcmp(fname
+ strlen(fname
) - 4, ".mp4")) {
626 memcpy(rt
->playpath
, "mp4:", 5);
630 strncat(rt
->playpath
, fname
, sizeof(rt
->playpath
) - 5);
632 av_log(LOG_CONTEXT
, AV_LOG_DEBUG
, "Proto = %s, path = %s, app = %s, fname = %s\n",
633 proto
, path
, app
, rt
->playpath
);
634 gen_connect(s
, rt
, proto
, hostname
, port
, app
);
637 ret
= get_packet(s
, 1);
638 } while (ret
== EAGAIN
);
641 // generate FLV header for demuxer
643 rt
->flv_data
= av_realloc(rt
->flv_data
, rt
->flv_size
);
645 memcpy(rt
->flv_data
, "FLV\1\5\0\0\0\011\0\0\0\0", rt
->flv_size
);
648 s
->max_packet_size
= url_get_max_packet_size(rt
->stream
);
657 static int rtmp_read(URLContext
*s
, uint8_t *buf
, int size
)
659 RTMPContext
*rt
= s
->priv_data
;
660 int orig_size
= size
;
664 int data_left
= rt
->flv_size
- rt
->flv_off
;
666 if (data_left
>= size
) {
667 memcpy(buf
, rt
->flv_data
+ rt
->flv_off
, size
);
672 memcpy(buf
, rt
->flv_data
+ rt
->flv_off
, data_left
);
675 rt
->flv_off
= rt
->flv_size
;
677 if ((ret
= get_packet(s
, 0)) < 0)
683 static int rtmp_write(URLContext
*h
, uint8_t *buf
, int size
)
688 URLProtocol rtmp_protocol
= {