mirror of
https://github.com/FFmpeg/FFmpeg.git
synced 2025-01-08 13:22:53 +02:00
libavformat: output cues for each subtitle block in MKV muxer
Signed-off-by: John Peebles <johnpeeb@gmail.com> Signed-off-by: Michael Niedermayer <michaelni@gmx.at>
This commit is contained in:
parent
9c22e909ed
commit
925f7980eb
@ -418,6 +418,7 @@ Muxers/Demuxers:
|
|||||||
matroska.c Aurelien Jacobs
|
matroska.c Aurelien Jacobs
|
||||||
matroskadec.c Aurelien Jacobs
|
matroskadec.c Aurelien Jacobs
|
||||||
matroskaenc.c David Conrad
|
matroskaenc.c David Conrad
|
||||||
|
matroska subtitles (matroskaenc.c) John Peebles
|
||||||
metadata* Aurelien Jacobs
|
metadata* Aurelien Jacobs
|
||||||
mgsts.c Paul B Mahol
|
mgsts.c Paul B Mahol
|
||||||
microdvd* Aurelien Jacobs
|
microdvd* Aurelien Jacobs
|
||||||
|
@ -427,8 +427,9 @@ static int mkv_add_cuepoint(mkv_cues *cues, int stream, int tracknum, int64_t ts
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int64_t mkv_write_cues(AVIOContext *pb, mkv_cues *cues, mkv_track *tracks, int num_tracks)
|
static int64_t mkv_write_cues(AVFormatContext *s, mkv_cues *cues, mkv_track *tracks, int num_tracks)
|
||||||
{
|
{
|
||||||
|
AVIOContext *pb = s->pb;
|
||||||
ebml_master cues_element;
|
ebml_master cues_element;
|
||||||
int64_t currentpos;
|
int64_t currentpos;
|
||||||
int i, j;
|
int i, j;
|
||||||
@ -451,7 +452,7 @@ static int64_t mkv_write_cues(AVIOContext *pb, mkv_cues *cues, mkv_track *tracks
|
|||||||
for (j = 0; j < cues->num_entries - i && entry[j].pts == pts; j++) {
|
for (j = 0; j < cues->num_entries - i && entry[j].pts == pts; j++) {
|
||||||
int tracknum = entry[j].stream_idx;
|
int tracknum = entry[j].stream_idx;
|
||||||
av_assert0(tracknum>=0 && tracknum<num_tracks);
|
av_assert0(tracknum>=0 && tracknum<num_tracks);
|
||||||
if (tracks[tracknum].has_cue)
|
if (tracks[tracknum].has_cue && s->streams[tracknum]->codec->codec_type != AVMEDIA_TYPE_SUBTITLE)
|
||||||
continue;
|
continue;
|
||||||
tracks[tracknum].has_cue = 1;
|
tracks[tracknum].has_cue = 1;
|
||||||
track_positions = start_ebml_master(pb, MATROSKA_ID_CUETRACKPOSITION, MAX_CUETRACKPOS_SIZE);
|
track_positions = start_ebml_master(pb, MATROSKA_ID_CUETRACKPOSITION, MAX_CUETRACKPOS_SIZE);
|
||||||
@ -1325,24 +1326,27 @@ static int ass_get_duration(const uint8_t *p)
|
|||||||
}
|
}
|
||||||
|
|
||||||
#if FF_API_ASS_SSA
|
#if FF_API_ASS_SSA
|
||||||
static int mkv_write_ass_blocks(AVFormatContext *s, AVIOContext *pb, AVPacket *pkt)
|
/* Writes the contents of pkt to a block, using the data starting at *datap.
|
||||||
|
* If pkt corresponds to more than one block, this writes the contents of the first block
|
||||||
|
* (starting from *datap) and updates *datap so it points to the beginning of the data
|
||||||
|
* corresponding to the next block.
|
||||||
|
*/
|
||||||
|
static int mkv_write_ass_block(AVFormatContext *s, AVIOContext *pb, AVPacket *pkt, uint8_t **datap)
|
||||||
{
|
{
|
||||||
MatroskaMuxContext *mkv = s->priv_data;
|
MatroskaMuxContext *mkv = s->priv_data;
|
||||||
int i, layer = 0, max_duration = 0, size, line_size, data_size = pkt->size;
|
int i, layer = 0, size, line_size, data_size = pkt->size - (*datap - pkt->data);
|
||||||
uint8_t *start, *end, *data = pkt->data;
|
uint8_t *start, *end, *data = *datap;
|
||||||
ebml_master blockgroup;
|
ebml_master blockgroup;
|
||||||
char buffer[2048];
|
char buffer[2048];
|
||||||
|
|
||||||
while (data_size) {
|
|
||||||
int duration = ass_get_duration(data);
|
int duration = ass_get_duration(data);
|
||||||
max_duration = FFMAX(duration, max_duration);
|
|
||||||
end = memchr(data, '\n', data_size);
|
end = memchr(data, '\n', data_size);
|
||||||
size = line_size = end ? end-data+1 : data_size;
|
size = line_size = end ? end-data+1 : data_size;
|
||||||
size -= end ? (end[-1]=='\r')+1 : 0;
|
size -= end ? (end[-1]=='\r')+1 : 0;
|
||||||
start = data;
|
start = data;
|
||||||
for (i=0; i<3; i++, start++)
|
for (i=0; i<3; i++, start++)
|
||||||
if (!(start = memchr(start, ',', size-(start-data))))
|
if (!(start = memchr(start, ',', size-(start-data))))
|
||||||
return max_duration;
|
return duration;
|
||||||
size -= start - data;
|
size -= start - data;
|
||||||
sscanf(data, "Dialogue: %d,", &layer);
|
sscanf(data, "Dialogue: %d,", &layer);
|
||||||
i = snprintf(buffer, sizeof(buffer), "%"PRId64",%d,",
|
i = snprintf(buffer, sizeof(buffer), "%"PRId64",%d,",
|
||||||
@ -1363,11 +1367,9 @@ static int mkv_write_ass_blocks(AVFormatContext *s, AVIOContext *pb, AVPacket *p
|
|||||||
put_ebml_uint(pb, MATROSKA_ID_BLOCKDURATION, duration);
|
put_ebml_uint(pb, MATROSKA_ID_BLOCKDURATION, duration);
|
||||||
end_ebml_master(pb, blockgroup);
|
end_ebml_master(pb, blockgroup);
|
||||||
|
|
||||||
data += line_size;
|
*datap += line_size;
|
||||||
data_size -= line_size;
|
|
||||||
}
|
|
||||||
|
|
||||||
return max_duration;
|
return duration;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@ -1613,6 +1615,8 @@ static int mkv_write_packet_internal(AVFormatContext *s, AVPacket *pkt, int add_
|
|||||||
int ret;
|
int ret;
|
||||||
int64_t ts = mkv->tracks[pkt->stream_index].write_dts ? pkt->dts : pkt->pts;
|
int64_t ts = mkv->tracks[pkt->stream_index].write_dts ? pkt->dts : pkt->pts;
|
||||||
int64_t relative_packet_pos;
|
int64_t relative_packet_pos;
|
||||||
|
uint8_t *data_offset = pkt->data;
|
||||||
|
int dash_tracknum = mkv->is_dash ? mkv->dash_track_number : pkt->stream_index + 1;
|
||||||
|
|
||||||
if (ts == AV_NOPTS_VALUE) {
|
if (ts == AV_NOPTS_VALUE) {
|
||||||
av_log(s, AV_LOG_ERROR, "Can't write packet with unknown timestamp\n");
|
av_log(s, AV_LOG_ERROR, "Can't write packet with unknown timestamp\n");
|
||||||
@ -1641,11 +1645,21 @@ static int mkv_write_packet_internal(AVFormatContext *s, AVPacket *pkt, int add_
|
|||||||
|
|
||||||
if (codec->codec_type != AVMEDIA_TYPE_SUBTITLE) {
|
if (codec->codec_type != AVMEDIA_TYPE_SUBTITLE) {
|
||||||
mkv_write_block(s, pb, MATROSKA_ID_SIMPLEBLOCK, pkt, keyframe << 7);
|
mkv_write_block(s, pb, MATROSKA_ID_SIMPLEBLOCK, pkt, keyframe << 7);
|
||||||
|
if (codec->codec_type == AVMEDIA_TYPE_VIDEO && keyframe) {
|
||||||
|
ret = mkv_add_cuepoint(mkv->cues, pkt->stream_index, dash_tracknum, ts, mkv->cluster_pos, relative_packet_pos, -1);
|
||||||
|
if (ret < 0) return ret;
|
||||||
|
}
|
||||||
#if FF_API_ASS_SSA
|
#if FF_API_ASS_SSA
|
||||||
} else if (codec->codec_id == AV_CODEC_ID_SSA) {
|
} else if (codec->codec_id == AV_CODEC_ID_SSA) {
|
||||||
duration = mkv_write_ass_blocks(s, pb, pkt);
|
while (data_offset < pkt->data + pkt->size) {
|
||||||
|
duration = mkv_write_ass_block(s, pb, pkt, &data_offset);
|
||||||
|
ret = mkv_add_cuepoint(mkv->cues, pkt->stream_index, dash_tracknum, ts, mkv->cluster_pos, relative_packet_pos, duration);
|
||||||
|
if (ret < 0) return ret;
|
||||||
|
relative_packet_pos = avio_tell(s->pb) - mkv->cluster.pos;
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
} else if (codec->codec_id == AV_CODEC_ID_SRT) {
|
} else {
|
||||||
|
if (codec->codec_id == AV_CODEC_ID_SRT) {
|
||||||
duration = mkv_write_srt_blocks(s, pb, pkt);
|
duration = mkv_write_srt_blocks(s, pb, pkt);
|
||||||
} else if (codec->codec_id == AV_CODEC_ID_WEBVTT) {
|
} else if (codec->codec_id == AV_CODEC_ID_WEBVTT) {
|
||||||
duration = mkv_write_vtt_blocks(s, pb, pkt);
|
duration = mkv_write_vtt_blocks(s, pb, pkt);
|
||||||
@ -1660,15 +1674,8 @@ static int mkv_write_packet_internal(AVFormatContext *s, AVPacket *pkt, int add_
|
|||||||
end_ebml_master(pb, blockgroup);
|
end_ebml_master(pb, blockgroup);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((codec->codec_type == AVMEDIA_TYPE_VIDEO && keyframe) ||
|
ret = mkv_add_cuepoint(mkv->cues, pkt->stream_index, dash_tracknum, ts, mkv->cluster_pos, relative_packet_pos, duration);
|
||||||
codec->codec_type == AVMEDIA_TYPE_SUBTITLE ||
|
if (ret < 0) return ret;
|
||||||
add_cue) {
|
|
||||||
ret = mkv_add_cuepoint(mkv->cues,
|
|
||||||
pkt->stream_index,
|
|
||||||
mkv->is_dash ? mkv->dash_track_number : pkt->stream_index + 1,
|
|
||||||
ts, mkv->cluster_pos, relative_packet_pos,
|
|
||||||
codec->codec_type == AVMEDIA_TYPE_SUBTITLE ? duration : -1);
|
|
||||||
if (ret < 0) return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
mkv->duration = FFMAX(mkv->duration, ts + duration);
|
mkv->duration = FFMAX(mkv->duration, ts + duration);
|
||||||
@ -1821,7 +1828,7 @@ static int mkv_write_trailer(AVFormatContext *s)
|
|||||||
currentpos = avio_tell(pb);
|
currentpos = avio_tell(pb);
|
||||||
avio_seek(pb, mkv->cues_pos, SEEK_SET);
|
avio_seek(pb, mkv->cues_pos, SEEK_SET);
|
||||||
|
|
||||||
cuespos = mkv_write_cues(pb, mkv->cues, mkv->tracks, s->nb_streams);
|
cuespos = mkv_write_cues(s, mkv->cues, mkv->tracks, s->nb_streams);
|
||||||
cues_end = avio_tell(pb);
|
cues_end = avio_tell(pb);
|
||||||
if (cues_end > cuespos + mkv->reserve_cues_space) {
|
if (cues_end > cuespos + mkv->reserve_cues_space) {
|
||||||
av_log(s, AV_LOG_ERROR, "Insufficient space reserved for cues: %d "
|
av_log(s, AV_LOG_ERROR, "Insufficient space reserved for cues: %d "
|
||||||
@ -1835,7 +1842,7 @@ static int mkv_write_trailer(AVFormatContext *s)
|
|||||||
|
|
||||||
avio_seek(pb, currentpos, SEEK_SET);
|
avio_seek(pb, currentpos, SEEK_SET);
|
||||||
} else {
|
} else {
|
||||||
cuespos = mkv_write_cues(pb, mkv->cues, mkv->tracks, s->nb_streams);
|
cuespos = mkv_write_cues(s, mkv->cues, mkv->tracks, s->nb_streams);
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = mkv_add_seekhead_entry(mkv->main_seekhead, MATROSKA_ID_CUES, cuespos);
|
ret = mkv_add_seekhead_entry(mkv->main_seekhead, MATROSKA_ID_CUES, cuespos);
|
||||||
|
Loading…
Reference in New Issue
Block a user