1cabdff1aSopenharmony_ci/*
2cabdff1aSopenharmony_ci * Copyright (c) 2011 Justin Ruggles
3cabdff1aSopenharmony_ci *
4cabdff1aSopenharmony_ci * This file is part of FFmpeg.
5cabdff1aSopenharmony_ci *
6cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or
7cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public
8cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either
9cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version.
10cabdff1aSopenharmony_ci *
11cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful,
12cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of
13cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14cabdff1aSopenharmony_ci * Lesser General Public License for more details.
15cabdff1aSopenharmony_ci *
16cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public
17cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software
18cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19cabdff1aSopenharmony_ci */
20cabdff1aSopenharmony_ci
21cabdff1aSopenharmony_ci/**
22cabdff1aSopenharmony_ci * @file
23cabdff1aSopenharmony_ci * CRI ADX demuxer
24cabdff1aSopenharmony_ci */
25cabdff1aSopenharmony_ci
26cabdff1aSopenharmony_ci#include "libavutil/intreadwrite.h"
27cabdff1aSopenharmony_ci#include "avformat.h"
28cabdff1aSopenharmony_ci#include "demux.h"
29cabdff1aSopenharmony_ci#include "internal.h"
30cabdff1aSopenharmony_ci
31cabdff1aSopenharmony_ci#define BLOCK_SIZE    18
32cabdff1aSopenharmony_ci#define BLOCK_SAMPLES 32
33cabdff1aSopenharmony_ci
34cabdff1aSopenharmony_citypedef struct ADXDemuxerContext {
35cabdff1aSopenharmony_ci    int header_size;
36cabdff1aSopenharmony_ci} ADXDemuxerContext;
37cabdff1aSopenharmony_ci
38cabdff1aSopenharmony_cistatic int adx_probe(const AVProbeData *p)
39cabdff1aSopenharmony_ci{
40cabdff1aSopenharmony_ci    int offset;
41cabdff1aSopenharmony_ci    if (AV_RB16(p->buf) != 0x8000)
42cabdff1aSopenharmony_ci        return 0;
43cabdff1aSopenharmony_ci    offset = AV_RB16(&p->buf[2]);
44cabdff1aSopenharmony_ci    if (   offset < 8
45cabdff1aSopenharmony_ci        || offset > p->buf_size - 4
46cabdff1aSopenharmony_ci        || memcmp(p->buf + offset - 2, "(c)CRI", 6))
47cabdff1aSopenharmony_ci        return 0;
48cabdff1aSopenharmony_ci    return AVPROBE_SCORE_MAX * 3 / 4;
49cabdff1aSopenharmony_ci}
50cabdff1aSopenharmony_ci
51cabdff1aSopenharmony_cistatic int adx_read_packet(AVFormatContext *s, AVPacket *pkt)
52cabdff1aSopenharmony_ci{
53cabdff1aSopenharmony_ci    ADXDemuxerContext *c = s->priv_data;
54cabdff1aSopenharmony_ci    AVCodecParameters *par = s->streams[0]->codecpar;
55cabdff1aSopenharmony_ci    int ret, size;
56cabdff1aSopenharmony_ci
57cabdff1aSopenharmony_ci    if (avio_feof(s->pb))
58cabdff1aSopenharmony_ci        return AVERROR_EOF;
59cabdff1aSopenharmony_ci
60cabdff1aSopenharmony_ci    if (par->ch_layout.nb_channels <= 0) {
61cabdff1aSopenharmony_ci        av_log(s, AV_LOG_ERROR, "invalid number of channels %d\n", par->ch_layout.nb_channels);
62cabdff1aSopenharmony_ci        return AVERROR_INVALIDDATA;
63cabdff1aSopenharmony_ci    }
64cabdff1aSopenharmony_ci
65cabdff1aSopenharmony_ci    size = BLOCK_SIZE * par->ch_layout.nb_channels;
66cabdff1aSopenharmony_ci
67cabdff1aSopenharmony_ci    pkt->pos = avio_tell(s->pb);
68cabdff1aSopenharmony_ci    pkt->stream_index = 0;
69cabdff1aSopenharmony_ci
70cabdff1aSopenharmony_ci    ret = av_get_packet(s->pb, pkt, size * 128);
71cabdff1aSopenharmony_ci    if (ret < 0)
72cabdff1aSopenharmony_ci        return ret;
73cabdff1aSopenharmony_ci    if ((ret % size) && ret >= size) {
74cabdff1aSopenharmony_ci        size = ret - (ret % size);
75cabdff1aSopenharmony_ci        av_shrink_packet(pkt, size);
76cabdff1aSopenharmony_ci        pkt->flags &= ~AV_PKT_FLAG_CORRUPT;
77cabdff1aSopenharmony_ci    } else if (ret < size) {
78cabdff1aSopenharmony_ci        return AVERROR(EIO);
79cabdff1aSopenharmony_ci    } else {
80cabdff1aSopenharmony_ci        size = ret;
81cabdff1aSopenharmony_ci    }
82cabdff1aSopenharmony_ci
83cabdff1aSopenharmony_ci    pkt->duration = size / (BLOCK_SIZE * par->ch_layout.nb_channels);
84cabdff1aSopenharmony_ci    pkt->pts      = (pkt->pos - c->header_size) / (BLOCK_SIZE * par->ch_layout.nb_channels);
85cabdff1aSopenharmony_ci
86cabdff1aSopenharmony_ci    return 0;
87cabdff1aSopenharmony_ci}
88cabdff1aSopenharmony_ci
89cabdff1aSopenharmony_cistatic int adx_read_header(AVFormatContext *s)
90cabdff1aSopenharmony_ci{
91cabdff1aSopenharmony_ci    ADXDemuxerContext *c = s->priv_data;
92cabdff1aSopenharmony_ci    AVCodecParameters *par;
93cabdff1aSopenharmony_ci    int ret;
94cabdff1aSopenharmony_ci    int channels;
95cabdff1aSopenharmony_ci
96cabdff1aSopenharmony_ci    AVStream *st = avformat_new_stream(s, NULL);
97cabdff1aSopenharmony_ci    if (!st)
98cabdff1aSopenharmony_ci        return AVERROR(ENOMEM);
99cabdff1aSopenharmony_ci    par = s->streams[0]->codecpar;
100cabdff1aSopenharmony_ci
101cabdff1aSopenharmony_ci    if (avio_rb16(s->pb) != 0x8000)
102cabdff1aSopenharmony_ci        return AVERROR_INVALIDDATA;
103cabdff1aSopenharmony_ci    c->header_size = avio_rb16(s->pb) + 4;
104cabdff1aSopenharmony_ci    avio_seek(s->pb, -4, SEEK_CUR);
105cabdff1aSopenharmony_ci
106cabdff1aSopenharmony_ci    if ((ret = ff_get_extradata(s, par, s->pb, c->header_size)) < 0)
107cabdff1aSopenharmony_ci        return ret;
108cabdff1aSopenharmony_ci
109cabdff1aSopenharmony_ci    if (par->extradata_size < 12) {
110cabdff1aSopenharmony_ci        av_log(s, AV_LOG_ERROR, "Invalid extradata size.\n");
111cabdff1aSopenharmony_ci        return AVERROR_INVALIDDATA;
112cabdff1aSopenharmony_ci    }
113cabdff1aSopenharmony_ci    channels = AV_RB8 (par->extradata + 7);
114cabdff1aSopenharmony_ci    par->sample_rate = AV_RB32(par->extradata + 8);
115cabdff1aSopenharmony_ci
116cabdff1aSopenharmony_ci    if (channels <= 0) {
117cabdff1aSopenharmony_ci        av_log(s, AV_LOG_ERROR, "invalid number of channels %d\n", channels);
118cabdff1aSopenharmony_ci        return AVERROR_INVALIDDATA;
119cabdff1aSopenharmony_ci    }
120cabdff1aSopenharmony_ci
121cabdff1aSopenharmony_ci    if (par->sample_rate <= 0) {
122cabdff1aSopenharmony_ci        av_log(s, AV_LOG_ERROR, "Invalid sample rate %d\n", par->sample_rate);
123cabdff1aSopenharmony_ci        return AVERROR_INVALIDDATA;
124cabdff1aSopenharmony_ci    }
125cabdff1aSopenharmony_ci
126cabdff1aSopenharmony_ci    par->ch_layout.nb_channels = channels;
127cabdff1aSopenharmony_ci    par->codec_type  = AVMEDIA_TYPE_AUDIO;
128cabdff1aSopenharmony_ci    par->codec_id    = s->iformat->raw_codec_id;
129cabdff1aSopenharmony_ci    par->bit_rate    = (int64_t)par->sample_rate * par->ch_layout.nb_channels * BLOCK_SIZE * 8LL / BLOCK_SAMPLES;
130cabdff1aSopenharmony_ci
131cabdff1aSopenharmony_ci    avpriv_set_pts_info(st, 64, BLOCK_SAMPLES, par->sample_rate);
132cabdff1aSopenharmony_ci
133cabdff1aSopenharmony_ci    return 0;
134cabdff1aSopenharmony_ci}
135cabdff1aSopenharmony_ci
136cabdff1aSopenharmony_ciconst AVInputFormat ff_adx_demuxer = {
137cabdff1aSopenharmony_ci    .name           = "adx",
138cabdff1aSopenharmony_ci    .long_name      = NULL_IF_CONFIG_SMALL("CRI ADX"),
139cabdff1aSopenharmony_ci    .read_probe     = adx_probe,
140cabdff1aSopenharmony_ci    .priv_data_size = sizeof(ADXDemuxerContext),
141cabdff1aSopenharmony_ci    .read_header    = adx_read_header,
142cabdff1aSopenharmony_ci    .read_packet    = adx_read_packet,
143cabdff1aSopenharmony_ci    .extensions     = "adx",
144cabdff1aSopenharmony_ci    .raw_codec_id   = AV_CODEC_ID_ADPCM_ADX,
145cabdff1aSopenharmony_ci    .flags          = AVFMT_GENERIC_INDEX,
146cabdff1aSopenharmony_ci};
147