yading@11
|
1 /*
|
yading@11
|
2 * Copyright (c) 2012 Clément Bœsch
|
yading@11
|
3 *
|
yading@11
|
4 * This file is part of FFmpeg.
|
yading@11
|
5 *
|
yading@11
|
6 * FFmpeg is free software; you can redistribute it and/or
|
yading@11
|
7 * modify it under the terms of the GNU Lesser General Public
|
yading@11
|
8 * License as published by the Free Software Foundation; either
|
yading@11
|
9 * version 2.1 of the License, or (at your option) any later version.
|
yading@11
|
10 *
|
yading@11
|
11 * FFmpeg is distributed in the hope that it will be useful,
|
yading@11
|
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
yading@11
|
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
yading@11
|
14 * Lesser General Public License for more details.
|
yading@11
|
15 *
|
yading@11
|
16 * You should have received a copy of the GNU Lesser General Public
|
yading@11
|
17 * License along with FFmpeg; if not, write to the Free Software
|
yading@11
|
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
yading@11
|
19 */
|
yading@11
|
20
|
yading@11
|
21 /**
|
yading@11
|
22 * @file
|
yading@11
|
23 * PJS (Phoenix Japanimation Society) subtitles format demuxer
|
yading@11
|
24 *
|
yading@11
|
25 * @see http://subs.com.ru/page.php?al=pjs
|
yading@11
|
26 */
|
yading@11
|
27
|
yading@11
|
28 #include "avformat.h"
|
yading@11
|
29 #include "internal.h"
|
yading@11
|
30 #include "subtitles.h"
|
yading@11
|
31
|
yading@11
|
32 typedef struct {
|
yading@11
|
33 FFDemuxSubtitlesQueue q;
|
yading@11
|
34 } PJSContext;
|
yading@11
|
35
|
yading@11
|
36 static int pjs_probe(AVProbeData *p)
|
yading@11
|
37 {
|
yading@11
|
38 char c;
|
yading@11
|
39 int64_t start, end;
|
yading@11
|
40 const unsigned char *ptr = p->buf;
|
yading@11
|
41
|
yading@11
|
42 if (sscanf(ptr, "%"SCNd64",%"SCNd64",%c", &start, &end, &c) == 3) {
|
yading@11
|
43 size_t q1pos = strcspn(ptr, "\"");
|
yading@11
|
44 size_t q2pos = q1pos + strcspn(ptr + q1pos + 1, "\"") + 1;
|
yading@11
|
45 if (strcspn(ptr, "\r\n") > q2pos)
|
yading@11
|
46 return AVPROBE_SCORE_MAX;
|
yading@11
|
47 }
|
yading@11
|
48 return 0;
|
yading@11
|
49 }
|
yading@11
|
50
|
yading@11
|
51 static int64_t read_ts(char **line, int *duration)
|
yading@11
|
52 {
|
yading@11
|
53 int64_t start, end;
|
yading@11
|
54
|
yading@11
|
55 if (sscanf(*line, "%"SCNd64",%"SCNd64, &start, &end) == 2) {
|
yading@11
|
56 *line += strcspn(*line, "\"") + 1;
|
yading@11
|
57 *duration = end - start;
|
yading@11
|
58 return start;
|
yading@11
|
59 }
|
yading@11
|
60 return AV_NOPTS_VALUE;
|
yading@11
|
61 }
|
yading@11
|
62
|
yading@11
|
63 static int pjs_read_header(AVFormatContext *s)
|
yading@11
|
64 {
|
yading@11
|
65 PJSContext *pjs = s->priv_data;
|
yading@11
|
66 AVStream *st = avformat_new_stream(s, NULL);
|
yading@11
|
67 int res = 0;
|
yading@11
|
68
|
yading@11
|
69 if (!st)
|
yading@11
|
70 return AVERROR(ENOMEM);
|
yading@11
|
71 avpriv_set_pts_info(st, 64, 1, 10);
|
yading@11
|
72 st->codec->codec_type = AVMEDIA_TYPE_SUBTITLE;
|
yading@11
|
73 st->codec->codec_id = AV_CODEC_ID_PJS;
|
yading@11
|
74
|
yading@11
|
75 while (!url_feof(s->pb)) {
|
yading@11
|
76 char line[4096];
|
yading@11
|
77 char *p = line;
|
yading@11
|
78 const int64_t pos = avio_tell(s->pb);
|
yading@11
|
79 int len = ff_get_line(s->pb, line, sizeof(line));
|
yading@11
|
80 int64_t pts_start;
|
yading@11
|
81 int duration;
|
yading@11
|
82
|
yading@11
|
83 if (!len)
|
yading@11
|
84 break;
|
yading@11
|
85
|
yading@11
|
86 line[strcspn(line, "\r\n")] = 0;
|
yading@11
|
87
|
yading@11
|
88 pts_start = read_ts(&p, &duration);
|
yading@11
|
89 if (pts_start != AV_NOPTS_VALUE) {
|
yading@11
|
90 AVPacket *sub;
|
yading@11
|
91
|
yading@11
|
92 p[strcspn(p, "\"")] = 0;
|
yading@11
|
93 sub = ff_subtitles_queue_insert(&pjs->q, p, strlen(p), 0);
|
yading@11
|
94 if (!sub)
|
yading@11
|
95 return AVERROR(ENOMEM);
|
yading@11
|
96 sub->pos = pos;
|
yading@11
|
97 sub->pts = pts_start;
|
yading@11
|
98 sub->duration = duration;
|
yading@11
|
99 }
|
yading@11
|
100 }
|
yading@11
|
101
|
yading@11
|
102 ff_subtitles_queue_finalize(&pjs->q);
|
yading@11
|
103 return res;
|
yading@11
|
104 }
|
yading@11
|
105
|
yading@11
|
106 static int pjs_read_packet(AVFormatContext *s, AVPacket *pkt)
|
yading@11
|
107 {
|
yading@11
|
108 PJSContext *pjs = s->priv_data;
|
yading@11
|
109 return ff_subtitles_queue_read_packet(&pjs->q, pkt);
|
yading@11
|
110 }
|
yading@11
|
111
|
yading@11
|
112 static int pjs_read_seek(AVFormatContext *s, int stream_index,
|
yading@11
|
113 int64_t min_ts, int64_t ts, int64_t max_ts, int flags)
|
yading@11
|
114 {
|
yading@11
|
115 PJSContext *pjs = s->priv_data;
|
yading@11
|
116 return ff_subtitles_queue_seek(&pjs->q, s, stream_index,
|
yading@11
|
117 min_ts, ts, max_ts, flags);
|
yading@11
|
118 }
|
yading@11
|
119
|
yading@11
|
120 static int pjs_read_close(AVFormatContext *s)
|
yading@11
|
121 {
|
yading@11
|
122 PJSContext *pjs = s->priv_data;
|
yading@11
|
123 ff_subtitles_queue_clean(&pjs->q);
|
yading@11
|
124 return 0;
|
yading@11
|
125 }
|
yading@11
|
126
|
yading@11
|
127 AVInputFormat ff_pjs_demuxer = {
|
yading@11
|
128 .name = "pjs",
|
yading@11
|
129 .long_name = NULL_IF_CONFIG_SMALL("PJS (Phoenix Japanimation Society) subtitles"),
|
yading@11
|
130 .priv_data_size = sizeof(PJSContext),
|
yading@11
|
131 .read_probe = pjs_probe,
|
yading@11
|
132 .read_header = pjs_read_header,
|
yading@11
|
133 .read_packet = pjs_read_packet,
|
yading@11
|
134 .read_seek2 = pjs_read_seek,
|
yading@11
|
135 .read_close = pjs_read_close,
|
yading@11
|
136 .extensions = "pjs",
|
yading@11
|
137 };
|