1cabdff1aSopenharmony_ci/*
2cabdff1aSopenharmony_ci * MD STUDIO audio demuxer
3cabdff1aSopenharmony_ci *
4cabdff1aSopenharmony_ci * Copyright (c) 2009 Benjamin Larsson
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 "libavutil/channel_layout.h"
24cabdff1aSopenharmony_ci#include "libavutil/intreadwrite.h"
25cabdff1aSopenharmony_ci#include "avformat.h"
26cabdff1aSopenharmony_ci#include "pcm.h"
27cabdff1aSopenharmony_ci
28cabdff1aSopenharmony_ci#define AT1_SU_SIZE     212
29cabdff1aSopenharmony_ci
30cabdff1aSopenharmony_cistatic int aea_read_probe(const AVProbeData *p)
31cabdff1aSopenharmony_ci{
32cabdff1aSopenharmony_ci    if (p->buf_size <= 2048+212)
33cabdff1aSopenharmony_ci        return 0;
34cabdff1aSopenharmony_ci
35cabdff1aSopenharmony_ci    /* Magic is '00 08 00 00' in little-endian*/
36cabdff1aSopenharmony_ci    if (AV_RL32(p->buf)==0x800) {
37cabdff1aSopenharmony_ci        int ch, i;
38cabdff1aSopenharmony_ci        ch = p->buf[264];
39cabdff1aSopenharmony_ci
40cabdff1aSopenharmony_ci        if (ch != 1 && ch != 2)
41cabdff1aSopenharmony_ci            return 0;
42cabdff1aSopenharmony_ci
43cabdff1aSopenharmony_ci        /* Check so that the redundant bsm bytes and info bytes are valid
44cabdff1aSopenharmony_ci         * the block size mode bytes have to be the same
45cabdff1aSopenharmony_ci         * the info bytes have to be the same
46cabdff1aSopenharmony_ci         */
47cabdff1aSopenharmony_ci        for (i = 2048; i + 211 < p->buf_size; i+= 212) {
48cabdff1aSopenharmony_ci            int bsm_s, bsm_e, inb_s, inb_e;
49cabdff1aSopenharmony_ci            bsm_s = p->buf[0];
50cabdff1aSopenharmony_ci            inb_s = p->buf[1];
51cabdff1aSopenharmony_ci            inb_e = p->buf[210];
52cabdff1aSopenharmony_ci            bsm_e = p->buf[211];
53cabdff1aSopenharmony_ci
54cabdff1aSopenharmony_ci            if (bsm_s != bsm_e || inb_s != inb_e)
55cabdff1aSopenharmony_ci                return 0;
56cabdff1aSopenharmony_ci        }
57cabdff1aSopenharmony_ci        return AVPROBE_SCORE_MAX / 4 + 1;
58cabdff1aSopenharmony_ci    }
59cabdff1aSopenharmony_ci    return 0;
60cabdff1aSopenharmony_ci}
61cabdff1aSopenharmony_ci
62cabdff1aSopenharmony_cistatic int aea_read_header(AVFormatContext *s)
63cabdff1aSopenharmony_ci{
64cabdff1aSopenharmony_ci    AVStream *st = avformat_new_stream(s, NULL);
65cabdff1aSopenharmony_ci    int channels;
66cabdff1aSopenharmony_ci    if (!st)
67cabdff1aSopenharmony_ci        return AVERROR(ENOMEM);
68cabdff1aSopenharmony_ci
69cabdff1aSopenharmony_ci    /* Parse the amount of channels and skip to pos 2048(0x800) */
70cabdff1aSopenharmony_ci    avio_skip(s->pb, 264);
71cabdff1aSopenharmony_ci    channels = avio_r8(s->pb);
72cabdff1aSopenharmony_ci    avio_skip(s->pb, 1783);
73cabdff1aSopenharmony_ci
74cabdff1aSopenharmony_ci
75cabdff1aSopenharmony_ci    st->codecpar->codec_type     = AVMEDIA_TYPE_AUDIO;
76cabdff1aSopenharmony_ci    st->codecpar->codec_id       = AV_CODEC_ID_ATRAC1;
77cabdff1aSopenharmony_ci    st->codecpar->sample_rate    = 44100;
78cabdff1aSopenharmony_ci    st->codecpar->bit_rate       = 292000;
79cabdff1aSopenharmony_ci
80cabdff1aSopenharmony_ci    if (channels != 1 && channels != 2) {
81cabdff1aSopenharmony_ci        av_log(s, AV_LOG_ERROR, "Channels %d not supported!\n", channels);
82cabdff1aSopenharmony_ci        return AVERROR_INVALIDDATA;
83cabdff1aSopenharmony_ci    }
84cabdff1aSopenharmony_ci
85cabdff1aSopenharmony_ci    av_channel_layout_default(&st->codecpar->ch_layout, channels);
86cabdff1aSopenharmony_ci
87cabdff1aSopenharmony_ci    st->codecpar->block_align = AT1_SU_SIZE * st->codecpar->ch_layout.nb_channels;
88cabdff1aSopenharmony_ci    return 0;
89cabdff1aSopenharmony_ci}
90cabdff1aSopenharmony_ci
91cabdff1aSopenharmony_cistatic int aea_read_packet(AVFormatContext *s, AVPacket *pkt)
92cabdff1aSopenharmony_ci{
93cabdff1aSopenharmony_ci    int ret = av_get_packet(s->pb, pkt, s->streams[0]->codecpar->block_align);
94cabdff1aSopenharmony_ci
95cabdff1aSopenharmony_ci    pkt->stream_index = 0;
96cabdff1aSopenharmony_ci    if (ret <= 0)
97cabdff1aSopenharmony_ci        return AVERROR(EIO);
98cabdff1aSopenharmony_ci
99cabdff1aSopenharmony_ci    return ret;
100cabdff1aSopenharmony_ci}
101cabdff1aSopenharmony_ci
102cabdff1aSopenharmony_ciconst AVInputFormat ff_aea_demuxer = {
103cabdff1aSopenharmony_ci    .name           = "aea",
104cabdff1aSopenharmony_ci    .long_name      = NULL_IF_CONFIG_SMALL("MD STUDIO audio"),
105cabdff1aSopenharmony_ci    .read_probe     = aea_read_probe,
106cabdff1aSopenharmony_ci    .read_header    = aea_read_header,
107cabdff1aSopenharmony_ci    .read_packet    = aea_read_packet,
108cabdff1aSopenharmony_ci    .read_seek      = ff_pcm_read_seek,
109cabdff1aSopenharmony_ci    .flags          = AVFMT_GENERIC_INDEX,
110cabdff1aSopenharmony_ci    .extensions     = "aea",
111cabdff1aSopenharmony_ci};
112