annotate ffmpeg/libavformat/bmv.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 * Discworld II BMV demuxer
yading@11 3 * Copyright (c) 2011 Konstantin Shishkov.
yading@11 4 *
yading@11 5 * This file is part of Libav.
yading@11 6 *
yading@11 7 * Libav 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 * Libav 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 Libav; 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 "avformat.h"
yading@11 24 #include "internal.h"
yading@11 25
yading@11 26 enum BMVFlags {
yading@11 27 BMV_NOP = 0,
yading@11 28 BMV_END,
yading@11 29 BMV_DELTA,
yading@11 30 BMV_INTRA,
yading@11 31
yading@11 32 BMV_AUDIO = 0x20,
yading@11 33 };
yading@11 34
yading@11 35 typedef struct BMVContext {
yading@11 36 uint8_t *packet;
yading@11 37 int size;
yading@11 38 int get_next;
yading@11 39 int64_t audio_pos;
yading@11 40 } BMVContext;
yading@11 41
yading@11 42 static int bmv_read_header(AVFormatContext *s)
yading@11 43 {
yading@11 44 AVStream *st, *ast;
yading@11 45 BMVContext *c = s->priv_data;
yading@11 46
yading@11 47 st = avformat_new_stream(s, 0);
yading@11 48 if (!st)
yading@11 49 return AVERROR(ENOMEM);
yading@11 50 st->codec->codec_type = AVMEDIA_TYPE_VIDEO;
yading@11 51 st->codec->codec_id = AV_CODEC_ID_BMV_VIDEO;
yading@11 52 st->codec->width = 640;
yading@11 53 st->codec->height = 429;
yading@11 54 st->codec->pix_fmt = AV_PIX_FMT_PAL8;
yading@11 55 avpriv_set_pts_info(st, 16, 1, 12);
yading@11 56 ast = avformat_new_stream(s, 0);
yading@11 57 if (!ast)
yading@11 58 return AVERROR(ENOMEM);
yading@11 59 ast->codec->codec_type = AVMEDIA_TYPE_AUDIO;
yading@11 60 ast->codec->codec_id = AV_CODEC_ID_BMV_AUDIO;
yading@11 61 ast->codec->channels = 2;
yading@11 62 ast->codec->channel_layout = AV_CH_LAYOUT_STEREO;
yading@11 63 ast->codec->sample_rate = 22050;
yading@11 64 avpriv_set_pts_info(ast, 16, 1, 22050);
yading@11 65
yading@11 66 c->get_next = 1;
yading@11 67 c->audio_pos = 0;
yading@11 68 return 0;
yading@11 69 }
yading@11 70
yading@11 71 static int bmv_read_packet(AVFormatContext *s, AVPacket *pkt)
yading@11 72 {
yading@11 73 BMVContext *c = s->priv_data;
yading@11 74 int type;
yading@11 75 void *tmp;
yading@11 76
yading@11 77 while (c->get_next) {
yading@11 78 if (s->pb->eof_reached)
yading@11 79 return AVERROR_EOF;
yading@11 80 type = avio_r8(s->pb);
yading@11 81 if (type == BMV_NOP)
yading@11 82 continue;
yading@11 83 if (type == BMV_END)
yading@11 84 return AVERROR_EOF;
yading@11 85 c->size = avio_rl24(s->pb);
yading@11 86 if (!c->size)
yading@11 87 return AVERROR_INVALIDDATA;
yading@11 88 tmp = av_realloc(c->packet, c->size + 1);
yading@11 89 if (!tmp)
yading@11 90 return AVERROR(ENOMEM);
yading@11 91 c->packet = tmp;
yading@11 92 c->packet[0] = type;
yading@11 93 if (avio_read(s->pb, c->packet + 1, c->size) != c->size)
yading@11 94 return AVERROR(EIO);
yading@11 95 if (type & BMV_AUDIO) {
yading@11 96 int audio_size = c->packet[1] * 65 + 1;
yading@11 97 if (audio_size >= c->size) {
yading@11 98 av_log(s, AV_LOG_ERROR, "Reported audio size %d is bigger than packet size (%d)\n",
yading@11 99 audio_size, c->size);
yading@11 100 return AVERROR_INVALIDDATA;
yading@11 101 }
yading@11 102 if (av_new_packet(pkt, audio_size) < 0)
yading@11 103 return AVERROR(ENOMEM);
yading@11 104 memcpy(pkt->data, c->packet + 1, pkt->size);
yading@11 105 pkt->stream_index = 1;
yading@11 106 pkt->pts = c->audio_pos;
yading@11 107 pkt->duration = c->packet[1] * 32;
yading@11 108 c->audio_pos += pkt->duration;
yading@11 109 c->get_next = 0;
yading@11 110 return pkt->size;
yading@11 111 } else
yading@11 112 break;
yading@11 113 }
yading@11 114 if (av_new_packet(pkt, c->size + 1) < 0)
yading@11 115 return AVERROR(ENOMEM);
yading@11 116 pkt->stream_index = 0;
yading@11 117 c->get_next = 1;
yading@11 118 memcpy(pkt->data, c->packet, pkt->size);
yading@11 119 return pkt->size;
yading@11 120 }
yading@11 121
yading@11 122 static int bmv_read_close(AVFormatContext *s)
yading@11 123 {
yading@11 124 BMVContext *c = s->priv_data;
yading@11 125
yading@11 126 av_freep(&c->packet);
yading@11 127
yading@11 128 return 0;
yading@11 129 }
yading@11 130
yading@11 131 AVInputFormat ff_bmv_demuxer = {
yading@11 132 .name = "bmv",
yading@11 133 .long_name = NULL_IF_CONFIG_SMALL("Discworld II BMV"),
yading@11 134 .priv_data_size = sizeof(BMVContext),
yading@11 135 .read_header = bmv_read_header,
yading@11 136 .read_packet = bmv_read_packet,
yading@11 137 .read_close = bmv_read_close,
yading@11 138 .extensions = "bmv",
yading@11 139 };