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 && size != prev_pkt[channel_id].size) {
240 size, prev_pkt[channel_id].size);
242 prev_pkt[channel_id].
read = 0;
246 if (!prev_pkt[channel_id].read) {
252 prev_pkt[channel_id].
ts_field = ts_field;
253 prev_pkt[channel_id].
timestamp = timestamp;
273 prev_pkt[channel_id].
extra = extra;
276 toread =
FFMIN(size, chunk_size);
294 prev_pkt[channel_id].
read = 0;
305 if (ret > 0 || ret !=
AVERROR(EAGAIN))
317 uint8_t pkt_hdr[16], *p = pkt_hdr;
329 prev_pkt = *prev_pkt_ptr;
340 if (timestamp >= 0xFFFFFF) {
358 bytestream_put_byte(&p, pkt->
channel_id | (mode << 6));
360 bytestream_put_byte(&p, 0 | (mode << 6));
361 bytestream_put_byte(&p, pkt->
channel_id - 64);
363 bytestream_put_byte(&p, 1 | (mode << 6));
364 bytestream_put_le16(&p, pkt->
channel_id - 64);
367 bytestream_put_be24(&p, pkt->
ts_field);
369 bytestream_put_be24(&p, pkt->
size);
370 bytestream_put_byte(&p, pkt->
type);
372 bytestream_put_le32(&p, pkt->
extra);
376 bytestream_put_be32(&p, timestamp);
385 if ((ret =
ffurl_write(h, pkt_hdr, p - pkt_hdr)) < 0)
387 written = p - pkt_hdr + pkt->
size;
388 while (off < pkt->
size) {
389 int towrite =
FFMIN(chunk_size, pkt->
size - off);
393 if (off < pkt->size) {
411 int timestamp,
int size)
445 type = bytestream2_get_byte(gb);
448 bytestream2_get_be64(gb);
451 bytestream2_get_byte(gb);
467 nb = bytestream2_get_be32(gb);
472 int size = bytestream2_get_be16(gb);
474 bytestream2_get_byte(gb);
496 if (data >= data_end)
511 int namelen = strlen(name);
521 bytestream2_get_byte(gb);
524 int size = bytestream2_get_be16(gb);
530 if (size == namelen && !memcmp(gb->
buffer-size, name, namelen)) {
531 switch (bytestream2_get_byte(gb)) {
536 snprintf(dst, dst_size,
"%s", bytestream2_get_byte(gb) ?
"true" :
"false");
539 len = bytestream2_get_be16(gb);
542 if (dst_size < len + 1)
564 if (data >= data_end)
589 default:
return "unknown";
596 unsigned int size, nb = -1;
601 if (data >= data_end)
603 switch ((type = *data++)) {
613 size = bytestream_get_be16(&data);
615 size = bytestream_get_be32(&data);
617 size =
FFMIN(size,
sizeof(buf) - 1);
618 memcpy(buf, data, size);
628 nb = bytestream_get_be32(&data);
634 size = bytestream_get_be16(&data);
635 size =
FFMIN(size,
sizeof(buf) - 1);
641 memcpy(buf, data, size);
643 if (size >= data_end - data)
650 if (t < 0 || t >= data_end - data)
665 av_log(ctx,
AV_LOG_DEBUG,
"RTMP packet type '%s'(%d) for channel %d, timestamp %d, extra field %d size %d\n",
669 while (src < src_end) {
683 for (i = 0; i < p->
size; i++)
691 int len = strlen(str);
704 if ((size -= 4 + 1) < 0)
706 amf_len = bytestream_get_be32(&data);
708 if ((size -= 2 + 1) < 0)
710 amf_len = bytestream_get_be16(&data);
719 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.
void * av_realloc(void *ptr, size_t size)
Allocate, reallocate, or free a block of memory.
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.
static av_always_inline void bytestream2_init(GetByteContext *g, const uint8_t *buf, int buf_size)
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
#define av_assert0(cond)
assert() equivalent, that is always enabled.
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.
void * av_realloc_array(void *ptr, size_t nmemb, size_t size)
int size
packet payload size
void ff_amf_write_bool(uint8_t **dst, int val)
Write boolean value in AMF format to buffer.
static int parse_key(DBEContext *s)
#define AV_LOG_ERROR
Something went wrong and cannot losslessly be recovered.
static av_always_inline void bytestream2_skip(GetByteContext *g, unsigned int size)
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.
static av_always_inline unsigned int bytestream2_get_bytes_left(GetByteContext *g)
int ff_amf_get_string(GetByteContext *bc, uint8_t *str, int strsize, int *length)
Get AMF string value.
packet has 12-byte header
static int amf_get_field_value2(GetByteContext *gb, const uint8_t *name, uint8_t *dst, int dst_size)
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.
static int amf_tag_skip(GetByteContext *gb)
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.
static av_always_inline int bytestream2_tell(GetByteContext *g)
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
channel
Use these values when setting the channel map with ebur128_set_channel().
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.
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...
mode
Use these values in ebur128_init (or'ed).
window acknowledgement size