annotate pcx.c @ 9473:e38284cd69dc libavcodec

Use memcpy instead of the very inefficient bytecopy where both are correct (i.e. no overlap of src and dst is possible).
author reimar
date Fri, 17 Apr 2009 17:20:48 +0000
parents 0dce4fe6e6f3
children aebbb6c631b6
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,
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
46 unsigned int bytes_per_scanline) {
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
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
50 while (i<bytes_per_scanline) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
51 run = 1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
52 value = *src++;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
53 if (value >= 0xc0) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
54 run = value & 0x3f;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
55 value = *src++;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
56 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
57 while (i<bytes_per_scanline && run--)
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
58 dst[i++] = value;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
59 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
60
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
61 return src;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
62 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
63
6244
michael
parents: 6075
diff changeset
64 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
65 unsigned int i;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
66
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
67 for (i=0; i<pallen; i++)
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
68 *dst++ = bytestream_get_be24(src);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
69 memset(dst, 0, (256 - pallen) * sizeof(*dst));
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
70 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
71
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
72 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
73 AVPacket *avpkt) {
54bc8a2727b0 Implement avcodec_decode_video2(), _audio3() and _subtitle2() which takes an
rbultje
parents: 7040
diff changeset
74 const uint8_t *buf = avpkt->data;
54bc8a2727b0 Implement avcodec_decode_video2(), _audio3() and _subtitle2() which takes an
rbultje
parents: 7040
diff changeset
75 int buf_size = avpkt->size;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
76 PCXContext * const s = avctx->priv_data;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
77 AVFrame *picture = data;
6075
4d43f2e3aaa5 remove useless cast
ivo
parents: 6073
diff changeset
78 AVFrame * const p = &s->picture;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
79 int xmin, ymin, xmax, ymax;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
80 unsigned int w, h, bits_per_pixel, bytes_per_line, nplanes, stride, y, x,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
81 bytes_per_scanline;
6244
michael
parents: 6075
diff changeset
82 uint8_t *ptr;
michael
parents: 6075
diff changeset
83 uint8_t const *bufstart = buf;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
84
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
85 if (buf[0] != 0x0a || buf[1] > 5 || buf[1] == 1 || buf[2] != 1) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
86 av_log(avctx, AV_LOG_ERROR, "this is not PCX encoded data\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
87 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
88 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
89
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
90 xmin = AV_RL16(buf+ 4);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
91 ymin = AV_RL16(buf+ 6);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
92 xmax = AV_RL16(buf+ 8);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
93 ymax = AV_RL16(buf+10);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
94
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
95 if (xmax < xmin || ymax < ymin) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
96 av_log(avctx, AV_LOG_ERROR, "invalid image dimensions\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
97 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
98 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
99
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
100 w = xmax - xmin + 1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
101 h = ymax - ymin + 1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
102
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
103 bits_per_pixel = buf[3];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
104 bytes_per_line = AV_RL16(buf+66);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
105 nplanes = buf[65];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
106 bytes_per_scanline = nplanes * bytes_per_line;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
107
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
108 if (bytes_per_scanline < w * bits_per_pixel * nplanes / 8) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
109 av_log(avctx, AV_LOG_ERROR, "PCX data is corrupted\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
110 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
111 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
112
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
113 switch ((nplanes<<8) + bits_per_pixel) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
114 case 0x0308:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
115 avctx->pix_fmt = PIX_FMT_RGB24;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
116 break;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
117 case 0x0108:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
118 case 0x0104:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
119 case 0x0102:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
120 case 0x0101:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
121 case 0x0401:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
122 case 0x0301:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
123 case 0x0201:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
124 avctx->pix_fmt = PIX_FMT_PAL8;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
125 break;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
126 default:
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
127 av_log(avctx, AV_LOG_ERROR, "invalid PCX file\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
128 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
129 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
130
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
131 buf += 128;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
132
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
133 if (p->data[0])
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
134 avctx->release_buffer(avctx, p);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
135
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
136 if (avcodec_check_dimensions(avctx, w, h))
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
137 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
138 if (w != avctx->width || h != avctx->height)
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
139 avcodec_set_dimensions(avctx, w, h);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
140 if (avctx->get_buffer(avctx, p) < 0) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
141 av_log(avctx, AV_LOG_ERROR, "get_buffer() failed\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
142 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
143 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
144
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
145 p->pict_type = FF_I_TYPE;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
146
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
147 ptr = p->data[0];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
148 stride = p->linesize[0];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
149
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
150 if (nplanes == 3 && bits_per_pixel == 8) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
151 uint8_t scanline[bytes_per_scanline];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
152
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
153 for (y=0; y<h; y++) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
154 buf = pcx_rle_decode(buf, scanline, bytes_per_scanline);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
155
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
156 for (x=0; x<w; x++) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
157 ptr[3*x ] = scanline[x ];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
158 ptr[3*x+1] = scanline[x+ bytes_per_line ];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
159 ptr[3*x+2] = scanline[x+(bytes_per_line<<1)];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
160 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
161
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
162 ptr += stride;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
163 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
164
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
165 } else if (nplanes == 1 && bits_per_pixel == 8) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
166 uint8_t scanline[bytes_per_scanline];
6244
michael
parents: 6075
diff changeset
167 const uint8_t *palstart = bufstart + buf_size - 769;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
168
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
169 for (y=0; y<h; y++, ptr+=stride) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
170 buf = pcx_rle_decode(buf, scanline, bytes_per_scanline);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
171 memcpy(ptr, scanline, w);
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 if (buf != palstart) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
175 av_log(avctx, AV_LOG_WARNING, "image data possibly corrupted\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
176 buf = palstart;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
177 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
178 if (*buf++ != 12) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
179 av_log(avctx, AV_LOG_ERROR, "expected palette after image data\n");
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
180 return -1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
181 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
182
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
183 } else if (nplanes == 1) { /* all packed formats, max. 16 colors */
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
184 uint8_t scanline[bytes_per_scanline];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
185 GetBitContext s;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
186
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
187 for (y=0; y<h; y++) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
188 init_get_bits(&s, scanline, bytes_per_scanline<<3);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
189
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
190 buf = pcx_rle_decode(buf, scanline, bytes_per_scanline);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
191
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
192 for (x=0; x<w; x++)
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
193 ptr[x] = get_bits(&s, bits_per_pixel);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
194 ptr += stride;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
195 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
196
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
197 } else { /* planar, 4, 8 or 16 colors */
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
198 uint8_t scanline[bytes_per_scanline];
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
199 int i;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
200
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
201 for (y=0; y<h; y++) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
202 buf = pcx_rle_decode(buf, scanline, bytes_per_scanline);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
203
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
204 for (x=0; x<w; x++) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
205 int m = 0x80 >> (x&7), v = 0;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
206 for (i=nplanes - 1; i>=0; i--) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
207 v <<= 1;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
208 v += !!(scanline[i*bytes_per_line + (x>>3)] & m);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
209 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
210 ptr[x] = v;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
211 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
212 ptr += stride;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
213 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
214 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
215
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
216 if (nplanes == 1 && bits_per_pixel == 8) {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
217 pcx_palette(&buf, (uint32_t *) p->data[1], 256);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
218 } else if (bits_per_pixel < 8) {
6244
michael
parents: 6075
diff changeset
219 const uint8_t *palette = bufstart+16;
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
220 pcx_palette(&palette, (uint32_t *) p->data[1], 16);
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 *picture = s->picture;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
224 *data_size = sizeof(AVFrame);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
225
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
226 return buf - bufstart;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
227 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
228
6517
48759bfbd073 Apply 'cold' attribute to init/uninit functions in libavcodec
zuxy
parents: 6244
diff changeset
229 static av_cold int pcx_end(AVCodecContext *avctx) {
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
230 PCXContext *s = avctx->priv_data;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
231
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
232 if(s->picture.data[0])
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
233 avctx->release_buffer(avctx, &s->picture);
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
234
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
235 return 0;
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
236 }
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
237
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
238 AVCodec pcx_decoder = {
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
239 "pcx",
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
240 CODEC_TYPE_VIDEO,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
241 CODEC_ID_PCX,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
242 sizeof(PCXContext),
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
243 pcx_init,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
244 NULL,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
245 pcx_end,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
246 pcx_decode_frame,
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
247 0,
6722
6eeb19edcee3 Add long names to some AVCodec declarations.
diego
parents: 6517
diff changeset
248 NULL,
7040
e943e1409077 Make AVCodec long_names definition conditional depending on CONFIG_SMALL.
stefano
parents: 6793
diff changeset
249 .long_name = NULL_IF_CONFIG_SMALL("PC Paintbrush PCX image"),
6073
60d30d88cf71 PC Paintbrush PCX image decoder
ivo
parents:
diff changeset
250 };