annotate pcx.c @ 12197:fbf4d5b1b664 libavcodec

Remove FF_MM_SSE2/3 flags for CPUs where this is generally not faster than regular MMX code. Examples of this are the Core1 CPU. Instead, set a new flag, FF_MM_SSE2/3SLOW, which can be checked for particular SSE2/3 functions that have been checked specifically on such CPUs and are actually faster than their MMX counterparts. In addition, use this flag to enable particular VP8 and LPC SSE2 functions that are faster than their MMX counterparts. Based on a patch by Loren Merritt <lorenm AT u washington edu>.
author rbultje
date Mon, 19 Jul 2010 22:38:23 +0000
parents 150c7c236e78
children 914f484bb476
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
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
25 #include "avcodec.h"
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
26 #include "bytestream.h"
9428
0dce4fe6e6f3 Rename bitstream.h to get_bits.h.
stefano
parents: 9355
diff changeset
27 #include "get_bits.h"
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
28
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
29 typedef struct PCXContext {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
30 AVFrame picture;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
31 } PCXContext;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
32
6517
48759bfbd073 Apply 'cold' attribute to init/uninit functions in libavcodec
zuxy
parents: 6244
diff changeset
33 static av_cold int pcx_init(AVCodecContext *avctx) {
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
34 PCXContext *s = avctx->priv_data;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
35
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
36 avcodec_get_frame_defaults(&s->picture);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
37 avctx->coded_frame= &s->picture;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
38
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
39 return 0;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
40 }
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 * @return advanced src pointer
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
44 */
6793
3a715cb7f4b2 Correct type for pcx_rle_decode().
cehoyos
parents: 6722
diff changeset
45 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
46 unsigned int bytes_per_scanline, int compressed) {
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
47 unsigned int i = 0;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
48 unsigned char run, value;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
49
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
50 if (compressed) {
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
51 while (i<bytes_per_scanline) {
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
52 run = 1;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
53 value = *src++;
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
54 if (value >= 0xc0) {
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
55 run = value & 0x3f;
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
56 value = *src++;
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
57 }
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
58 while (i<bytes_per_scanline && run--)
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
59 dst[i++] = value;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
60 }
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
61 } else {
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
62 memcpy(dst, src, bytes_per_scanline);
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
63 src += bytes_per_scanline;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
64 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
65
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
66 return src;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
67 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
68
6244
michael
parents: 6075
diff changeset
69 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
70 unsigned int i;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
71
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
72 for (i=0; i<pallen; i++)
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
73 *dst++ = bytestream_get_be24(src);
10455
0609da72112c Only memset for palettes of 16 elements.
benoit
parents: 10351
diff changeset
74 if (pallen < 256)
10456
2422415a296c Fix indentation.
benoit
parents: 10455
diff changeset
75 memset(dst, 0, (256 - pallen) * sizeof(*dst));
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
76 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
77
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
78 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
79 AVPacket *avpkt) {
54bc8a2727b0 Implement avcodec_decode_video2(), _audio3() and _subtitle2() which takes an
rbultje
parents: 7040
diff changeset
80 const uint8_t *buf = avpkt->data;
54bc8a2727b0 Implement avcodec_decode_video2(), _audio3() and _subtitle2() which takes an
rbultje
parents: 7040
diff changeset
81 int buf_size = avpkt->size;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
82 PCXContext * const s = avctx->priv_data;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
83 AVFrame *picture = data;
6075
4d43f2e3aaa5 remove useless cast
ivo
parents: 6073
diff changeset
84 AVFrame * const p = &s->picture;
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
85 int compressed, xmin, ymin, xmax, ymax;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
86 unsigned int w, h, bits_per_pixel, bytes_per_line, nplanes, stride, y, x,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
87 bytes_per_scanline;
6244
michael
parents: 6075
diff changeset
88 uint8_t *ptr;
michael
parents: 6075
diff changeset
89 uint8_t const *bufstart = buf;
11962
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
90 uint8_t *scanline;
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
91 int ret = -1;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
92
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
93 if (buf[0] != 0x0a || buf[1] > 5) {
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
94 av_log(avctx, AV_LOG_ERROR, "this is not PCX encoded data\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
95 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
96 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
97
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
98 compressed = buf[2];
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
99 xmin = AV_RL16(buf+ 4);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
100 ymin = AV_RL16(buf+ 6);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
101 xmax = AV_RL16(buf+ 8);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
102 ymax = AV_RL16(buf+10);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
103
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
104 if (xmax < xmin || ymax < ymin) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
105 av_log(avctx, AV_LOG_ERROR, "invalid image dimensions\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
106 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
107 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
108
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
109 w = xmax - xmin + 1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
110 h = ymax - ymin + 1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
111
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
112 bits_per_pixel = buf[3];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
113 bytes_per_line = AV_RL16(buf+66);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
114 nplanes = buf[65];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
115 bytes_per_scanline = nplanes * bytes_per_line;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
116
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
117 if (bytes_per_scanline < w * bits_per_pixel * nplanes / 8) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
118 av_log(avctx, AV_LOG_ERROR, "PCX data is corrupted\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
119 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
120 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
121
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
122 switch ((nplanes<<8) + bits_per_pixel) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
123 case 0x0308:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
124 avctx->pix_fmt = PIX_FMT_RGB24;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
125 break;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
126 case 0x0108:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
127 case 0x0104:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
128 case 0x0102:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
129 case 0x0101:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
130 case 0x0401:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
131 case 0x0301:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
132 case 0x0201:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
133 avctx->pix_fmt = PIX_FMT_PAL8;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
134 break;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
135 default:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
136 av_log(avctx, AV_LOG_ERROR, "invalid PCX file\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
137 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
138 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
139
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
140 buf += 128;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
141
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
142 if (p->data[0])
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
143 avctx->release_buffer(avctx, p);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
144
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
145 if (avcodec_check_dimensions(avctx, w, h))
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
146 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
147 if (w != avctx->width || h != avctx->height)
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
148 avcodec_set_dimensions(avctx, w, h);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
149 if (avctx->get_buffer(avctx, p) < 0) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
150 av_log(avctx, AV_LOG_ERROR, "get_buffer() failed\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
151 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
152 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
153
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
154 p->pict_type = FF_I_TYPE;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
155
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
156 ptr = p->data[0];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
157 stride = p->linesize[0];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
158
11962
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
159 scanline = av_malloc(bytes_per_scanline);
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
160 if (!scanline)
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
161 return AVERROR(ENOMEM);
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
162
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
163 if (nplanes == 3 && bits_per_pixel == 8) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
164 for (y=0; y<h; y++) {
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
165 buf = pcx_rle_decode(buf, scanline, bytes_per_scanline, compressed);
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
166
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
167 for (x=0; x<w; x++) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
168 ptr[3*x ] = scanline[x ];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
169 ptr[3*x+1] = scanline[x+ bytes_per_line ];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
170 ptr[3*x+2] = scanline[x+(bytes_per_line<<1)];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
171 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
172
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
173 ptr += stride;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
174 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
175
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
176 } else if (nplanes == 1 && bits_per_pixel == 8) {
6244
michael
parents: 6075
diff changeset
177 const uint8_t *palstart = bufstart + buf_size - 769;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
178
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
179 for (y=0; y<h; y++, ptr+=stride) {
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
180 buf = pcx_rle_decode(buf, scanline, bytes_per_scanline, compressed);
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
181 memcpy(ptr, scanline, w);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
182 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
183
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
184 if (buf != palstart) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
185 av_log(avctx, AV_LOG_WARNING, "image data possibly corrupted\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
186 buf = palstart;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
187 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
188 if (*buf++ != 12) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
189 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
190 goto end;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
191 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
192
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
193 } else if (nplanes == 1) { /* all packed formats, max. 16 colors */
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
194 GetBitContext s;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
195
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
196 for (y=0; y<h; y++) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
197 init_get_bits(&s, scanline, bytes_per_scanline<<3);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
198
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
199 buf = pcx_rle_decode(buf, scanline, bytes_per_scanline, compressed);
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
200
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
201 for (x=0; x<w; x++)
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
202 ptr[x] = get_bits(&s, bits_per_pixel);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
203 ptr += stride;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
204 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
205
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
206 } else { /* planar, 4, 8 or 16 colors */
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
207 int i;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
208
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
209 for (y=0; y<h; y++) {
10351
997bdf43a4ed Support decoding of uncompressed PCX scanlines
pross
parents: 9802
diff changeset
210 buf = pcx_rle_decode(buf, scanline, bytes_per_scanline, compressed);
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
211
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
212 for (x=0; x<w; x++) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
213 int m = 0x80 >> (x&7), v = 0;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
214 for (i=nplanes - 1; i>=0; i--) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
215 v <<= 1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
216 v += !!(scanline[i*bytes_per_line + (x>>3)] & m);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
217 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
218 ptr[x] = v;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
219 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
220 ptr += stride;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
221 }
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 if (nplanes == 1 && bits_per_pixel == 8) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
225 pcx_palette(&buf, (uint32_t *) p->data[1], 256);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
226 } else if (bits_per_pixel < 8) {
6244
michael
parents: 6075
diff changeset
227 const uint8_t *palette = bufstart+16;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
228 pcx_palette(&palette, (uint32_t *) p->data[1], 16);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
229 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
230
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
231 *picture = s->picture;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
232 *data_size = sizeof(AVFrame);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
233
11962
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
234 ret = buf - bufstart;
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
235 end:
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
236 av_free(scanline);
150c7c236e78 pcx: convert VLAs to malloc/free
mru
parents: 11560
diff changeset
237 return ret;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
238 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
239
6517
48759bfbd073 Apply 'cold' attribute to init/uninit functions in libavcodec
zuxy
parents: 6244
diff changeset
240 static av_cold int pcx_end(AVCodecContext *avctx) {
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
241 PCXContext *s = avctx->priv_data;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
242
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
243 if(s->picture.data[0])
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
244 avctx->release_buffer(avctx, &s->picture);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
245
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
246 return 0;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
247 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
248
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
249 AVCodec pcx_decoder = {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
250 "pcx",
11560
8a4984c5cacc Define AVMediaType enum, and use it instead of enum CodecType, which
stefano
parents: 10456
diff changeset
251 AVMEDIA_TYPE_VIDEO,
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
252 CODEC_ID_PCX,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
253 sizeof(PCXContext),
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
254 pcx_init,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
255 NULL,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
256 pcx_end,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
257 pcx_decode_frame,
9802
aebbb6c631b6 pcx decoder use get_buffer, set CODEC_CAP_DR1
bcoudurier
parents: 9428
diff changeset
258 CODEC_CAP_DR1,
6722
6eeb19edcee3 Add long names to some AVCodec declarations.
diego
parents: 6517
diff changeset
259 NULL,
7040
e943e1409077 Make AVCodec long_names definition conditional depending on CONFIG_SMALL.
stefano
parents: 6793
diff changeset
260 .long_name = NULL_IF_CONFIG_SMALL("PC Paintbrush PCX image"),
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
261 };