diff --git a/Changelog b/Changelog index 729ca5d449..ee3678e57a 100644 --- a/Changelog +++ b/Changelog @@ -81,6 +81,7 @@ version - Bethsoft VID demuxer and video decoder - CRYO APC demuxer - Atrac3 decoder +- V.Flash PTX decoder version 0.4.9-pre1: diff --git a/MAINTAINERS b/MAINTAINERS index 9fa8a86eeb..d16cbda337 100644 --- a/MAINTAINERS +++ b/MAINTAINERS @@ -152,6 +152,7 @@ Codecs: msvideo1.c Mike Melanson nuv.c Reimar Doeffinger oggtheora.c Mans Rullgard + ptx.c Ivo van Poorten qdm2.c, qdm2data.h Roberto Togni qdrw.c Kostya Shishkov qpeg.c Kostya Shishkov diff --git a/libavcodec/Makefile b/libavcodec/Makefile index f886d28d95..14b70adfc3 100644 --- a/libavcodec/Makefile +++ b/libavcodec/Makefile @@ -116,6 +116,7 @@ OBJS-$(CONFIG_MSZH_DECODER) += lcl.o OBJS-$(CONFIG_NUV_DECODER) += nuv.o rtjpeg.o OBJS-$(CONFIG_PNG_DECODER) += png.o OBJS-$(CONFIG_PNG_ENCODER) += png.o +OBJS-$(CONFIG_PTX_DECODER) += ptx.o OBJS-$(CONFIG_QDM2_DECODER) += qdm2.o OBJS-$(CONFIG_QDRAW_DECODER) += qdrw.o OBJS-$(CONFIG_QPEG_DECODER) += qpeg.o diff --git a/libavcodec/allcodecs.c b/libavcodec/allcodecs.c index 99c046457d..8c1afb1e01 100644 --- a/libavcodec/allcodecs.c +++ b/libavcodec/allcodecs.c @@ -116,6 +116,7 @@ void avcodec_register_all(void) REGISTER_ENCODER(PGMYUV, pgmyuv); REGISTER_ENCDEC (PNG, png); REGISTER_ENCODER(PPM, ppm); + REGISTER_DECODER(PTX, ptx); REGISTER_DECODER(QDRAW, qdraw); REGISTER_DECODER(QPEG, qpeg); REGISTER_DECODER(QTRLE, qtrle); diff --git a/libavcodec/avcodec.h b/libavcodec/avcodec.h index b96307c813..55a37ca874 100644 --- a/libavcodec/avcodec.h +++ b/libavcodec/avcodec.h @@ -162,6 +162,7 @@ enum CodecID { CODEC_ID_SGI, CODEC_ID_C93, CODEC_ID_BETHSOFTVID, + CODEC_ID_PTX, /* various PCM "codecs" */ CODEC_ID_PCM_S16LE= 0x10000, @@ -2321,6 +2322,7 @@ extern AVCodec mszh_decoder; extern AVCodec nuv_decoder; extern AVCodec oggvorbis_decoder; extern AVCodec png_decoder; +extern AVCodec ptx_decoder; extern AVCodec qdm2_decoder; extern AVCodec qdraw_decoder; extern AVCodec qpeg_decoder; diff --git a/libavcodec/ptx.c b/libavcodec/ptx.c new file mode 100644 index 0000000000..897e7ec6b2 --- /dev/null +++ b/libavcodec/ptx.c @@ -0,0 +1,119 @@ +/* + * V.Flash PTX (.ptx) image decoder + * Copyright (c) 2007 Ivo van Poorten + * + * This file is part of FFmpeg. + * + * FFmpeg 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. + * + * FFmpeg 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 FFmpeg; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA + * + */ + +#include "avcodec.h" + +typedef struct PTXContext { + AVFrame picture; +} PTXContext; + +static int ptx_init(AVCodecContext *avctx) { + PTXContext *s = avctx->priv_data; + + avcodec_get_frame_defaults((AVFrame*)&s->picture); + avctx->coded_frame= (AVFrame*)&s->picture; + s->picture.data[0] = NULL; + + return 0; +} + +static int ptx_decode_frame(AVCodecContext *avctx, void *data, int *data_size, + uint8_t *buf, int buf_size) { + PTXContext * const s = avctx->priv_data; + AVFrame *picture = data; + AVFrame * const p = (AVFrame *)&s->picture; + unsigned int offset, w, h, y, stride, bytes_per_pixel; + uint8_t *ptr; + + offset = AV_RL16(buf); + w = AV_RL16(buf+8); + h = AV_RL16(buf+10); + bytes_per_pixel = AV_RL16(buf+12) >> 3; + + if (bytes_per_pixel != 2) { + av_log(avctx, AV_LOG_ERROR, "image format is not rgb15, please report on ffmpeg-users mailing list\n"); + return -1; + } + + avctx->pix_fmt = PIX_FMT_RGB555; + + if (offset != 0x2c) + av_log(avctx, AV_LOG_WARNING, "offset != 0x2c, untested due to lack of sample files\n"); + + buf += offset; + + if (p->data[0]) + avctx->release_buffer(avctx, p); + + if (avcodec_check_dimensions(avctx, w, h)) + return -1; + if (w != avctx->width || h != avctx->height) + avcodec_set_dimensions(avctx, w, h); + if (avctx->get_buffer(avctx, p) < 0) { + av_log(avctx, AV_LOG_ERROR, "get_buffer() failed\n"); + return -1; + } + + p->pict_type = FF_I_TYPE; + + ptr = p->data[0]; + stride = p->linesize[0]; + + for (y=0; ypicture; + *data_size = sizeof(AVPicture); + + return offset + w*h*bytes_per_pixel; +} + +static int ptx_end(AVCodecContext *avctx) { + PTXContext *s = avctx->priv_data; + + if(s->picture.data[0]) + avctx->release_buffer(avctx, &s->picture); + + return 0; +} + +AVCodec ptx_decoder = { + "ptx", + CODEC_TYPE_VIDEO, + CODEC_ID_PTX, + sizeof(PTXContext), + ptx_init, + NULL, + ptx_end, + ptx_decode_frame, + 0, + NULL +};