1cabdff1aSopenharmony_ci/* 2cabdff1aSopenharmony_ci * RedSpark demuxer 3cabdff1aSopenharmony_ci * Copyright (c) 2013 James Almer 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 "libavcodec/bytestream.h" 23cabdff1aSopenharmony_ci#include "libavutil/intreadwrite.h" 24cabdff1aSopenharmony_ci#include "avformat.h" 25cabdff1aSopenharmony_ci#include "avio.h" 26cabdff1aSopenharmony_ci#include "internal.h" 27cabdff1aSopenharmony_ci 28cabdff1aSopenharmony_ci#define HEADER_SIZE 4096 29cabdff1aSopenharmony_ci#define rol(value, bits) (((value) << (bits)) | ((value) >> (32 - (bits)))) 30cabdff1aSopenharmony_ci 31cabdff1aSopenharmony_citypedef struct RedSparkContext { 32cabdff1aSopenharmony_ci int samples_count; 33cabdff1aSopenharmony_ci} RedSparkContext; 34cabdff1aSopenharmony_ci 35cabdff1aSopenharmony_cistatic int redspark_probe(const AVProbeData *p) 36cabdff1aSopenharmony_ci{ 37cabdff1aSopenharmony_ci uint32_t key, data; 38cabdff1aSopenharmony_ci uint8_t header[8]; 39cabdff1aSopenharmony_ci 40cabdff1aSopenharmony_ci /* Decrypt first 8 bytes of the header */ 41cabdff1aSopenharmony_ci data = AV_RB32(p->buf); 42cabdff1aSopenharmony_ci key = data ^ 0x52656453; 43cabdff1aSopenharmony_ci data ^= key; 44cabdff1aSopenharmony_ci AV_WB32(header, data); 45cabdff1aSopenharmony_ci key = rol(key, 11); 46cabdff1aSopenharmony_ci 47cabdff1aSopenharmony_ci key += rol(key, 3); 48cabdff1aSopenharmony_ci data = AV_RB32(p->buf + 4) ^ key; 49cabdff1aSopenharmony_ci AV_WB32(header + 4, data); 50cabdff1aSopenharmony_ci 51cabdff1aSopenharmony_ci if (AV_RB64(header) == AV_RB64("RedSpark")) 52cabdff1aSopenharmony_ci return AVPROBE_SCORE_MAX; 53cabdff1aSopenharmony_ci 54cabdff1aSopenharmony_ci return 0; 55cabdff1aSopenharmony_ci} 56cabdff1aSopenharmony_ci 57cabdff1aSopenharmony_cistatic int redspark_read_header(AVFormatContext *s) 58cabdff1aSopenharmony_ci{ 59cabdff1aSopenharmony_ci AVIOContext *pb = s->pb; 60cabdff1aSopenharmony_ci RedSparkContext *redspark = s->priv_data; 61cabdff1aSopenharmony_ci AVCodecParameters *par; 62cabdff1aSopenharmony_ci GetByteContext gbc; 63cabdff1aSopenharmony_ci int i, coef_off, ret = 0; 64cabdff1aSopenharmony_ci uint32_t key, data; 65cabdff1aSopenharmony_ci uint8_t header[HEADER_SIZE]; 66cabdff1aSopenharmony_ci AVStream *st; 67cabdff1aSopenharmony_ci 68cabdff1aSopenharmony_ci st = avformat_new_stream(s, NULL); 69cabdff1aSopenharmony_ci if (!st) 70cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 71cabdff1aSopenharmony_ci par = st->codecpar; 72cabdff1aSopenharmony_ci 73cabdff1aSopenharmony_ci /* Decrypt header */ 74cabdff1aSopenharmony_ci data = avio_rb32(pb); 75cabdff1aSopenharmony_ci key = data ^ 0x52656453; 76cabdff1aSopenharmony_ci data ^= key; 77cabdff1aSopenharmony_ci AV_WB32(header, data); 78cabdff1aSopenharmony_ci key = rol(key, 11); 79cabdff1aSopenharmony_ci 80cabdff1aSopenharmony_ci for (i = 4; i < HEADER_SIZE; i += 4) { 81cabdff1aSopenharmony_ci key += rol(key, 3); 82cabdff1aSopenharmony_ci data = avio_rb32(pb) ^ key; 83cabdff1aSopenharmony_ci AV_WB32(header + i, data); 84cabdff1aSopenharmony_ci } 85cabdff1aSopenharmony_ci 86cabdff1aSopenharmony_ci par->codec_id = AV_CODEC_ID_ADPCM_THP; 87cabdff1aSopenharmony_ci par->codec_type = AVMEDIA_TYPE_AUDIO; 88cabdff1aSopenharmony_ci 89cabdff1aSopenharmony_ci bytestream2_init(&gbc, header, HEADER_SIZE); 90cabdff1aSopenharmony_ci bytestream2_seek(&gbc, 0x3c, SEEK_SET); 91cabdff1aSopenharmony_ci par->sample_rate = bytestream2_get_be32u(&gbc); 92cabdff1aSopenharmony_ci if (par->sample_rate <= 0 || par->sample_rate > 96000) { 93cabdff1aSopenharmony_ci av_log(s, AV_LOG_ERROR, "Invalid sample rate: %d\n", par->sample_rate); 94cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 95cabdff1aSopenharmony_ci } 96cabdff1aSopenharmony_ci 97cabdff1aSopenharmony_ci st->duration = bytestream2_get_be32u(&gbc) * 14; 98cabdff1aSopenharmony_ci redspark->samples_count = 0; 99cabdff1aSopenharmony_ci bytestream2_skipu(&gbc, 10); 100cabdff1aSopenharmony_ci par->ch_layout.nb_channels = bytestream2_get_byteu(&gbc); 101cabdff1aSopenharmony_ci if (!par->ch_layout.nb_channels) { 102cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 103cabdff1aSopenharmony_ci } 104cabdff1aSopenharmony_ci 105cabdff1aSopenharmony_ci coef_off = 0x54 + par->ch_layout.nb_channels * 8; 106cabdff1aSopenharmony_ci if (bytestream2_get_byteu(&gbc)) // Loop flag 107cabdff1aSopenharmony_ci coef_off += 16; 108cabdff1aSopenharmony_ci 109cabdff1aSopenharmony_ci if (coef_off + par->ch_layout.nb_channels * (32 + 14) > HEADER_SIZE) { 110cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 111cabdff1aSopenharmony_ci } 112cabdff1aSopenharmony_ci 113cabdff1aSopenharmony_ci if (ff_alloc_extradata(par, 32 * par->ch_layout.nb_channels)) { 114cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 115cabdff1aSopenharmony_ci } 116cabdff1aSopenharmony_ci 117cabdff1aSopenharmony_ci /* Get the ADPCM table */ 118cabdff1aSopenharmony_ci bytestream2_seek(&gbc, coef_off, SEEK_SET); 119cabdff1aSopenharmony_ci for (i = 0; i < par->ch_layout.nb_channels; i++) { 120cabdff1aSopenharmony_ci if (bytestream2_get_bufferu(&gbc, par->extradata + i * 32, 32) != 32) { 121cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 122cabdff1aSopenharmony_ci } 123cabdff1aSopenharmony_ci bytestream2_skipu(&gbc, 14); 124cabdff1aSopenharmony_ci } 125cabdff1aSopenharmony_ci 126cabdff1aSopenharmony_ci avpriv_set_pts_info(st, 64, 1, par->sample_rate); 127cabdff1aSopenharmony_ci 128cabdff1aSopenharmony_ci return ret; 129cabdff1aSopenharmony_ci} 130cabdff1aSopenharmony_ci 131cabdff1aSopenharmony_cistatic int redspark_read_packet(AVFormatContext *s, AVPacket *pkt) 132cabdff1aSopenharmony_ci{ 133cabdff1aSopenharmony_ci AVCodecParameters *par = s->streams[0]->codecpar; 134cabdff1aSopenharmony_ci RedSparkContext *redspark = s->priv_data; 135cabdff1aSopenharmony_ci uint32_t size = 8 * par->ch_layout.nb_channels; 136cabdff1aSopenharmony_ci int ret; 137cabdff1aSopenharmony_ci 138cabdff1aSopenharmony_ci if (avio_feof(s->pb) || redspark->samples_count == s->streams[0]->duration) 139cabdff1aSopenharmony_ci return AVERROR_EOF; 140cabdff1aSopenharmony_ci 141cabdff1aSopenharmony_ci ret = av_get_packet(s->pb, pkt, size); 142cabdff1aSopenharmony_ci if (ret != size) { 143cabdff1aSopenharmony_ci return AVERROR(EIO); 144cabdff1aSopenharmony_ci } 145cabdff1aSopenharmony_ci 146cabdff1aSopenharmony_ci pkt->duration = 14; 147cabdff1aSopenharmony_ci redspark->samples_count += pkt->duration; 148cabdff1aSopenharmony_ci pkt->stream_index = 0; 149cabdff1aSopenharmony_ci 150cabdff1aSopenharmony_ci return ret; 151cabdff1aSopenharmony_ci} 152cabdff1aSopenharmony_ci 153cabdff1aSopenharmony_ciconst AVInputFormat ff_redspark_demuxer = { 154cabdff1aSopenharmony_ci .name = "redspark", 155cabdff1aSopenharmony_ci .long_name = NULL_IF_CONFIG_SMALL("RedSpark"), 156cabdff1aSopenharmony_ci .priv_data_size = sizeof(RedSparkContext), 157cabdff1aSopenharmony_ci .read_probe = redspark_probe, 158cabdff1aSopenharmony_ci .read_header = redspark_read_header, 159cabdff1aSopenharmony_ci .read_packet = redspark_read_packet, 160cabdff1aSopenharmony_ci .extensions = "rsd", 161cabdff1aSopenharmony_ci}; 162