2 * V.Flash PTX (.ptx) image decoder
3 * Copyright (c) 2007 Ivo van Poorten
5 * This file is part of Libav.
7 * Libav is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
12 * Libav is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with Libav; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
22 #include "libavutil/common.h"
23 #include "libavutil/intreadwrite.h"
24 #include "libavutil/imgutils.h"
28 typedef struct PTXContext
{
32 static av_cold
int ptx_init(AVCodecContext
*avctx
) {
33 PTXContext
*s
= avctx
->priv_data
;
35 avcodec_get_frame_defaults(&s
->picture
);
36 avctx
->coded_frame
= &s
->picture
;
41 static int ptx_decode_frame(AVCodecContext
*avctx
, void *data
, int *got_frame
,
43 const uint8_t *buf
= avpkt
->data
;
44 const uint8_t *buf_end
= avpkt
->data
+ avpkt
->size
;
45 PTXContext
* const s
= avctx
->priv_data
;
46 AVFrame
*picture
= data
;
47 AVFrame
* const p
= &s
->picture
;
48 unsigned int offset
, w
, h
, y
, stride
, bytes_per_pixel
;
52 if (buf_end
- buf
< 14)
53 return AVERROR_INVALIDDATA
;
54 offset
= AV_RL16(buf
);
57 bytes_per_pixel
= AV_RL16(buf
+12) >> 3;
59 if (bytes_per_pixel
!= 2) {
60 av_log_ask_for_sample(avctx
, "Image format is not RGB15.\n");
61 return AVERROR_PATCHWELCOME
;
64 avctx
->pix_fmt
= AV_PIX_FMT_RGB555
;
66 if (buf_end
- buf
< offset
)
67 return AVERROR_INVALIDDATA
;
69 av_log_ask_for_sample(avctx
, "offset != 0x2c\n");
74 avctx
->release_buffer(avctx
, p
);
76 if ((ret
= av_image_check_size(w
, h
, 0, avctx
)) < 0)
78 if (w
!= avctx
->width
|| h
!= avctx
->height
)
79 avcodec_set_dimensions(avctx
, w
, h
);
80 if ((ret
= ff_get_buffer(avctx
, p
)) < 0) {
81 av_log(avctx
, AV_LOG_ERROR
, "get_buffer() failed\n");
85 p
->pict_type
= AV_PICTURE_TYPE_I
;
88 stride
= p
->linesize
[0];
90 for (y
= 0; y
< h
&& buf_end
- buf
>= w
* bytes_per_pixel
; y
++) {
93 for (x
=0; x
<w
*bytes_per_pixel
; x
+=bytes_per_pixel
)
94 AV_WN16(ptr
+x
, AV_RL16(buf
+x
));
96 memcpy(ptr
, buf
, w
*bytes_per_pixel
);
99 buf
+= w
*bytes_per_pixel
;
102 *picture
= s
->picture
;
106 av_log(avctx
, AV_LOG_WARNING
, "incomplete packet\n");
110 return offset
+ w
*h
*bytes_per_pixel
;
113 static av_cold
int ptx_end(AVCodecContext
*avctx
) {
114 PTXContext
*s
= avctx
->priv_data
;
116 if(s
->picture
.data
[0])
117 avctx
->release_buffer(avctx
, &s
->picture
);
122 AVCodec ff_ptx_decoder
= {
124 .type
= AVMEDIA_TYPE_VIDEO
,
125 .id
= AV_CODEC_ID_PTX
,
126 .priv_data_size
= sizeof(PTXContext
),
129 .decode
= ptx_decode_frame
,
130 .capabilities
= CODEC_CAP_DR1
,
131 .long_name
= NULL_IF_CONFIG_SMALL("V.Flash PTX image"),