1cabdff1aSopenharmony_ci/* 2cabdff1aSopenharmony_ci * SCC muxer 3cabdff1aSopenharmony_ci * Copyright (c) 2017 Paul B Mahol 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#include "avformat.h" 23cabdff1aSopenharmony_ci#include "internal.h" 24cabdff1aSopenharmony_ci#include "libavutil/log.h" 25cabdff1aSopenharmony_ci#include "libavutil/intreadwrite.h" 26cabdff1aSopenharmony_ci 27cabdff1aSopenharmony_citypedef struct SCCContext { 28cabdff1aSopenharmony_ci int prev_h, prev_m, prev_s, prev_f; 29cabdff1aSopenharmony_ci int inside; 30cabdff1aSopenharmony_ci int n; 31cabdff1aSopenharmony_ci} SCCContext; 32cabdff1aSopenharmony_ci 33cabdff1aSopenharmony_cistatic int scc_write_header(AVFormatContext *avf) 34cabdff1aSopenharmony_ci{ 35cabdff1aSopenharmony_ci SCCContext *scc = avf->priv_data; 36cabdff1aSopenharmony_ci 37cabdff1aSopenharmony_ci if (avf->nb_streams != 1 || 38cabdff1aSopenharmony_ci avf->streams[0]->codecpar->codec_type != AVMEDIA_TYPE_SUBTITLE) { 39cabdff1aSopenharmony_ci av_log(avf, AV_LOG_ERROR, 40cabdff1aSopenharmony_ci "SCC supports only a single subtitles stream.\n"); 41cabdff1aSopenharmony_ci return AVERROR(EINVAL); 42cabdff1aSopenharmony_ci } 43cabdff1aSopenharmony_ci if (avf->streams[0]->codecpar->codec_id != AV_CODEC_ID_EIA_608) { 44cabdff1aSopenharmony_ci av_log(avf, AV_LOG_ERROR, 45cabdff1aSopenharmony_ci "Unsupported subtitles codec: %s\n", 46cabdff1aSopenharmony_ci avcodec_get_name(avf->streams[0]->codecpar->codec_id)); 47cabdff1aSopenharmony_ci return AVERROR(EINVAL); 48cabdff1aSopenharmony_ci } 49cabdff1aSopenharmony_ci avpriv_set_pts_info(avf->streams[0], 64, 1, 1000); 50cabdff1aSopenharmony_ci avio_printf(avf->pb, "Scenarist_SCC V1.0\n"); 51cabdff1aSopenharmony_ci 52cabdff1aSopenharmony_ci scc->prev_h = scc->prev_m = scc->prev_s = scc->prev_f = -1; 53cabdff1aSopenharmony_ci scc->inside = 0; 54cabdff1aSopenharmony_ci 55cabdff1aSopenharmony_ci return 0; 56cabdff1aSopenharmony_ci} 57cabdff1aSopenharmony_ci 58cabdff1aSopenharmony_cistatic int scc_write_packet(AVFormatContext *avf, AVPacket *pkt) 59cabdff1aSopenharmony_ci{ 60cabdff1aSopenharmony_ci SCCContext *scc = avf->priv_data; 61cabdff1aSopenharmony_ci int64_t pts = pkt->pts; 62cabdff1aSopenharmony_ci int i, h, m, s, f; 63cabdff1aSopenharmony_ci 64cabdff1aSopenharmony_ci if (pts == AV_NOPTS_VALUE) { 65cabdff1aSopenharmony_ci av_log(avf, AV_LOG_WARNING, 66cabdff1aSopenharmony_ci "Insufficient timestamps.\n"); 67cabdff1aSopenharmony_ci return 0; 68cabdff1aSopenharmony_ci } 69cabdff1aSopenharmony_ci 70cabdff1aSopenharmony_ci h = (int)(pts / (3600000)); 71cabdff1aSopenharmony_ci m = (int)(pts / (60000)) % 60; 72cabdff1aSopenharmony_ci s = (int)(pts / 1000) % 60; 73cabdff1aSopenharmony_ci f = (int)(pts % 1000) / 33; 74cabdff1aSopenharmony_ci 75cabdff1aSopenharmony_ci for (i = 0; i < pkt->size - 2; i+=3) { 76cabdff1aSopenharmony_ci if (pkt->data[i] == 0xfc && ((pkt->data[i + 1] != 0x80 || pkt->data[i + 2] != 0x80))) 77cabdff1aSopenharmony_ci break; 78cabdff1aSopenharmony_ci } 79cabdff1aSopenharmony_ci if (i >= pkt->size - 2) 80cabdff1aSopenharmony_ci return 0; 81cabdff1aSopenharmony_ci 82cabdff1aSopenharmony_ci if (!scc->inside && (scc->prev_h != h || scc->prev_m != m || scc->prev_s != s || scc->prev_f != f)) { 83cabdff1aSopenharmony_ci avio_printf(avf->pb, "\n%02d:%02d:%02d:%02d\t", h, m, s, f); 84cabdff1aSopenharmony_ci scc->inside = 1; 85cabdff1aSopenharmony_ci } 86cabdff1aSopenharmony_ci for (i = 0; i < pkt->size; i+=3) { 87cabdff1aSopenharmony_ci if (i + 3 > pkt->size) 88cabdff1aSopenharmony_ci break; 89cabdff1aSopenharmony_ci 90cabdff1aSopenharmony_ci if (pkt->data[i] != 0xfc || (pkt->data[i + 1] == 0x80 && pkt->data[i + 2] == 0x80)) 91cabdff1aSopenharmony_ci continue; 92cabdff1aSopenharmony_ci if (!scc->inside) { 93cabdff1aSopenharmony_ci avio_printf(avf->pb, "\n%02d:%02d:%02d:%02d\t", h, m, s, f); 94cabdff1aSopenharmony_ci scc->inside = 1; 95cabdff1aSopenharmony_ci } 96cabdff1aSopenharmony_ci if (scc->n > 0) 97cabdff1aSopenharmony_ci avio_w8(avf->pb, ' '); 98cabdff1aSopenharmony_ci avio_printf(avf->pb, "%02x%02x", pkt->data[i + 1], pkt->data[i + 2]); 99cabdff1aSopenharmony_ci scc->n++; 100cabdff1aSopenharmony_ci } 101cabdff1aSopenharmony_ci if (scc->inside && (scc->prev_h != h || scc->prev_m != m || scc->prev_s != s || scc->prev_f != f)) { 102cabdff1aSopenharmony_ci avio_w8(avf->pb, '\n'); 103cabdff1aSopenharmony_ci scc->n = 0; 104cabdff1aSopenharmony_ci scc->inside = 0; 105cabdff1aSopenharmony_ci } 106cabdff1aSopenharmony_ci 107cabdff1aSopenharmony_ci scc->prev_h = h; 108cabdff1aSopenharmony_ci scc->prev_m = m; 109cabdff1aSopenharmony_ci scc->prev_s = s; 110cabdff1aSopenharmony_ci scc->prev_f = f; 111cabdff1aSopenharmony_ci return 0; 112cabdff1aSopenharmony_ci} 113cabdff1aSopenharmony_ci 114cabdff1aSopenharmony_ciconst AVOutputFormat ff_scc_muxer = { 115cabdff1aSopenharmony_ci .name = "scc", 116cabdff1aSopenharmony_ci .long_name = NULL_IF_CONFIG_SMALL("Scenarist Closed Captions"), 117cabdff1aSopenharmony_ci .extensions = "scc", 118cabdff1aSopenharmony_ci .priv_data_size = sizeof(SCCContext), 119cabdff1aSopenharmony_ci .write_header = scc_write_header, 120cabdff1aSopenharmony_ci .write_packet = scc_write_packet, 121cabdff1aSopenharmony_ci .flags = AVFMT_GLOBALHEADER | AVFMT_VARIABLE_FPS | AVFMT_TS_NONSTRICT, 122cabdff1aSopenharmony_ci .subtitle_codec = AV_CODEC_ID_EIA_608, 123cabdff1aSopenharmony_ci}; 124