diff options
author | Thilo Borgmann <thilo.borgmann@googlemail.com> | 2009-04-10 17:16:19 +0000 |
---|---|---|
committer | Stefano Sabatini <stefano.sabatini-lala@poste.it> | 2009-04-10 17:16:19 +0000 |
commit | ee30cdabadc055ae49d9b92e800839b5d8925598 (patch) | |
tree | 3df0aa6804850d417a789576d03a811c3cbfb3d4 /libavcodec/pngdec.c | |
parent | cc7b62afd2d9b9f3f37f93536a8e69c627f81409 (diff) | |
download | ffmpeg-ee30cdabadc055ae49d9b92e800839b5d8925598.tar.gz |
Add support to CorePNG P-frames.
Patch by Thilo Borgmann thilo DOT borgmann A googlemail com.
Originally committed as revision 18421 to svn://svn.ffmpeg.org/ffmpeg/trunk
Diffstat (limited to 'libavcodec/pngdec.c')
-rw-r--r-- | libavcodec/pngdec.c | 34 |
1 files changed, 29 insertions, 5 deletions
diff --git a/libavcodec/pngdec.c b/libavcodec/pngdec.c index fadbcd07c9..bf719b2382 100644 --- a/libavcodec/pngdec.c +++ b/libavcodec/pngdec.c @@ -37,7 +37,8 @@ typedef struct PNGDecContext { const uint8_t *bytestream; const uint8_t *bytestream_start; const uint8_t *bytestream_end; - AVFrame picture; + AVFrame picture1, picture2; + AVFrame *current_picture, *last_picture; int state; int width, height; @@ -385,10 +386,14 @@ static int decode_frame(AVCodecContext *avctx, int buf_size = avpkt->size; PNGDecContext * const s = avctx->priv_data; AVFrame *picture = data; - AVFrame * const p= &s->picture; + AVFrame *p; uint32_t tag, length; int ret, crc; + FFSWAP(AVFrame *, s->current_picture, s->last_picture); + avctx->coded_frame= s->current_picture; + p = s->current_picture; + s->bytestream_start= s->bytestream= buf; s->bytestream_end= buf + buf_size; @@ -584,7 +589,24 @@ static int decode_frame(AVCodecContext *avctx, } } exit_loop: - *picture= s->picture; + /* handle p-frames only if a predecessor frame is available */ + if(s->last_picture->data[0] != NULL) { + if(!(avpkt->flags & PKT_FLAG_KEY)) { + int i, j; + uint8_t *pd = s->current_picture->data[0]; + uint8_t *pd_last = s->last_picture->data[0]; + + for(j=0; j < s->height; j++) { + for(i=0; i < s->width * s->bpp; i++) { + pd[i] += pd_last[i]; + } + pd += s->image_linesize; + pd_last += s->image_linesize; + } + } + } + + *picture= *s->current_picture; *data_size = sizeof(AVFrame); ret = s->bytestream - s->bytestream_start; @@ -602,8 +624,10 @@ static int decode_frame(AVCodecContext *avctx, static av_cold int png_dec_init(AVCodecContext *avctx){ PNGDecContext *s = avctx->priv_data; - avcodec_get_frame_defaults(&s->picture); - avctx->coded_frame= &s->picture; + s->current_picture = &s->picture1; + s->last_picture = &s->picture2; + avcodec_get_frame_defaults(&s->picture1); + avcodec_get_frame_defaults(&s->picture2); dsputil_init(&s->dsp, avctx); return 0; |