annotate pcx.c @ 12530:63edd10ad4bc libavcodec tip

Try to fix crashes introduced by r25218 r25218 made assumptions about the existence of past reference frames that weren't necessarily true.
author darkshikari
date Tue, 28 Sep 2010 09:06:22 +0000
parents ffb3668ff7af
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
1 /*
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
2 * PC Paintbrush PCX (.pcx) image decoder
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
3 * Copyright (c) 2007, 2008 Ivo van Poorten
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
4 *
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
5 * This decoder does not support CGA palettes. I am unable to find samples
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
6 * and Netpbm cannot generate them.
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
7 *
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
8 * This file is part of FFmpeg.
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
9 *
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
10 * FFmpeg is free software; you can redistribute it and/or
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
11 * modify it under the terms of the GNU Lesser General Public
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
12 * License as published by the Free Software Foundation; either
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
13 * version 2.1 of the License, or (at your option) any later version.
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
14 *
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
15 * FFmpeg is distributed in the hope that it will be useful,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
18 * Lesser General Public License for more details.
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
19 *
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
20 * You should have received a copy of the GNU Lesser General Public
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
21 * License along with FFmpeg; if not, write to the Free Software
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
22 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
23 */
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
24
12372
914f484bb476 Remove use of the deprecated function avcodec_check_dimensions(), use
stefano
parents: 11962
diff changeset
25 #include "libavcore/imgutils.h"
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
26 #include "avcodec.h"
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
27 #include "bytestream.h"
9428
0dce4fe6e6f3 Rename bitstream.h to get_bits.h.
stefano
parents: 9355
diff changeset
28 #include "get_bits.h"
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
29
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
30 typedef struct PCXContext {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
31 AVFrame picture;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
32 } PCXContext;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
33
6517
48759bfbd073 Apply 'cold' attribute to init/uninit functions in libavcodec
zuxy
parents: 6244
diff changeset
34 static av_cold int pcx_init(AVCodecContext *avctx) {
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
35 PCXContext *s = avctx->priv_data;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
36
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
37 avcodec_get_frame_defaults(&s->picture);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
38 avctx->coded_frame= &s->picture;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
39
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
40 return 0;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
41 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
42
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
43 /**
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
44 * @return advanced src pointer
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
45 */
6793
3a715cb7f4b2 Correct type for pcx_rle_decode().
cehoyos
parents: 6722
diff changeset
46 static const uint8_t *pcx_rle_decode(const uint8_t *src, uint8_t *dst,
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
47 unsigned int bytes_per_scanline, int compressed) {
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
48 unsigned int i = 0;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
49 unsigned char run, value;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
50
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
51 if (compressed) {
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
52 while (i<bytes_per_scanline) {
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
53 run = 1;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
54 value = *src++;
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
55 if (value >= 0xc0) {
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
56 run = value & 0x3f;
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
57 value = *src++;
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
58 }
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
59 while (i<bytes_per_scanline && run--)
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
60 dst[i++] = value;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
61 }
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
62 } else {
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
63 memcpy(dst, src, bytes_per_scanline);
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
64 src += bytes_per_scanline;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
65 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
66
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
67 return src;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
68 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
69
6244
michael
parents: 6075
diff changeset
70 static void pcx_palette(const uint8_t **src, uint32_t *dst, unsigned int pallen) {
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
71 unsigned int i;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
72
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
73 for (i=0; i<pallen; i++)
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
74 *dst++ = bytestream_get_be24(src);
10455
0609da72112c Only memset for palettes of 16 elements.
benoit
parents: 10351
diff changeset
75 if (pallen < 256)
10456
2422415a296c Fix indentation.
benoit
parents: 10455
diff changeset
76 memset(dst, 0, (256 - pallen) * sizeof(*dst));
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
77 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
78
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
79 static int pcx_decode_frame(AVCodecContext *avctx, void *data, int *data_size,
9355
54bc8a2727b0 Implement avcodec_decode_video2(), _audio3() and _subtitle2() which takes an
rbultje
parents: 7040
diff changeset
80 AVPacket *avpkt) {
54bc8a2727b0 Implement avcodec_decode_video2(), _audio3() and _subtitle2() which takes an
rbultje
parents: 7040
diff changeset
81 const uint8_t *buf = avpkt->data;
54bc8a2727b0 Implement avcodec_decode_video2(), _audio3() and _subtitle2() which takes an
rbultje
parents: 7040
diff changeset
82 int buf_size = avpkt->size;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
83 PCXContext * const s = avctx->priv_data;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
84 AVFrame *picture = data;
6075
4d43f2e3aaa5 remove useless cast
ivo
parents: 6073
diff changeset
85 AVFrame * const p = &s->picture;
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
86 int compressed, xmin, ymin, xmax, ymax;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
87 unsigned int w, h, bits_per_pixel, bytes_per_line, nplanes, stride, y, x,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
88 bytes_per_scanline;
6244
michael
parents: 6075
diff changeset
89 uint8_t *ptr;
michael
parents: 6075
diff changeset
90 uint8_t const *bufstart = buf;
11962
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
91 uint8_t *scanline;
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
92 int ret = -1;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
93
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
94 if (buf[0] != 0x0a || buf[1] > 5) {
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
95 av_log(avctx, AV_LOG_ERROR, "this is not PCX encoded data\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
96 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
97 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
98
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
99 compressed = buf[2];
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
100 xmin = AV_RL16(buf+ 4);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
101 ymin = AV_RL16(buf+ 6);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
102 xmax = AV_RL16(buf+ 8);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
103 ymax = AV_RL16(buf+10);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
104
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
105 if (xmax < xmin || ymax < ymin) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
106 av_log(avctx, AV_LOG_ERROR, "invalid image dimensions\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
107 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
108 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
109
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
110 w = xmax - xmin + 1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
111 h = ymax - ymin + 1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
112
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
113 bits_per_pixel = buf[3];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
114 bytes_per_line = AV_RL16(buf+66);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
115 nplanes = buf[65];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
116 bytes_per_scanline = nplanes * bytes_per_line;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
117
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
118 if (bytes_per_scanline < w * bits_per_pixel * nplanes / 8) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
119 av_log(avctx, AV_LOG_ERROR, "PCX data is corrupted\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
120 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
121 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
122
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
123 switch ((nplanes<<8) + bits_per_pixel) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
124 case 0x0308:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
125 avctx->pix_fmt = PIX_FMT_RGB24;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
126 break;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
127 case 0x0108:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
128 case 0x0104:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
129 case 0x0102:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
130 case 0x0101:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
131 case 0x0401:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
132 case 0x0301:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
133 case 0x0201:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
134 avctx->pix_fmt = PIX_FMT_PAL8;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
135 break;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
136 default:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
137 av_log(avctx, AV_LOG_ERROR, "invalid PCX file\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
138 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
139 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
140
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
141 buf += 128;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
142
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
143 if (p->data[0])
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
144 avctx->release_buffer(avctx, p);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
145
12462
ffb3668ff7af Use new imgutils.h API names, fix deprecation warnings.
stefano
parents: 12372
diff changeset
146 if (av_image_check_size(w, h, 0, avctx))
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
147 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
148 if (w != avctx->width || h != avctx->height)
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
149 avcodec_set_dimensions(avctx, w, h);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
150 if (avctx->get_buffer(avctx, p) < 0) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
151 av_log(avctx, AV_LOG_ERROR, "get_buffer() failed\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
152 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
153 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
154
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
155 p->pict_type = FF_I_TYPE;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
156
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
157 ptr = p->data[0];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
158 stride = p->linesize[0];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
159
11962
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
160 scanline = av_malloc(bytes_per_scanline);
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
161 if (!scanline)
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
162 return AVERROR(ENOMEM);
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
163
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
164 if (nplanes == 3 && bits_per_pixel == 8) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
165 for (y=0; y<h; y++) {
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
166 buf = pcx_rle_decode(buf, scanline, bytes_per_scanline, compressed);
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
167
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
168 for (x=0; x<w; x++) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
169 ptr[3*x ] = scanline[x ];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
170 ptr[3*x+1] = scanline[x+ bytes_per_line ];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
171 ptr[3*x+2] = scanline[x+(bytes_per_line<<1)];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
172 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
173
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
174 ptr += stride;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
175 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
176
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
177 } else if (nplanes == 1 && bits_per_pixel == 8) {
6244
michael
parents: 6075
diff changeset
178 const uint8_t *palstart = bufstart + buf_size - 769;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
179
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
180 for (y=0; y<h; y++, ptr+=stride) {
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
181 buf = pcx_rle_decode(buf, scanline, bytes_per_scanline, compressed);
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
182 memcpy(ptr, scanline, w);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
183 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
184
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
185 if (buf != palstart) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
186 av_log(avctx, AV_LOG_WARNING, "image data possibly corrupted\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
187 buf = palstart;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
188 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
189 if (*buf++ != 12) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
190 av_log(avctx, AV_LOG_ERROR, "expected palette after image data\n");
11962
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
191 goto end;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
192 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
193
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
194 } else if (nplanes == 1) { /* all packed formats, max. 16 colors */
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
195 GetBitContext s;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
196
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
197 for (y=0; y<h; y++) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
198 init_get_bits(&s, scanline, bytes_per_scanline<<3);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
199
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
200 buf = pcx_rle_decode(buf, scanline, bytes_per_scanline, compressed);
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
201
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
202 for (x=0; x<w; x++)
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
203 ptr[x] = get_bits(&s, bits_per_pixel);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
204 ptr += stride;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
205 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
206
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
207 } else { /* planar, 4, 8 or 16 colors */
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
208 int i;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
209
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
210 for (y=0; y<h; y++) {
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
211 buf = pcx_rle_decode(buf, scanline, bytes_per_scanline, compressed);
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
212
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
213 for (x=0; x<w; x++) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
214 int m = 0x80 >> (x&7), v = 0;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
215 for (i=nplanes - 1; i>=0; i--) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
216 v <<= 1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
217 v += !!(scanline[i*bytes_per_line + (x>>3)] & m);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
218 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
219 ptr[x] = v;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
220 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
221 ptr += stride;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
222 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
223 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
224
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
225 if (nplanes == 1 && bits_per_pixel == 8) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
226 pcx_palette(&buf, (uint32_t *) p->data[1], 256);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
227 } else if (bits_per_pixel < 8) {
6244
michael
parents: 6075
diff changeset
228 const uint8_t *palette = bufstart+16;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
229 pcx_palette(&palette, (uint32_t *) p->data[1], 16);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
230 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
231
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
232 *picture = s->picture;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
233 *data_size = sizeof(AVFrame);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
234
11962
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
235 ret = buf - bufstart;
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
236 end:
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
237 av_free(scanline);
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
238 return ret;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
239 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
240
6517
48759bfbd073 Apply 'cold' attribute to init/uninit functions in libavcodec
zuxy
parents: 6244
diff changeset
241 static av_cold int pcx_end(AVCodecContext *avctx) {
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
242 PCXContext *s = avctx->priv_data;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
243
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
244 if(s->picture.data[0])
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
245 avctx->release_buffer(avctx, &s->picture);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
246
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
247 return 0;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
248 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
249
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
250 AVCodec pcx_decoder = {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
251 "pcx",
11560
8a4984c5cacc Define AVMediaType enum, and use it instead of enum CodecType, which
stefano
parents: 10456
diff changeset
252 AVMEDIA_TYPE_VIDEO,
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
253 CODEC_ID_PCX,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
254 sizeof(PCXContext),
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
255 pcx_init,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
256 NULL,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
257 pcx_end,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
258 pcx_decode_frame,
9802
aebbb6c631b6 pcx decoder use get_buffer, set CODEC_CAP_DR1
bcoudurier
parents: 9428
diff changeset
259 CODEC_CAP_DR1,
6722
6eeb19edcee3 Add long names to some AVCodec declarations.
diego
parents: 6517
diff changeset
260 NULL,
7040
e943e1409077 Make AVCodec long_names definition conditional depending on CONFIG_SMALL.
stefano
parents: 6793
diff changeset
261 .long_name = NULL_IF_CONFIG_SMALL("PC Paintbrush PCX image"),
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
262 };