annotate ffmpeg/libavformat/astdec.c @ 13:844d341cf643 tip

Back up before ISMIR
author Yading Song <yading.song@eecs.qmul.ac.uk>
date Thu, 31 Oct 2013 13:17:06 +0000
parents f445c3017523
children
rev   line source
yading@11 1 /*
yading@11 2 * AST demuxer
yading@11 3 * Copyright (c) 2012 Paul B Mahol
yading@11 4 *
yading@11 5 * This file is part of FFmpeg.
yading@11 6 *
yading@11 7 * FFmpeg is free software; you can redistribute it and/or
yading@11 8 * modify it under the terms of the GNU Lesser General Public
yading@11 9 * License as published by the Free Software Foundation; either
yading@11 10 * version 2.1 of the License, or (at your option) any later version.
yading@11 11 *
yading@11 12 * FFmpeg is distributed in the hope that it will be useful,
yading@11 13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
yading@11 14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
yading@11 15 * Lesser General Public License for more details.
yading@11 16 *
yading@11 17 * You should have received a copy of the GNU Lesser General Public
yading@11 18 * License along with FFmpeg; if not, write to the Free Software
yading@11 19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
yading@11 20 */
yading@11 21
yading@11 22 #include "libavutil/channel_layout.h"
yading@11 23 #include "libavutil/intreadwrite.h"
yading@11 24 #include "avformat.h"
yading@11 25 #include "internal.h"
yading@11 26 #include "ast.h"
yading@11 27
yading@11 28 static int ast_probe(AVProbeData *p)
yading@11 29 {
yading@11 30 if (AV_RL32(p->buf) == MKTAG('S','T','R','M') &&
yading@11 31 AV_RB16(p->buf + 10) &&
yading@11 32 AV_RB16(p->buf + 12) &&
yading@11 33 AV_RB32(p->buf + 16))
yading@11 34 return AVPROBE_SCORE_MAX / 3 * 2;
yading@11 35 return 0;
yading@11 36 }
yading@11 37
yading@11 38 static int ast_read_header(AVFormatContext *s)
yading@11 39 {
yading@11 40 int depth;
yading@11 41 AVStream *st;
yading@11 42
yading@11 43 st = avformat_new_stream(s, NULL);
yading@11 44 if (!st)
yading@11 45 return AVERROR(ENOMEM);
yading@11 46
yading@11 47 avio_skip(s->pb, 8);
yading@11 48 st->codec->codec_type = AVMEDIA_TYPE_AUDIO;
yading@11 49 st->codec->codec_id = ff_codec_get_id(ff_codec_ast_tags, avio_rb16(s->pb));
yading@11 50
yading@11 51 depth = avio_rb16(s->pb);
yading@11 52 if (depth != 16) {
yading@11 53 avpriv_request_sample(s, "depth %d", depth);
yading@11 54 return AVERROR_INVALIDDATA;
yading@11 55 }
yading@11 56
yading@11 57 st->codec->channels = avio_rb16(s->pb);
yading@11 58 if (!st->codec->channels)
yading@11 59 return AVERROR_INVALIDDATA;
yading@11 60
yading@11 61 if (st->codec->channels == 2)
yading@11 62 st->codec->channel_layout = AV_CH_LAYOUT_STEREO;
yading@11 63 else if (st->codec->channels == 4)
yading@11 64 st->codec->channel_layout = AV_CH_LAYOUT_4POINT0;
yading@11 65
yading@11 66 avio_skip(s->pb, 2);
yading@11 67 st->codec->sample_rate = avio_rb32(s->pb);
yading@11 68 if (st->codec->sample_rate <= 0)
yading@11 69 return AVERROR_INVALIDDATA;
yading@11 70 st->start_time = 0;
yading@11 71 st->duration = avio_rb32(s->pb);
yading@11 72 avio_skip(s->pb, 40);
yading@11 73 avpriv_set_pts_info(st, 64, 1, st->codec->sample_rate);
yading@11 74
yading@11 75 return 0;
yading@11 76 }
yading@11 77
yading@11 78 static int ast_read_packet(AVFormatContext *s, AVPacket *pkt)
yading@11 79 {
yading@11 80 uint32_t type, size;
yading@11 81 int64_t pos;
yading@11 82 int ret;
yading@11 83
yading@11 84 if (url_feof(s->pb))
yading@11 85 return AVERROR_EOF;
yading@11 86
yading@11 87 pos = avio_tell(s->pb);
yading@11 88 type = avio_rl32(s->pb);
yading@11 89 size = avio_rb32(s->pb);
yading@11 90 if (size > INT_MAX / s->streams[0]->codec->channels)
yading@11 91 return AVERROR_INVALIDDATA;
yading@11 92
yading@11 93 size *= s->streams[0]->codec->channels;
yading@11 94 if ((ret = avio_skip(s->pb, 24)) < 0) // padding
yading@11 95 return ret;
yading@11 96
yading@11 97 if (type == MKTAG('B','L','C','K')) {
yading@11 98 ret = av_get_packet(s->pb, pkt, size);
yading@11 99 pkt->stream_index = 0;
yading@11 100 pkt->pos = pos;
yading@11 101 } else {
yading@11 102 av_log(s, AV_LOG_ERROR, "unknown chunk %x\n", type);
yading@11 103 avio_skip(s->pb, size);
yading@11 104 ret = AVERROR_INVALIDDATA;
yading@11 105 }
yading@11 106
yading@11 107 return ret;
yading@11 108 }
yading@11 109
yading@11 110 AVInputFormat ff_ast_demuxer = {
yading@11 111 .name = "ast",
yading@11 112 .long_name = NULL_IF_CONFIG_SMALL("AST (Audio Stream)"),
yading@11 113 .read_probe = ast_probe,
yading@11 114 .read_header = ast_read_header,
yading@11 115 .read_packet = ast_read_packet,
yading@11 116 .extensions = "ast",
yading@11 117 .flags = AVFMT_GENERIC_INDEX,
yading@11 118 .codec_tag = (const AVCodecTag* const []){ff_codec_ast_tags, 0},
yading@11 119 };