Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
V
vlc-2-2
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Redmine
Redmine
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Operations
Operations
Metrics
Environments
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
videolan
vlc-2-2
Commits
291c6479
Commit
291c6479
authored
Mar 07, 2005
by
Steve Lhomme
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
mkv.cpp: prepare the code to handle seemless segment switching
parent
cb48de28
Changes
1
Show whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
335 additions
and
244 deletions
+335
-244
modules/demux/mkv.cpp
modules/demux/mkv.cpp
+335
-244
No files found.
modules/demux/mkv.cpp
View file @
291c6479
...
...
@@ -362,13 +362,11 @@ public:
bool
b_ordered
;
};
class
demux_sys
_t
class
matroska_segment
_t
{
public:
demux_sys_t
()
:
in
(
NULL
)
,
es
(
NULL
)
,
ep
(
NULL
)
matroska_segment_t
()
:
segment
(
NULL
)
,
i_timescale
(
0
)
,
f_duration
(
0.0
)
,
i_track
(
0
)
...
...
@@ -376,11 +374,7 @@ public:
,
i_cues_position
(
0
)
,
i_chapters_position
(
0
)
,
i_tags_position
(
0
)
,
segment
(
NULL
)
,
cluster
(
NULL
)
,
i_pts
(
0
)
,
i_start_pts
(
0
)
,
i_chapter_time
(
0
)
,
b_cues
(
false
)
,
i_index
(
0
)
,
i_index_max
(
0
)
...
...
@@ -390,15 +384,11 @@ public:
,
psz_segment_filename
(
NULL
)
,
psz_title
(
NULL
)
,
psz_date_utc
(
NULL
)
,
meta
(
NULL
)
,
title
(
NULL
)
,
i_current_edition
(
0
)
,
psz_current_chapter
(
NULL
)
{}
vlc_stream_io_callback
*
in
;
EbmlStream
*
es
;
EbmlParser
*
ep
;
KaxSegment
*
segment
;
/* time scale */
uint64_t
i_timescale
;
...
...
@@ -415,15 +405,9 @@ public:
int64_t
i_chapters_position
;
int64_t
i_tags_position
;
/* current data */
KaxSegment
*
segment
;
KaxCluster
*
cluster
;
KaxSegmentUID
segment_uid
;
mtime_t
i_pts
;
mtime_t
i_start_pts
;
mtime_t
i_chapter_time
;
vlc_bool_t
b_cues
;
int
i_index
;
int
i_index_max
;
...
...
@@ -436,16 +420,93 @@ public:
char
*
psz_title
;
char
*
psz_date_utc
;
std
::
vector
<
chapter_edition_t
>
editions
;
int
i_current_edition
;
const
chapter_item_t
*
psz_current_chapter
;
std
::
vector
<
KaxSegmentFamily
>
families
;
chapter_edition_t
*
Edition
()
{
if
(
i_current_edition
!=
-
1
&&
i_current_edition
<
editions
.
size
()
)
return
&
editions
[
i_current_edition
];
return
NULL
;
}
};
class
matroska_stream_t
{
public:
matroska_stream_t
()
:
in
(
NULL
)
,
es
(
NULL
)
,
ep
(
NULL
)
,
i_current_segment
(
-
1
)
{}
~
matroska_stream_t
()
{
for
(
int
i
=
0
;
i
<
segments
.
size
();
i
++
)
delete
segments
[
i
];
if
(
in
)
delete
in
;
if
(
es
)
delete
es
;
if
(
ep
)
delete
ep
;
}
vlc_stream_io_callback
*
in
;
EbmlStream
*
es
;
EbmlParser
*
ep
;
std
::
vector
<
matroska_segment_t
*>
segments
;
int
i_current_segment
;
matroska_segment_t
*
Segment
()
{
if
(
i_current_segment
!=
-
1
&&
i_current_segment
<
segments
.
size
()
)
return
segments
[
i_current_segment
];
return
NULL
;
}
};
class
demux_sys_t
{
public:
demux_sys_t
()
:
i_pts
(
0
)
,
i_start_pts
(
0
)
,
i_chapter_time
(
0
)
,
meta
(
NULL
)
,
title
(
NULL
)
,
i_current_stream
(
-
1
)
{}
~
demux_sys_t
()
{
for
(
int
i
=
0
;
i
<
streams
.
size
();
i
++
)
delete
streams
[
i
];
}
/* current data */
mtime_t
i_pts
;
mtime_t
i_start_pts
;
mtime_t
i_chapter_time
;
vlc_meta_t
*
meta
;
input_title_t
*
title
;
std
::
vector
<
KaxSegmentFamily
>
familie
s
;
std
::
vector
<
KaxSegment
*>
family_members
;
std
::
vector
<
matroska_stream_t
*>
stream
s
;
int
i_current_stream
;
std
::
vector
<
chapter_edition_t
>
editions
;
int
i_current_edition
;
const
chapter_item_t
*
psz_current_chapter
;
matroska_stream_t
*
Stream
()
{
if
(
i_current_stream
!=
-
1
&&
i_current_stream
<
streams
.
size
()
)
return
streams
[
i_current_stream
];
return
NULL
;
}
};
static
int
Demux
(
demux_t
*
);
...
...
@@ -473,6 +534,8 @@ static int Open( vlc_object_t * p_this )
{
demux_t
*
p_demux
=
(
demux_t
*
)
p_this
;
demux_sys_t
*
p_sys
;
matroska_stream_t
*
p_stream
;
matroska_segment_t
*
p_segment
;
uint8_t
*
p_peek
;
std
::
string
s_path
,
s_filename
;
int
i_upper_lvl
;
...
...
@@ -494,40 +557,48 @@ static int Open( vlc_object_t * p_this )
p_demux
->
pf_control
=
Control
;
p_demux
->
p_sys
=
p_sys
=
new
demux_sys_t
;
p_sys
->
in
=
new
vlc_stream_io_callback
(
p_demux
->
s
);
p_sys
->
es
=
new
EbmlStream
(
*
p_sys
->
in
);
p_sys
->
f_duration
=
-
1
;
p_sys
->
i_timescale
=
MKVD_TIMECODESCALE
;
p_sys
->
i_track
=
0
;
p_sys
->
track
=
(
mkv_track_t
*
)
malloc
(
sizeof
(
mkv_track_t
)
);
p_stream
=
new
matroska_stream_t
;
p_segment
=
new
matroska_segment_t
;
p_sys
->
streams
.
push_back
(
p_stream
);
p_sys
->
i_current_stream
=
0
;
p_stream
->
segments
.
push_back
(
p_segment
);
p_stream
->
i_current_segment
=
0
;
p_stream
->
in
=
new
vlc_stream_io_callback
(
p_demux
->
s
);
p_stream
->
es
=
new
EbmlStream
(
*
p_stream
->
in
);
p_segment
->
f_duration
=
-
1
;
p_segment
->
i_timescale
=
MKVD_TIMECODESCALE
;
p_segment
->
i_track
=
0
;
p_segment
->
track
=
(
mkv_track_t
*
)
malloc
(
sizeof
(
mkv_track_t
)
);
p_sys
->
i_pts
=
0
;
p_s
ys
->
i_cues_position
=
-
1
;
p_s
ys
->
i_chapters_position
=
-
1
;
p_s
ys
->
i_tags_position
=
-
1
;
p_s
ys
->
b_cues
=
VLC_FALSE
;
p_s
ys
->
i_index
=
0
;
p_s
ys
->
i_index_max
=
1024
;
p_s
ys
->
index
=
(
mkv_index_t
*
)
malloc
(
sizeof
(
mkv_index_t
)
*
p_s
ys
->
i_index_max
);
p_s
ys
->
psz_muxing_application
=
NULL
;
p_s
ys
->
psz_writing_application
=
NULL
;
p_s
ys
->
psz_segment_filename
=
NULL
;
p_s
ys
->
psz_title
=
NULL
;
p_s
ys
->
psz_date_utc
=
NULL
;;
p_s
egment
->
i_cues_position
=
-
1
;
p_s
egment
->
i_chapters_position
=
-
1
;
p_s
egment
->
i_tags_position
=
-
1
;
p_s
egment
->
b_cues
=
VLC_FALSE
;
p_s
egment
->
i_index
=
0
;
p_s
egment
->
i_index_max
=
1024
;
p_s
egment
->
index
=
(
mkv_index_t
*
)
malloc
(
sizeof
(
mkv_index_t
)
*
p_s
egment
->
i_index_max
);
p_s
egment
->
psz_muxing_application
=
NULL
;
p_s
egment
->
psz_writing_application
=
NULL
;
p_s
egment
->
psz_segment_filename
=
NULL
;
p_s
egment
->
psz_title
=
NULL
;
p_s
egment
->
psz_date_utc
=
NULL
;;
p_sys
->
meta
=
NULL
;
p_sys
->
title
=
NULL
;
if
(
p_s
ys
->
es
==
NULL
)
if
(
p_s
tream
->
es
==
NULL
)
{
msg_Err
(
p_demux
,
"failed to create EbmlStream"
);
delete
p_sys
->
in
;
delete
p_sys
;
return
VLC_EGENERIC
;
}
/* Find the EbmlHead element */
el
=
p_s
ys
->
es
->
FindNextID
(
EbmlHead
::
ClassInfos
,
0xFFFFFFFFL
);
el
=
p_s
tream
->
es
->
FindNextID
(
EbmlHead
::
ClassInfos
,
0xFFFFFFFFL
);
if
(
el
==
NULL
)
{
msg_Err
(
p_demux
,
"cannot find EbmlHead"
);
...
...
@@ -535,23 +606,23 @@ static int Open( vlc_object_t * p_this )
}
msg_Dbg
(
p_demux
,
"EbmlHead"
);
/* skip it */
el
->
SkipData
(
*
p_s
ys
->
es
,
el
->
Generic
().
Context
);
el
->
SkipData
(
*
p_s
tream
->
es
,
el
->
Generic
().
Context
);
delete
el
;
/* Find a segment */
el
=
p_s
ys
->
es
->
FindNextID
(
KaxSegment
::
ClassInfos
,
0xFFFFFFFFL
);
el
=
p_s
tream
->
es
->
FindNextID
(
KaxSegment
::
ClassInfos
,
0xFFFFFFFFL
);
if
(
el
==
NULL
)
{
msg_Err
(
p_demux
,
"cannot find KaxSegment"
);
goto
error
;
}
MkvTree
(
p_demux
,
0
,
"Segment"
);
p_s
ys
->
segment
=
(
KaxSegment
*
)
el
;
p_s
ys
->
cluster
=
NULL
;
p_s
egment
->
segment
=
(
KaxSegment
*
)
el
;
p_s
egment
->
cluster
=
NULL
;
p_s
ys
->
ep
=
new
EbmlParser
(
p_sys
->
es
,
el
);
p_s
tream
->
ep
=
new
EbmlParser
(
p_stream
->
es
,
el
);
while
(
(
el1
=
p_s
ys
->
ep
->
Get
()
)
!=
NULL
)
while
(
(
el1
=
p_s
tream
->
ep
->
Get
()
)
!=
NULL
)
{
if
(
MKV_IS_ID
(
el1
,
KaxInfo
)
)
{
...
...
@@ -573,9 +644,9 @@ static int Open( vlc_object_t * p_this )
{
msg_Dbg
(
p_demux
,
"| + Cluster"
);
p_s
ys
->
cluster
=
(
KaxCluster
*
)
el1
;
p_s
egment
->
cluster
=
(
KaxCluster
*
)
el1
;
p_s
ys
->
ep
->
Down
();
p_s
tream
->
ep
->
Down
();
/* stop parsing the stream */
break
;
}
...
...
@@ -692,15 +763,15 @@ static int Open( vlc_object_t * p_this )
if
(
MKV_IS_ID
(
l
,
KaxSegmentUID
)
)
{
p_uid
=
static_cast
<
KaxSegmentUID
*>
(
l
);
if
(
p_s
ys
->
segment_uid
==
*
p_uid
)
if
(
p_s
egment
->
segment_uid
==
*
p_uid
)
break
;
}
else
if
(
MKV_IS_ID
(
l
,
KaxSegmentFamily
)
)
{
KaxSegmentFamily
*
p_fam
=
static_cast
<
KaxSegmentFamily
*>
(
l
);
std
::
vector
<
KaxSegmentFamily
>::
iterator
iter
;
for
(
iter
=
p_s
ys
->
families
.
begin
();
iter
!=
p_s
ys
->
families
.
end
();
for
(
iter
=
p_s
egment
->
families
.
begin
();
iter
!=
p_s
egment
->
families
.
end
();
iter
++
)
{
if
(
*
iter
==
*
p_fam
)
...
...
@@ -739,14 +810,14 @@ static int Open( vlc_object_t * p_this )
}
if
(
p_s
ys
->
cluster
==
NULL
)
if
(
p_s
egment
->
cluster
==
NULL
)
{
msg_Err
(
p_demux
,
"cannot find any cluster, damaged file ?"
);
goto
error
;
}
/* *** Load the cue if found *** */
if
(
p_s
ys
->
i_cues_position
>=
0
)
if
(
p_s
egment
->
i_cues_position
>=
0
)
{
vlc_bool_t
b_seekable
;
...
...
@@ -757,20 +828,20 @@ static int Open( vlc_object_t * p_this )
}
}
if
(
!
p_s
ys
->
b_cues
||
p_sys
->
i_index
<=
0
)
if
(
!
p_s
egment
->
b_cues
||
p_segment
->
i_index
<=
0
)
{
msg_Warn
(
p_demux
,
"no cues/empty cues found->seek won't be precise"
);
IndexAppendCluster
(
p_demux
,
p_s
ys
->
cluster
);
IndexAppendCluster
(
p_demux
,
p_s
egment
->
cluster
);
p_s
ys
->
b_cues
=
VLC_FALSE
;
p_s
egment
->
b_cues
=
VLC_FALSE
;
}
/* add all es */
msg_Dbg
(
p_demux
,
"found %d es"
,
p_s
ys
->
i_track
);
for
(
i_track
=
0
;
i_track
<
p_s
ys
->
i_track
;
i_track
++
)
msg_Dbg
(
p_demux
,
"found %d es"
,
p_s
egment
->
i_track
);
for
(
i_track
=
0
;
i_track
<
p_s
egment
->
i_track
;
i_track
++
)
{
#define tk p_s
ys
->track[i_track]
#define tk p_s
egment
->track[i_track]
if
(
tk
.
fmt
.
i_cat
==
UNKNOWN_ES
)
{
msg_Warn
(
p_demux
,
"invalid track[%d, n=%d]"
,
i_track
,
tk
.
i_number
);
...
...
@@ -1060,8 +1131,6 @@ static int Open( vlc_object_t * p_this )
return
VLC_SUCCESS
;
error:
delete
p_sys
->
es
;
delete
p_sys
->
in
;
delete
p_sys
;
return
VLC_EGENERIC
;
}
...
...
@@ -1073,11 +1142,13 @@ static void Close( vlc_object_t *p_this )
{
demux_t
*
p_demux
=
(
demux_t
*
)
p_this
;
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
matroska_stream_t
*
p_stream
=
p_sys
->
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
int
i_track
;
for
(
i_track
=
0
;
i_track
<
p_s
ys
->
i_track
;
i_track
++
)
for
(
i_track
=
0
;
i_track
<
p_s
egment
->
i_track
;
i_track
++
)
{
#define tk p_s
ys
->track[i_track]
#define tk p_s
egment
->track[i_track]
if
(
tk
.
fmt
.
psz_description
)
{
free
(
tk
.
fmt
.
psz_description
);
...
...
@@ -1092,22 +1163,19 @@ static void Close( vlc_object_t *p_this )
}
#undef tk
}
free
(
p_s
ys
->
track
);
free
(
p_s
egment
->
track
);
if
(
p_s
ys
->
psz_writing_application
)
if
(
p_s
egment
->
psz_writing_application
)
{
free
(
p_s
ys
->
psz_writing_application
);
free
(
p_s
egment
->
psz_writing_application
);
}
if
(
p_s
ys
->
psz_muxing_application
)
if
(
p_s
egment
->
psz_muxing_application
)
{
free
(
p_s
ys
->
psz_muxing_application
);
free
(
p_s
egment
->
psz_muxing_application
);
}
delete
p_s
ys
->
segment
;
delete
p_s
egment
->
segment
;
delete
p_sys
->
ep
;
delete
p_sys
->
es
;
delete
p_sys
->
in
;
delete
p_sys
;
}
...
...
@@ -1117,10 +1185,11 @@ static void Close( vlc_object_t *p_this )
static
int
Control
(
demux_t
*
p_demux
,
int
i_query
,
va_list
args
)
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
matroska_stream_t
*
p_stream
=
p_sys
->
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
int64_t
*
pi64
;
double
*
pf
,
f
;
int
i_skp
;
mtime_t
*
i_sk_time
;
vlc_meta_t
**
pp_meta
;
...
...
@@ -1133,19 +1202,16 @@ static int Control( demux_t *p_demux, int i_query, va_list args )
case
DEMUX_GET_LENGTH
:
pi64
=
(
int64_t
*
)
va_arg
(
args
,
int64_t
*
);
if
(
p_s
ys
->
f_duration
>
0.0
)
if
(
p_s
egment
->
f_duration
>
0.0
)
{
*
pi64
=
(
int64_t
)(
p_s
ys
->
f_duration
*
1000
);
*
pi64
=
(
int64_t
)(
p_s
egment
->
f_duration
*
1000
);
return
VLC_SUCCESS
;
}
return
VLC_EGENERIC
;
case
DEMUX_GET_POSITION
:
pf
=
(
double
*
)
va_arg
(
args
,
double
*
);
/* if (p_sys->i_pts < p_sys->i_start_pts)
*pf = (double)p_sys->i_start_pts / (1000.0 * p_sys->f_duration);
else*/
*
pf
=
(
double
)
p_sys
->
i_pts
/
(
1000.0
*
p_sys
->
f_duration
);
*
pf
=
(
double
)
p_sys
->
i_pts
/
(
1000.0
*
p_segment
->
f_duration
);
return
VLC_SUCCESS
;
case
DEMUX_SET_POSITION
:
...
...
@@ -1155,9 +1221,6 @@ static int Control( demux_t *p_demux, int i_query, va_list args )
case
DEMUX_GET_TIME
:
pi64
=
(
int64_t
*
)
va_arg
(
args
,
int64_t
*
);
/* if (p_sys->i_pts < p_sys->i_start_pts)
*pi64 = p_sys->i_start_pts;
else*/
*
pi64
=
p_sys
->
i_pts
;
return
VLC_SUCCESS
;
...
...
@@ -1207,6 +1270,8 @@ static int Control( demux_t *p_demux, int i_query, va_list args )
static
int
BlockGet
(
demux_t
*
p_demux
,
KaxBlock
**
pp_block
,
int64_t
*
pi_ref1
,
int64_t
*
pi_ref2
,
int64_t
*
pi_duration
)
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
matroska_stream_t
*
p_stream
=
p_sys
->
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
*
pp_block
=
NULL
;
*
pi_ref1
=
-
1
;
...
...
@@ -1222,14 +1287,14 @@ static int BlockGet( demux_t *p_demux, KaxBlock **pp_block, int64_t *pi_ref1, in
return
VLC_EGENERIC
;
}
el
=
p_s
ys
->
ep
->
Get
();
i_level
=
p_s
ys
->
ep
->
GetLevel
();
el
=
p_s
tream
->
ep
->
Get
();
i_level
=
p_s
tream
->
ep
->
GetLevel
();
if
(
el
==
NULL
&&
*
pp_block
!=
NULL
)
{
/* update the index */
#define idx p_s
ys->index[p_sys
->i_index - 1]
if
(
p_s
ys
->
i_index
>
0
&&
idx
.
i_time
==
-
1
)
#define idx p_s
egment->index[p_segment
->i_index - 1]
if
(
p_s
egment
->
i_index
>
0
&&
idx
.
i_time
==
-
1
)
{
idx
.
i_time
=
(
*
pp_block
)
->
GlobalTimecode
()
/
(
mtime_t
)
1000
;
idx
.
b_key
=
*
pi_ref1
==
-
1
?
VLC_TRUE
:
VLC_FALSE
;
...
...
@@ -1240,9 +1305,9 @@ static int BlockGet( demux_t *p_demux, KaxBlock **pp_block, int64_t *pi_ref1, in
if
(
el
==
NULL
)
{
if
(
p_s
ys
->
ep
->
GetLevel
()
>
1
)
if
(
p_s
tream
->
ep
->
GetLevel
()
>
1
)
{
p_s
ys
->
ep
->
Up
();
p_s
tream
->
ep
->
Up
();
continue
;
}
msg_Warn
(
p_demux
,
"EOF"
);
...
...
@@ -1254,16 +1319,16 @@ static int BlockGet( demux_t *p_demux, KaxBlock **pp_block, int64_t *pi_ref1, in
{
if
(
MKV_IS_ID
(
el
,
KaxCluster
)
)
{
p_s
ys
->
cluster
=
(
KaxCluster
*
)
el
;
p_s
egment
->
cluster
=
(
KaxCluster
*
)
el
;
/* add it to the index */
if
(
p_s
ys
->
i_index
==
0
||
(
p_s
ys
->
i_index
>
0
&&
p_sys
->
index
[
p_sys
->
i_index
-
1
].
i_position
<
(
int64_t
)
p_sys
->
cluster
->
GetElementPosition
()
)
)
if
(
p_s
egment
->
i_index
==
0
||
(
p_s
egment
->
i_index
>
0
&&
p_segment
->
index
[
p_segment
->
i_index
-
1
].
i_position
<
(
int64_t
)
p_segment
->
cluster
->
GetElementPosition
()
)
)
{
IndexAppendCluster
(
p_demux
,
p_s
ys
->
cluster
);
IndexAppendCluster
(
p_demux
,
p_s
egment
->
cluster
);
}
p_s
ys
->
ep
->
Down
();
p_s
tream
->
ep
->
Down
();
}
else
if
(
MKV_IS_ID
(
el
,
KaxCues
)
)
{
...
...
@@ -1281,12 +1346,12 @@ static int BlockGet( demux_t *p_demux, KaxBlock **pp_block, int64_t *pi_ref1, in
{
KaxClusterTimecode
&
ctc
=
*
(
KaxClusterTimecode
*
)
el
;
ctc
.
ReadData
(
p_s
ys
->
es
->
I_O
(),
SCOPE_ALL_DATA
);
p_s
ys
->
cluster
->
InitTimecode
(
uint64
(
ctc
),
p_sys
->
i_timescale
);
ctc
.
ReadData
(
p_s
tream
->
es
->
I_O
(),
SCOPE_ALL_DATA
);
p_s
egment
->
cluster
->
InitTimecode
(
uint64
(
ctc
),
p_segment
->
i_timescale
);
}
else
if
(
MKV_IS_ID
(
el
,
KaxBlockGroup
)
)
{
p_s
ys
->
ep
->
Down
();
p_s
tream
->
ep
->
Down
();
}
}
else
if
(
i_level
==
3
)
...
...
@@ -1295,23 +1360,23 @@ static int BlockGet( demux_t *p_demux, KaxBlock **pp_block, int64_t *pi_ref1, in
{
*
pp_block
=
(
KaxBlock
*
)
el
;
(
*
pp_block
)
->
ReadData
(
p_s
ys
->
es
->
I_O
()
);
(
*
pp_block
)
->
SetParent
(
*
p_s
ys
->
cluster
);
(
*
pp_block
)
->
ReadData
(
p_s
tream
->
es
->
I_O
()
);
(
*
pp_block
)
->
SetParent
(
*
p_s
egment
->
cluster
);
p_s
ys
->
ep
->
Keep
();
p_s
tream
->
ep
->
Keep
();
}
else
if
(
MKV_IS_ID
(
el
,
KaxBlockDuration
)
)
{
KaxBlockDuration
&
dur
=
*
(
KaxBlockDuration
*
)
el
;
dur
.
ReadData
(
p_s
ys
->
es
->
I_O
()
);
dur
.
ReadData
(
p_s
tream
->
es
->
I_O
()
);
*
pi_duration
=
uint64
(
dur
);
}
else
if
(
MKV_IS_ID
(
el
,
KaxReferenceBlock
)
)
{
KaxReferenceBlock
&
ref
=
*
(
KaxReferenceBlock
*
)
el
;
ref
.
ReadData
(
p_s
ys
->
es
->
I_O
()
);
ref
.
ReadData
(
p_s
tream
->
es
->
I_O
()
);
if
(
*
pi_ref1
==
-
1
)
{
*
pi_ref1
=
int64
(
ref
);
...
...
@@ -1343,13 +1408,15 @@ static void BlockDecode( demux_t *p_demux, KaxBlock *block, mtime_t i_pts,
mtime_t
i_duration
)
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
matroska_stream_t
*
p_stream
=
p_sys
->
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
int
i_track
;
unsigned
int
i
;
vlc_bool_t
b
;
#define tk p_s
ys
->track[i_track]
for
(
i_track
=
0
;
i_track
<
p_s
ys
->
i_track
;
i_track
++
)
#define tk p_s
egment
->track[i_track]
for
(
i_track
=
0
;
i_track
<
p_s
egment
->
i_track
;
i_track
++
)
{
if
(
tk
.
i_number
==
block
->
TrackNum
()
)
{
...
...
@@ -1357,7 +1424,7 @@ static void BlockDecode( demux_t *p_demux, KaxBlock *block, mtime_t i_pts,
}
}
if
(
i_track
>=
p_s
ys
->
i_track
)
if
(
i_track
>=
p_s
egment
->
i_track
)
{
msg_Err
(
p_demux
,
"invalid track number=%d"
,
block
->
TrackNum
()
);
return
;
...
...
@@ -1437,42 +1504,46 @@ static void BlockDecode( demux_t *p_demux, KaxBlock *block, mtime_t i_pts,
static
void
UpdateCurrentToChapter
(
demux_t
&
demux
)
{
demux_sys_t
&
sys
=
*
demux
.
p_sys
;
matroska_stream_t
*
p_stream
=
sys
.
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
const
chapter_item_t
*
psz_curr_chapter
;
/* update current chapter/seekpoint */
if
(
sys
.
editions
.
size
())
if
(
p_segment
->
editions
.
size
())
{
/* 1st, we need to know in which chapter we are */
psz_curr_chapter
=
sys
.
editions
[
sys
.
i_current_edition
].
FindTimecode
(
sys
.
i_pts
);
psz_curr_chapter
=
p_segment
->
editions
[
p_segment
->
i_current_edition
].
FindTimecode
(
sys
.
i_pts
);
/* we have moved to a new chapter */
if
(
sys
.
psz_current_chapter
!=
NULL
&&
psz_curr_chapter
!=
NULL
&&
sys
.
psz_current_chapter
!=
psz_curr_chapter
)
if
(
p_segment
->
psz_current_chapter
!=
NULL
&&
psz_curr_chapter
!=
NULL
&&
p_segment
->
psz_current_chapter
!=
psz_curr_chapter
)
{
if
(
sys
.
psz_current_chapter
->
i_seekpoint_num
!=
psz_curr_chapter
->
i_seekpoint_num
&&
psz_curr_chapter
->
i_seekpoint_num
>
0
)
if
(
p_segment
->
psz_current_chapter
->
i_seekpoint_num
!=
psz_curr_chapter
->
i_seekpoint_num
&&
psz_curr_chapter
->
i_seekpoint_num
>
0
)
{
demux
.
info
.
i_update
|=
INPUT_UPDATE_SEEKPOINT
;
demux
.
info
.
i_seekpoint
=
psz_curr_chapter
->
i_seekpoint_num
-
1
;
}
if
(
sys
.
editions
[
sys
.
i_current_edition
].
b_ordered
)
if
(
p_segment
->
editions
[
p_segment
->
i_current_edition
].
b_ordered
)
{
/* TODO check if we need to silently seek to a new location in the stream (switch to another chapter) */
if
(
sys
.
psz_current_chapter
->
i_end_time
!=
psz_curr_chapter
->
i_start_time
)
if
(
p_segment
->
psz_current_chapter
->
i_end_time
!=
psz_curr_chapter
->
i_start_time
)
Seek
(
&
demux
,
sys
.
i_pts
,
-
1
,
psz_curr_chapter
);
/* count the last duration time found for each track in a table (-1 not found, -2 silent) */
/* only seek after each duration >= end timecode of the current chapter */
}
//
sys.
i_user_time = psz_curr_chapter->i_user_start_time - psz_curr_chapter->i_start_time;
//
sys.
i_start_pts = psz_curr_chapter->i_user_start_time;
//
p_segment->
i_user_time = psz_curr_chapter->i_user_start_time - psz_curr_chapter->i_start_time;
//
p_segment->
i_start_pts = psz_curr_chapter->i_user_start_time;
}
sys
.
psz_current_chapter
=
psz_curr_chapter
;
p_segment
->
psz_current_chapter
=
psz_curr_chapter
;
}
}
static
void
Seek
(
demux_t
*
p_demux
,
mtime_t
i_date
,
double
f_percent
,
const
chapter_item_t
*
psz_chapter
)
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
matroska_stream_t
*
p_stream
=
p_sys
->
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
mtime_t
i_time_offset
=
0
;
KaxBlock
*
block
;
...
...
@@ -1496,38 +1567,38 @@ static void Seek( demux_t *p_demux, mtime_t i_date, double f_percent, const chap
return
;
}
delete
p_s
ys
->
ep
;
p_s
ys
->
ep
=
new
EbmlParser
(
p_sys
->
es
,
p_sys
->
segment
);
p_s
ys
->
cluster
=
NULL
;
delete
p_s
tream
->
ep
;
p_s
tream
->
ep
=
new
EbmlParser
(
p_stream
->
es
,
p_segment
->
segment
);
p_s
egment
->
cluster
=
NULL
;
/* seek without index or without date */
if
(
f_percent
>=
0
&&
(
config_GetInt
(
p_demux
,
"mkv-seek-percent"
)
||
!
p_s
ys
->
b_cues
||
i_date
<
0
))
if
(
f_percent
>=
0
&&
(
config_GetInt
(
p_demux
,
"mkv-seek-percent"
)
||
!
p_s
egment
->
b_cues
||
i_date
<
0
))
{
if
(
p_s
ys
->
f_duration
>=
0
)
if
(
p_s
egment
->
f_duration
>=
0
)
{
i_date
=
int64_t
(
f_percent
*
p_s
ys
->
f_duration
*
1000.0
);
i_date
=
int64_t
(
f_percent
*
p_s
egment
->
f_duration
*
1000.0
);
}
else
{
int64_t
i_pos
=
int64_t
(
f_percent
*
stream_Size
(
p_demux
->
s
)
);
msg_Dbg
(
p_demux
,
"inacurate way of seeking"
);
for
(
i_index
=
0
;
i_index
<
p_s
ys
->
i_index
;
i_index
++
)
for
(
i_index
=
0
;
i_index
<
p_s
egment
->
i_index
;
i_index
++
)
{
if
(
p_s
ys
->
index
[
i_index
].
i_position
>=
i_pos
)
if
(
p_s
egment
->
index
[
i_index
].
i_position
>=
i_pos
)
{
break
;
}
}
if
(
i_index
==
p_s
ys
->
i_index
)
if
(
i_index
==
p_s
egment
->
i_index
)
{
i_index
--
;
}
i_date
=
p_s
ys
->
index
[
i_index
].
i_time
;
i_date
=
p_s
egment
->
index
[
i_index
].
i_time
;
#if 0
if( p_s
ys
->index[i_index].i_position < i_pos )
if( p_s
egment
->index[i_index].i_position < i_pos )
{
EbmlElement *el;
...
...
@@ -1559,24 +1630,24 @@ static void Seek( demux_t *p_demux, mtime_t i_date, double f_percent, const chap
// find the actual time for an ordered edition
if
(
psz_chapter
==
NULL
)
{
if
(
p_s
ys
->
editions
.
size
()
&&
p_sys
->
editions
[
p_sys
->
i_current_edition
].
b_ordered
)
if
(
p_s
egment
->
editions
.
size
()
&&
p_segment
->
editions
[
p_segment
->
i_current_edition
].
b_ordered
)
{
/* 1st, we need to know in which chapter we are */
psz_chapter
=
p_s
ys
->
editions
[
p_sys
->
i_current_edition
].
FindTimecode
(
i_date
);
psz_chapter
=
p_s
egment
->
editions
[
p_segment
->
i_current_edition
].
FindTimecode
(
i_date
);
}
}
if
(
psz_chapter
!=
NULL
)
{
p_s
ys
->
psz_current_chapter
=
psz_chapter
;
p_s
egment
->
psz_current_chapter
=
psz_chapter
;
p_sys
->
i_chapter_time
=
i_time_offset
=
psz_chapter
->
i_user_start_time
-
psz_chapter
->
i_start_time
;
p_demux
->
info
.
i_update
|=
INPUT_UPDATE_SEEKPOINT
;
p_demux
->
info
.
i_seekpoint
=
psz_chapter
->
i_seekpoint_num
-
1
;
}
for
(
;
i_index
<
p_s
ys
->
i_index
;
i_index
++
)
for
(
;
i_index
<
p_s
egment
->
i_index
;
i_index
++
)
{
if
(
p_s
ys
->
index
[
i_index
].
i_time
+
i_time_offset
>
i_date
)
if
(
p_s
egment
->
index
[
i_index
].
i_time
+
i_time_offset
>
i_date
)
{
break
;
}
...
...
@@ -1588,11 +1659,11 @@ static void Seek( demux_t *p_demux, mtime_t i_date, double f_percent, const chap
}
msg_Dbg
(
p_demux
,
"seek got "
I64Fd
" (%d%%)"
,
p_s
ys
->
index
[
i_index
].
i_time
,
(
int
)(
100
*
p_s
ys
->
index
[
i_index
].
i_position
/
p_s
egment
->
index
[
i_index
].
i_time
,
(
int
)(
100
*
p_s
egment
->
index
[
i_index
].
i_position
/
stream_Size
(
p_demux
->
s
)
)
);
p_s
ys
->
in
->
setFilePointer
(
p_sys
->
index
[
i_index
].
i_position
,
p_s
tream
->
in
->
setFilePointer
(
p_segment
->
index
[
i_index
].
i_position
,
seek_beginning
);
p_sys
->
i_start_pts
=
i_date
;
...
...
@@ -1600,9 +1671,9 @@ static void Seek( demux_t *p_demux, mtime_t i_date, double f_percent, const chap
es_out_Control
(
p_demux
->
out
,
ES_OUT_RESET_PCR
);
/* now parse until key frame */
#define tk p_s
ys
->track[i_track]
#define tk p_s
egment
->track[i_track]
i_track_skipping
=
0
;
for
(
i_track
=
0
;
i_track
<
p_s
ys
->
i_track
;
i_track
++
)
for
(
i_track
=
0
;
i_track
<
p_s
egment
->
i_track
;
i_track
++
)
{
if
(
tk
.
fmt
.
i_cat
==
VIDEO_ES
)
{
...
...
@@ -1622,7 +1693,7 @@ static void Seek( demux_t *p_demux, mtime_t i_date, double f_percent, const chap
return
;
}
for
(
i_track
=
0
;
i_track
<
p_s
ys
->
i_track
;
i_track
++
)
for
(
i_track
=
0
;
i_track
<
p_s
egment
->
i_track
;
i_track
++
)
{
if
(
tk
.
i_number
==
block
->
TrackNum
()
)
{
...
...
@@ -1632,7 +1703,7 @@ static void Seek( demux_t *p_demux, mtime_t i_date, double f_percent, const chap
p_sys
->
i_pts
=
p_sys
->
i_chapter_time
+
block
->
GlobalTimecode
()
/
(
mtime_t
)
1000
;
if
(
i_track
<
p_s
ys
->
i_track
)
if
(
i_track
<
p_s
egment
->
i_track
)
{
if
(
tk
.
fmt
.
i_cat
==
VIDEO_ES
)
{
...
...
@@ -1661,6 +1732,8 @@ static void Seek( demux_t *p_demux, mtime_t i_date, double f_percent, const chap
static
int
Demux
(
demux_t
*
p_demux
)
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
matroska_stream_t
*
p_stream
=
p_sys
->
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
int
i_block_count
=
0
;
KaxBlock
*
block
;
...
...
@@ -1673,7 +1746,7 @@ static int Demux( demux_t *p_demux)
if
(
p_sys
->
i_pts
>=
p_sys
->
i_start_pts
)
UpdateCurrentToChapter
(
*
p_demux
);
if
(
p_s
ys
->
editions
.
size
()
&&
p_sys
->
editions
[
p_sys
->
i_current_edition
].
b_ordered
&&
p_sys
->
psz_current_chapter
==
NULL
)
if
(
p_s
egment
->
editions
.
size
()
&&
p_segment
->
editions
[
p_segment
->
i_current_edition
].
b_ordered
&&
p_segment
->
psz_current_chapter
==
NULL
)
{
/* nothing left to read in this ordered edition */
return
0
;
...
...
@@ -1681,12 +1754,12 @@ static int Demux( demux_t *p_demux)
if
(
BlockGet
(
p_demux
,
&
block
,
&
i_block_ref1
,
&
i_block_ref2
,
&
i_block_duration
)
)
{
if
(
p_s
ys
->
editions
.
size
()
&&
p_sys
->
editions
[
p_sys
->
i_current_edition
].
b_ordered
)
if
(
p_s
egment
->
editions
.
size
()
&&
p_segment
->
editions
[
p_segment
->
i_current_edition
].
b_ordered
)
{
// check if there are more chapters to read
if
(
p_s
ys
->
psz_current_chapter
!=
NULL
)
if
(
p_s
egment
->
psz_current_chapter
!=
NULL
)
{
p_sys
->
i_pts
=
p_s
ys
->
psz_current_chapter
->
i_user_end_time
;
p_sys
->
i_pts
=
p_s
egment
->
psz_current_chapter
->
i_user_end_time
;
return
1
;
}
...
...
@@ -1908,27 +1981,29 @@ EbmlElement *EbmlParser::Get( void )
static
void
LoadCues
(
demux_t
*
p_demux
)
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
int64_t
i_sav_position
=
p_sys
->
in
->
getFilePointer
();
matroska_stream_t
*
p_stream
=
p_sys
->
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
int64_t
i_sav_position
=
p_stream
->
in
->
getFilePointer
();
EbmlParser
*
ep
;
EbmlElement
*
el
,
*
cues
;
msg_Dbg
(
p_demux
,
"loading cues"
);
p_s
ys
->
in
->
setFilePointer
(
p_sys
->
i_cues_position
,
seek_beginning
);
cues
=
p_s
ys
->
es
->
FindNextID
(
KaxCues
::
ClassInfos
,
0xFFFFFFFFL
);
p_s
tream
->
in
->
setFilePointer
(
p_segment
->
i_cues_position
,
seek_beginning
);
cues
=
p_s
tream
->
es
->
FindNextID
(
KaxCues
::
ClassInfos
,
0xFFFFFFFFL
);
if
(
cues
==
NULL
)
{
msg_Err
(
p_demux
,
"cannot load cues (broken seekhead or file)"
);
p_s
ys
->
in
->
setFilePointer
(
i_sav_position
,
seek_beginning
);
p_s
tream
->
in
->
setFilePointer
(
i_sav_position
,
seek_beginning
);
return
;
}
ep
=
new
EbmlParser
(
p_s
ys
->
es
,
cues
);
ep
=
new
EbmlParser
(
p_s
tream
->
es
,
cues
);
while
(
(
el
=
ep
->
Get
()
)
!=
NULL
)
{
if
(
MKV_IS_ID
(
el
,
KaxCuePoint
)
)
{
#define idx p_s
ys->index[p_sys
->i_index]
#define idx p_s
egment->index[p_segment
->i_index]
idx
.
i_track
=
-
1
;
idx
.
i_block_number
=
-
1
;
...
...
@@ -1943,9 +2018,9 @@ static void LoadCues( demux_t *p_demux )
{
KaxCueTime
&
ctime
=
*
(
KaxCueTime
*
)
el
;
ctime
.
ReadData
(
p_s
ys
->
es
->
I_O
()
);
ctime
.
ReadData
(
p_s
tream
->
es
->
I_O
()
);
idx
.
i_time
=
uint64
(
ctime
)
*
p_s
ys
->
i_timescale
/
(
mtime_t
)
1000
;
idx
.
i_time
=
uint64
(
ctime
)
*
p_s
egment
->
i_timescale
/
(
mtime_t
)
1000
;
}
else
if
(
MKV_IS_ID
(
el
,
KaxCueTrackPositions
)
)
{
...
...
@@ -1956,21 +2031,21 @@ static void LoadCues( demux_t *p_demux )
{
KaxCueTrack
&
ctrack
=
*
(
KaxCueTrack
*
)
el
;
ctrack
.
ReadData
(
p_s
ys
->
es
->
I_O
()
);
ctrack
.
ReadData
(
p_s
tream
->
es
->
I_O
()
);
idx
.
i_track
=
uint16
(
ctrack
);
}
else
if
(
MKV_IS_ID
(
el
,
KaxCueClusterPosition
)
)
{
KaxCueClusterPosition
&
ccpos
=
*
(
KaxCueClusterPosition
*
)
el
;
ccpos
.
ReadData
(
p_s
ys
->
es
->
I_O
()
);
idx
.
i_position
=
p_s
ys
->
segment
->
GetGlobalPosition
(
uint64
(
ccpos
)
);
ccpos
.
ReadData
(
p_s
tream
->
es
->
I_O
()
);
idx
.
i_position
=
p_s
egment
->
segment
->
GetGlobalPosition
(
uint64
(
ccpos
)
);
}
else
if
(
MKV_IS_ID
(
el
,
KaxCueBlockNumber
)
)
{
KaxCueBlockNumber
&
cbnum
=
*
(
KaxCueBlockNumber
*
)
el
;
cbnum
.
ReadData
(
p_s
ys
->
es
->
I_O
()
);
cbnum
.
ReadData
(
p_s
tream
->
es
->
I_O
()
);
idx
.
i_block_number
=
uint32
(
cbnum
);
}
else
...
...
@@ -1993,11 +2068,11 @@ static void LoadCues( demux_t *p_demux )
idx.i_track, idx.i_block_number );
#endif
p_s
ys
->
i_index
++
;
if
(
p_s
ys
->
i_index
>=
p_sys
->
i_index_max
)
p_s
egment
->
i_index
++
;
if
(
p_s
egment
->
i_index
>=
p_segment
->
i_index_max
)
{
p_s
ys
->
i_index_max
+=
1024
;
p_s
ys
->
index
=
(
mkv_index_t
*
)
realloc
(
p_sys
->
index
,
sizeof
(
mkv_index_t
)
*
p_sys
->
i_index_max
);
p_s
egment
->
i_index_max
+=
1024
;
p_s
egment
->
index
=
(
mkv_index_t
*
)
realloc
(
p_segment
->
index
,
sizeof
(
mkv_index_t
)
*
p_segment
->
i_index_max
);
}
#undef idx
}
...
...
@@ -2009,32 +2084,34 @@ static void LoadCues( demux_t *p_demux )
delete
ep
;
delete
cues
;
p_s
ys
->
b_cues
=
VLC_TRUE
;
p_s
egment
->
b_cues
=
VLC_TRUE
;
msg_Dbg
(
p_demux
,
"loading cues done."
);
p_s
ys
->
in
->
setFilePointer
(
i_sav_position
,
seek_beginning
);
p_s
tream
->
in
->
setFilePointer
(
i_sav_position
,
seek_beginning
);
}
static
void
LoadTags
(
demux_t
*
p_demux
)
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
int64_t
i_sav_position
=
p_sys
->
in
->
getFilePointer
();
matroska_stream_t
*
p_stream
=
p_sys
->
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
int64_t
i_sav_position
=
p_stream
->
in
->
getFilePointer
();
EbmlParser
*
ep
;
EbmlElement
*
el
,
*
tags
;
msg_Dbg
(
p_demux
,
"loading tags"
);
p_s
ys
->
in
->
setFilePointer
(
p_sys
->
i_tags_position
,
seek_beginning
);
tags
=
p_s
ys
->
es
->
FindNextID
(
KaxTags
::
ClassInfos
,
0xFFFFFFFFL
);
p_s
tream
->
in
->
setFilePointer
(
p_segment
->
i_tags_position
,
seek_beginning
);
tags
=
p_s
tream
->
es
->
FindNextID
(
KaxTags
::
ClassInfos
,
0xFFFFFFFFL
);
if
(
tags
==
NULL
)
{
msg_Err
(
p_demux
,
"cannot load tags (broken seekhead or file)"
);
p_s
ys
->
in
->
setFilePointer
(
i_sav_position
,
seek_beginning
);
p_s
tream
->
in
->
setFilePointer
(
i_sav_position
,
seek_beginning
);
return
;
}
msg_Dbg
(
p_demux
,
"Tags"
);
ep
=
new
EbmlParser
(
p_s
ys
->
es
,
tags
);
ep
=
new
EbmlParser
(
p_s
tream
->
es
,
tags
);
while
(
(
el
=
ep
->
Get
()
)
!=
NULL
)
{
if
(
MKV_IS_ID
(
el
,
KaxTag
)
)
...
...
@@ -2137,7 +2214,7 @@ static void LoadTags( demux_t *p_demux )
delete
tags
;
msg_Dbg
(
p_demux
,
"loading tags done."
);
p_s
ys
->
in
->
setFilePointer
(
i_sav_position
,
seek_beginning
);
p_s
tream
->
in
->
setFilePointer
(
i_sav_position
,
seek_beginning
);
}
/*****************************************************************************
...
...
@@ -2146,6 +2223,8 @@ static void LoadTags( demux_t *p_demux )
static
void
ParseSeekHead
(
demux_t
*
p_demux
,
EbmlElement
*
seekhead
)
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
matroska_stream_t
*
p_stream
=
p_sys
->
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
EbmlElement
*
el
;
EbmlMaster
*
m
;
unsigned
int
i
;
...
...
@@ -2155,7 +2234,7 @@ static void ParseSeekHead( demux_t *p_demux, EbmlElement *seekhead )
/* Master elements */
m
=
static_cast
<
EbmlMaster
*>
(
seekhead
);
m
->
Read
(
*
p_s
ys
->
es
,
seekhead
->
Generic
().
Context
,
i_upper_level
,
el
,
true
);
m
->
Read
(
*
p_s
tream
->
es
,
seekhead
->
Generic
().
Context
,
i_upper_level
,
el
,
true
);
for
(
i
=
0
;
i
<
m
->
ListSize
();
i
++
)
{
...
...
@@ -2194,17 +2273,17 @@ static void ParseSeekHead( demux_t *p_demux, EbmlElement *seekhead )
if
(
id
==
KaxCues
::
ClassInfos
.
GlobalId
)
{
msg_Dbg
(
p_demux
,
"| | | = cues at "
I64Fd
,
i_pos
);
p_s
ys
->
i_cues_position
=
p_sys
->
segment
->
GetGlobalPosition
(
i_pos
);
p_s
egment
->
i_cues_position
=
p_segment
->
segment
->
GetGlobalPosition
(
i_pos
);
}
else
if
(
id
==
KaxChapters
::
ClassInfos
.
GlobalId
)
{
msg_Dbg
(
p_demux
,
"| | | = chapters at "
I64Fd
,
i_pos
);
p_s
ys
->
i_chapters_position
=
p_sys
->
segment
->
GetGlobalPosition
(
i_pos
);
p_s
egment
->
i_chapters_position
=
p_segment
->
segment
->
GetGlobalPosition
(
i_pos
);
}
else
if
(
id
==
KaxTags
::
ClassInfos
.
GlobalId
)
{
msg_Dbg
(
p_demux
,
"| | | = tags at "
I64Fd
,
i_pos
);
p_s
ys
->
i_tags_position
=
p_sys
->
segment
->
GetGlobalPosition
(
i_pos
);
p_s
egment
->
i_tags_position
=
p_segment
->
segment
->
GetGlobalPosition
(
i_pos
);
}
}
}
...
...
@@ -2221,17 +2300,19 @@ static void ParseSeekHead( demux_t *p_demux, EbmlElement *seekhead )
static
void
ParseTrackEntry
(
demux_t
*
p_demux
,
EbmlMaster
*
m
)
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
matroska_stream_t
*
p_stream
=
p_sys
->
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
unsigned
int
i
;
mkv_track_t
*
tk
;
msg_Dbg
(
p_demux
,
"| | + Track Entry"
);
p_s
ys
->
i_track
++
;
p_s
ys
->
track
=
(
mkv_track_t
*
)
realloc
(
p_sys
->
track
,
sizeof
(
mkv_track_t
)
*
(
p_sys
->
i_track
+
1
)
);
p_s
egment
->
i_track
++
;
p_s
egment
->
track
=
(
mkv_track_t
*
)
realloc
(
p_segment
->
track
,
sizeof
(
mkv_track_t
)
*
(
p_segment
->
i_track
+
1
)
);
/* Init the track */
tk
=
&
p_s
ys
->
track
[
p_sys
->
i_track
-
1
];
tk
=
&
p_s
egment
->
track
[
p_segment
->
i_track
-
1
];
memset
(
tk
,
0
,
sizeof
(
mkv_track_t
)
);
...
...
@@ -2241,7 +2322,7 @@ static void ParseTrackEntry( demux_t *p_demux, EbmlMaster *m )
tk
->
b_default
=
VLC_TRUE
;
tk
->
b_enabled
=
VLC_TRUE
;
tk
->
i_number
=
p_s
ys
->
i_track
-
1
;
tk
->
i_number
=
p_s
egment
->
i_track
-
1
;
tk
->
i_extra_data
=
0
;
tk
->
p_extra_data
=
NULL
;
tk
->
psz_codec
=
NULL
;
...
...
@@ -2625,6 +2706,8 @@ static void ParseTrackEntry( demux_t *p_demux, EbmlMaster *m )
static
void
ParseTracks
(
demux_t
*
p_demux
,
EbmlElement
*
tracks
)
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
matroska_stream_t
*
p_stream
=
p_sys
->
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
EbmlElement
*
el
;
EbmlMaster
*
m
;
unsigned
int
i
;
...
...
@@ -2634,7 +2717,7 @@ static void ParseTracks( demux_t *p_demux, EbmlElement *tracks )
/* Master elements */
m
=
static_cast
<
EbmlMaster
*>
(
tracks
);
m
->
Read
(
*
p_s
ys
->
es
,
tracks
->
Generic
().
Context
,
i_upper_level
,
el
,
true
);
m
->
Read
(
*
p_s
tream
->
es
,
tracks
->
Generic
().
Context
,
i_upper_level
,
el
,
true
);
for
(
i
=
0
;
i
<
m
->
ListSize
();
i
++
)
{
...
...
@@ -2657,6 +2740,8 @@ static void ParseTracks( demux_t *p_demux, EbmlElement *tracks )
static
void
ParseInfo
(
demux_t
*
p_demux
,
EbmlElement
*
info
)
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
matroska_stream_t
*
p_stream
=
p_sys
->
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
EbmlElement
*
el
;
EbmlMaster
*
m
;
unsigned
int
i
;
...
...
@@ -2666,7 +2751,7 @@ static void ParseInfo( demux_t *p_demux, EbmlElement *info )
/* Master elements */
m
=
static_cast
<
EbmlMaster
*>
(
info
);
m
->
Read
(
*
p_s
ys
->
es
,
info
->
Generic
().
Context
,
i_upper_level
,
el
,
true
);
m
->
Read
(
*
p_s
tream
->
es
,
info
->
Generic
().
Context
,
i_upper_level
,
el
,
true
);
for
(
i
=
0
;
i
<
m
->
ListSize
();
i
++
)
{
...
...
@@ -2674,68 +2759,68 @@ static void ParseInfo( demux_t *p_demux, EbmlElement *info )
if
(
MKV_IS_ID
(
l
,
KaxSegmentUID
)
)
{
p_s
ys
->
segment_uid
=
*
(
new
KaxSegmentUID
(
*
static_cast
<
KaxSegmentUID
*>
(
l
)));
p_s
egment
->
segment_uid
=
*
(
new
KaxSegmentUID
(
*
static_cast
<
KaxSegmentUID
*>
(
l
)));
msg_Dbg
(
p_demux
,
"| | + UID=%d"
,
*
(
uint32
*
)
p_s
ys
->
segment_uid
.
GetBuffer
()
);
msg_Dbg
(
p_demux
,
"| | + UID=%d"
,
*
(
uint32
*
)
p_s
egment
->
segment_uid
.
GetBuffer
()
);
}
else
if
(
MKV_IS_ID
(
l
,
KaxTimecodeScale
)
)
{
KaxTimecodeScale
&
tcs
=
*
(
KaxTimecodeScale
*
)
l
;
p_s
ys
->
i_timescale
=
uint64
(
tcs
);
p_s
egment
->
i_timescale
=
uint64
(
tcs
);
msg_Dbg
(
p_demux
,
"| | + TimecodeScale="
I64Fd
,
p_s
ys
->
i_timescale
);
p_s
egment
->
i_timescale
);
}
else
if
(
MKV_IS_ID
(
l
,
KaxDuration
)
)
{
KaxDuration
&
dur
=
*
(
KaxDuration
*
)
l
;
p_s
ys
->
f_duration
=
float
(
dur
);
p_s
egment
->
f_duration
=
float
(
dur
);
msg_Dbg
(
p_demux
,
"| | + Duration=%f"
,
p_s
ys
->
f_duration
);
p_s
egment
->
f_duration
);
}
else
if
(
MKV_IS_ID
(
l
,
KaxMuxingApp
)
)
{
KaxMuxingApp
&
mapp
=
*
(
KaxMuxingApp
*
)
l
;
p_s
ys
->
psz_muxing_application
=
UTF8ToStr
(
UTFstring
(
mapp
)
);
p_s
egment
->
psz_muxing_application
=
UTF8ToStr
(
UTFstring
(
mapp
)
);
msg_Dbg
(
p_demux
,
"| | + Muxing Application=%s"
,
p_s
ys
->
psz_muxing_application
);
p_s
egment
->
psz_muxing_application
);
}
else
if
(
MKV_IS_ID
(
l
,
KaxWritingApp
)
)
{
KaxWritingApp
&
wapp
=
*
(
KaxWritingApp
*
)
l
;
p_s
ys
->
psz_writing_application
=
UTF8ToStr
(
UTFstring
(
wapp
)
);
p_s
egment
->
psz_writing_application
=
UTF8ToStr
(
UTFstring
(
wapp
)
);
msg_Dbg
(
p_demux
,
"| | + Writing Application=%s"
,
p_s
ys
->
psz_writing_application
);
p_s
egment
->
psz_writing_application
);
}
else
if
(
MKV_IS_ID
(
l
,
KaxSegmentFilename
)
)
{
KaxSegmentFilename
&
sfn
=
*
(
KaxSegmentFilename
*
)
l
;
p_s
ys
->
psz_segment_filename
=
UTF8ToStr
(
UTFstring
(
sfn
)
);
p_s
egment
->
psz_segment_filename
=
UTF8ToStr
(
UTFstring
(
sfn
)
);
msg_Dbg
(
p_demux
,
"| | + Segment Filename=%s"
,
p_s
ys
->
psz_segment_filename
);
p_s
egment
->
psz_segment_filename
);
}
else
if
(
MKV_IS_ID
(
l
,
KaxTitle
)
)
{
KaxTitle
&
title
=
*
(
KaxTitle
*
)
l
;
p_s
ys
->
psz_title
=
UTF8ToStr
(
UTFstring
(
title
)
);
p_s
egment
->
psz_title
=
UTF8ToStr
(
UTFstring
(
title
)
);
msg_Dbg
(
p_demux
,
"| | + Title=%s"
,
p_s
ys
->
psz_title
);
msg_Dbg
(
p_demux
,
"| | + Title=%s"
,
p_s
egment
->
psz_title
);
}
if
(
MKV_IS_ID
(
l
,
KaxSegmentFamily
)
)
{
KaxSegmentFamily
*
uid
=
static_cast
<
KaxSegmentFamily
*>
(
l
);
p_s
ys
->
families
.
push_back
(
*
uid
);
p_s
egment
->
families
.
push_back
(
*
uid
);
msg_Dbg
(
p_demux
,
"| | + family=%d"
,
*
(
uint32
*
)
uid
->
GetBuffer
()
);
}
...
...
@@ -2753,8 +2838,8 @@ static void ParseInfo( demux_t *p_demux, EbmlElement *info )
asctime_r
(
&
tmres
,
buffer
)
)
{
buffer
[
strlen
(
buffer
)
-
1
]
=
'\0'
;
p_s
ys
->
psz_date_utc
=
strdup
(
buffer
);
msg_Dbg
(
p_demux
,
"| | + Date=%s"
,
p_s
ys
->
psz_date_utc
);
p_s
egment
->
psz_date_utc
=
strdup
(
buffer
);
msg_Dbg
(
p_demux
,
"| | + Date=%s"
,
p_s
egment
->
psz_date_utc
);
}
}
#endif
...
...
@@ -2764,7 +2849,7 @@ static void ParseInfo( demux_t *p_demux, EbmlElement *info )
}
}
p_s
ys
->
f_duration
*=
p_sys
->
i_timescale
/
1000000.0
;
p_s
egment
->
f_duration
*=
p_segment
->
i_timescale
/
1000000.0
;
}
...
...
@@ -2866,6 +2951,8 @@ static void ParseChapterAtom( demux_t *p_demux, int i_level, EbmlMaster *ca, cha
static
void
ParseChapters
(
demux_t
*
p_demux
,
EbmlElement
*
chapters
)
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
matroska_stream_t
*
p_stream
=
p_sys
->
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
EbmlElement
*
el
;
EbmlMaster
*
m
;
unsigned
int
i
;
...
...
@@ -2875,7 +2962,7 @@ static void ParseChapters( demux_t *p_demux, EbmlElement *chapters )
/* Master elements */
m
=
static_cast
<
EbmlMaster
*>
(
chapters
);
m
->
Read
(
*
p_s
ys
->
es
,
chapters
->
Generic
().
Context
,
i_upper_level
,
el
,
true
);
m
->
Read
(
*
p_s
tream
->
es
,
chapters
->
Generic
().
Context
,
i_upper_level
,
el
,
true
);
for
(
i
=
0
;
i
<
m
->
ListSize
();
i
++
)
{
...
...
@@ -2909,14 +2996,14 @@ static void ParseChapters( demux_t *p_demux, EbmlElement *chapters )
else
if
(
MKV_IS_ID
(
l
,
KaxEditionFlagDefault
)
)
{
if
(
uint8
(
*
static_cast
<
KaxEditionFlagDefault
*>
(
l
))
!=
0
)
i_default_edition
=
p_s
ys
->
editions
.
size
();
i_default_edition
=
p_s
egment
->
editions
.
size
();
}
else
{
msg_Dbg
(
p_demux
,
"| | | + Unknown (%s)"
,
typeid
(
*
l
).
name
()
);
}
}
p_s
ys
->
editions
.
push_back
(
edition
);
p_s
egment
->
editions
.
push_back
(
edition
);
}
else
{
...
...
@@ -2924,19 +3011,19 @@ static void ParseChapters( demux_t *p_demux, EbmlElement *chapters )
}
}
for
(
i
=
0
;
i
<
p_s
ys
->
editions
.
size
();
i
++
)
for
(
i
=
0
;
i
<
p_s
egment
->
editions
.
size
();
i
++
)
{
p_s
ys
->
editions
[
i
].
RefreshChapters
(
*
p_sys
->
title
);
p_s
egment
->
editions
[
i
].
RefreshChapters
(
*
p_sys
->
title
);
}
p_s
ys
->
i_current_edition
=
i_default_edition
;
p_s
egment
->
i_current_edition
=
i_default_edition
;
if
(
p_s
ys
->
editions
[
i_default_edition
].
b_ordered
)
if
(
p_s
egment
->
editions
[
i_default_edition
].
b_ordered
)
{
/* update the duration of the segment according to the sum of all sub chapters */
f_duration
=
p_s
ys
->
editions
[
i_default_edition
].
Duration
()
/
I64C
(
1000
);
f_duration
=
p_s
egment
->
editions
[
i_default_edition
].
Duration
()
/
I64C
(
1000
);
if
(
f_duration
>
0.0
)
p_s
ys
->
f_duration
=
f_duration
;
p_s
egment
->
f_duration
=
f_duration
;
}
}
...
...
@@ -2946,34 +3033,36 @@ static void ParseChapters( demux_t *p_demux, EbmlElement *chapters )
static
void
InformationCreate
(
demux_t
*
p_demux
)
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
matroska_stream_t
*
p_stream
=
p_sys
->
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
int
i_track
;
p_sys
->
meta
=
vlc_meta_New
();
if
(
p_s
ys
->
psz_title
)
if
(
p_s
egment
->
psz_title
)
{
vlc_meta_Add
(
p_sys
->
meta
,
VLC_META_TITLE
,
p_s
ys
->
psz_title
);
vlc_meta_Add
(
p_sys
->
meta
,
VLC_META_TITLE
,
p_s
egment
->
psz_title
);
}
if
(
p_s
ys
->
psz_date_utc
)
if
(
p_s
egment
->
psz_date_utc
)
{
vlc_meta_Add
(
p_sys
->
meta
,
VLC_META_DATE
,
p_s
ys
->
psz_date_utc
);
vlc_meta_Add
(
p_sys
->
meta
,
VLC_META_DATE
,
p_s
egment
->
psz_date_utc
);
}
if
(
p_s
ys
->
psz_segment_filename
)
if
(
p_s
egment
->
psz_segment_filename
)
{
vlc_meta_Add
(
p_sys
->
meta
,
_
(
"Segment filename"
),
p_s
ys
->
psz_segment_filename
);
vlc_meta_Add
(
p_sys
->
meta
,
_
(
"Segment filename"
),
p_s
egment
->
psz_segment_filename
);
}
if
(
p_s
ys
->
psz_muxing_application
)
if
(
p_s
egment
->
psz_muxing_application
)
{
vlc_meta_Add
(
p_sys
->
meta
,
_
(
"Muxing application"
),
p_s
ys
->
psz_muxing_application
);
vlc_meta_Add
(
p_sys
->
meta
,
_
(
"Muxing application"
),
p_s
egment
->
psz_muxing_application
);
}
if
(
p_s
ys
->
psz_writing_application
)
if
(
p_s
egment
->
psz_writing_application
)
{
vlc_meta_Add
(
p_sys
->
meta
,
_
(
"Writing application"
),
p_s
ys
->
psz_writing_application
);
vlc_meta_Add
(
p_sys
->
meta
,
_
(
"Writing application"
),
p_s
egment
->
psz_writing_application
);
}
for
(
i_track
=
0
;
i_track
<
p_s
ys
->
i_track
;
i_track
++
)
for
(
i_track
=
0
;
i_track
<
p_s
egment
->
i_track
;
i_track
++
)
{
mkv_track_t
*
tk
=
&
p_s
ys
->
track
[
i_track
];
mkv_track_t
*
tk
=
&
p_s
egment
->
track
[
i_track
];
vlc_meta_t
*
mtk
=
vlc_meta_New
();
p_sys
->
meta
->
track
=
(
vlc_meta_t
**
)
realloc
(
p_sys
->
meta
->
track
,
...
...
@@ -3002,7 +3091,7 @@ static void InformationCreate( demux_t *p_demux )
}
}
if
(
p_s
ys
->
i_tags_position
>=
0
)
if
(
p_s
egment
->
i_tags_position
>=
0
)
{
vlc_bool_t
b_seekable
;
...
...
@@ -3022,19 +3111,21 @@ static void InformationCreate( demux_t *p_demux )
static
void
IndexAppendCluster
(
demux_t
*
p_demux
,
KaxCluster
*
cluster
)
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
matroska_stream_t
*
p_stream
=
p_sys
->
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
#define idx p_s
ys->index[p_sys
->i_index]
#define idx p_s
egment->index[p_segment
->i_index]
idx
.
i_track
=
-
1
;
idx
.
i_block_number
=
-
1
;
idx
.
i_position
=
cluster
->
GetElementPosition
();
idx
.
i_time
=
-
1
;
idx
.
b_key
=
VLC_TRUE
;
p_s
ys
->
i_index
++
;
if
(
p_s
ys
->
i_index
>=
p_sys
->
i_index_max
)
p_s
egment
->
i_index
++
;
if
(
p_s
egment
->
i_index
>=
p_segment
->
i_index_max
)
{
p_s
ys
->
i_index_max
+=
1024
;
p_s
ys
->
index
=
(
mkv_index_t
*
)
realloc
(
p_sys
->
index
,
sizeof
(
mkv_index_t
)
*
p_sys
->
i_index_max
);
p_s
egment
->
i_index_max
+=
1024
;
p_s
egment
->
index
=
(
mkv_index_t
*
)
realloc
(
p_segment
->
index
,
sizeof
(
mkv_index_t
)
*
p_segment
->
i_index_max
);
}
#undef idx
}
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment