1cabdff1aSopenharmony_ci/* 2cabdff1aSopenharmony_ci * Copyright (C) 2008 David Conrad 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#include "libavutil/imgutils.h" 22cabdff1aSopenharmony_ci#include "libavutil/intreadwrite.h" 23cabdff1aSopenharmony_ci#include "libavcodec/dirac.h" 24cabdff1aSopenharmony_ci#include "avformat.h" 25cabdff1aSopenharmony_ci#include "internal.h" 26cabdff1aSopenharmony_ci#include "oggdec.h" 27cabdff1aSopenharmony_ci 28cabdff1aSopenharmony_cistatic int dirac_header(AVFormatContext *s, int idx) 29cabdff1aSopenharmony_ci{ 30cabdff1aSopenharmony_ci struct ogg *ogg = s->priv_data; 31cabdff1aSopenharmony_ci struct ogg_stream *os = ogg->streams + idx; 32cabdff1aSopenharmony_ci AVStream *st = s->streams[idx]; 33cabdff1aSopenharmony_ci AVDiracSeqHeader *dsh; 34cabdff1aSopenharmony_ci int ret; 35cabdff1aSopenharmony_ci 36cabdff1aSopenharmony_ci // already parsed the header 37cabdff1aSopenharmony_ci if (st->codecpar->codec_id == AV_CODEC_ID_DIRAC) 38cabdff1aSopenharmony_ci return 0; 39cabdff1aSopenharmony_ci 40cabdff1aSopenharmony_ci ret = av_dirac_parse_sequence_header(&dsh, os->buf + os->pstart + 13, (os->psize - 13), s); 41cabdff1aSopenharmony_ci if (ret < 0) 42cabdff1aSopenharmony_ci return ret; 43cabdff1aSopenharmony_ci 44cabdff1aSopenharmony_ci st->codecpar->codec_type = AVMEDIA_TYPE_VIDEO; 45cabdff1aSopenharmony_ci st->codecpar->codec_id = AV_CODEC_ID_DIRAC; 46cabdff1aSopenharmony_ci st->codecpar->width = dsh->width; 47cabdff1aSopenharmony_ci st->codecpar->height = dsh->height; 48cabdff1aSopenharmony_ci st->codecpar->format = dsh->pix_fmt; 49cabdff1aSopenharmony_ci st->codecpar->color_range = dsh->color_range; 50cabdff1aSopenharmony_ci st->codecpar->color_trc = dsh->color_trc; 51cabdff1aSopenharmony_ci st->codecpar->color_primaries = dsh->color_primaries; 52cabdff1aSopenharmony_ci st->codecpar->color_space = dsh->colorspace; 53cabdff1aSopenharmony_ci st->codecpar->profile = dsh->profile; 54cabdff1aSopenharmony_ci st->codecpar->level = dsh->level; 55cabdff1aSopenharmony_ci if (av_image_check_sar(st->codecpar->width, st->codecpar->height, dsh->sample_aspect_ratio) >= 0) 56cabdff1aSopenharmony_ci st->sample_aspect_ratio = dsh->sample_aspect_ratio; 57cabdff1aSopenharmony_ci 58cabdff1aSopenharmony_ci // Dirac in Ogg always stores timestamps as though the video were interlaced 59cabdff1aSopenharmony_ci avpriv_set_pts_info(st, 64, dsh->framerate.den, 2 * dsh->framerate.num); 60cabdff1aSopenharmony_ci 61cabdff1aSopenharmony_ci av_freep(&dsh); 62cabdff1aSopenharmony_ci return 1; 63cabdff1aSopenharmony_ci} 64cabdff1aSopenharmony_ci 65cabdff1aSopenharmony_ci// various undocumented things: granule is signed (only for Dirac!) 66cabdff1aSopenharmony_cistatic uint64_t dirac_gptopts(AVFormatContext *s, int idx, uint64_t granule, 67cabdff1aSopenharmony_ci int64_t *dts_out) 68cabdff1aSopenharmony_ci{ 69cabdff1aSopenharmony_ci int64_t gp = granule; 70cabdff1aSopenharmony_ci struct ogg *ogg = s->priv_data; 71cabdff1aSopenharmony_ci struct ogg_stream *os = ogg->streams + idx; 72cabdff1aSopenharmony_ci 73cabdff1aSopenharmony_ci unsigned dist = ((gp >> 14) & 0xff00) | (gp & 0xff); 74cabdff1aSopenharmony_ci int64_t dts = (gp >> 31); 75cabdff1aSopenharmony_ci int64_t pts = dts + ((gp >> 9) & 0x1fff); 76cabdff1aSopenharmony_ci 77cabdff1aSopenharmony_ci if (!dist) 78cabdff1aSopenharmony_ci os->pflags |= AV_PKT_FLAG_KEY; 79cabdff1aSopenharmony_ci 80cabdff1aSopenharmony_ci if (dts_out) 81cabdff1aSopenharmony_ci *dts_out = dts; 82cabdff1aSopenharmony_ci 83cabdff1aSopenharmony_ci return pts; 84cabdff1aSopenharmony_ci} 85cabdff1aSopenharmony_ci 86cabdff1aSopenharmony_cistatic int old_dirac_header(AVFormatContext *s, int idx) 87cabdff1aSopenharmony_ci{ 88cabdff1aSopenharmony_ci struct ogg *ogg = s->priv_data; 89cabdff1aSopenharmony_ci struct ogg_stream *os = ogg->streams + idx; 90cabdff1aSopenharmony_ci AVStream *st = s->streams[idx]; 91cabdff1aSopenharmony_ci uint8_t *buf = os->buf + os->pstart; 92cabdff1aSopenharmony_ci 93cabdff1aSopenharmony_ci if (buf[0] != 'K') 94cabdff1aSopenharmony_ci return 0; 95cabdff1aSopenharmony_ci 96cabdff1aSopenharmony_ci st->codecpar->codec_type = AVMEDIA_TYPE_VIDEO; 97cabdff1aSopenharmony_ci st->codecpar->codec_id = AV_CODEC_ID_DIRAC; 98cabdff1aSopenharmony_ci avpriv_set_pts_info(st, 64, AV_RB32(buf+12), AV_RB32(buf+8)); 99cabdff1aSopenharmony_ci return 1; 100cabdff1aSopenharmony_ci} 101cabdff1aSopenharmony_ci 102cabdff1aSopenharmony_cistatic uint64_t old_dirac_gptopts(AVFormatContext *s, int idx, uint64_t gp, 103cabdff1aSopenharmony_ci int64_t *dts) 104cabdff1aSopenharmony_ci{ 105cabdff1aSopenharmony_ci struct ogg *ogg = s->priv_data; 106cabdff1aSopenharmony_ci struct ogg_stream *os = ogg->streams + idx; 107cabdff1aSopenharmony_ci uint64_t iframe = gp >> 30; 108cabdff1aSopenharmony_ci uint64_t pframe = gp & 0x3fffffff; 109cabdff1aSopenharmony_ci 110cabdff1aSopenharmony_ci if (!pframe) 111cabdff1aSopenharmony_ci os->pflags |= AV_PKT_FLAG_KEY; 112cabdff1aSopenharmony_ci 113cabdff1aSopenharmony_ci return iframe + pframe; 114cabdff1aSopenharmony_ci} 115cabdff1aSopenharmony_ci 116cabdff1aSopenharmony_ciconst struct ogg_codec ff_dirac_codec = { 117cabdff1aSopenharmony_ci .magic = "BBCD\0", 118cabdff1aSopenharmony_ci .magicsize = 5, 119cabdff1aSopenharmony_ci .header = dirac_header, 120cabdff1aSopenharmony_ci .gptopts = dirac_gptopts, 121cabdff1aSopenharmony_ci .granule_is_start = 1, 122cabdff1aSopenharmony_ci .nb_header = 1, 123cabdff1aSopenharmony_ci}; 124cabdff1aSopenharmony_ci 125cabdff1aSopenharmony_ciconst struct ogg_codec ff_old_dirac_codec = { 126cabdff1aSopenharmony_ci .magic = "KW-DIRAC", 127cabdff1aSopenharmony_ci .magicsize = 8, 128cabdff1aSopenharmony_ci .header = old_dirac_header, 129cabdff1aSopenharmony_ci .gptopts = old_dirac_gptopts, 130cabdff1aSopenharmony_ci .granule_is_start = 1, 131cabdff1aSopenharmony_ci .nb_header = 1, 132cabdff1aSopenharmony_ci}; 133