1cabdff1aSopenharmony_ci/* 2cabdff1aSopenharmony_ci * 3DO STR demuxer 3cabdff1aSopenharmony_ci * Copyright (c) 2015 Paul B Mahol 4cabdff1aSopenharmony_ci * 5cabdff1aSopenharmony_ci * This file is part of FFmpeg. 6cabdff1aSopenharmony_ci * 7cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or 8cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public 9cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either 10cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version. 11cabdff1aSopenharmony_ci * 12cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful, 13cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of 14cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 15cabdff1aSopenharmony_ci * Lesser General Public License for more details. 16cabdff1aSopenharmony_ci * 17cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public 18cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software 19cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA 20cabdff1aSopenharmony_ci */ 21cabdff1aSopenharmony_ci 22cabdff1aSopenharmony_ci#include "libavutil/intreadwrite.h" 23cabdff1aSopenharmony_ci#include "avformat.h" 24cabdff1aSopenharmony_ci#include "internal.h" 25cabdff1aSopenharmony_ci 26cabdff1aSopenharmony_cistatic int threedostr_probe(const AVProbeData *p) 27cabdff1aSopenharmony_ci{ 28cabdff1aSopenharmony_ci for (int i = 0; i < p->buf_size;) { 29cabdff1aSopenharmony_ci unsigned chunk = AV_RL32(p->buf + i); 30cabdff1aSopenharmony_ci unsigned size = AV_RB32(p->buf + i + 4); 31cabdff1aSopenharmony_ci 32cabdff1aSopenharmony_ci if (size < 8 || p->buf_size - i < size) 33cabdff1aSopenharmony_ci return 0; 34cabdff1aSopenharmony_ci i += 8; 35cabdff1aSopenharmony_ci size -= 8; 36cabdff1aSopenharmony_ci switch (chunk) { 37cabdff1aSopenharmony_ci case MKTAG('C','T','R','L'): 38cabdff1aSopenharmony_ci break; 39cabdff1aSopenharmony_ci case MKTAG('S','N','D','S'): 40cabdff1aSopenharmony_ci if (size < 56) 41cabdff1aSopenharmony_ci return 0; 42cabdff1aSopenharmony_ci i += 8; 43cabdff1aSopenharmony_ci if (AV_RL32(p->buf + i) != MKTAG('S','H','D','R')) 44cabdff1aSopenharmony_ci return 0; 45cabdff1aSopenharmony_ci i += 28; 46cabdff1aSopenharmony_ci 47cabdff1aSopenharmony_ci if (AV_RB32(p->buf + i) <= 0) 48cabdff1aSopenharmony_ci return 0; 49cabdff1aSopenharmony_ci i += 4; 50cabdff1aSopenharmony_ci if (AV_RB32(p->buf + i) <= 0) 51cabdff1aSopenharmony_ci return 0; 52cabdff1aSopenharmony_ci i += 4; 53cabdff1aSopenharmony_ci if (AV_RL32(p->buf + i) == MKTAG('S','D','X','2')) 54cabdff1aSopenharmony_ci return AVPROBE_SCORE_MAX; 55cabdff1aSopenharmony_ci else 56cabdff1aSopenharmony_ci return 0; 57cabdff1aSopenharmony_ci break; 58cabdff1aSopenharmony_ci case MKTAG('S','H','D','R'): 59cabdff1aSopenharmony_ci if (size > 0x78) { 60cabdff1aSopenharmony_ci i += 0x78; 61cabdff1aSopenharmony_ci size -= 0x78; 62cabdff1aSopenharmony_ci } 63cabdff1aSopenharmony_ci break; 64cabdff1aSopenharmony_ci default: 65cabdff1aSopenharmony_ci break; 66cabdff1aSopenharmony_ci } 67cabdff1aSopenharmony_ci 68cabdff1aSopenharmony_ci i += size; 69cabdff1aSopenharmony_ci } 70cabdff1aSopenharmony_ci 71cabdff1aSopenharmony_ci return 0; 72cabdff1aSopenharmony_ci} 73cabdff1aSopenharmony_ci 74cabdff1aSopenharmony_cistatic int threedostr_read_header(AVFormatContext *s) 75cabdff1aSopenharmony_ci{ 76cabdff1aSopenharmony_ci unsigned chunk, codec = 0, size, ctrl_size = -1, found_shdr = 0; 77cabdff1aSopenharmony_ci AVStream *st; 78cabdff1aSopenharmony_ci 79cabdff1aSopenharmony_ci while (!avio_feof(s->pb) && !found_shdr) { 80cabdff1aSopenharmony_ci chunk = avio_rl32(s->pb); 81cabdff1aSopenharmony_ci size = avio_rb32(s->pb); 82cabdff1aSopenharmony_ci 83cabdff1aSopenharmony_ci if (size < 8) 84cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 85cabdff1aSopenharmony_ci size -= 8; 86cabdff1aSopenharmony_ci 87cabdff1aSopenharmony_ci switch (chunk) { 88cabdff1aSopenharmony_ci case MKTAG('C','T','R','L'): 89cabdff1aSopenharmony_ci ctrl_size = size; 90cabdff1aSopenharmony_ci break; 91cabdff1aSopenharmony_ci case MKTAG('S','N','D','S'): 92cabdff1aSopenharmony_ci if (size < 56) 93cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 94cabdff1aSopenharmony_ci avio_skip(s->pb, 8); 95cabdff1aSopenharmony_ci if (avio_rl32(s->pb) != MKTAG('S','H','D','R')) 96cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 97cabdff1aSopenharmony_ci avio_skip(s->pb, 24); 98cabdff1aSopenharmony_ci 99cabdff1aSopenharmony_ci st = avformat_new_stream(s, NULL); 100cabdff1aSopenharmony_ci if (!st) 101cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 102cabdff1aSopenharmony_ci 103cabdff1aSopenharmony_ci st->codecpar->codec_type = AVMEDIA_TYPE_AUDIO; 104cabdff1aSopenharmony_ci st->codecpar->sample_rate = avio_rb32(s->pb); 105cabdff1aSopenharmony_ci st->codecpar->ch_layout.nb_channels = avio_rb32(s->pb); 106cabdff1aSopenharmony_ci if (st->codecpar->ch_layout.nb_channels <= 0 || st->codecpar->sample_rate <= 0) 107cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 108cabdff1aSopenharmony_ci codec = avio_rl32(s->pb); 109cabdff1aSopenharmony_ci avio_skip(s->pb, 4); 110cabdff1aSopenharmony_ci if (ctrl_size == 20 || ctrl_size == 3 || ctrl_size == -1) 111cabdff1aSopenharmony_ci st->duration = (avio_rb32(s->pb) - 1) / st->codecpar->ch_layout.nb_channels; 112cabdff1aSopenharmony_ci else 113cabdff1aSopenharmony_ci st->duration = avio_rb32(s->pb) * 16 / st->codecpar->ch_layout.nb_channels; 114cabdff1aSopenharmony_ci size -= 56; 115cabdff1aSopenharmony_ci found_shdr = 1; 116cabdff1aSopenharmony_ci break; 117cabdff1aSopenharmony_ci case MKTAG('S','H','D','R'): 118cabdff1aSopenharmony_ci if (size > 0x78) { 119cabdff1aSopenharmony_ci avio_skip(s->pb, 0x74); 120cabdff1aSopenharmony_ci size -= 0x78; 121cabdff1aSopenharmony_ci if (avio_rl32(s->pb) == MKTAG('C','T','R','L') && size > 4) { 122cabdff1aSopenharmony_ci ctrl_size = avio_rb32(s->pb); 123cabdff1aSopenharmony_ci size -= 4; 124cabdff1aSopenharmony_ci } 125cabdff1aSopenharmony_ci } 126cabdff1aSopenharmony_ci break; 127cabdff1aSopenharmony_ci default: 128cabdff1aSopenharmony_ci av_log(s, AV_LOG_DEBUG, "skipping unknown chunk: %X\n", chunk); 129cabdff1aSopenharmony_ci break; 130cabdff1aSopenharmony_ci } 131cabdff1aSopenharmony_ci 132cabdff1aSopenharmony_ci avio_skip(s->pb, size); 133cabdff1aSopenharmony_ci } 134cabdff1aSopenharmony_ci 135cabdff1aSopenharmony_ci switch (codec) { 136cabdff1aSopenharmony_ci case MKTAG('S','D','X','2'): 137cabdff1aSopenharmony_ci st->codecpar->codec_id = AV_CODEC_ID_SDX2_DPCM; 138cabdff1aSopenharmony_ci st->codecpar->block_align = 1 * st->codecpar->ch_layout.nb_channels; 139cabdff1aSopenharmony_ci break; 140cabdff1aSopenharmony_ci default: 141cabdff1aSopenharmony_ci avpriv_request_sample(s, "codec %X", codec); 142cabdff1aSopenharmony_ci return AVERROR_PATCHWELCOME; 143cabdff1aSopenharmony_ci } 144cabdff1aSopenharmony_ci 145cabdff1aSopenharmony_ci avpriv_set_pts_info(st, 64, 1, st->codecpar->sample_rate); 146cabdff1aSopenharmony_ci 147cabdff1aSopenharmony_ci return 0; 148cabdff1aSopenharmony_ci} 149cabdff1aSopenharmony_ci 150cabdff1aSopenharmony_cistatic int threedostr_read_packet(AVFormatContext *s, AVPacket *pkt) 151cabdff1aSopenharmony_ci{ 152cabdff1aSopenharmony_ci unsigned chunk, size; 153cabdff1aSopenharmony_ci AVStream *st = s->streams[0]; 154cabdff1aSopenharmony_ci int64_t pos; 155cabdff1aSopenharmony_ci int ret = 0; 156cabdff1aSopenharmony_ci 157cabdff1aSopenharmony_ci while (!avio_feof(s->pb)) { 158cabdff1aSopenharmony_ci pos = avio_tell(s->pb); 159cabdff1aSopenharmony_ci chunk = avio_rl32(s->pb); 160cabdff1aSopenharmony_ci size = avio_rb32(s->pb); 161cabdff1aSopenharmony_ci 162cabdff1aSopenharmony_ci if (!size) 163cabdff1aSopenharmony_ci continue; 164cabdff1aSopenharmony_ci 165cabdff1aSopenharmony_ci if (size < 8) 166cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 167cabdff1aSopenharmony_ci size -= 8; 168cabdff1aSopenharmony_ci 169cabdff1aSopenharmony_ci switch (chunk) { 170cabdff1aSopenharmony_ci case MKTAG('S','N','D','S'): 171cabdff1aSopenharmony_ci if (size <= 16) 172cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 173cabdff1aSopenharmony_ci avio_skip(s->pb, 8); 174cabdff1aSopenharmony_ci if (avio_rl32(s->pb) != MKTAG('S','S','M','P')) 175cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 176cabdff1aSopenharmony_ci avio_skip(s->pb, 4); 177cabdff1aSopenharmony_ci size -= 16; 178cabdff1aSopenharmony_ci ret = av_get_packet(s->pb, pkt, size); 179cabdff1aSopenharmony_ci pkt->pos = pos; 180cabdff1aSopenharmony_ci pkt->stream_index = 0; 181cabdff1aSopenharmony_ci pkt->duration = size / st->codecpar->ch_layout.nb_channels; 182cabdff1aSopenharmony_ci return ret; 183cabdff1aSopenharmony_ci default: 184cabdff1aSopenharmony_ci av_log(s, AV_LOG_DEBUG, "skipping unknown chunk: %X\n", chunk); 185cabdff1aSopenharmony_ci break; 186cabdff1aSopenharmony_ci } 187cabdff1aSopenharmony_ci 188cabdff1aSopenharmony_ci avio_skip(s->pb, size); 189cabdff1aSopenharmony_ci } 190cabdff1aSopenharmony_ci 191cabdff1aSopenharmony_ci return AVERROR_EOF; 192cabdff1aSopenharmony_ci} 193cabdff1aSopenharmony_ci 194cabdff1aSopenharmony_ciconst AVInputFormat ff_threedostr_demuxer = { 195cabdff1aSopenharmony_ci .name = "3dostr", 196cabdff1aSopenharmony_ci .long_name = NULL_IF_CONFIG_SMALL("3DO STR"), 197cabdff1aSopenharmony_ci .read_probe = threedostr_probe, 198cabdff1aSopenharmony_ci .read_header = threedostr_read_header, 199cabdff1aSopenharmony_ci .read_packet = threedostr_read_packet, 200cabdff1aSopenharmony_ci .extensions = "str", 201cabdff1aSopenharmony_ci .flags = AVFMT_GENERIC_INDEX, 202cabdff1aSopenharmony_ci}; 203