mirror of https://github.com/FFmpeg/FFmpeg.git
* qatar/master: (21 commits) CDXL demuxer and decoder hls: Re-add legacy applehttp name to preserve interface compatibility. hlsproto: Rename the functions and context hlsproto: Encourage users to try the hls demuxer instead of the proto doc: Move the hls protocol section into the right place libavformat: Rename the applehttp protocol to hls hls: Rename the functions and context libavformat: Rename the applehttp demuxer to hls rtpdec: Support H263 in RFC 2190 format rv30: check block type validity ttadec: CRC checking movenc: Support muxing VC1 avconv: Don't split out inline sequence headers when stream copying VC1 rv34: handle size changes during frame multithreading rv40: prevent undefined signed overflow in rv40_loop_filter() rv34: use AVERROR return values in ff_rv34_decode_frame() rv34: use uint16_t for RV34DecContext.deblock_coefs librtmp: Add "lib" prefix to librtmp URLProtocol declarations. movenc: Use defines instead of hardcoded numbers for RTCP types smjpegdec: implement seeking ... Conflicts: Changelog doc/general.texi libavcodec/avcodec.h libavcodec/rv30.c libavcodec/tta.c libavcodec/version.h libavformat/Makefile libavformat/allformats.c libavformat/version.h libswscale/x86/swscale_mmx.c Merged-by: Michael Niedermayer <michaelni@gmx.at>pull/3/merge
commit
c980be9e3a
28 changed files with 1084 additions and 156 deletions
@ -0,0 +1,278 @@ |
||||
/*
|
||||
* CDXL video decoder |
||||
* Copyright (c) 2011-2012 Paul B Mahol |
||||
* |
||||
* This file is part of Libav. |
||||
* |
||||
* Libav is free software; you can redistribute it and/or |
||||
* modify it under the terms of the GNU Lesser General Public |
||||
* License as published by the Free Software Foundation; either |
||||
* version 2.1 of the License, or (at your option) any later version. |
||||
* |
||||
* Libav is distributed in the hope that it will be useful, |
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
||||
* Lesser General Public License for more details. |
||||
* |
||||
* You should have received a copy of the GNU Lesser General Public |
||||
* License along with Libav; if not, write to the Free Software |
||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
||||
*/ |
||||
|
||||
#include "libavutil/intreadwrite.h" |
||||
#include "libavutil/imgutils.h" |
||||
#include "avcodec.h" |
||||
#include "get_bits.h" |
||||
|
||||
typedef struct { |
||||
AVCodecContext *avctx; |
||||
AVFrame frame; |
||||
int bpp; |
||||
const uint8_t *palette; |
||||
int palette_size; |
||||
const uint8_t *video; |
||||
int video_size; |
||||
uint8_t *new_video; |
||||
int new_video_size; |
||||
} CDXLVideoContext; |
||||
|
||||
static av_cold int cdxl_decode_init(AVCodecContext *avctx) |
||||
{ |
||||
CDXLVideoContext *c = avctx->priv_data; |
||||
|
||||
avcodec_get_frame_defaults(&c->frame); |
||||
c->new_video_size = 0; |
||||
c->avctx = avctx; |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static void import_palette(CDXLVideoContext *c, uint32_t *new_palette) |
||||
{ |
||||
int i; |
||||
|
||||
for (i = 0; i < c->palette_size / 2; i++) { |
||||
unsigned rgb = AV_RB16(&c->palette[i * 2]); |
||||
unsigned r = ((rgb >> 8) & 0xF) * 0x11; |
||||
unsigned g = ((rgb >> 4) & 0xF) * 0x11; |
||||
unsigned b = (rgb & 0xF) * 0x11; |
||||
AV_WN32(&new_palette[i], (r << 16) | (g << 8) | b); |
||||
} |
||||
} |
||||
|
||||
static void bitplanar2chunky(CDXLVideoContext *c, int width, |
||||
int linesize, uint8_t *out) |
||||
{ |
||||
GetBitContext gb; |
||||
int x, y, plane; |
||||
|
||||
init_get_bits(&gb, c->video, c->video_size * 8); |
||||
memset(out, 0, linesize * c->avctx->height); |
||||
for (plane = 0; plane < c->bpp; plane++) |
||||
for (y = 0; y < c->avctx->height; y++) |
||||
for (x = 0; x < width; x++) |
||||
out[linesize * y + x] |= get_bits1(&gb) << plane; |
||||
} |
||||
|
||||
static void cdxl_decode_rgb(CDXLVideoContext *c) |
||||
{ |
||||
uint32_t *new_palette = (uint32_t *)c->frame.data[1]; |
||||
int padded_width = FFALIGN(c->avctx->width, 16); |
||||
|
||||
import_palette(c, new_palette); |
||||
bitplanar2chunky(c, padded_width, c->frame.linesize[0], c->frame.data[0]); |
||||
} |
||||
|
||||
static void cdxl_decode_ham6(CDXLVideoContext *c) |
||||
{ |
||||
AVCodecContext *avctx = c->avctx; |
||||
uint32_t new_palette[16], r, g, b; |
||||
uint8_t *ptr, *out, index, op; |
||||
int x, y; |
||||
|
||||
ptr = c->new_video; |
||||
out = c->frame.data[0]; |
||||
|
||||
import_palette(c, new_palette); |
||||
bitplanar2chunky(c, avctx->width, avctx->width, c->new_video); |
||||
|
||||
for (y = 0; y < avctx->height; y++) { |
||||
r = new_palette[0] & 0xFF0000; |
||||
g = new_palette[0] & 0xFF00; |
||||
b = new_palette[0] & 0xFF; |
||||
for (x = 0; x < avctx->width; x++) { |
||||
index = *ptr++; |
||||
op = index >> 4; |
||||
index &= 15; |
||||
switch (op) { |
||||
case 0: |
||||
r = new_palette[index] & 0xFF0000; |
||||
g = new_palette[index] & 0xFF00; |
||||
b = new_palette[index] & 0xFF; |
||||
break; |
||||
case 1: |
||||
b = index * 0x11; |
||||
break; |
||||
case 2: |
||||
r = index * 0x11 << 16; |
||||
break; |
||||
case 3: |
||||
g = index * 0x11 << 8; |
||||
break; |
||||
} |
||||
AV_WN32(out + x * 3, r | g | b); |
||||
} |
||||
out += c->frame.linesize[0]; |
||||
} |
||||
} |
||||
|
||||
static void cdxl_decode_ham8(CDXLVideoContext *c) |
||||
{ |
||||
AVCodecContext *avctx = c->avctx; |
||||
uint32_t new_palette[64], r, g, b; |
||||
uint8_t *ptr, *out, index, op; |
||||
int x, y; |
||||
|
||||
ptr = c->new_video; |
||||
out = c->frame.data[0]; |
||||
|
||||
import_palette(c, new_palette); |
||||
bitplanar2chunky(c, avctx->width, avctx->width, c->new_video); |
||||
|
||||
for (y = 0; y < avctx->height; y++) { |
||||
r = new_palette[0] & 0xFF0000; |
||||
g = new_palette[0] & 0xFF00; |
||||
b = new_palette[0] & 0xFF; |
||||
for (x = 0; x < avctx->width; x++) { |
||||
index = *ptr++; |
||||
op = index >> 6; |
||||
index &= 63; |
||||
switch (op) { |
||||
case 0: |
||||
r = new_palette[index] & 0xFF0000; |
||||
g = new_palette[index] & 0xFF00; |
||||
b = new_palette[index] & 0xFF; |
||||
break; |
||||
case 1: |
||||
b = (index << 2) | (b & 3); |
||||
break; |
||||
case 2: |
||||
r = (index << 18) | (r & (3 << 16)); |
||||
break; |
||||
case 3: |
||||
g = (index << 10) | (g & (3 << 8)); |
||||
break; |
||||
} |
||||
AV_WN32(out + x * 3, r | g | b); |
||||
} |
||||
out += c->frame.linesize[0]; |
||||
} |
||||
} |
||||
|
||||
static int cdxl_decode_frame(AVCodecContext *avctx, void *data, |
||||
int *data_size, AVPacket *pkt) |
||||
{ |
||||
CDXLVideoContext *c = avctx->priv_data; |
||||
AVFrame * const p = &c->frame; |
||||
int ret, w, h, encoding, format, buf_size = pkt->size; |
||||
const uint8_t *buf = pkt->data; |
||||
|
||||
if (buf_size < 32) |
||||
return AVERROR_INVALIDDATA; |
||||
encoding = buf[1] & 7; |
||||
format = buf[1] & 0xE0; |
||||
w = AV_RB16(&buf[14]); |
||||
h = AV_RB16(&buf[16]); |
||||
c->bpp = buf[19]; |
||||
c->palette_size = AV_RB16(&buf[20]); |
||||
c->palette = buf + 32; |
||||
c->video = c->palette + c->palette_size; |
||||
c->video_size = buf_size - c->palette_size - 32; |
||||
|
||||
if (c->palette_size > 512) |
||||
return AVERROR_INVALIDDATA; |
||||
if (buf_size < c->palette_size + 32) |
||||
return AVERROR_INVALIDDATA; |
||||
if (c->bpp < 1) |
||||
return AVERROR_INVALIDDATA; |
||||
if (c->bpp > 8) { |
||||
av_log_ask_for_sample(avctx, "unsupported pixel size: %d\n", c->bpp); |
||||
return AVERROR_PATCHWELCOME; |
||||
} |
||||
if (format) { |
||||
av_log_ask_for_sample(avctx, "unsupported pixel format: %d\n", format); |
||||
return AVERROR_PATCHWELCOME; |
||||
} |
||||
|
||||
if ((ret = av_image_check_size(w, h, 0, avctx)) < 0) |
||||
return ret; |
||||
if (w != avctx->width || h != avctx->height) |
||||
avcodec_set_dimensions(avctx, w, h); |
||||
|
||||
if (encoding == 0) { |
||||
if (c->video_size < FFALIGN(avctx->width, 16) * |
||||
avctx->height * c->bpp / 8) |
||||
return AVERROR_INVALIDDATA; |
||||
avctx->pix_fmt = PIX_FMT_PAL8; |
||||
} else if (encoding == 1 && (c->bpp == 6 || c->bpp == 8)) { |
||||
if (c->palette_size != (1 << (c->bpp - 1))) |
||||
return AVERROR_INVALIDDATA; |
||||
if (c->video_size < avctx->width * avctx->height * c->bpp / 8) |
||||
return AVERROR_INVALIDDATA; |
||||
avctx->pix_fmt = PIX_FMT_BGR24; |
||||
} else { |
||||
av_log_ask_for_sample(avctx, "unsupported encoding %d and bpp %d\n", |
||||
encoding, c->bpp); |
||||
return AVERROR_PATCHWELCOME; |
||||
} |
||||
|
||||
if (p->data[0]) |
||||
avctx->release_buffer(avctx, p); |
||||
|
||||
p->reference = 0; |
||||
if ((ret = avctx->get_buffer(avctx, p)) < 0) { |
||||
av_log(avctx, AV_LOG_ERROR, "get_buffer() failed\n"); |
||||
return ret; |
||||
} |
||||
p->pict_type = AV_PICTURE_TYPE_I; |
||||
|
||||
if (encoding) { |
||||
av_fast_padded_malloc(&c->new_video, &c->new_video_size, |
||||
h * w + FF_INPUT_BUFFER_PADDING_SIZE); |
||||
if (!c->new_video) |
||||
return AVERROR(ENOMEM); |
||||
if (c->bpp == 8) |
||||
cdxl_decode_ham8(c); |
||||
else |
||||
cdxl_decode_ham6(c); |
||||
} else { |
||||
cdxl_decode_rgb(c); |
||||
} |
||||
*data_size = sizeof(AVFrame); |
||||
*(AVFrame*)data = c->frame; |
||||
|
||||
return buf_size; |
||||
} |
||||
|
||||
static av_cold int cdxl_decode_end(AVCodecContext *avctx) |
||||
{ |
||||
CDXLVideoContext *c = avctx->priv_data; |
||||
|
||||
av_free(c->new_video); |
||||
if (c->frame.data[0]) |
||||
avctx->release_buffer(avctx, &c->frame); |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
AVCodec ff_cdxl_decoder = { |
||||
.name = "cdxl", |
||||
.type = AVMEDIA_TYPE_VIDEO, |
||||
.id = CODEC_ID_CDXL, |
||||
.priv_data_size = sizeof(CDXLVideoContext), |
||||
.init = cdxl_decode_init, |
||||
.close = cdxl_decode_end, |
||||
.decode = cdxl_decode_frame, |
||||
.capabilities = CODEC_CAP_DR1, |
||||
.long_name = NULL_IF_CONFIG_SMALL("Commodore CDXL video"), |
||||
}; |
@ -0,0 +1,170 @@ |
||||
/*
|
||||
* CDXL demuxer |
||||
* Copyright (c) 2011-2012 Paul B Mahol |
||||
* |
||||
* This file is part of Libav. |
||||
* |
||||
* Libav is free software; you can redistribute it and/or |
||||
* modify it under the terms of the GNU Lesser General Public |
||||
* License as published by the Free Software Foundation; either |
||||
* version 2.1 of the License, or (at your option) any later version. |
||||
* |
||||
* Libav is distributed in the hope that it will be useful, |
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
||||
* Lesser General Public License for more details. |
||||
* |
||||
* You should have received a copy of the GNU Lesser General Public |
||||
* License along with Libav; if not, write to the Free Software |
||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
||||
*/ |
||||
|
||||
#include "libavutil/intreadwrite.h" |
||||
#include "libavutil/parseutils.h" |
||||
#include "libavutil/opt.h" |
||||
#include "avformat.h" |
||||
#include "internal.h" |
||||
|
||||
#define CDXL_HEADER_SIZE 32 |
||||
|
||||
typedef struct CDXLDemuxContext { |
||||
AVClass *class; |
||||
int sample_rate; |
||||
char *framerate; |
||||
AVRational fps; |
||||
int read_chunk; |
||||
uint8_t header[CDXL_HEADER_SIZE]; |
||||
int video_stream_index; |
||||
int audio_stream_index; |
||||
} CDXLDemuxContext; |
||||
|
||||
static int cdxl_read_header(AVFormatContext *s) |
||||
{ |
||||
CDXLDemuxContext *cdxl = s->priv_data; |
||||
int ret; |
||||
|
||||
if ((ret = av_parse_video_rate(&cdxl->fps, cdxl->framerate)) < 0) { |
||||
av_log(s, AV_LOG_ERROR, |
||||
"Could not parse framerate: %s.\n", cdxl->framerate); |
||||
return ret; |
||||
} |
||||
|
||||
cdxl->read_chunk = 0; |
||||
cdxl->video_stream_index = -1; |
||||
cdxl->audio_stream_index = -1; |
||||
|
||||
s->ctx_flags |= AVFMTCTX_NOHEADER; |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int cdxl_read_packet(AVFormatContext *s, AVPacket *pkt) |
||||
{ |
||||
CDXLDemuxContext *cdxl = s->priv_data; |
||||
AVIOContext *pb = s->pb; |
||||
uint32_t current_size; |
||||
uint16_t audio_size, palette_size; |
||||
int32_t video_size; |
||||
int64_t pos; |
||||
int ret; |
||||
|
||||
if (pb->eof_reached) |
||||
return AVERROR_EOF; |
||||
|
||||
pos = avio_tell(pb); |
||||
if (!cdxl->read_chunk && |
||||
avio_read(pb, cdxl->header, CDXL_HEADER_SIZE) != CDXL_HEADER_SIZE) |
||||
return AVERROR_EOF; |
||||
if (cdxl->header[0] != 1) { |
||||
av_log(s, AV_LOG_ERROR, "non-standard cdxl file\n"); |
||||
return AVERROR_INVALIDDATA; |
||||
} |
||||
|
||||
current_size = AV_RB32(&cdxl->header[2]); |
||||
palette_size = AV_RB16(&cdxl->header[20]); |
||||
audio_size = AV_RB16(&cdxl->header[22]); |
||||
|
||||
if (palette_size > 512) |
||||
return AVERROR_INVALIDDATA; |
||||
if (current_size < audio_size + palette_size + CDXL_HEADER_SIZE) |
||||
return AVERROR_INVALIDDATA; |
||||
video_size = current_size - audio_size - CDXL_HEADER_SIZE; |
||||
|
||||
if (cdxl->read_chunk && audio_size) { |
||||
if (cdxl->audio_stream_index == -1) { |
||||
AVStream *st = avformat_new_stream(s, NULL); |
||||
if (!st) |
||||
return AVERROR(ENOMEM); |
||||
|
||||
st->codec->codec_type = AVMEDIA_TYPE_AUDIO; |
||||
st->codec->codec_tag = 0; |
||||
st->codec->codec_id = CODEC_ID_PCM_S8; |
||||
st->codec->channels = cdxl->header[1] & 0x10 ? 2 : 1; |
||||
st->codec->sample_rate = cdxl->sample_rate; |
||||
cdxl->audio_stream_index = st->index; |
||||
avpriv_set_pts_info(st, 32, 1, cdxl->sample_rate); |
||||
} |
||||
|
||||
ret = av_get_packet(pb, pkt, audio_size); |
||||
if (ret < 0) |
||||
return ret; |
||||
pkt->stream_index = cdxl->audio_stream_index; |
||||
pkt->pos = pos; |
||||
pkt->duration = audio_size; |
||||
cdxl->read_chunk = 0; |
||||
} else { |
||||
if (cdxl->video_stream_index == -1) { |
||||
AVStream *st = avformat_new_stream(s, NULL); |
||||
if (!st) |
||||
return AVERROR(ENOMEM); |
||||
|
||||
st->codec->codec_type = AVMEDIA_TYPE_VIDEO; |
||||
st->codec->codec_tag = 0; |
||||
st->codec->codec_id = CODEC_ID_CDXL; |
||||
st->codec->width = AV_RB16(&cdxl->header[14]); |
||||
st->codec->height = AV_RB16(&cdxl->header[16]); |
||||
cdxl->video_stream_index = st->index; |
||||
avpriv_set_pts_info(st, 63, cdxl->fps.den, cdxl->fps.num); |
||||
} |
||||
|
||||
if (av_new_packet(pkt, video_size + CDXL_HEADER_SIZE) < 0) |
||||
return AVERROR(ENOMEM); |
||||
memcpy(pkt->data, cdxl->header, CDXL_HEADER_SIZE); |
||||
ret = avio_read(pb, pkt->data + CDXL_HEADER_SIZE, video_size); |
||||
if (ret < 0) { |
||||
av_free_packet(pkt); |
||||
return ret; |
||||
} |
||||
pkt->stream_index = cdxl->video_stream_index; |
||||
pkt->flags |= AV_PKT_FLAG_KEY; |
||||
pkt->pos = pos; |
||||
cdxl->read_chunk = audio_size; |
||||
} |
||||
|
||||
return ret; |
||||
} |
||||
|
||||
#define OFFSET(x) offsetof(CDXLDemuxContext, x) |
||||
static const AVOption cdxl_options[] = { |
||||
{ "sample_rate", "", OFFSET(sample_rate), AV_OPT_TYPE_INT, { .dbl = 11025 }, 1, INT_MAX, AV_OPT_FLAG_DECODING_PARAM }, |
||||
{ "framerate", "", OFFSET(framerate), AV_OPT_TYPE_STRING, { .str = "10" }, 0, 0, AV_OPT_FLAG_DECODING_PARAM }, |
||||
{ NULL }, |
||||
}; |
||||
|
||||
static const AVClass cdxl_demuxer_class = { |
||||
.class_name = "CDXL demuxer", |
||||
.item_name = av_default_item_name, |
||||
.option = cdxl_options, |
||||
.version = LIBAVUTIL_VERSION_INT, |
||||
}; |
||||
|
||||
AVInputFormat ff_cdxl_demuxer = { |
||||
.name = "cdxl", |
||||
.long_name = NULL_IF_CONFIG_SMALL("Commodore CDXL video format"), |
||||
.priv_data_size = sizeof(CDXLDemuxContext), |
||||
.read_header = cdxl_read_header, |
||||
.read_packet = cdxl_read_packet, |
||||
.extensions = "cdxl,xl", |
||||
.flags = AVFMT_GENERIC_INDEX, |
||||
.priv_class = &cdxl_demuxer_class, |
||||
}; |
@ -0,0 +1,184 @@ |
||||
/*
|
||||
* RTP H.263 Depacketizer, RFC 2190 |
||||
* Copyright (c) 2012 Martin Storsjo |
||||
* Based on the GStreamer H.263 Depayloder: |
||||
* Copyright 2005 Wim Taymans |
||||
* Copyright 2007 Edward Hervey |
||||
* Copyright 2007 Nokia Corporation |
||||
* Copyright 2007 Collabora Ltd, Philippe Kalaf |
||||
* Copyright 2010 Mark Nauwelaerts |
||||
* |
||||
* This file is part of Libav. |
||||
* |
||||
* Libav is free software; you can redistribute it and/or |
||||
* modify it under the terms of the GNU Lesser General Public |
||||
* License as published by the Free Software Foundation; either |
||||
* version 2.1 of the License, or (at your option) any later version. |
||||
* |
||||
* Libav is distributed in the hope that it will be useful, |
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
||||
* Lesser General Public License for more details. |
||||
* |
||||
* You should have received a copy of the GNU Lesser General Public |
||||
* License along with Libav; if not, write to the Free Software |
||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
||||
*/ |
||||
|
||||
#include "avformat.h" |
||||
#include "rtpdec_formats.h" |
||||
#include "libavutil/intreadwrite.h" |
||||
#include "libavcodec/get_bits.h" |
||||
|
||||
struct PayloadContext { |
||||
AVIOContext *buf; |
||||
uint8_t endbyte; |
||||
int endbyte_bits; |
||||
uint32_t timestamp; |
||||
}; |
||||
|
||||
static PayloadContext *h263_new_context(void) |
||||
{ |
||||
return av_mallocz(sizeof(PayloadContext)); |
||||
} |
||||
|
||||
static void h263_free_context(PayloadContext *data) |
||||
{ |
||||
if (!data) |
||||
return; |
||||
if (data->buf) { |
||||
uint8_t *p; |
||||
avio_close_dyn_buf(data->buf, &p); |
||||
av_free(p); |
||||
} |
||||
av_free(data); |
||||
} |
||||
|
||||
static int h263_handle_packet(AVFormatContext *ctx, PayloadContext *data, |
||||
AVStream *st, AVPacket *pkt, uint32_t *timestamp, |
||||
const uint8_t *buf, int len, int flags) |
||||
{ |
||||
int f, p, i, sbit, ebit; /* Corresponding to header fields in the RFC */ |
||||
int header_size; |
||||
|
||||
if (data->buf && data->timestamp != *timestamp) { |
||||
/* Dropping old buffered, unfinished data */ |
||||
uint8_t *p; |
||||
avio_close_dyn_buf(data->buf, &p); |
||||
av_free(p); |
||||
data->buf = NULL; |
||||
} |
||||
|
||||
if (len < 4) { |
||||
av_log(ctx, AV_LOG_ERROR, "Too short H.263 RTP packet: %d\n", len); |
||||
return AVERROR_INVALIDDATA; |
||||
} |
||||
|
||||
f = buf[0] & 0x80; |
||||
p = buf[0] & 0x40; |
||||
if (!f) { |
||||
/* Mode A */ |
||||
header_size = 4; |
||||
i = buf[1] & 0x10; |
||||
} else if (!p) { |
||||
/* Mode B */ |
||||
header_size = 8; |
||||
if (len < header_size) { |
||||
av_log(ctx, AV_LOG_ERROR, |
||||
"Too short H.263 RTP packet: %d bytes, %d header bytes\n", |
||||
len, header_size); |
||||
return AVERROR_INVALIDDATA; |
||||
} |
||||
i = buf[4] & 0x80; |
||||
} else { |
||||
/* Mode C */ |
||||
header_size = 12; |
||||
if (len < header_size) { |
||||
av_log(ctx, AV_LOG_ERROR, |
||||
"Too short H.263 RTP packet: %d bytes, %d header bytes\n", |
||||
len, header_size); |
||||
return AVERROR_INVALIDDATA; |
||||
} |
||||
i = buf[4] & 0x80; |
||||
} |
||||
sbit = (buf[0] >> 3) & 0x7; |
||||
ebit = buf[0] & 0x7; |
||||
|
||||
buf += header_size; |
||||
len -= header_size; |
||||
|
||||
if (!data->buf) { |
||||
/* Check the picture start code, only start buffering a new frame
|
||||
* if this is correct */ |
||||
if (!f && len > 4 && AV_RB32(buf) >> 10 == 0x20) { |
||||
int ret = avio_open_dyn_buf(&data->buf); |
||||
if (ret < 0) |
||||
return ret; |
||||
data->timestamp = *timestamp; |
||||
} else { |
||||
/* Frame not started yet, skipping */ |
||||
return AVERROR(EAGAIN); |
||||
} |
||||
} |
||||
|
||||
if (data->endbyte_bits || sbit) { |
||||
if (data->endbyte_bits == sbit) { |
||||
data->endbyte |= buf[0] & (0xff >> sbit); |
||||
data->endbyte_bits = 0; |
||||
buf++; |
||||
len--; |
||||
avio_w8(data->buf, data->endbyte); |
||||
} else { |
||||
/* Start/end skip bits not matching - missed packets? */ |
||||
GetBitContext gb; |
||||
init_get_bits(&gb, buf, len*8 - ebit); |
||||
skip_bits(&gb, sbit); |
||||
if (data->endbyte_bits) { |
||||
data->endbyte |= get_bits(&gb, 8 - data->endbyte_bits); |
||||
avio_w8(data->buf, data->endbyte); |
||||
} |
||||
while (get_bits_left(&gb) >= 8) |
||||
avio_w8(data->buf, get_bits(&gb, 8)); |
||||
data->endbyte_bits = get_bits_left(&gb); |
||||
if (data->endbyte_bits) |
||||
data->endbyte = get_bits(&gb, data->endbyte_bits) << |
||||
(8 - data->endbyte_bits); |
||||
ebit = 0; |
||||
len = 0; |
||||
} |
||||
} |
||||
if (ebit) { |
||||
if (len > 0) |
||||
avio_write(data->buf, buf, len - 1); |
||||
data->endbyte_bits = 8 - ebit; |
||||
data->endbyte = buf[len - 1] & (0xff << ebit); |
||||
} else { |
||||
avio_write(data->buf, buf, len); |
||||
} |
||||
|
||||
if (!(flags & RTP_FLAG_MARKER)) |
||||
return AVERROR(EAGAIN); |
||||
|
||||
if (data->endbyte_bits) |
||||
avio_w8(data->buf, data->endbyte); |
||||
data->endbyte_bits = 0; |
||||
|
||||
av_init_packet(pkt); |
||||
pkt->size = avio_close_dyn_buf(data->buf, &pkt->data); |
||||
pkt->destruct = av_destruct_packet; |
||||
pkt->stream_index = st->index; |
||||
if (!i) |
||||
pkt->flags |= AV_PKT_FLAG_KEY; |
||||
data->buf = NULL; |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
RTPDynamicProtocolHandler ff_h263_rfc2190_dynamic_handler = { |
||||
.codec_type = AVMEDIA_TYPE_VIDEO, |
||||
.codec_id = CODEC_ID_H263, |
||||
.parse_packet = h263_handle_packet, |
||||
.alloc = h263_new_context, |
||||
.free = h263_free_context, |
||||
.static_payload_id = 34, |
||||
}; |
Loading…
Reference in new issue