1cabdff1aSopenharmony_ci/* 2cabdff1aSopenharmony_ci * Simbiosis game demuxer 3cabdff1aSopenharmony_ci * 4cabdff1aSopenharmony_ci * Copyright (C) 2021 Paul B Mahol 5cabdff1aSopenharmony_ci * 6cabdff1aSopenharmony_ci * This file is part of FFmpeg. 7cabdff1aSopenharmony_ci * 8cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or 9cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public 10cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either 11cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version. 12cabdff1aSopenharmony_ci * 13cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful, 14cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of 15cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 16cabdff1aSopenharmony_ci * Lesser General Public License for more details. 17cabdff1aSopenharmony_ci * 18cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public 19cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software 20cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA 21cabdff1aSopenharmony_ci */ 22cabdff1aSopenharmony_ci 23cabdff1aSopenharmony_ci#include "avformat.h" 24cabdff1aSopenharmony_ci#include "internal.h" 25cabdff1aSopenharmony_ci#include "libavutil/intreadwrite.h" 26cabdff1aSopenharmony_ci#include "libavutil/channel_layout.h" 27cabdff1aSopenharmony_ci#include "libavutil/internal.h" 28cabdff1aSopenharmony_ci 29cabdff1aSopenharmony_ci#define IMX_TAG MKTAG('I', 'M', 'A', 'X') 30cabdff1aSopenharmony_ci 31cabdff1aSopenharmony_citypedef struct SimbiosisIMXDemuxContext { 32cabdff1aSopenharmony_ci uint8_t pal[AVPALETTE_SIZE]; 33cabdff1aSopenharmony_ci int pal_changed; 34cabdff1aSopenharmony_ci int64_t first_video_packet_pos; 35cabdff1aSopenharmony_ci} SimbiosisIMXDemuxContext; 36cabdff1aSopenharmony_ci 37cabdff1aSopenharmony_cistatic int simbiosis_imx_probe(const AVProbeData *p) 38cabdff1aSopenharmony_ci{ 39cabdff1aSopenharmony_ci if (AV_RL32(p->buf) != IMX_TAG) 40cabdff1aSopenharmony_ci return 0; 41cabdff1aSopenharmony_ci if (AV_RN32(p->buf+4) == 0) 42cabdff1aSopenharmony_ci return 0; 43cabdff1aSopenharmony_ci if (AV_RN16(p->buf+8) == 0) 44cabdff1aSopenharmony_ci return 0; 45cabdff1aSopenharmony_ci if (AV_RL16(p->buf+10) != 0x102) 46cabdff1aSopenharmony_ci return 0; 47cabdff1aSopenharmony_ci 48cabdff1aSopenharmony_ci return AVPROBE_SCORE_EXTENSION + 10; 49cabdff1aSopenharmony_ci} 50cabdff1aSopenharmony_ci 51cabdff1aSopenharmony_cistatic int simbiosis_imx_read_header(AVFormatContext *s) 52cabdff1aSopenharmony_ci{ 53cabdff1aSopenharmony_ci AVIOContext *pb = s->pb; 54cabdff1aSopenharmony_ci AVStream *vst, *ast; 55cabdff1aSopenharmony_ci int rate; 56cabdff1aSopenharmony_ci 57cabdff1aSopenharmony_ci vst = avformat_new_stream(s, NULL); 58cabdff1aSopenharmony_ci ast = avformat_new_stream(s, NULL); 59cabdff1aSopenharmony_ci if (!vst || !ast) 60cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 61cabdff1aSopenharmony_ci 62cabdff1aSopenharmony_ci avio_skip(pb, 4); 63cabdff1aSopenharmony_ci 64cabdff1aSopenharmony_ci vst->codecpar->codec_type = AVMEDIA_TYPE_VIDEO; 65cabdff1aSopenharmony_ci vst->codecpar->codec_tag = 0; 66cabdff1aSopenharmony_ci vst->codecpar->format = AV_PIX_FMT_PAL8; 67cabdff1aSopenharmony_ci vst->codecpar->codec_id = AV_CODEC_ID_SIMBIOSIS_IMX; 68cabdff1aSopenharmony_ci vst->start_time = 0; 69cabdff1aSopenharmony_ci vst->duration = 70cabdff1aSopenharmony_ci vst->nb_frames = avio_rl32(pb); 71cabdff1aSopenharmony_ci rate = avio_rl16(pb); 72cabdff1aSopenharmony_ci avio_skip(pb, 12); 73cabdff1aSopenharmony_ci 74cabdff1aSopenharmony_ci avpriv_set_pts_info(vst, 64, 1, rate); 75cabdff1aSopenharmony_ci 76cabdff1aSopenharmony_ci ast->codecpar->codec_type = AVMEDIA_TYPE_AUDIO; 77cabdff1aSopenharmony_ci ast->codecpar->codec_tag = 0; 78cabdff1aSopenharmony_ci ast->codecpar->codec_id = AV_CODEC_ID_PCM_U8; 79cabdff1aSopenharmony_ci ast->codecpar->ch_layout = (AVChannelLayout)AV_CHANNEL_LAYOUT_MONO; 80cabdff1aSopenharmony_ci ast->codecpar->sample_rate = 22050; 81cabdff1aSopenharmony_ci ast->start_time = 0; 82cabdff1aSopenharmony_ci 83cabdff1aSopenharmony_ci avpriv_set_pts_info(ast, 64, 1, 22050); 84cabdff1aSopenharmony_ci 85cabdff1aSopenharmony_ci return 0; 86cabdff1aSopenharmony_ci} 87cabdff1aSopenharmony_ci 88cabdff1aSopenharmony_cistatic int simbiosis_imx_read_packet(AVFormatContext *s, AVPacket *pkt) 89cabdff1aSopenharmony_ci{ 90cabdff1aSopenharmony_ci AVIOContext *pb = s->pb; 91cabdff1aSopenharmony_ci SimbiosisIMXDemuxContext *imx = s->priv_data; 92cabdff1aSopenharmony_ci uint32_t chunk_size, chunk_type; 93cabdff1aSopenharmony_ci int64_t pos = avio_tell(pb); 94cabdff1aSopenharmony_ci int ret, idx = -1; 95cabdff1aSopenharmony_ci 96cabdff1aSopenharmony_ciretry: 97cabdff1aSopenharmony_ci if (avio_feof(pb)) 98cabdff1aSopenharmony_ci return AVERROR_EOF; 99cabdff1aSopenharmony_ci 100cabdff1aSopenharmony_ci chunk_size = avio_rl32(pb); 101cabdff1aSopenharmony_ci chunk_type = avio_rl32(pb); 102cabdff1aSopenharmony_ci 103cabdff1aSopenharmony_ci switch (chunk_type) { 104cabdff1aSopenharmony_ci case 0xAAFF: 105cabdff1aSopenharmony_ci return AVERROR_EOF; 106cabdff1aSopenharmony_ci case 0xAA99: 107cabdff1aSopenharmony_ci idx = 1; 108cabdff1aSopenharmony_ci break; 109cabdff1aSopenharmony_ci case 0xAA97: 110cabdff1aSopenharmony_ci idx = 0; 111cabdff1aSopenharmony_ci if (!imx->first_video_packet_pos) 112cabdff1aSopenharmony_ci imx->first_video_packet_pos = pos; 113cabdff1aSopenharmony_ci break; 114cabdff1aSopenharmony_ci case 0xAA98: 115cabdff1aSopenharmony_ci if (chunk_size > 256 * 3) 116cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 117cabdff1aSopenharmony_ci for (int i = 0; i < chunk_size / 3; i++) { 118cabdff1aSopenharmony_ci unsigned r = avio_r8(pb) << 18; 119cabdff1aSopenharmony_ci unsigned g = avio_r8(pb) << 10; 120cabdff1aSopenharmony_ci unsigned b = avio_r8(pb) << 2; 121cabdff1aSopenharmony_ci 122cabdff1aSopenharmony_ci AV_WL32(imx->pal + i * 4, (0xFFU << 24) | r | g | b); 123cabdff1aSopenharmony_ci } 124cabdff1aSopenharmony_ci imx->pal_changed = 1; 125cabdff1aSopenharmony_ci idx = -1; 126cabdff1aSopenharmony_ci break; 127cabdff1aSopenharmony_ci default: 128cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 129cabdff1aSopenharmony_ci } 130cabdff1aSopenharmony_ci 131cabdff1aSopenharmony_ci if (idx == -1) 132cabdff1aSopenharmony_ci goto retry; 133cabdff1aSopenharmony_ci 134cabdff1aSopenharmony_ci ret = av_get_packet(pb, pkt, chunk_size); 135cabdff1aSopenharmony_ci if (ret < 0) 136cabdff1aSopenharmony_ci return ret; 137cabdff1aSopenharmony_ci 138cabdff1aSopenharmony_ci if (imx->pal_changed && idx == 0) { 139cabdff1aSopenharmony_ci uint8_t *pal = av_packet_new_side_data(pkt, AV_PKT_DATA_PALETTE, 140cabdff1aSopenharmony_ci AVPALETTE_SIZE); 141cabdff1aSopenharmony_ci if (!pal) 142cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 143cabdff1aSopenharmony_ci memcpy(pal, imx->pal, AVPALETTE_SIZE); 144cabdff1aSopenharmony_ci imx->pal_changed = 0; 145cabdff1aSopenharmony_ci if (pos <= imx->first_video_packet_pos) 146cabdff1aSopenharmony_ci pkt->flags |= AV_PKT_FLAG_KEY; 147cabdff1aSopenharmony_ci } else if (idx == 1) { 148cabdff1aSopenharmony_ci pkt->flags |= AV_PKT_FLAG_KEY; 149cabdff1aSopenharmony_ci } 150cabdff1aSopenharmony_ci 151cabdff1aSopenharmony_ci pkt->pos = pos; 152cabdff1aSopenharmony_ci pkt->stream_index = idx; 153cabdff1aSopenharmony_ci pkt->duration = idx ? chunk_size : 1; 154cabdff1aSopenharmony_ci 155cabdff1aSopenharmony_ci return ret; 156cabdff1aSopenharmony_ci} 157cabdff1aSopenharmony_ci 158cabdff1aSopenharmony_ciconst AVInputFormat ff_simbiosis_imx_demuxer = { 159cabdff1aSopenharmony_ci .name = "simbiosis_imx", 160cabdff1aSopenharmony_ci .long_name = NULL_IF_CONFIG_SMALL("Simbiosis Interactive IMX"), 161cabdff1aSopenharmony_ci .priv_data_size = sizeof(SimbiosisIMXDemuxContext), 162cabdff1aSopenharmony_ci .read_probe = simbiosis_imx_probe, 163cabdff1aSopenharmony_ci .read_header = simbiosis_imx_read_header, 164cabdff1aSopenharmony_ci .read_packet = simbiosis_imx_read_packet, 165cabdff1aSopenharmony_ci .extensions = "imx", 166cabdff1aSopenharmony_ci .flags = AVFMT_GENERIC_INDEX, 167cabdff1aSopenharmony_ci}; 168