1cabdff1aSopenharmony_ci/* 2cabdff1aSopenharmony_ci * Silicon Graphics Movie demuxer 3cabdff1aSopenharmony_ci * Copyright (c) 2012 Peter Ross 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/** 23cabdff1aSopenharmony_ci * @file 24cabdff1aSopenharmony_ci * Silicon Graphics Movie demuxer 25cabdff1aSopenharmony_ci */ 26cabdff1aSopenharmony_ci 27cabdff1aSopenharmony_ci#include "libavutil/channel_layout.h" 28cabdff1aSopenharmony_ci#include "libavutil/eval.h" 29cabdff1aSopenharmony_ci#include "libavutil/intfloat.h" 30cabdff1aSopenharmony_ci#include "libavutil/intreadwrite.h" 31cabdff1aSopenharmony_ci#include "libavutil/rational.h" 32cabdff1aSopenharmony_ci 33cabdff1aSopenharmony_ci#include "avformat.h" 34cabdff1aSopenharmony_ci#include "internal.h" 35cabdff1aSopenharmony_ci 36cabdff1aSopenharmony_citypedef struct MvContext { 37cabdff1aSopenharmony_ci int nb_video_tracks; 38cabdff1aSopenharmony_ci int nb_audio_tracks; 39cabdff1aSopenharmony_ci 40cabdff1aSopenharmony_ci int eof_count; ///< number of streams that have finished 41cabdff1aSopenharmony_ci int stream_index; ///< current stream index 42cabdff1aSopenharmony_ci int frame[2]; ///< frame nb for current stream 43cabdff1aSopenharmony_ci 44cabdff1aSopenharmony_ci int acompression; ///< compression level for audio stream 45cabdff1aSopenharmony_ci int aformat; ///< audio format 46cabdff1aSopenharmony_ci} MvContext; 47cabdff1aSopenharmony_ci 48cabdff1aSopenharmony_ci/* these magic numbers are defined in moviefile.h on Silicon Grahpics IRIX */ 49cabdff1aSopenharmony_ci#define MOVIE_SOUND 1 50cabdff1aSopenharmony_ci#define MOVIE_SILENT 2 51cabdff1aSopenharmony_ci 52cabdff1aSopenharmony_ci#define AUDIO_FORMAT_SIGNED 401 53cabdff1aSopenharmony_ci 54cabdff1aSopenharmony_cistatic int mv_probe(const AVProbeData *p) 55cabdff1aSopenharmony_ci{ 56cabdff1aSopenharmony_ci if (AV_RB32(p->buf) == MKBETAG('M', 'O', 'V', 'I') && 57cabdff1aSopenharmony_ci AV_RB16(p->buf + 4) < 3) 58cabdff1aSopenharmony_ci return AVPROBE_SCORE_MAX; 59cabdff1aSopenharmony_ci return 0; 60cabdff1aSopenharmony_ci} 61cabdff1aSopenharmony_ci 62cabdff1aSopenharmony_cistatic char *var_read_string(AVIOContext *pb, int size) 63cabdff1aSopenharmony_ci{ 64cabdff1aSopenharmony_ci int n; 65cabdff1aSopenharmony_ci char *str; 66cabdff1aSopenharmony_ci 67cabdff1aSopenharmony_ci if (size < 0 || size == INT_MAX) 68cabdff1aSopenharmony_ci return NULL; 69cabdff1aSopenharmony_ci 70cabdff1aSopenharmony_ci str = av_malloc(size + 1); 71cabdff1aSopenharmony_ci if (!str) 72cabdff1aSopenharmony_ci return NULL; 73cabdff1aSopenharmony_ci n = avio_get_str(pb, size, str, size + 1); 74cabdff1aSopenharmony_ci if (n < size) 75cabdff1aSopenharmony_ci avio_skip(pb, size - n); 76cabdff1aSopenharmony_ci return str; 77cabdff1aSopenharmony_ci} 78cabdff1aSopenharmony_ci 79cabdff1aSopenharmony_cistatic int var_read_int(AVIOContext *pb, int size) 80cabdff1aSopenharmony_ci{ 81cabdff1aSopenharmony_ci int v; 82cabdff1aSopenharmony_ci char *s = var_read_string(pb, size); 83cabdff1aSopenharmony_ci if (!s) 84cabdff1aSopenharmony_ci return 0; 85cabdff1aSopenharmony_ci v = strtol(s, NULL, 10); 86cabdff1aSopenharmony_ci av_free(s); 87cabdff1aSopenharmony_ci return v; 88cabdff1aSopenharmony_ci} 89cabdff1aSopenharmony_ci 90cabdff1aSopenharmony_cistatic AVRational var_read_float(AVIOContext *pb, int size) 91cabdff1aSopenharmony_ci{ 92cabdff1aSopenharmony_ci AVRational v; 93cabdff1aSopenharmony_ci char *s = var_read_string(pb, size); 94cabdff1aSopenharmony_ci if (!s) 95cabdff1aSopenharmony_ci return (AVRational) { 0, 0 }; 96cabdff1aSopenharmony_ci v = av_d2q(av_strtod(s, NULL), INT_MAX); 97cabdff1aSopenharmony_ci av_free(s); 98cabdff1aSopenharmony_ci return v; 99cabdff1aSopenharmony_ci} 100cabdff1aSopenharmony_ci 101cabdff1aSopenharmony_cistatic void var_read_metadata(AVFormatContext *avctx, const char *tag, int size) 102cabdff1aSopenharmony_ci{ 103cabdff1aSopenharmony_ci char *value = var_read_string(avctx->pb, size); 104cabdff1aSopenharmony_ci if (value) 105cabdff1aSopenharmony_ci av_dict_set(&avctx->metadata, tag, value, AV_DICT_DONT_STRDUP_VAL); 106cabdff1aSopenharmony_ci} 107cabdff1aSopenharmony_ci 108cabdff1aSopenharmony_cistatic int set_channels(AVFormatContext *avctx, AVStream *st, int channels) 109cabdff1aSopenharmony_ci{ 110cabdff1aSopenharmony_ci if (channels <= 0) { 111cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "Channel count %d invalid.\n", channels); 112cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 113cabdff1aSopenharmony_ci } 114cabdff1aSopenharmony_ci av_channel_layout_default(&st->codecpar->ch_layout, channels); 115cabdff1aSopenharmony_ci return 0; 116cabdff1aSopenharmony_ci} 117cabdff1aSopenharmony_ci 118cabdff1aSopenharmony_ci/** 119cabdff1aSopenharmony_ci * Parse global variable 120cabdff1aSopenharmony_ci * @return < 0 if unknown 121cabdff1aSopenharmony_ci */ 122cabdff1aSopenharmony_cistatic int parse_global_var(AVFormatContext *avctx, AVStream *st, 123cabdff1aSopenharmony_ci const char *name, int size) 124cabdff1aSopenharmony_ci{ 125cabdff1aSopenharmony_ci MvContext *mv = avctx->priv_data; 126cabdff1aSopenharmony_ci AVIOContext *pb = avctx->pb; 127cabdff1aSopenharmony_ci if (!strcmp(name, "__NUM_I_TRACKS")) { 128cabdff1aSopenharmony_ci mv->nb_video_tracks = var_read_int(pb, size); 129cabdff1aSopenharmony_ci } else if (!strcmp(name, "__NUM_A_TRACKS")) { 130cabdff1aSopenharmony_ci mv->nb_audio_tracks = var_read_int(pb, size); 131cabdff1aSopenharmony_ci } else if (!strcmp(name, "COMMENT") || !strcmp(name, "TITLE")) { 132cabdff1aSopenharmony_ci var_read_metadata(avctx, name, size); 133cabdff1aSopenharmony_ci } else if (!strcmp(name, "LOOP_MODE") || !strcmp(name, "NUM_LOOPS") || 134cabdff1aSopenharmony_ci !strcmp(name, "OPTIMIZED")) { 135cabdff1aSopenharmony_ci avio_skip(pb, size); // ignore 136cabdff1aSopenharmony_ci } else 137cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 138cabdff1aSopenharmony_ci 139cabdff1aSopenharmony_ci return 0; 140cabdff1aSopenharmony_ci} 141cabdff1aSopenharmony_ci 142cabdff1aSopenharmony_ci/** 143cabdff1aSopenharmony_ci * Parse audio variable 144cabdff1aSopenharmony_ci * @return < 0 if unknown 145cabdff1aSopenharmony_ci */ 146cabdff1aSopenharmony_cistatic int parse_audio_var(AVFormatContext *avctx, AVStream *st, 147cabdff1aSopenharmony_ci const char *name, int size) 148cabdff1aSopenharmony_ci{ 149cabdff1aSopenharmony_ci MvContext *mv = avctx->priv_data; 150cabdff1aSopenharmony_ci AVIOContext *pb = avctx->pb; 151cabdff1aSopenharmony_ci if (!strcmp(name, "__DIR_COUNT")) { 152cabdff1aSopenharmony_ci st->nb_frames = var_read_int(pb, size); 153cabdff1aSopenharmony_ci } else if (!strcmp(name, "AUDIO_FORMAT")) { 154cabdff1aSopenharmony_ci mv->aformat = var_read_int(pb, size); 155cabdff1aSopenharmony_ci } else if (!strcmp(name, "COMPRESSION")) { 156cabdff1aSopenharmony_ci mv->acompression = var_read_int(pb, size); 157cabdff1aSopenharmony_ci } else if (!strcmp(name, "DEFAULT_VOL")) { 158cabdff1aSopenharmony_ci var_read_metadata(avctx, name, size); 159cabdff1aSopenharmony_ci } else if (!strcmp(name, "NUM_CHANNELS")) { 160cabdff1aSopenharmony_ci return set_channels(avctx, st, var_read_int(pb, size)); 161cabdff1aSopenharmony_ci } else if (!strcmp(name, "SAMPLE_RATE")) { 162cabdff1aSopenharmony_ci int sample_rate = var_read_int(pb, size); 163cabdff1aSopenharmony_ci if (sample_rate <= 0) 164cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 165cabdff1aSopenharmony_ci st->codecpar->sample_rate = sample_rate; 166cabdff1aSopenharmony_ci avpriv_set_pts_info(st, 33, 1, st->codecpar->sample_rate); 167cabdff1aSopenharmony_ci } else if (!strcmp(name, "SAMPLE_WIDTH")) { 168cabdff1aSopenharmony_ci uint64_t bpc = var_read_int(pb, size) * (uint64_t)8; 169cabdff1aSopenharmony_ci if (bpc > 16) 170cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 171cabdff1aSopenharmony_ci st->codecpar->bits_per_coded_sample = bpc; 172cabdff1aSopenharmony_ci } else 173cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 174cabdff1aSopenharmony_ci 175cabdff1aSopenharmony_ci return 0; 176cabdff1aSopenharmony_ci} 177cabdff1aSopenharmony_ci 178cabdff1aSopenharmony_ci/** 179cabdff1aSopenharmony_ci * Parse video variable 180cabdff1aSopenharmony_ci * @return < 0 if unknown 181cabdff1aSopenharmony_ci */ 182cabdff1aSopenharmony_cistatic int parse_video_var(AVFormatContext *avctx, AVStream *st, 183cabdff1aSopenharmony_ci const char *name, int size) 184cabdff1aSopenharmony_ci{ 185cabdff1aSopenharmony_ci AVIOContext *pb = avctx->pb; 186cabdff1aSopenharmony_ci if (!strcmp(name, "__DIR_COUNT")) { 187cabdff1aSopenharmony_ci st->nb_frames = st->duration = var_read_int(pb, size); 188cabdff1aSopenharmony_ci } else if (!strcmp(name, "COMPRESSION")) { 189cabdff1aSopenharmony_ci char *str = var_read_string(pb, size); 190cabdff1aSopenharmony_ci if (!str) 191cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 192cabdff1aSopenharmony_ci if (!strcmp(str, "1")) { 193cabdff1aSopenharmony_ci st->codecpar->codec_id = AV_CODEC_ID_MVC1; 194cabdff1aSopenharmony_ci } else if (!strcmp(str, "2")) { 195cabdff1aSopenharmony_ci st->codecpar->format = AV_PIX_FMT_ABGR; 196cabdff1aSopenharmony_ci st->codecpar->codec_id = AV_CODEC_ID_RAWVIDEO; 197cabdff1aSopenharmony_ci } else if (!strcmp(str, "3")) { 198cabdff1aSopenharmony_ci st->codecpar->codec_id = AV_CODEC_ID_SGIRLE; 199cabdff1aSopenharmony_ci } else if (!strcmp(str, "10")) { 200cabdff1aSopenharmony_ci st->codecpar->codec_id = AV_CODEC_ID_MJPEG; 201cabdff1aSopenharmony_ci } else if (!strcmp(str, "MVC2")) { 202cabdff1aSopenharmony_ci st->codecpar->codec_id = AV_CODEC_ID_MVC2; 203cabdff1aSopenharmony_ci } else { 204cabdff1aSopenharmony_ci avpriv_request_sample(avctx, "Video compression %s", str); 205cabdff1aSopenharmony_ci } 206cabdff1aSopenharmony_ci av_free(str); 207cabdff1aSopenharmony_ci } else if (!strcmp(name, "FPS")) { 208cabdff1aSopenharmony_ci AVRational fps = var_read_float(pb, size); 209cabdff1aSopenharmony_ci avpriv_set_pts_info(st, 64, fps.den, fps.num); 210cabdff1aSopenharmony_ci st->avg_frame_rate = fps; 211cabdff1aSopenharmony_ci } else if (!strcmp(name, "HEIGHT")) { 212cabdff1aSopenharmony_ci st->codecpar->height = var_read_int(pb, size); 213cabdff1aSopenharmony_ci } else if (!strcmp(name, "PIXEL_ASPECT")) { 214cabdff1aSopenharmony_ci st->sample_aspect_ratio = var_read_float(pb, size); 215cabdff1aSopenharmony_ci av_reduce(&st->sample_aspect_ratio.num, &st->sample_aspect_ratio.den, 216cabdff1aSopenharmony_ci st->sample_aspect_ratio.num, st->sample_aspect_ratio.den, 217cabdff1aSopenharmony_ci INT_MAX); 218cabdff1aSopenharmony_ci } else if (!strcmp(name, "WIDTH")) { 219cabdff1aSopenharmony_ci st->codecpar->width = var_read_int(pb, size); 220cabdff1aSopenharmony_ci } else if (!strcmp(name, "ORIENTATION")) { 221cabdff1aSopenharmony_ci if (var_read_int(pb, size) == 1101) { 222cabdff1aSopenharmony_ci if (!st->codecpar->extradata) { 223cabdff1aSopenharmony_ci st->codecpar->extradata = av_strdup("BottomUp"); 224cabdff1aSopenharmony_ci if (!st->codecpar->extradata) 225cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 226cabdff1aSopenharmony_ci st->codecpar->extradata_size = 9; 227cabdff1aSopenharmony_ci } 228cabdff1aSopenharmony_ci } 229cabdff1aSopenharmony_ci } else if (!strcmp(name, "Q_SPATIAL") || !strcmp(name, "Q_TEMPORAL")) { 230cabdff1aSopenharmony_ci var_read_metadata(avctx, name, size); 231cabdff1aSopenharmony_ci } else if (!strcmp(name, "INTERLACING") || !strcmp(name, "PACKING")) { 232cabdff1aSopenharmony_ci avio_skip(pb, size); // ignore 233cabdff1aSopenharmony_ci } else 234cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 235cabdff1aSopenharmony_ci 236cabdff1aSopenharmony_ci return 0; 237cabdff1aSopenharmony_ci} 238cabdff1aSopenharmony_ci 239cabdff1aSopenharmony_cistatic int read_table(AVFormatContext *avctx, AVStream *st, 240cabdff1aSopenharmony_ci int (*parse)(AVFormatContext *avctx, AVStream *st, 241cabdff1aSopenharmony_ci const char *name, int size)) 242cabdff1aSopenharmony_ci{ 243cabdff1aSopenharmony_ci unsigned count; 244cabdff1aSopenharmony_ci int i; 245cabdff1aSopenharmony_ci 246cabdff1aSopenharmony_ci AVIOContext *pb = avctx->pb; 247cabdff1aSopenharmony_ci avio_skip(pb, 4); 248cabdff1aSopenharmony_ci count = avio_rb32(pb); 249cabdff1aSopenharmony_ci avio_skip(pb, 4); 250cabdff1aSopenharmony_ci for (i = 0; i < count; i++) { 251cabdff1aSopenharmony_ci char name[17]; 252cabdff1aSopenharmony_ci int size; 253cabdff1aSopenharmony_ci 254cabdff1aSopenharmony_ci if (avio_feof(pb)) 255cabdff1aSopenharmony_ci return AVERROR_EOF; 256cabdff1aSopenharmony_ci 257cabdff1aSopenharmony_ci avio_read(pb, name, 16); 258cabdff1aSopenharmony_ci name[sizeof(name) - 1] = 0; 259cabdff1aSopenharmony_ci size = avio_rb32(pb); 260cabdff1aSopenharmony_ci if (size < 0) { 261cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "entry size %d is invalid\n", size); 262cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 263cabdff1aSopenharmony_ci } 264cabdff1aSopenharmony_ci if (parse(avctx, st, name, size) < 0) { 265cabdff1aSopenharmony_ci avpriv_request_sample(avctx, "Variable %s", name); 266cabdff1aSopenharmony_ci avio_skip(pb, size); 267cabdff1aSopenharmony_ci } 268cabdff1aSopenharmony_ci } 269cabdff1aSopenharmony_ci return 0; 270cabdff1aSopenharmony_ci} 271cabdff1aSopenharmony_ci 272cabdff1aSopenharmony_cistatic void read_index(AVIOContext *pb, AVStream *st) 273cabdff1aSopenharmony_ci{ 274cabdff1aSopenharmony_ci uint64_t timestamp = 0; 275cabdff1aSopenharmony_ci int i; 276cabdff1aSopenharmony_ci for (i = 0; i < st->nb_frames; i++) { 277cabdff1aSopenharmony_ci uint32_t pos = avio_rb32(pb); 278cabdff1aSopenharmony_ci uint32_t size = avio_rb32(pb); 279cabdff1aSopenharmony_ci avio_skip(pb, 8); 280cabdff1aSopenharmony_ci if (avio_feof(pb)) 281cabdff1aSopenharmony_ci return ; 282cabdff1aSopenharmony_ci av_add_index_entry(st, pos, timestamp, size, 0, AVINDEX_KEYFRAME); 283cabdff1aSopenharmony_ci if (st->codecpar->codec_type == AVMEDIA_TYPE_AUDIO) { 284cabdff1aSopenharmony_ci timestamp += size / (st->codecpar->ch_layout.nb_channels * 2LL); 285cabdff1aSopenharmony_ci } else { 286cabdff1aSopenharmony_ci timestamp++; 287cabdff1aSopenharmony_ci } 288cabdff1aSopenharmony_ci } 289cabdff1aSopenharmony_ci} 290cabdff1aSopenharmony_ci 291cabdff1aSopenharmony_cistatic int mv_read_header(AVFormatContext *avctx) 292cabdff1aSopenharmony_ci{ 293cabdff1aSopenharmony_ci MvContext *mv = avctx->priv_data; 294cabdff1aSopenharmony_ci AVIOContext *pb = avctx->pb; 295cabdff1aSopenharmony_ci AVStream *ast = NULL, *vst = NULL; //initialization to suppress warning 296cabdff1aSopenharmony_ci int version, i; 297cabdff1aSopenharmony_ci int ret; 298cabdff1aSopenharmony_ci 299cabdff1aSopenharmony_ci avio_skip(pb, 4); 300cabdff1aSopenharmony_ci 301cabdff1aSopenharmony_ci version = avio_rb16(pb); 302cabdff1aSopenharmony_ci if (version == 2) { 303cabdff1aSopenharmony_ci uint64_t timestamp; 304cabdff1aSopenharmony_ci int v; 305cabdff1aSopenharmony_ci uint32_t bytes_per_sample; 306cabdff1aSopenharmony_ci AVRational fps; 307cabdff1aSopenharmony_ci 308cabdff1aSopenharmony_ci avio_skip(pb, 10); 309cabdff1aSopenharmony_ci 310cabdff1aSopenharmony_ci fps = av_d2q(av_int2double(avio_rb64(pb)), INT_MAX); 311cabdff1aSopenharmony_ci 312cabdff1aSopenharmony_ci /* allocate audio track first to prevent unnecessary seeking 313cabdff1aSopenharmony_ci * (audio packet always precede video packet for a given frame) */ 314cabdff1aSopenharmony_ci v = avio_rb16(pb); 315cabdff1aSopenharmony_ci if (v == MOVIE_SOUND) { 316cabdff1aSopenharmony_ci /* movie has sound so allocate an audio stream */ 317cabdff1aSopenharmony_ci ast = avformat_new_stream(avctx, NULL); 318cabdff1aSopenharmony_ci if (!ast) 319cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 320cabdff1aSopenharmony_ci } else if (v != MOVIE_SILENT) 321cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 322cabdff1aSopenharmony_ci 323cabdff1aSopenharmony_ci avio_skip(pb, 2); 324cabdff1aSopenharmony_ci 325cabdff1aSopenharmony_ci vst = avformat_new_stream(avctx, NULL); 326cabdff1aSopenharmony_ci if (!vst) 327cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 328cabdff1aSopenharmony_ci avpriv_set_pts_info(vst, 64, fps.den, fps.num); 329cabdff1aSopenharmony_ci vst->codecpar->codec_type = AVMEDIA_TYPE_VIDEO; 330cabdff1aSopenharmony_ci vst->avg_frame_rate = fps; 331cabdff1aSopenharmony_ci vst->duration = vst->nb_frames = avio_rb32(pb); 332cabdff1aSopenharmony_ci v = avio_rb32(pb); 333cabdff1aSopenharmony_ci switch (v) { 334cabdff1aSopenharmony_ci case 1: 335cabdff1aSopenharmony_ci vst->codecpar->codec_id = AV_CODEC_ID_MVC1; 336cabdff1aSopenharmony_ci break; 337cabdff1aSopenharmony_ci case 2: 338cabdff1aSopenharmony_ci vst->codecpar->format = AV_PIX_FMT_ARGB; 339cabdff1aSopenharmony_ci vst->codecpar->codec_id = AV_CODEC_ID_RAWVIDEO; 340cabdff1aSopenharmony_ci break; 341cabdff1aSopenharmony_ci default: 342cabdff1aSopenharmony_ci avpriv_request_sample(avctx, "Video compression %i", v); 343cabdff1aSopenharmony_ci break; 344cabdff1aSopenharmony_ci } 345cabdff1aSopenharmony_ci vst->codecpar->codec_tag = 0; 346cabdff1aSopenharmony_ci vst->codecpar->width = avio_rb32(pb); 347cabdff1aSopenharmony_ci vst->codecpar->height = avio_rb32(pb); 348cabdff1aSopenharmony_ci avio_skip(pb, 12); 349cabdff1aSopenharmony_ci 350cabdff1aSopenharmony_ci if (ast) { 351cabdff1aSopenharmony_ci ast->codecpar->codec_type = AVMEDIA_TYPE_AUDIO; 352cabdff1aSopenharmony_ci ast->nb_frames = vst->nb_frames; 353cabdff1aSopenharmony_ci ast->codecpar->sample_rate = avio_rb32(pb); 354cabdff1aSopenharmony_ci if (ast->codecpar->sample_rate <= 0) { 355cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "Invalid sample rate %d\n", ast->codecpar->sample_rate); 356cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 357cabdff1aSopenharmony_ci } 358cabdff1aSopenharmony_ci avpriv_set_pts_info(ast, 33, 1, ast->codecpar->sample_rate); 359cabdff1aSopenharmony_ci 360cabdff1aSopenharmony_ci bytes_per_sample = avio_rb32(pb); 361cabdff1aSopenharmony_ci 362cabdff1aSopenharmony_ci v = avio_rb32(pb); 363cabdff1aSopenharmony_ci if (v == AUDIO_FORMAT_SIGNED) { 364cabdff1aSopenharmony_ci switch (bytes_per_sample) { 365cabdff1aSopenharmony_ci case 1: 366cabdff1aSopenharmony_ci ast->codecpar->codec_id = AV_CODEC_ID_PCM_S8; 367cabdff1aSopenharmony_ci break; 368cabdff1aSopenharmony_ci case 2: 369cabdff1aSopenharmony_ci ast->codecpar->codec_id = AV_CODEC_ID_PCM_S16BE; 370cabdff1aSopenharmony_ci break; 371cabdff1aSopenharmony_ci default: 372cabdff1aSopenharmony_ci avpriv_request_sample(avctx, "Audio sample size %i bytes", bytes_per_sample); 373cabdff1aSopenharmony_ci break; 374cabdff1aSopenharmony_ci } 375cabdff1aSopenharmony_ci } else { 376cabdff1aSopenharmony_ci avpriv_request_sample(avctx, "Audio compression (format %i)", v); 377cabdff1aSopenharmony_ci } 378cabdff1aSopenharmony_ci 379cabdff1aSopenharmony_ci if (bytes_per_sample == 0) 380cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 381cabdff1aSopenharmony_ci 382cabdff1aSopenharmony_ci if (set_channels(avctx, ast, avio_rb32(pb)) < 0) 383cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 384cabdff1aSopenharmony_ci 385cabdff1aSopenharmony_ci avio_skip(pb, 8); 386cabdff1aSopenharmony_ci } else 387cabdff1aSopenharmony_ci avio_skip(pb, 24); /* skip meaningless audio metadata */ 388cabdff1aSopenharmony_ci 389cabdff1aSopenharmony_ci var_read_metadata(avctx, "title", 0x80); 390cabdff1aSopenharmony_ci var_read_metadata(avctx, "comment", 0x100); 391cabdff1aSopenharmony_ci avio_skip(pb, 0x80); 392cabdff1aSopenharmony_ci 393cabdff1aSopenharmony_ci timestamp = 0; 394cabdff1aSopenharmony_ci for (i = 0; i < vst->nb_frames; i++) { 395cabdff1aSopenharmony_ci uint32_t pos = avio_rb32(pb); 396cabdff1aSopenharmony_ci uint32_t asize = avio_rb32(pb); 397cabdff1aSopenharmony_ci uint32_t vsize = avio_rb32(pb); 398cabdff1aSopenharmony_ci if (avio_feof(pb)) 399cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 400cabdff1aSopenharmony_ci avio_skip(pb, 8); 401cabdff1aSopenharmony_ci if (ast) { 402cabdff1aSopenharmony_ci av_add_index_entry(ast, pos, timestamp, asize, 0, AVINDEX_KEYFRAME); 403cabdff1aSopenharmony_ci timestamp += asize / (ast->codecpar->ch_layout.nb_channels * (uint64_t)bytes_per_sample); 404cabdff1aSopenharmony_ci } 405cabdff1aSopenharmony_ci av_add_index_entry(vst, pos + asize, i, vsize, 0, AVINDEX_KEYFRAME); 406cabdff1aSopenharmony_ci } 407cabdff1aSopenharmony_ci } else if (!version && avio_rb16(pb) == 3) { 408cabdff1aSopenharmony_ci avio_skip(pb, 4); 409cabdff1aSopenharmony_ci 410cabdff1aSopenharmony_ci if ((ret = read_table(avctx, NULL, parse_global_var)) < 0) 411cabdff1aSopenharmony_ci return ret; 412cabdff1aSopenharmony_ci 413cabdff1aSopenharmony_ci if (mv->nb_audio_tracks < 0 || mv->nb_video_tracks < 0 || 414cabdff1aSopenharmony_ci (mv->nb_audio_tracks == 0 && mv->nb_video_tracks == 0)) { 415cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "Stream count is invalid.\n"); 416cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 417cabdff1aSopenharmony_ci } 418cabdff1aSopenharmony_ci 419cabdff1aSopenharmony_ci if (mv->nb_audio_tracks > 1) { 420cabdff1aSopenharmony_ci avpriv_request_sample(avctx, "Multiple audio streams support"); 421cabdff1aSopenharmony_ci return AVERROR_PATCHWELCOME; 422cabdff1aSopenharmony_ci } else if (mv->nb_audio_tracks) { 423cabdff1aSopenharmony_ci ast = avformat_new_stream(avctx, NULL); 424cabdff1aSopenharmony_ci if (!ast) 425cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 426cabdff1aSopenharmony_ci ast->codecpar->codec_type = AVMEDIA_TYPE_AUDIO; 427cabdff1aSopenharmony_ci if ((ret = read_table(avctx, ast, parse_audio_var)) < 0) 428cabdff1aSopenharmony_ci return ret; 429cabdff1aSopenharmony_ci if (mv->acompression == 100 && 430cabdff1aSopenharmony_ci mv->aformat == AUDIO_FORMAT_SIGNED && 431cabdff1aSopenharmony_ci ast->codecpar->bits_per_coded_sample == 16) { 432cabdff1aSopenharmony_ci ast->codecpar->codec_id = AV_CODEC_ID_PCM_S16BE; 433cabdff1aSopenharmony_ci } else { 434cabdff1aSopenharmony_ci avpriv_request_sample(avctx, 435cabdff1aSopenharmony_ci "Audio compression %i (format %i, sr %i)", 436cabdff1aSopenharmony_ci mv->acompression, mv->aformat, 437cabdff1aSopenharmony_ci ast->codecpar->bits_per_coded_sample); 438cabdff1aSopenharmony_ci ast->codecpar->codec_id = AV_CODEC_ID_NONE; 439cabdff1aSopenharmony_ci } 440cabdff1aSopenharmony_ci if (ast->codecpar->ch_layout.nb_channels <= 0) { 441cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "No valid channel count found.\n"); 442cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 443cabdff1aSopenharmony_ci } 444cabdff1aSopenharmony_ci } 445cabdff1aSopenharmony_ci 446cabdff1aSopenharmony_ci if (mv->nb_video_tracks > 1) { 447cabdff1aSopenharmony_ci avpriv_request_sample(avctx, "Multiple video streams support"); 448cabdff1aSopenharmony_ci return AVERROR_PATCHWELCOME; 449cabdff1aSopenharmony_ci } else if (mv->nb_video_tracks) { 450cabdff1aSopenharmony_ci vst = avformat_new_stream(avctx, NULL); 451cabdff1aSopenharmony_ci if (!vst) 452cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 453cabdff1aSopenharmony_ci vst->codecpar->codec_type = AVMEDIA_TYPE_VIDEO; 454cabdff1aSopenharmony_ci if ((ret = read_table(avctx, vst, parse_video_var))<0) 455cabdff1aSopenharmony_ci return ret; 456cabdff1aSopenharmony_ci } 457cabdff1aSopenharmony_ci 458cabdff1aSopenharmony_ci if (mv->nb_audio_tracks) 459cabdff1aSopenharmony_ci read_index(pb, ast); 460cabdff1aSopenharmony_ci 461cabdff1aSopenharmony_ci if (mv->nb_video_tracks) 462cabdff1aSopenharmony_ci read_index(pb, vst); 463cabdff1aSopenharmony_ci } else { 464cabdff1aSopenharmony_ci avpriv_request_sample(avctx, "Version %i", version); 465cabdff1aSopenharmony_ci return AVERROR_PATCHWELCOME; 466cabdff1aSopenharmony_ci } 467cabdff1aSopenharmony_ci 468cabdff1aSopenharmony_ci return 0; 469cabdff1aSopenharmony_ci} 470cabdff1aSopenharmony_ci 471cabdff1aSopenharmony_cistatic int mv_read_packet(AVFormatContext *avctx, AVPacket *pkt) 472cabdff1aSopenharmony_ci{ 473cabdff1aSopenharmony_ci MvContext *mv = avctx->priv_data; 474cabdff1aSopenharmony_ci AVIOContext *pb = avctx->pb; 475cabdff1aSopenharmony_ci AVStream *st = avctx->streams[mv->stream_index]; 476cabdff1aSopenharmony_ci FFStream *const sti = ffstream(st); 477cabdff1aSopenharmony_ci const AVIndexEntry *index; 478cabdff1aSopenharmony_ci int frame = mv->frame[mv->stream_index]; 479cabdff1aSopenharmony_ci int64_t ret; 480cabdff1aSopenharmony_ci uint64_t pos; 481cabdff1aSopenharmony_ci 482cabdff1aSopenharmony_ci if (frame < sti->nb_index_entries) { 483cabdff1aSopenharmony_ci index = &sti->index_entries[frame]; 484cabdff1aSopenharmony_ci pos = avio_tell(pb); 485cabdff1aSopenharmony_ci if (index->pos > pos) 486cabdff1aSopenharmony_ci avio_skip(pb, index->pos - pos); 487cabdff1aSopenharmony_ci else if (index->pos < pos) { 488cabdff1aSopenharmony_ci if (!(pb->seekable & AVIO_SEEKABLE_NORMAL)) 489cabdff1aSopenharmony_ci return AVERROR(EIO); 490cabdff1aSopenharmony_ci ret = avio_seek(pb, index->pos, SEEK_SET); 491cabdff1aSopenharmony_ci if (ret < 0) 492cabdff1aSopenharmony_ci return ret; 493cabdff1aSopenharmony_ci } 494cabdff1aSopenharmony_ci ret = av_get_packet(pb, pkt, index->size); 495cabdff1aSopenharmony_ci if (ret < 0) 496cabdff1aSopenharmony_ci return ret; 497cabdff1aSopenharmony_ci 498cabdff1aSopenharmony_ci pkt->stream_index = mv->stream_index; 499cabdff1aSopenharmony_ci pkt->pts = index->timestamp; 500cabdff1aSopenharmony_ci pkt->flags |= AV_PKT_FLAG_KEY; 501cabdff1aSopenharmony_ci 502cabdff1aSopenharmony_ci mv->frame[mv->stream_index]++; 503cabdff1aSopenharmony_ci mv->eof_count = 0; 504cabdff1aSopenharmony_ci } else { 505cabdff1aSopenharmony_ci mv->eof_count++; 506cabdff1aSopenharmony_ci if (mv->eof_count >= avctx->nb_streams) 507cabdff1aSopenharmony_ci return AVERROR_EOF; 508cabdff1aSopenharmony_ci 509cabdff1aSopenharmony_ci // avoid returning 0 without a packet 510cabdff1aSopenharmony_ci return AVERROR(EAGAIN); 511cabdff1aSopenharmony_ci } 512cabdff1aSopenharmony_ci 513cabdff1aSopenharmony_ci mv->stream_index++; 514cabdff1aSopenharmony_ci if (mv->stream_index >= avctx->nb_streams) 515cabdff1aSopenharmony_ci mv->stream_index = 0; 516cabdff1aSopenharmony_ci 517cabdff1aSopenharmony_ci return 0; 518cabdff1aSopenharmony_ci} 519cabdff1aSopenharmony_ci 520cabdff1aSopenharmony_cistatic int mv_read_seek(AVFormatContext *avctx, int stream_index, 521cabdff1aSopenharmony_ci int64_t timestamp, int flags) 522cabdff1aSopenharmony_ci{ 523cabdff1aSopenharmony_ci MvContext *mv = avctx->priv_data; 524cabdff1aSopenharmony_ci AVStream *st = avctx->streams[stream_index]; 525cabdff1aSopenharmony_ci int frame, i; 526cabdff1aSopenharmony_ci 527cabdff1aSopenharmony_ci if ((flags & AVSEEK_FLAG_FRAME) || (flags & AVSEEK_FLAG_BYTE)) 528cabdff1aSopenharmony_ci return AVERROR(ENOSYS); 529cabdff1aSopenharmony_ci 530cabdff1aSopenharmony_ci if (!(avctx->pb->seekable & AVIO_SEEKABLE_NORMAL)) 531cabdff1aSopenharmony_ci return AVERROR(EIO); 532cabdff1aSopenharmony_ci 533cabdff1aSopenharmony_ci frame = av_index_search_timestamp(st, timestamp, flags); 534cabdff1aSopenharmony_ci if (frame < 0) 535cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 536cabdff1aSopenharmony_ci 537cabdff1aSopenharmony_ci for (i = 0; i < avctx->nb_streams; i++) 538cabdff1aSopenharmony_ci mv->frame[i] = frame; 539cabdff1aSopenharmony_ci return 0; 540cabdff1aSopenharmony_ci} 541cabdff1aSopenharmony_ci 542cabdff1aSopenharmony_ciconst AVInputFormat ff_mv_demuxer = { 543cabdff1aSopenharmony_ci .name = "mv", 544cabdff1aSopenharmony_ci .long_name = NULL_IF_CONFIG_SMALL("Silicon Graphics Movie"), 545cabdff1aSopenharmony_ci .priv_data_size = sizeof(MvContext), 546cabdff1aSopenharmony_ci .read_probe = mv_probe, 547cabdff1aSopenharmony_ci .read_header = mv_read_header, 548cabdff1aSopenharmony_ci .read_packet = mv_read_packet, 549cabdff1aSopenharmony_ci .read_seek = mv_read_seek, 550cabdff1aSopenharmony_ci}; 551