34 bytestream_put_byte(dst, val);
46 bytestream_put_be16(dst, strlen(str));
52 int len1 = 0, len2 = 0;
58 bytestream_put_be16(dst, len1 + len2);
75 bytestream_put_be16(dst, strlen(str));
91 *val = bytestream2_get_byte(bc);
100 read = bytestream2_get_be64(bc);
110 stringlen = bytestream2_get_be16(bc);
111 if (stringlen + 1 > strsize)
114 if (readsize != stringlen) {
116 "Unable to read as many bytes as AMF string signaled\n");
118 str[readsize] =
'\0';
119 *length =
FFMIN(stringlen, readsize);
143 if (channel < *nb_prev_pkt)
146 nb_alloc = channel + 16;
153 memset(ptr + *nb_prev_pkt, 0, (nb_alloc - *nb_prev_pkt) *
sizeof(*ptr));
155 *nb_prev_pkt = nb_alloc;
160 int chunk_size,
RTMPPacket **prev_pkt,
int *nb_prev_pkt)
177 int channel_id, timestamp,
size;
186 channel_id = hdr & 0x3F;
188 if (channel_id < 2) {
192 written += channel_id + 1;
193 channel_id =
AV_RL16(buf) + 64;
198 prev_pkt = *prev_pkt_ptr;
199 size = prev_pkt[channel_id].
size;
200 type = prev_pkt[channel_id].
type;
201 extra = prev_pkt[channel_id].
extra;
205 ts_field = prev_pkt[channel_id].
ts_field;
228 if (ts_field == 0xFFFFFF) {
233 timestamp = ts_field;
236 timestamp += prev_pkt[channel_id].
timestamp;
238 if (!prev_pkt[channel_id].read) {
244 prev_pkt[channel_id].
ts_field = ts_field;
245 prev_pkt[channel_id].
timestamp = timestamp;
265 prev_pkt[channel_id].
extra = extra;
268 toread =
FFMIN(size, chunk_size);
286 prev_pkt[channel_id].
read = 0;
297 if (ret > 0 || ret !=
AVERROR(EAGAIN))
309 uint8_t pkt_hdr[16], *p = pkt_hdr;
321 prev_pkt = *prev_pkt_ptr;
332 if (timestamp >= 0xFFFFFF) {
350 bytestream_put_byte(&p, pkt->
channel_id | (mode << 6));
352 bytestream_put_byte(&p, 0 | (mode << 6));
353 bytestream_put_byte(&p, pkt->
channel_id - 64);
355 bytestream_put_byte(&p, 1 | (mode << 6));
356 bytestream_put_le16(&p, pkt->
channel_id - 64);
359 bytestream_put_be24(&p, pkt->
ts_field);
361 bytestream_put_be24(&p, pkt->
size);
362 bytestream_put_byte(&p, pkt->
type);
364 bytestream_put_le32(&p, pkt->
extra);
368 bytestream_put_be32(&p, timestamp);
377 if ((ret =
ffurl_write(h, pkt_hdr, p - pkt_hdr)) < 0)
379 written = p - pkt_hdr + pkt->
size;
380 while (off < pkt->
size) {
381 int towrite =
FFMIN(chunk_size, pkt->
size - off);
385 if (off < pkt->size) {
403 int timestamp,
int size)
435 if (data >= data_end)
437 switch ((type = *data++)) {
446 nb = bytestream_get_be32(&data);
451 int size = bytestream_get_be16(&data);
461 if (t < 0 || t >= data_end - data)
474 int namelen = strlen(name);
480 len = data_end -
data;
483 if (data_end - data < 3)
487 int size = bytestream_get_be16(&data);
493 if (size == namelen && !memcmp(data-size, name, namelen)) {
499 snprintf(dst, dst_size,
"%s", *data ?
"true" :
"false");
502 len = bytestream_get_be16(&data);
511 if (len < 0 || len >= data_end - data)
535 default:
return "unknown";
542 unsigned int size, nb = -1;
547 if (data >= data_end)
549 switch ((type = *data++)) {
559 size = bytestream_get_be16(&data);
561 size = bytestream_get_be32(&data);
563 size =
FFMIN(size,
sizeof(buf) - 1);
564 memcpy(buf, data, size);
574 nb = bytestream_get_be32(&data);
580 size = bytestream_get_be16(&data);
581 size =
FFMIN(size,
sizeof(buf) - 1);
587 memcpy(buf, data, size);
589 if (size >= data_end - data)
596 if (t < 0 || t >= data_end - data)
611 av_log(ctx,
AV_LOG_DEBUG,
"RTMP packet type '%s'(%d) for channel %d, timestamp %d, extra field %d size %d\n",
615 while (src < src_end) {
629 for (i = 0; i < p->
size; i++)
637 int len = strlen(str);
650 if ((size -= 4 + 1) < 0)
652 amf_len = bytestream_get_be32(&data);
654 if ((size -= 2 + 1) < 0)
656 amf_len = bytestream_get_be16(&data);
665 return !memcmp(data, str, len);
int ff_amf_match_string(const uint8_t *data, int size, const char *str)
Match AMF string with a NULL-terminated string.
int ff_rtmp_packet_read_internal(URLContext *h, RTMPPacket *p, int chunk_size, RTMPPacket **prev_pkt, int *nb_prev_pkt, uint8_t hdr)
Read internal RTMP packet sent by the server.
const char const char void * val
#define AVERROR_INVALIDDATA
Invalid data found when processing input.
int ff_amf_read_null(GetByteContext *bc)
Read AMF NULL value.
ptrdiff_t const GLvoid * data
int ff_rtmp_packet_read(URLContext *h, RTMPPacket *p, int chunk_size, RTMPPacket **prev_pkt, int *nb_prev_pkt)
Read RTMP packet sent by the server.
#define AV_LOG_WARNING
Something somehow does not look correct.
int ffurl_write(URLContext *h, const unsigned char *buf, int size)
Write size bytes from buf to the resource accessed by h.
void ff_amf_write_field_name(uint8_t **dst, const char *str)
Write string used as field name in AMF object to buffer.
int ff_amf_tag_size(const uint8_t *data, const uint8_t *data_end)
Calculate number of bytes taken by first AMF entry in data.
static av_always_inline uint64_t av_double2int(double f)
Reinterpret a double as a 64-bit integer.
uint64_t_TMPL AV_WL64 unsigned int_TMPL AV_WL32 unsigned int_TMPL AV_WL24 unsigned int_TMPL AV_WL16 uint64_t_TMPL AV_WB64 unsigned int_TMPL AV_WB32 unsigned int_TMPL AV_WB24 unsigned int_TMPL AV_RB16
uint64_t_TMPL AV_WL64 unsigned int_TMPL AV_WL32 unsigned int_TMPL AV_WL24 unsigned int_TMPL AV_RL16
RTMPPacketType type
packet payload type
static void amf_tag_contents(void *ctx, const uint8_t *data, const uint8_t *data_end)
static av_always_inline double av_int2double(uint64_t i)
Reinterpret a 64-bit integer as a double.
int read
amount read, including headers
uint32_t extra
probably an additional channel ID used during streaming data
uint64_t_TMPL AV_WL64 unsigned int_TMPL AV_WL32 unsigned int_TMPL AV_WL24 unsigned int_TMPL AV_WL16 uint64_t_TMPL AV_WB64 unsigned int_TMPL AV_RB32
void ff_amf_write_string(uint8_t **dst, const char *str)
Write string in AMF format to buffer.
void ff_rtmp_packet_dump(void *ctx, RTMPPacket *p)
Print information and contents of RTMP packet.
void ff_amf_write_object_end(uint8_t **dst)
Write marker for end of AMF object to buffer.
int size
packet payload size
void ff_amf_write_bool(uint8_t **dst, int val)
Write boolean value in AMF format to buffer.
static av_always_inline unsigned int bytestream2_get_buffer(GetByteContext *g, uint8_t *dst, unsigned int size)
#define AV_LOG_DEBUG
Stuff which is only useful for libav* developers.
int ff_amf_get_string(GetByteContext *bc, uint8_t *str, int strsize, int *length)
Get AMF string value.
packet has 12-byte header
size_t av_strlcpy(char *dst, const char *src, size_t size)
Copy the string src to dst, but no more than size - 1 bytes, and null-terminate dst.
static int rtmp_packet_read_one_chunk(URLContext *h, RTMPPacket *p, int chunk_size, RTMPPacket **prev_pkt_ptr, int *nb_prev_pkt, uint8_t hdr)
void ff_rtmp_packet_destroy(RTMPPacket *pkt)
Free RTMP packet.
RTMPPacketType
known RTMP packet types
int ff_amf_get_field_value(const uint8_t *data, const uint8_t *data_end, const uint8_t *name, uint8_t *dst, int dst_size)
Retrieve value of given AMF object field in string form.
int ff_rtmp_packet_write(URLContext *h, RTMPPacket *pkt, int chunk_size, RTMPPacket **prev_pkt_ptr, int *nb_prev_pkt)
Send RTMP packet to the server.
uint64_t_TMPL AV_WL64 unsigned int_TMPL AV_WL32 unsigned int_TMPL AV_WL24 unsigned int_TMPL AV_WL16 uint64_t_TMPL AV_WB64 unsigned int_TMPL AV_WB32 unsigned int_TMPL AV_RB24
uint32_t ts_field
24-bit timestamp or increment to the previous one, in milliseconds (latter only for media packets)...
int ff_rtmp_packet_create(RTMPPacket *pkt, int channel_id, RTMPPacketType type, int timestamp, int size)
Create new RTMP packet with given attributes.
int ff_amf_read_number(GetByteContext *bc, double *val)
Read AMF number value.
int channel_id
RTMP channel ID (nothing to do with audio/video channels though)
int ff_amf_read_bool(GetByteContext *bc, int *val)
Read AMF boolean value.
void ff_amf_write_null(uint8_t **dst)
Write AMF NULL value to buffer.
uint32_t timestamp
packet full timestamp
uint8_t * data
packet payload
static const char * rtmp_packet_type(int type)
int offset
amount of data read so far
uint64_t_TMPL AV_WL64 unsigned int_TMPL AV_WL32 unsigned int_TMPL AV_WL24 unsigned int_TMPL AV_WL16 uint64_t_TMPL AV_RB64
void * av_realloc(void *ptr, size_t size)
Allocate or reallocate a block of memory.
int ffurl_read_complete(URLContext *h, unsigned char *buf, int size)
Read as many bytes as possible (up to size), calling the read function multiple times if necessary...
int ff_rtmp_check_alloc_array(RTMPPacket **prev_pkt, int *nb_prev_pkt, int channel)
Enlarge the prev_pkt array to fit the given channel.
void * av_realloc_array(void *ptr, size_t nmemb, size_t size)
int ff_amf_read_string(GetByteContext *bc, uint8_t *str, int strsize, int *length)
Read AMF string value.
static av_always_inline void bytestream_put_buffer(uint8_t **b, const uint8_t *src, unsigned int size)
packet is really a next chunk of a packet
void ff_amf_write_number(uint8_t **dst, double val)
Write number in AMF format to buffer.
void ff_amf_write_object_start(uint8_t **dst)
Write marker for AMF object to buffer.
structure for holding RTMP packets
void ff_amf_write_string2(uint8_t **dst, const char *str1, const char *str2)
Write a string consisting of two parts in AMF format to a buffer.
unbuffered private I/O API
invoke some stream action
uint64_t_TMPL AV_WL64 unsigned int_TMPL AV_RL32
int ffurl_read(URLContext *h, unsigned char *buf, int size)
Read up to size bytes from the resource accessed by h, and store the read bytes in buf...