FFmpeg coverage


Directory: ../../../ffmpeg/
File: src/libavcodec/mscc.c
Date: 2024-04-23 16:28:37
Exec Total Coverage
Lines: 0 142 0.0%
Functions: 0 4 0.0%
Branches: 0 70 0.0%

Line Branch Exec Source
1 /*
2 * Mandsoft Screen Capture Codec decoder
3 *
4 * Copyright (c) 2017 Paul B Mahol
5 *
6 * This file is part of FFmpeg.
7 *
8 * FFmpeg is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License as published by the Free Software Foundation; either
11 * version 2.1 of the License, or (at your option) any later version.
12 *
13 * FFmpeg is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Lesser General Public License for more details.
17 *
18 * You should have received a copy of the GNU Lesser General Public
19 * License along with FFmpeg; if not, write to the Free Software
20 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
21 */
22
23 #include <stdio.h>
24 #include <string.h>
25
26 #include "libavutil/mem.h"
27 #include "avcodec.h"
28 #include "bytestream.h"
29 #include "codec_internal.h"
30 #include "decode.h"
31 #include "zlib_wrapper.h"
32
33 #include <zlib.h>
34
35 typedef struct MSCCContext {
36 unsigned bpp;
37 unsigned int decomp_size;
38 uint8_t *decomp_buf;
39 unsigned int uncomp_size;
40 uint8_t *uncomp_buf;
41 FFZStream zstream;
42
43 uint32_t pal[256];
44 } MSCCContext;
45
46 static int rle_uncompress(AVCodecContext *avctx, GetByteContext *gb, PutByteContext *pb)
47 {
48 MSCCContext *s = avctx->priv_data;
49 unsigned x = 0, y = 0;
50
51 while (bytestream2_get_bytes_left(gb) > 0) {
52 uint32_t fill;
53 int j;
54 unsigned run = bytestream2_get_byte(gb);
55
56 if (run) {
57 switch (avctx->bits_per_coded_sample) {
58 case 8:
59 fill = bytestream2_get_byte(gb);
60 break;
61 case 16:
62 fill = bytestream2_get_le16(gb);
63 break;
64 case 24:
65 fill = bytestream2_get_le24(gb);
66 break;
67 case 32:
68 fill = bytestream2_get_le32(gb);
69 break;
70 }
71
72 for (j = 0; j < run; j++) {
73 switch (avctx->bits_per_coded_sample) {
74 case 8:
75 bytestream2_put_byte(pb, fill);
76 break;
77 case 16:
78 bytestream2_put_le16(pb, fill);
79 break;
80 case 24:
81 bytestream2_put_le24(pb, fill);
82 break;
83 case 32:
84 bytestream2_put_le32(pb, fill);
85 break;
86 }
87 }
88 x += run;
89 } else {
90 unsigned copy = bytestream2_get_byte(gb);
91
92 if (copy == 0) {
93 x = 0;
94 y++;
95 bytestream2_seek_p(pb, y * avctx->width * s->bpp, SEEK_SET);
96 } else if (copy == 1) {
97 return 0;
98 } else if (copy == 2) {
99
100 x += bytestream2_get_byte(gb);
101 y += bytestream2_get_byte(gb);
102
103 bytestream2_seek_p(pb, y * avctx->width * s->bpp + x * s->bpp, SEEK_SET);
104 } else {
105 for (j = 0; j < copy; j++) {
106 switch (avctx->bits_per_coded_sample) {
107 case 8:
108 bytestream2_put_byte(pb, bytestream2_get_byte(gb));
109 break;
110 case 16:
111 bytestream2_put_le16(pb, bytestream2_get_le16(gb));
112 break;
113 case 24:
114 bytestream2_put_le24(pb, bytestream2_get_le24(gb));
115 break;
116 case 32:
117 bytestream2_put_le32(pb, bytestream2_get_le32(gb));
118 break;
119 }
120 }
121
122 if (s->bpp == 1 && (copy & 1))
123 bytestream2_skip(gb, 1);
124 x += copy;
125 }
126 }
127 }
128
129 return AVERROR_INVALIDDATA;
130 }
131
132 static int decode_frame(AVCodecContext *avctx, AVFrame *frame,
133 int *got_frame, AVPacket *avpkt)
134 {
135 MSCCContext *s = avctx->priv_data;
136 z_stream *const zstream = &s->zstream.zstream;
137 const uint8_t *buf = avpkt->data;
138 int buf_size = avpkt->size;
139 GetByteContext gb;
140 PutByteContext pb;
141 int ret, j;
142
143 if (avpkt->size < 3)
144 return buf_size;
145
146 if ((ret = ff_get_buffer(avctx, frame, 0)) < 0)
147 return ret;
148
149 if (avctx->pix_fmt == AV_PIX_FMT_PAL8) {
150 size_t size;
151 const uint8_t *pal = av_packet_get_side_data(avpkt, AV_PKT_DATA_PALETTE, &size);
152
153 if (pal && size == AVPALETTE_SIZE) {
154 #if FF_API_PALETTE_HAS_CHANGED
155 FF_DISABLE_DEPRECATION_WARNINGS
156 frame->palette_has_changed = 1;
157 FF_ENABLE_DEPRECATION_WARNINGS
158 #endif
159 for (j = 0; j < 256; j++)
160 s->pal[j] = 0xFF000000 | AV_RL32(pal + j * 4);
161 } else if (pal) {
162 av_log(avctx, AV_LOG_ERROR,
163 "Palette size %"SIZE_SPECIFIER" is wrong\n", size);
164 }
165 memcpy(frame->data[1], s->pal, AVPALETTE_SIZE);
166 }
167
168 ret = inflateReset(zstream);
169 if (ret != Z_OK) {
170 av_log(avctx, AV_LOG_ERROR, "Inflate reset error: %d\n", ret);
171 return AVERROR_UNKNOWN;
172 }
173 zstream->next_out = s->decomp_buf;
174 zstream->avail_out = s->decomp_size;
175 if (avctx->codec_id == AV_CODEC_ID_MSCC) {
176 const uint8_t start = avpkt->data[2] ^ avpkt->data[0];
177
178 zstream->next_in = &start;
179 zstream->avail_in = 1;
180 ret = inflate(zstream, Z_NO_FLUSH);
181 if (ret != Z_OK || zstream->avail_in != 0)
182 goto inflate_error;
183
184 buf += 3;
185 buf_size -= 3;
186 }
187 zstream->next_in = buf;
188 zstream->avail_in = buf_size;
189 ret = inflate(zstream, Z_FINISH);
190 if (ret != Z_STREAM_END) {
191 inflate_error:
192 av_log(avctx, AV_LOG_ERROR, "Inflate error: %d\n", ret);
193 return AVERROR_UNKNOWN;
194 }
195
196 bytestream2_init(&gb, s->decomp_buf, zstream->total_out);
197 bytestream2_init_writer(&pb, s->uncomp_buf, s->uncomp_size);
198
199 ret = rle_uncompress(avctx, &gb, &pb);
200 if (ret)
201 return ret;
202
203 for (j = 0; j < avctx->height; j++) {
204 memcpy(frame->data[0] + (avctx->height - j - 1) * frame->linesize[0],
205 s->uncomp_buf + s->bpp * j * avctx->width, s->bpp * avctx->width);
206 }
207
208 frame->flags |= AV_FRAME_FLAG_KEY;
209 frame->pict_type = AV_PICTURE_TYPE_I;
210
211 *got_frame = 1;
212
213 return avpkt->size;
214 }
215
216 static av_cold int decode_init(AVCodecContext *avctx)
217 {
218 MSCCContext *s = avctx->priv_data;
219 int stride;
220
221 switch (avctx->bits_per_coded_sample) {
222 case 8: avctx->pix_fmt = AV_PIX_FMT_PAL8; break;
223 case 16: avctx->pix_fmt = AV_PIX_FMT_RGB555; break;
224 case 24: avctx->pix_fmt = AV_PIX_FMT_BGR24; break;
225 case 32: avctx->pix_fmt = AV_PIX_FMT_BGRA; break;
226 default:
227 av_log(avctx, AV_LOG_ERROR, "Unsupported bitdepth %i\n", avctx->bits_per_coded_sample);
228 return AVERROR_INVALIDDATA;
229 }
230
231 s->bpp = avctx->bits_per_coded_sample >> 3;
232 stride = 4 * ((avctx->width * avctx->bits_per_coded_sample + 31) / 32);
233
234 s->decomp_size = 2 * avctx->height * stride;
235 if (!(s->decomp_buf = av_malloc(s->decomp_size)))
236 return AVERROR(ENOMEM);
237
238 s->uncomp_size = avctx->height * stride;
239 if (!(s->uncomp_buf = av_malloc(s->uncomp_size)))
240 return AVERROR(ENOMEM);
241
242 return ff_inflate_init(&s->zstream, avctx);
243 }
244
245 static av_cold int decode_close(AVCodecContext *avctx)
246 {
247 MSCCContext *s = avctx->priv_data;
248
249 av_freep(&s->decomp_buf);
250 s->decomp_size = 0;
251 av_freep(&s->uncomp_buf);
252 s->uncomp_size = 0;
253 ff_inflate_end(&s->zstream);
254
255 return 0;
256 }
257
258 const FFCodec ff_mscc_decoder = {
259 .p.name = "mscc",
260 CODEC_LONG_NAME("Mandsoft Screen Capture Codec"),
261 .p.type = AVMEDIA_TYPE_VIDEO,
262 .p.id = AV_CODEC_ID_MSCC,
263 .priv_data_size = sizeof(MSCCContext),
264 .init = decode_init,
265 .close = decode_close,
266 FF_CODEC_DECODE_CB(decode_frame),
267 .p.capabilities = AV_CODEC_CAP_DR1,
268 .caps_internal = FF_CODEC_CAP_INIT_CLEANUP,
269 };
270
271 const FFCodec ff_srgc_decoder = {
272 .p.name = "srgc",
273 CODEC_LONG_NAME("Screen Recorder Gold Codec"),
274 .p.type = AVMEDIA_TYPE_VIDEO,
275 .p.id = AV_CODEC_ID_SRGC,
276 .priv_data_size = sizeof(MSCCContext),
277 .init = decode_init,
278 .close = decode_close,
279 FF_CODEC_DECODE_CB(decode_frame),
280 .p.capabilities = AV_CODEC_CAP_DR1,
281 .caps_internal = FF_CODEC_CAP_INIT_CLEANUP,
282 };
283