Spaces:
Runtime error
Runtime error
/* | |
* 3DO STR demuxer | |
* Copyright (c) 2015 Paul B Mahol | |
* | |
* This file is part of FFmpeg. | |
* | |
* FFmpeg is free software; you can redistribute it and/or | |
* modify it under the terms of the GNU Lesser General Public | |
* License as published by the Free Software Foundation; either | |
* version 2.1 of the License, or (at your option) any later version. | |
* | |
* FFmpeg is distributed in the hope that it will be useful, | |
* but WITHOUT ANY WARRANTY; without even the implied warranty of | |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | |
* Lesser General Public License for more details. | |
* | |
* You should have received a copy of the GNU Lesser General Public | |
* License along with FFmpeg; if not, write to the Free Software | |
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA | |
*/ | |
static int threedostr_probe(const AVProbeData *p) | |
{ | |
for (int i = 0; i < p->buf_size;) { | |
unsigned chunk = AV_RL32(p->buf + i); | |
unsigned size = AV_RB32(p->buf + i + 4); | |
if (size < 8 || p->buf_size - i < size) | |
return 0; | |
i += 8; | |
size -= 8; | |
switch (chunk) { | |
case MKTAG('C','T','R','L'): | |
break; | |
case MKTAG('S','N','D','S'): | |
if (size < 56) | |
return 0; | |
i += 8; | |
if (AV_RL32(p->buf + i) != MKTAG('S','H','D','R')) | |
return 0; | |
i += 28; | |
if (AV_RB32(p->buf + i) <= 0) | |
return 0; | |
i += 4; | |
if (AV_RB32(p->buf + i) <= 0) | |
return 0; | |
i += 4; | |
if (AV_RL32(p->buf + i) == MKTAG('S','D','X','2')) | |
return AVPROBE_SCORE_MAX; | |
else | |
return 0; | |
break; | |
case MKTAG('S','H','D','R'): | |
if (size > 0x78) { | |
i += 0x78; | |
size -= 0x78; | |
} | |
break; | |
default: | |
break; | |
} | |
i += size; | |
} | |
return 0; | |
} | |
static int threedostr_read_header(AVFormatContext *s) | |
{ | |
unsigned chunk, codec = 0, size, ctrl_size = -1, found_shdr = 0; | |
AVStream *st; | |
while (!avio_feof(s->pb) && !found_shdr) { | |
chunk = avio_rl32(s->pb); | |
size = avio_rb32(s->pb); | |
if (size < 8) | |
return AVERROR_INVALIDDATA; | |
size -= 8; | |
switch (chunk) { | |
case MKTAG('C','T','R','L'): | |
ctrl_size = size; | |
break; | |
case MKTAG('S','N','D','S'): | |
if (size < 56) | |
return AVERROR_INVALIDDATA; | |
avio_skip(s->pb, 8); | |
if (avio_rl32(s->pb) != MKTAG('S','H','D','R')) | |
return AVERROR_INVALIDDATA; | |
avio_skip(s->pb, 24); | |
st = avformat_new_stream(s, NULL); | |
if (!st) | |
return AVERROR(ENOMEM); | |
st->codecpar->codec_type = AVMEDIA_TYPE_AUDIO; | |
st->codecpar->sample_rate = avio_rb32(s->pb); | |
st->codecpar->ch_layout.nb_channels = avio_rb32(s->pb); | |
if (st->codecpar->ch_layout.nb_channels <= 0 || st->codecpar->sample_rate <= 0) | |
return AVERROR_INVALIDDATA; | |
codec = avio_rl32(s->pb); | |
avio_skip(s->pb, 4); | |
if (ctrl_size == 20 || ctrl_size == 3 || ctrl_size == -1) | |
st->duration = (avio_rb32(s->pb) - 1) / st->codecpar->ch_layout.nb_channels; | |
else | |
st->duration = avio_rb32(s->pb) * 16 / st->codecpar->ch_layout.nb_channels; | |
size -= 56; | |
found_shdr = 1; | |
break; | |
case MKTAG('S','H','D','R'): | |
if (size > 0x78) { | |
avio_skip(s->pb, 0x74); | |
size -= 0x78; | |
if (avio_rl32(s->pb) == MKTAG('C','T','R','L') && size > 4) { | |
ctrl_size = avio_rb32(s->pb); | |
size -= 4; | |
} | |
} | |
break; | |
default: | |
av_log(s, AV_LOG_DEBUG, "skipping unknown chunk: %X\n", chunk); | |
break; | |
} | |
avio_skip(s->pb, size); | |
} | |
switch (codec) { | |
case MKTAG('S','D','X','2'): | |
st->codecpar->codec_id = AV_CODEC_ID_SDX2_DPCM; | |
st->codecpar->block_align = 1 * st->codecpar->ch_layout.nb_channels; | |
break; | |
default: | |
avpriv_request_sample(s, "codec %X", codec); | |
return AVERROR_PATCHWELCOME; | |
} | |
avpriv_set_pts_info(st, 64, 1, st->codecpar->sample_rate); | |
return 0; | |
} | |
static int threedostr_read_packet(AVFormatContext *s, AVPacket *pkt) | |
{ | |
unsigned chunk, size; | |
AVStream *st = s->streams[0]; | |
int64_t pos; | |
int ret = 0; | |
while (!avio_feof(s->pb)) { | |
pos = avio_tell(s->pb); | |
chunk = avio_rl32(s->pb); | |
size = avio_rb32(s->pb); | |
if (!size) | |
continue; | |
if (size < 8) | |
return AVERROR_INVALIDDATA; | |
size -= 8; | |
switch (chunk) { | |
case MKTAG('S','N','D','S'): | |
if (size <= 16) | |
return AVERROR_INVALIDDATA; | |
avio_skip(s->pb, 8); | |
if (avio_rl32(s->pb) != MKTAG('S','S','M','P')) | |
return AVERROR_INVALIDDATA; | |
avio_skip(s->pb, 4); | |
size -= 16; | |
ret = av_get_packet(s->pb, pkt, size); | |
pkt->pos = pos; | |
pkt->stream_index = 0; | |
pkt->duration = size / st->codecpar->ch_layout.nb_channels; | |
return ret; | |
default: | |
av_log(s, AV_LOG_DEBUG, "skipping unknown chunk: %X\n", chunk); | |
break; | |
} | |
avio_skip(s->pb, size); | |
} | |
return AVERROR_EOF; | |
} | |
const AVInputFormat ff_threedostr_demuxer = { | |
.name = "3dostr", | |
.long_name = NULL_IF_CONFIG_SMALL("3DO STR"), | |
.read_probe = threedostr_probe, | |
.read_header = threedostr_read_header, | |
.read_packet = threedostr_read_packet, | |
.extensions = "str", | |
.flags = AVFMT_GENERIC_INDEX, | |
}; | |