FFmpeg coverage


Directory: ../../../ffmpeg/
File: src/libavformat/sga.c
Date: 2025-01-20 09:27:23
Exec Total Coverage
Lines: 72 293 24.6%
Functions: 1 10 10.0%
Branches: 75 230 32.6%

Line Branch Exec Source
1 /*
2 * Digital Pictures SGA game demuxer
3 *
4 * Copyright (C) 2021 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 "libavutil/intreadwrite.h"
24 #include "libavutil/avassert.h"
25 #include "libavutil/channel_layout.h"
26 #include "libavutil/internal.h"
27 #include "avformat.h"
28 #include "demux.h"
29 #include "internal.h"
30 #include "avio_internal.h"
31
32 #define SEGA_CD_PCM_NUM 12500000
33 #define SEGA_CD_PCM_DEN 786432
34
35 typedef struct SGADemuxContext {
36 int video_stream_index;
37 int audio_stream_index;
38
39 uint8_t sector[65536 * 2];
40 int sector_headers;
41 int sample_rate;
42 int first_audio_size;
43 int payload_size;
44 int packet_type;
45 int flags;
46 int idx;
47 int left;
48 int64_t pkt_pos;
49 } SGADemuxContext;
50
51 7203 static int sga_probe(const AVProbeData *p)
52 {
53 7203 const uint8_t *src = p->buf;
54 7203 int score = 0, sectors = 1;
55 7203 int last_left = 0;
56 7203 int sample_rate = -1;
57
58
2/2
✓ Branch 0 taken 367 times.
✓ Branch 1 taken 6836 times.
7203 if (p->buf_size < 2048)
59 367 return 0;
60
61
2/2
✓ Branch 0 taken 8766 times.
✓ Branch 1 taken 1393 times.
10159 for (int i = 0; i + 2 < p->buf_size; i += 2048) {
62 8766 int header = AV_RB16(src + i);
63
64
6/6
✓ Branch 0 taken 5854 times.
✓ Branch 1 taken 2912 times.
✓ Branch 2 taken 590 times.
✓ Branch 3 taken 5264 times.
✓ Branch 4 taken 590 times.
✓ Branch 5 taken 2912 times.
8766 if ((header > 0x07FE && header < 0x8100) ||
65
4/4
✓ Branch 0 taken 488 times.
✓ Branch 1 taken 102 times.
✓ Branch 2 taken 488 times.
✓ Branch 3 taken 2912 times.
3502 (header > 0x8200 && header < 0xA100) ||
66
2/2
✓ Branch 0 taken 77 times.
✓ Branch 1 taken 411 times.
488 (header > 0xA200 && header < 0xC100)) {
67 5443 sectors = 0;
68 5443 break;
69 }
70 }
71
72
2/2
✓ Branch 0 taken 6876 times.
✓ Branch 1 taken 610 times.
7486 for (int i = 0; i + 4 < p->buf_size;) {
73 6876 int header = AV_RB16(src + i);
74 6876 int left = AV_RB16(src + i + 2);
75 int offset, type, size;
76
77
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6876 times.
6876 if (last_left < 0)
78 return 0;
79
6/6
✓ Branch 0 taken 1433 times.
✓ Branch 1 taken 5443 times.
✓ Branch 2 taken 271 times.
✓ Branch 3 taken 1162 times.
✓ Branch 4 taken 269 times.
✓ Branch 5 taken 2 times.
6876 if (sectors && header && last_left == 0) {
80
2/2
✓ Branch 0 taken 114 times.
✓ Branch 1 taken 155 times.
269 if (header >> 12) {
81 114 last_left = left;
82 } else {
83 155 last_left = left = header;
84 }
85
4/4
✓ Branch 0 taken 1164 times.
✓ Branch 1 taken 5443 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 1162 times.
6607 } else if (sectors && header) {
86 2 left = header;
87 2 last_left -= left;
88
2/4
✓ Branch 0 taken 2 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 2 times.
2 if (header != 0x7FE && left < 7)
89 return 0;
90
2/2
✓ Branch 0 taken 1162 times.
✓ Branch 1 taken 5443 times.
6605 } else if (sectors) {
91
2/2
✓ Branch 0 taken 521 times.
✓ Branch 1 taken 641 times.
1162 if (left <= 8)
92 521 return 0;
93
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 641 times.
641 i += sectors ? 2048 : left + 4;
94 641 last_left = 0;
95 641 continue;
96 }
97
98
6/6
✓ Branch 0 taken 271 times.
✓ Branch 1 taken 5443 times.
✓ Branch 2 taken 16 times.
✓ Branch 3 taken 255 times.
✓ Branch 4 taken 7 times.
✓ Branch 5 taken 9 times.
5714 if (sectors && (i > 0 && left < 0x7fe) &&
99
1/2
✓ Branch 0 taken 7 times.
✗ Branch 1 not taken.
7 (i + left + 14 < p->buf_size)) {
100 7 offset = i + left + 2;
101
4/4
✓ Branch 0 taken 264 times.
✓ Branch 1 taken 5443 times.
✓ Branch 2 taken 9 times.
✓ Branch 3 taken 255 times.
5707 } else if (sectors && i > 0) {
102 9 i += 2048;
103 9 last_left -= FFMIN(last_left, 2046);
104 9 continue;
105 } else {
106 5698 offset = 0;
107 5698 last_left = left;
108 }
109
110 5705 header = AV_RB16(src + offset);
111 5705 size = AV_RB16(src + offset + 2) + 4;
112
113
2/2
✓ Branch 0 taken 104816 times.
✓ Branch 1 taken 5705 times.
110521 while ((header & 0xFF00) == 0) {
114 104816 offset++;
115
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 104816 times.
104816 if (offset + 4 >= p->buf_size)
116 break;
117 104816 header = AV_RB16(src + offset);
118 104816 size = AV_RB16(src + offset + 2) + 4;
119 }
120
121
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 5705 times.
5705 if (offset + 12 >= p->buf_size)
122 break;
123
2/2
✓ Branch 0 taken 5439 times.
✓ Branch 1 taken 266 times.
5705 if ((header & 0xFF) > 1)
124 5439 return 0;
125 266 type = header >> 8;
126
127
2/4
✓ Branch 0 taken 266 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 266 times.
✗ Branch 3 not taken.
266 if (type == 0xAA ||
128
1/2
✓ Branch 0 taken 266 times.
✗ Branch 1 not taken.
266 type == 0xA1 ||
129
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 266 times.
266 type == 0xA2 ||
130 type == 0xA3) {
131 int new_rate;
132
133 if (size <= 12)
134 return 0;
135 new_rate = AV_RB16(src + offset + 8);
136 if (sample_rate < 0)
137 sample_rate = new_rate;
138 if (sample_rate == 0 || new_rate != sample_rate)
139 return 0;
140 if (src[offset + 10] != 1)
141 return 0;
142
143 score += 10;
144
2/4
✓ Branch 0 taken 266 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 266 times.
✗ Branch 3 not taken.
266 } else if (type == 0xC1 ||
145
2/2
✓ Branch 0 taken 265 times.
✓ Branch 1 taken 1 times.
266 type == 0xC6 ||
146
1/2
✓ Branch 0 taken 265 times.
✗ Branch 1 not taken.
265 type == 0xC7 ||
147
2/2
✓ Branch 0 taken 264 times.
✓ Branch 1 taken 1 times.
265 type == 0xC8 ||
148
1/2
✓ Branch 0 taken 264 times.
✗ Branch 1 not taken.
264 type == 0xC9 ||
149
1/2
✓ Branch 0 taken 264 times.
✗ Branch 1 not taken.
264 type == 0xCB ||
150
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 264 times.
264 type == 0xCD ||
151 type == 0xE7) {
152 2 int nb_pals = src[offset + 9];
153 2 int tiles_w = src[offset + 10];
154 2 int tiles_h = src[offset + 11];
155
156
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 2 times.
2 if (size <= 12)
157 return 0;
158
2/4
✓ Branch 0 taken 2 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 2 times.
2 if (nb_pals == 0 || nb_pals > 4)
159 return 0;
160
2/4
✓ Branch 0 taken 2 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 2 times.
✗ Branch 3 not taken.
2 if (tiles_w == 0 || tiles_w > 80)
161 2 return 0;
162 if (tiles_h == 0 || tiles_h > 60)
163 return 0;
164
165 score += 10;
166
1/2
✓ Branch 0 taken 264 times.
✗ Branch 1 not taken.
264 } else if (header == 0x7FE) {
167 ;
168 } else {
169 264 return 0;
170 }
171
172 i += sectors ? 2048 : size + 4;
173 last_left -= FFMIN(last_left, 2046);
174
175 if (score < 0)
176 break;
177 }
178
179 610 return av_clip(score, 0, AVPROBE_SCORE_MAX);
180 }
181
182 static int sga_read_header(AVFormatContext *s)
183 {
184 SGADemuxContext *sga = s->priv_data;
185 AVIOContext *pb = s->pb;
186
187 sga->sector_headers = 1;
188 sga->first_audio_size = 0;
189 sga->video_stream_index = -1;
190 sga->audio_stream_index = -1;
191 sga->left = 2048;
192 sga->idx = 0;
193
194 s->ctx_flags |= AVFMTCTX_NOHEADER;
195
196 if (pb->seekable & AVIO_SEEKABLE_NORMAL) {
197 while (!avio_feof(pb)) {
198 int header = avio_rb16(pb);
199 int type = header >> 8;
200 int skip = 2046;
201 int clock;
202
203 if (!sga->first_audio_size &&
204 (type == 0xAA ||
205 type == 0xA1 ||
206 type == 0xA2 ||
207 type == 0xA3)) {
208 sga->first_audio_size = avio_rb16(pb);
209 avio_skip(pb, 4);
210 clock = avio_rb16(pb);
211 sga->sample_rate = av_rescale(clock,
212 SEGA_CD_PCM_NUM,
213 SEGA_CD_PCM_DEN);
214 skip -= 8;
215 }
216 if ((header > 0x07FE && header < 0x8100) ||
217 (header > 0x8200 && header < 0xA100) ||
218 (header > 0xA200 && header < 0xC100)) {
219 sga->sector_headers = 0;
220 break;
221 }
222
223 avio_skip(pb, skip);
224 }
225
226 avio_seek(pb, 0, SEEK_SET);
227 }
228
229 return 0;
230 }
231
232 static void print_stats(AVFormatContext *s, const char *where)
233 {
234 SGADemuxContext *sga = s->priv_data;
235
236 av_log(s, AV_LOG_DEBUG, "START %s\n", where);
237 av_log(s, AV_LOG_DEBUG, "pos: %"PRIX64"\n", avio_tell(s->pb));
238 av_log(s, AV_LOG_DEBUG, "idx: %X\n", sga->idx);
239 av_log(s, AV_LOG_DEBUG, "packet_type: %X\n", sga->packet_type);
240 av_log(s, AV_LOG_DEBUG, "payload_size: %X\n", sga->payload_size);
241 av_log(s, AV_LOG_DEBUG, "SECTOR: %016"PRIX64"\n", AV_RB64(sga->sector));
242 av_log(s, AV_LOG_DEBUG, "stream: %X\n", sga->sector[1]);
243 av_log(s, AV_LOG_DEBUG, "END %s\n", where);
244 }
245
246 static void update_type_size(AVFormatContext *s)
247 {
248 SGADemuxContext *sga = s->priv_data;
249
250 if (sga->idx >= 4) {
251 sga->packet_type = sga->sector[0];
252 sga->payload_size = AV_RB16(sga->sector + 2);
253 } else {
254 sga->packet_type = 0;
255 sga->payload_size = 0;
256 }
257 }
258
259 static int sga_video_packet(AVFormatContext *s, AVPacket *pkt)
260 {
261 SGADemuxContext *sga = s->priv_data;
262 int ret;
263
264 if (sga->payload_size <= 8)
265 return AVERROR_INVALIDDATA;
266
267 if (sga->video_stream_index == -1) {
268 AVRational frame_rate;
269
270 AVStream *st = avformat_new_stream(s, NULL);
271 if (!st)
272 return AVERROR(ENOMEM);
273
274 st->start_time = 0;
275 st->codecpar->codec_type = AVMEDIA_TYPE_VIDEO;
276 st->codecpar->codec_tag = 0;
277 st->codecpar->codec_id = AV_CODEC_ID_SGA_VIDEO;
278 sga->video_stream_index = st->index;
279
280 if (sga->first_audio_size > 0 && sga->sample_rate > 0) {
281 frame_rate.num = sga->sample_rate;
282 frame_rate.den = sga->first_audio_size;
283 } else {
284 frame_rate.num = 15;
285 frame_rate.den = 1;
286 }
287 avpriv_set_pts_info(st, 64, frame_rate.den, frame_rate.num);
288 }
289
290 ret = av_new_packet(pkt, sga->payload_size + 4);
291 if (ret < 0)
292 return AVERROR(ENOMEM);
293 memcpy(pkt->data, sga->sector, sga->payload_size + 4);
294 av_assert0(sga->idx >= sga->payload_size + 4);
295 memmove(sga->sector, sga->sector + sga->payload_size + 4, sga->idx - sga->payload_size - 4);
296
297 pkt->stream_index = sga->video_stream_index;
298 pkt->duration = 1;
299 pkt->pos = sga->pkt_pos;
300 pkt->flags |= sga->flags;
301 sga->idx -= sga->payload_size + 4;
302 sga->flags = 0;
303 update_type_size(s);
304
305 av_log(s, AV_LOG_DEBUG, "VIDEO PACKET: %d:%016"PRIX64" i:%X\n", pkt->size, AV_RB64(sga->sector), sga->idx);
306
307 return 0;
308 }
309
310 static int sga_audio_packet(AVFormatContext *s, AVPacket *pkt)
311 {
312 SGADemuxContext *sga = s->priv_data;
313 int ret;
314
315 if (sga->payload_size <= 8)
316 return AVERROR_INVALIDDATA;
317
318 if (sga->audio_stream_index == -1) {
319 AVStream *st = avformat_new_stream(s, NULL);
320 if (!st)
321 return AVERROR(ENOMEM);
322
323 st->start_time = 0;
324 st->codecpar->codec_type = AVMEDIA_TYPE_AUDIO;
325 st->codecpar->codec_tag = 0;
326 st->codecpar->codec_id = AV_CODEC_ID_PCM_SGA;
327 st->codecpar->ch_layout = (AVChannelLayout)AV_CHANNEL_LAYOUT_MONO;
328 st->codecpar->sample_rate = av_rescale(AV_RB16(sga->sector + 8),
329 SEGA_CD_PCM_NUM,
330 SEGA_CD_PCM_DEN);
331 sga->audio_stream_index = st->index;
332
333 avpriv_set_pts_info(st, 64, 1, st->codecpar->sample_rate);
334 }
335
336 ret = av_new_packet(pkt, sga->payload_size - 8);
337 if (ret < 0)
338 return AVERROR(ENOMEM);
339 memcpy(pkt->data, sga->sector + 12, sga->payload_size - 8);
340 av_assert0(sga->idx >= sga->payload_size + 4);
341 memmove(sga->sector, sga->sector + sga->payload_size + 4, sga->idx - sga->payload_size - 4);
342
343 pkt->stream_index = sga->audio_stream_index;
344 pkt->duration = pkt->size;
345 pkt->pos = sga->pkt_pos;
346 pkt->flags |= sga->flags;
347 sga->idx -= sga->payload_size + 4;
348 sga->flags = 0;
349 update_type_size(s);
350
351 av_log(s, AV_LOG_DEBUG, "AUDIO PACKET: %d:%016"PRIX64" i:%X\n", pkt->size, AV_RB64(sga->sector), sga->idx);
352
353 return 0;
354 }
355
356 static int sga_packet(AVFormatContext *s, AVPacket *pkt)
357 {
358 SGADemuxContext *sga = s->priv_data;
359 int ret = 0;
360
361 if (sga->packet_type == 0xCD ||
362 sga->packet_type == 0xCB ||
363 sga->packet_type == 0xC9 ||
364 sga->packet_type == 0xC8 ||
365 sga->packet_type == 0xC7 ||
366 sga->packet_type == 0xC6 ||
367 sga->packet_type == 0xC1 ||
368 sga->packet_type == 0xE7) {
369 ret = sga_video_packet(s, pkt);
370 } else if (sga->packet_type == 0xA1 ||
371 sga->packet_type == 0xA2 ||
372 sga->packet_type == 0xA3 ||
373 sga->packet_type == 0xAA) {
374 ret = sga_audio_packet(s, pkt);
375 } else {
376 if (sga->idx == 0)
377 return AVERROR_EOF;
378 if (sga->sector[0])
379 return AVERROR_INVALIDDATA;
380 memmove(sga->sector, sga->sector + 1, sga->idx - 1);
381 sga->idx--;
382 return AVERROR(EAGAIN);
383 }
384
385 return ret;
386 }
387
388 static int try_packet(AVFormatContext *s, AVPacket *pkt)
389 {
390 SGADemuxContext *sga = s->priv_data;
391 int ret = AVERROR(EAGAIN);
392
393 update_type_size(s);
394 if (sga->idx >= sga->payload_size + 4) {
395 print_stats(s, "before sga_packet");
396 ret = sga_packet(s, pkt);
397 print_stats(s, "after sga_packet");
398 if (ret != AVERROR(EAGAIN))
399 return ret;
400 }
401
402 return sga->idx < sga->payload_size + 4 ? AVERROR(EAGAIN) : ret;
403 }
404
405 static int sga_read_packet(AVFormatContext *s, AVPacket *pkt)
406 {
407 SGADemuxContext *sga = s->priv_data;
408 AVIOContext *pb = s->pb;
409 int header, ret = 0;
410
411 sga->pkt_pos = avio_tell(pb);
412
413 retry:
414 update_type_size(s);
415
416 print_stats(s, "start");
417 if (avio_feof(pb) &&
418 (!sga->payload_size || sga->idx < sga->payload_size + 4))
419 return AVERROR_EOF;
420
421 if (sga->idx < sga->payload_size + 4) {
422 ret = ffio_ensure_seekback(pb, 2);
423 if (ret < 0)
424 return ret;
425
426 print_stats(s, "before read header");
427 header = avio_rb16(pb);
428 if (!header) {
429 avio_skip(pb, 2046);
430 sga->left = 0;
431 } else if (!avio_feof(pb) &&
432 ((header >> 15) ||
433 !sga->sector_headers)) {
434 avio_seek(pb, -2, SEEK_CUR);
435 sga->flags = AV_PKT_FLAG_KEY;
436 sga->left = 2048;
437 } else {
438 sga->left = 2046;
439 }
440
441 av_assert0(sga->idx + sga->left < sizeof(sga->sector));
442 ret = avio_read(pb, sga->sector + sga->idx, sga->left);
443 if (ret > 0)
444 sga->idx += ret;
445 else if (ret != AVERROR_EOF && ret)
446 return ret;
447 print_stats(s, "after read header");
448
449 update_type_size(s);
450 }
451
452 ret = try_packet(s, pkt);
453 if (ret == AVERROR(EAGAIN))
454 goto retry;
455
456 return ret;
457 }
458
459 static int sga_seek(AVFormatContext *s, int stream_index,
460 int64_t timestamp, int flags)
461 {
462 SGADemuxContext *sga = s->priv_data;
463
464 sga->packet_type = sga->payload_size = sga->idx = 0;
465 memset(sga->sector, 0, sizeof(sga->sector));
466
467 return -1;
468 }
469
470 const FFInputFormat ff_sga_demuxer = {
471 .p.name = "sga",
472 .p.long_name = NULL_IF_CONFIG_SMALL("Digital Pictures SGA"),
473 .p.extensions = "sga",
474 .p.flags = AVFMT_GENERIC_INDEX,
475 .priv_data_size = sizeof(SGADemuxContext),
476 .read_probe = sga_probe,
477 .read_header = sga_read_header,
478 .read_packet = sga_read_packet,
479 .read_seek = sga_seek,
480 };
481