FFmpeg coverage


Directory: ../../../ffmpeg/
File: src/libavformat/realtextdec.c
Date: 2025-01-20 09:27:23
Exec Total Coverage
Lines: 50 60 83.3%
Functions: 3 3 100.0%
Branches: 30 42 71.4%

Line Branch Exec Source
1 /*
2 * Copyright (c) 2012 Clément Bœsch
3 *
4 * This file is part of FFmpeg.
5 *
6 * FFmpeg is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
10 *
11 * FFmpeg is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with FFmpeg; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19 */
20
21 /**
22 * @file
23 * RealText subtitle demuxer
24 * @see http://service.real.com/help/library/guides/ProductionGuide/prodguide/htmfiles/realtext.htm
25 */
26
27 #include "avformat.h"
28 #include "demux.h"
29 #include "internal.h"
30 #include "subtitles.h"
31 #include "libavutil/avstring.h"
32 #include "libavutil/bprint.h"
33 #include "libavutil/mem.h"
34
35 typedef struct {
36 FFDemuxSubtitlesQueue q;
37 } RealTextContext;
38
39 7203 static int realtext_probe(const AVProbeData *p)
40 {
41 char buf[7];
42 FFTextReader tr;
43 7203 ff_text_init_buf(&tr, p->buf, p->buf_size);
44 7203 ff_text_read(&tr, buf, sizeof(buf));
45
46
2/2
✓ Branch 1 taken 1 times.
✓ Branch 2 taken 7202 times.
7203 return !av_strncasecmp(buf, "<window", 7) ? AVPROBE_SCORE_EXTENSION : 0;
47 }
48
49 7 static int64_t read_ts(const char *s)
50 {
51 int hh, mm, ss, ms;
52
53
2/2
✓ Branch 0 taken 1 times.
✓ Branch 1 taken 6 times.
7 if (sscanf(s, "%u:%u:%u.%u", &hh, &mm, &ss, &ms) == 4) return (hh*3600LL + mm*60LL + ss) * 100LL + ms;
54
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6 times.
6 if (sscanf(s, "%u:%u:%u" , &hh, &mm, &ss ) == 3) return (hh*3600LL + mm*60LL + ss) * 100LL;
55
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6 times.
6 if (sscanf(s, "%u:%u.%u", &mm, &ss, &ms) == 3) return ( mm*60LL + ss) * 100LL + ms;
56
2/2
✓ Branch 0 taken 1 times.
✓ Branch 1 taken 5 times.
6 if (sscanf(s, "%u:%u" , &mm, &ss ) == 2) return ( mm*60LL + ss) * 100LL;
57
2/2
✓ Branch 0 taken 1 times.
✓ Branch 1 taken 4 times.
5 if (sscanf(s, "%u.%u", &ss, &ms) == 2) return ( ss) * 100LL + ms;
58 4 return strtoll(s, NULL, 10) * 100ULL;
59 }
60
61 1 static int realtext_read_header(AVFormatContext *s)
62 {
63 1 RealTextContext *rt = s->priv_data;
64 1 AVStream *st = avformat_new_stream(s, NULL);
65 AVBPrint buf;
66 1 char c = 0;
67 1 int res = 0, duration = read_ts("60"); // default duration is 60 seconds
68 FFTextReader tr;
69 1 ff_text_init_avio(s, &tr, s->pb);
70
71
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
1 if (!st)
72 return AVERROR(ENOMEM);
73 1 avpriv_set_pts_info(st, 64, 1, 100);
74 1 st->codecpar->codec_type = AVMEDIA_TYPE_SUBTITLE;
75 1 st->codecpar->codec_id = AV_CODEC_ID_REALTEXT;
76
77 1 av_bprint_init(&buf, 0, AV_BPRINT_SIZE_UNLIMITED);
78
79
2/2
✓ Branch 1 taken 16 times.
✓ Branch 2 taken 1 times.
17 while (!ff_text_eof(&tr)) {
80 AVPacket *sub;
81 16 const int64_t pos = ff_text_pos(&tr) - (c != 0);
82 16 int n = ff_smil_extract_next_text_chunk(&tr, &buf, &c);
83
84
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 16 times.
16 if (n < 0) {
85 res = n;
86 goto end;
87 }
88
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 16 times.
16 if (n == 0)
89 break;
90
91
2/2
✓ Branch 1 taken 1 times.
✓ Branch 2 taken 15 times.
16 if (!av_strncasecmp(buf.str, "<window", 7)) {
92 /* save header to extradata */
93 1 const char *p = ff_smil_get_attr_ptr(buf.str, "duration");
94
95
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
1 if (st->codecpar->extradata) {
96 res = AVERROR_INVALIDDATA;
97 goto end;
98 }
99
1/2
✓ Branch 0 taken 1 times.
✗ Branch 1 not taken.
1 if (p)
100 1 duration = read_ts(p);
101 1 st->codecpar->extradata = av_strdup(buf.str);
102
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
1 if (!st->codecpar->extradata) {
103 res = AVERROR(ENOMEM);
104 goto end;
105 }
106 1 st->codecpar->extradata_size = buf.len + 1;
107 } else {
108 /* if we just read a <time> tag, introduce a new event, otherwise merge
109 * with the previous one */
110 15 int merge = !av_strncasecmp(buf.str, "<time", 5) ? 0 : 1;
111 15 sub = ff_subtitles_queue_insert_bprint(&rt->q, &buf, merge);
112
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 15 times.
15 if (!sub) {
113 res = AVERROR(ENOMEM);
114 goto end;
115 }
116
2/2
✓ Branch 0 taken 4 times.
✓ Branch 1 taken 11 times.
15 if (!merge) {
117 4 const char *begin = ff_smil_get_attr_ptr(buf.str, "begin");
118 4 const char *end = ff_smil_get_attr_ptr(buf.str, "end");
119
2/2
✓ Branch 0 taken 1 times.
✓ Branch 1 taken 3 times.
4 int64_t endi = end ? read_ts(end) : 0;
120
121 4 sub->pos = pos;
122
1/2
✓ Branch 0 taken 4 times.
✗ Branch 1 not taken.
4 sub->pts = begin ? read_ts(begin) : 0;
123
4/6
✓ Branch 0 taken 1 times.
✓ Branch 1 taken 3 times.
✓ Branch 2 taken 1 times.
✗ Branch 3 not taken.
✓ Branch 4 taken 1 times.
✗ Branch 5 not taken.
4 sub->duration = (end && endi > sub->pts && endi - (uint64_t)sub->pts <= INT64_MAX) ? endi - sub->pts : duration;
124 }
125 }
126 16 av_bprint_clear(&buf);
127 }
128 1 ff_subtitles_queue_finalize(s, &rt->q);
129
130 1 end:
131 1 av_bprint_finalize(&buf, NULL);
132 1 return res;
133 }
134
135 const FFInputFormat ff_realtext_demuxer = {
136 .p.name = "realtext",
137 .p.long_name = NULL_IF_CONFIG_SMALL("RealText subtitle format"),
138 .p.extensions = "rt",
139 .priv_data_size = sizeof(RealTextContext),
140 .flags_internal = FF_INFMT_FLAG_INIT_CLEANUP,
141 .read_probe = realtext_probe,
142 .read_header = realtext_read_header,
143 .read_packet = ff_subtitles_read_packet,
144 .read_seek2 = ff_subtitles_read_seek,
145 .read_close = ff_subtitles_read_close,
146 };
147