Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
V
vlc
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
Commits
291c6479
Commit
291c6479
authored
19 years ago
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
Hide 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:
...
@@ -362,13 +362,11 @@ public:
bool
b_ordered
;
bool
b_ordered
;
};
};
class
demux_sys
_t
class
matroska_segment
_t
{
{
public:
public:
demux_sys_t
()
matroska_segment_t
()
:
in
(
NULL
)
:
segment
(
NULL
)
,
es
(
NULL
)
,
ep
(
NULL
)
,
i_timescale
(
0
)
,
i_timescale
(
0
)
,
f_duration
(
0.0
)
,
f_duration
(
0.0
)
,
i_track
(
0
)
,
i_track
(
0
)
...
@@ -376,11 +374,7 @@ public:
...
@@ -376,11 +374,7 @@ public:
,
i_cues_position
(
0
)
,
i_cues_position
(
0
)
,
i_chapters_position
(
0
)
,
i_chapters_position
(
0
)
,
i_tags_position
(
0
)
,
i_tags_position
(
0
)
,
segment
(
NULL
)
,
cluster
(
NULL
)
,
cluster
(
NULL
)
,
i_pts
(
0
)
,
i_start_pts
(
0
)
,
i_chapter_time
(
0
)
,
b_cues
(
false
)
,
b_cues
(
false
)
,
i_index
(
0
)
,
i_index
(
0
)
,
i_index_max
(
0
)
,
i_index_max
(
0
)
...
@@ -390,15 +384,11 @@ public:
...
@@ -390,15 +384,11 @@ public:
,
psz_segment_filename
(
NULL
)
,
psz_segment_filename
(
NULL
)
,
psz_title
(
NULL
)
,
psz_title
(
NULL
)
,
psz_date_utc
(
NULL
)
,
psz_date_utc
(
NULL
)
,
meta
(
NULL
)
,
title
(
NULL
)
,
i_current_edition
(
0
)
,
i_current_edition
(
0
)
,
psz_current_chapter
(
NULL
)
,
psz_current_chapter
(
NULL
)
{}
{}
vlc_stream_io_callback
*
in
;
KaxSegment
*
segment
;
EbmlStream
*
es
;
EbmlParser
*
ep
;
/* time scale */
/* time scale */
uint64_t
i_timescale
;
uint64_t
i_timescale
;
...
@@ -415,15 +405,9 @@ public:
...
@@ -415,15 +405,9 @@ public:
int64_t
i_chapters_position
;
int64_t
i_chapters_position
;
int64_t
i_tags_position
;
int64_t
i_tags_position
;
/* current data */
KaxSegment
*
segment
;
KaxCluster
*
cluster
;
KaxCluster
*
cluster
;
KaxSegmentUID
segment_uid
;
KaxSegmentUID
segment_uid
;
mtime_t
i_pts
;
mtime_t
i_start_pts
;
mtime_t
i_chapter_time
;
vlc_bool_t
b_cues
;
vlc_bool_t
b_cues
;
int
i_index
;
int
i_index
;
int
i_index_max
;
int
i_index_max
;
...
@@ -436,16 +420,93 @@ public:
...
@@ -436,16 +420,93 @@ public:
char
*
psz_title
;
char
*
psz_title
;
char
*
psz_date_utc
;
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
;
vlc_meta_t
*
meta
;
input_title_t
*
title
;
input_title_t
*
title
;
std
::
vector
<
KaxSegmentFamily
>
families
;
std
::
vector
<
matroska_stream_t
*>
streams
;
std
::
vector
<
KaxSegment
*>
family_members
;
int
i_current_stream
;
std
::
vector
<
chapter_edition_t
>
editions
;
matroska_stream_t
*
Stream
()
int
i_current_edition
;
{
const
chapter_item_t
*
psz_current_chapter
;
if
(
i_current_stream
!=
-
1
&&
i_current_stream
<
streams
.
size
()
)
return
streams
[
i_current_stream
];
return
NULL
;
}
};
};
static
int
Demux
(
demux_t
*
);
static
int
Demux
(
demux_t
*
);
...
@@ -471,14 +532,16 @@ static void ParseChapters( demux_t *, EbmlElement *chapters );
...
@@ -471,14 +532,16 @@ static void ParseChapters( demux_t *, EbmlElement *chapters );
*****************************************************************************/
*****************************************************************************/
static
int
Open
(
vlc_object_t
*
p_this
)
static
int
Open
(
vlc_object_t
*
p_this
)
{
{
demux_t
*
p_demux
=
(
demux_t
*
)
p_this
;
demux_t
*
p_demux
=
(
demux_t
*
)
p_this
;
demux_sys_t
*
p_sys
;
demux_sys_t
*
p_sys
;
uint8_t
*
p_peek
;
matroska_stream_t
*
p_stream
;
std
::
string
s_path
,
s_filename
;
matroska_segment_t
*
p_segment
;
int
i_upper_lvl
;
uint8_t
*
p_peek
;
size_t
i
;
std
::
string
s_path
,
s_filename
;
int
i_upper_lvl
;
size_t
i
;
int
i_track
;
int
i_track
;
EbmlElement
*
el
=
NULL
,
*
el1
=
NULL
;
EbmlElement
*
el
=
NULL
,
*
el1
=
NULL
;
...
@@ -494,40 +557,48 @@ static int Open( vlc_object_t * p_this )
...
@@ -494,40 +557,48 @@ static int Open( vlc_object_t * p_this )
p_demux
->
pf_control
=
Control
;
p_demux
->
pf_control
=
Control
;
p_demux
->
p_sys
=
p_sys
=
new
demux_sys_t
;
p_demux
->
p_sys
=
p_sys
=
new
demux_sys_t
;
p_sys
->
in
=
new
vlc_stream_io_callback
(
p_demux
->
s
);
p_stream
=
new
matroska_stream_t
;
p_sys
->
es
=
new
EbmlStream
(
*
p_sys
->
in
);
p_segment
=
new
matroska_segment_t
;
p_sys
->
f_duration
=
-
1
;
p_sys
->
i_timescale
=
MKVD_TIMECODESCALE
;
p_sys
->
streams
.
push_back
(
p_stream
);
p_sys
->
i_track
=
0
;
p_sys
->
i_current_stream
=
0
;
p_sys
->
track
=
(
mkv_track_t
*
)
malloc
(
sizeof
(
mkv_track_t
)
);
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_sys
->
i_pts
=
0
;
p_s
ys
->
i_cues_position
=
-
1
;
p_s
egment
->
i_cues_position
=
-
1
;
p_s
ys
->
i_chapters_position
=
-
1
;
p_s
egment
->
i_chapters_position
=
-
1
;
p_s
ys
->
i_tags_position
=
-
1
;
p_s
egment
->
i_tags_position
=
-
1
;
p_s
ys
->
b_cues
=
VLC_FALSE
;
p_s
egment
->
b_cues
=
VLC_FALSE
;
p_s
ys
->
i_index
=
0
;
p_s
egment
->
i_index
=
0
;
p_s
ys
->
i_index_max
=
1024
;
p_s
egment
->
i_index_max
=
1024
;
p_s
ys
->
index
=
(
mkv_index_t
*
)
malloc
(
sizeof
(
mkv_index_t
)
*
p_s
egment
->
index
=
(
mkv_index_t
*
)
malloc
(
sizeof
(
mkv_index_t
)
*
p_s
ys
->
i_index_max
);
p_s
egment
->
i_index_max
);
p_s
ys
->
psz_muxing_application
=
NULL
;
p_s
egment
->
psz_muxing_application
=
NULL
;
p_s
ys
->
psz_writing_application
=
NULL
;
p_s
egment
->
psz_writing_application
=
NULL
;
p_s
ys
->
psz_segment_filename
=
NULL
;
p_s
egment
->
psz_segment_filename
=
NULL
;
p_s
ys
->
psz_title
=
NULL
;
p_s
egment
->
psz_title
=
NULL
;
p_s
ys
->
psz_date_utc
=
NULL
;;
p_s
egment
->
psz_date_utc
=
NULL
;;
p_sys
->
meta
=
NULL
;
p_sys
->
meta
=
NULL
;
p_sys
->
title
=
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"
);
msg_Err
(
p_demux
,
"failed to create EbmlStream"
);
delete
p_sys
->
in
;
delete
p_sys
;
delete
p_sys
;
return
VLC_EGENERIC
;
return
VLC_EGENERIC
;
}
}
/* Find the EbmlHead element */
/* 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
)
if
(
el
==
NULL
)
{
{
msg_Err
(
p_demux
,
"cannot find EbmlHead"
);
msg_Err
(
p_demux
,
"cannot find EbmlHead"
);
...
@@ -535,23 +606,23 @@ static int Open( vlc_object_t * p_this )
...
@@ -535,23 +606,23 @@ static int Open( vlc_object_t * p_this )
}
}
msg_Dbg
(
p_demux
,
"EbmlHead"
);
msg_Dbg
(
p_demux
,
"EbmlHead"
);
/* skip it */
/* skip it */
el
->
SkipData
(
*
p_s
ys
->
es
,
el
->
Generic
().
Context
);
el
->
SkipData
(
*
p_s
tream
->
es
,
el
->
Generic
().
Context
);
delete
el
;
delete
el
;
/* Find a segment */
/* Find a segment */
el
=
p_s
ys
->
es
->
FindNextID
(
KaxSegment
::
ClassInfos
,
0xFFFFFFFFL
);
el
=
p_s
tream
->
es
->
FindNextID
(
KaxSegment
::
ClassInfos
,
0xFFFFFFFFL
);
if
(
el
==
NULL
)
if
(
el
==
NULL
)
{
{
msg_Err
(
p_demux
,
"cannot find KaxSegment"
);
msg_Err
(
p_demux
,
"cannot find KaxSegment"
);
goto
error
;
goto
error
;
}
}
MkvTree
(
p_demux
,
0
,
"Segment"
);
MkvTree
(
p_demux
,
0
,
"Segment"
);
p_s
ys
->
segment
=
(
KaxSegment
*
)
el
;
p_s
egment
->
segment
=
(
KaxSegment
*
)
el
;
p_s
ys
->
cluster
=
NULL
;
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
)
)
if
(
MKV_IS_ID
(
el1
,
KaxInfo
)
)
{
{
...
@@ -573,9 +644,9 @@ static int Open( vlc_object_t * p_this )
...
@@ -573,9 +644,9 @@ static int Open( vlc_object_t * p_this )
{
{
msg_Dbg
(
p_demux
,
"| + Cluster"
);
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 */
/* stop parsing the stream */
break
;
break
;
}
}
...
@@ -692,15 +763,15 @@ static int Open( vlc_object_t * p_this )
...
@@ -692,15 +763,15 @@ static int Open( vlc_object_t * p_this )
if
(
MKV_IS_ID
(
l
,
KaxSegmentUID
)
)
if
(
MKV_IS_ID
(
l
,
KaxSegmentUID
)
)
{
{
p_uid
=
static_cast
<
KaxSegmentUID
*>
(
l
);
p_uid
=
static_cast
<
KaxSegmentUID
*>
(
l
);
if
(
p_s
ys
->
segment_uid
==
*
p_uid
)
if
(
p_s
egment
->
segment_uid
==
*
p_uid
)
break
;
break
;
}
}
else
if
(
MKV_IS_ID
(
l
,
KaxSegmentFamily
)
)
else
if
(
MKV_IS_ID
(
l
,
KaxSegmentFamily
)
)
{
{
KaxSegmentFamily
*
p_fam
=
static_cast
<
KaxSegmentFamily
*>
(
l
);
KaxSegmentFamily
*
p_fam
=
static_cast
<
KaxSegmentFamily
*>
(
l
);
std
::
vector
<
KaxSegmentFamily
>::
iterator
iter
;
std
::
vector
<
KaxSegmentFamily
>::
iterator
iter
;
for
(
iter
=
p_s
ys
->
families
.
begin
();
for
(
iter
=
p_s
egment
->
families
.
begin
();
iter
!=
p_s
ys
->
families
.
end
();
iter
!=
p_s
egment
->
families
.
end
();
iter
++
)
iter
++
)
{
{
if
(
*
iter
==
*
p_fam
)
if
(
*
iter
==
*
p_fam
)
...
@@ -739,14 +810,14 @@ static int Open( vlc_object_t * p_this )
...
@@ -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 ?"
);
msg_Err
(
p_demux
,
"cannot find any cluster, damaged file ?"
);
goto
error
;
goto
error
;
}
}
/* *** Load the cue if found *** */
/* *** 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
;
vlc_bool_t
b_seekable
;
...
@@ -757,20 +828,20 @@ static int Open( vlc_object_t * p_this )
...
@@ -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"
);
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 */
/* add all es */
msg_Dbg
(
p_demux
,
"found %d es"
,
p_s
ys
->
i_track
);
msg_Dbg
(
p_demux
,
"found %d es"
,
p_s
egment
->
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
.
i_cat
==
UNKNOWN_ES
)
if
(
tk
.
fmt
.
i_cat
==
UNKNOWN_ES
)
{
{
msg_Warn
(
p_demux
,
"invalid track[%d, n=%d]"
,
i_track
,
tk
.
i_number
);
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 )
...
@@ -1060,8 +1131,6 @@ static int Open( vlc_object_t * p_this )
return
VLC_SUCCESS
;
return
VLC_SUCCESS
;
error:
error:
delete
p_sys
->
es
;
delete
p_sys
->
in
;
delete
p_sys
;
delete
p_sys
;
return
VLC_EGENERIC
;
return
VLC_EGENERIC
;
}
}
...
@@ -1073,11 +1142,13 @@ static void Close( vlc_object_t *p_this )
...
@@ -1073,11 +1142,13 @@ static void Close( vlc_object_t *p_this )
{
{
demux_t
*
p_demux
=
(
demux_t
*
)
p_this
;
demux_t
*
p_demux
=
(
demux_t
*
)
p_this
;
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
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
;
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
)
if
(
tk
.
fmt
.
psz_description
)
{
{
free
(
tk
.
fmt
.
psz_description
);
free
(
tk
.
fmt
.
psz_description
);
...
@@ -1092,22 +1163,19 @@ static void Close( vlc_object_t *p_this )
...
@@ -1092,22 +1163,19 @@ static void Close( vlc_object_t *p_this )
}
}
#undef tk
#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
;
delete
p_sys
;
}
}
...
@@ -1116,11 +1184,12 @@ static void Close( vlc_object_t *p_this )
...
@@ -1116,11 +1184,12 @@ static void Close( vlc_object_t *p_this )
*****************************************************************************/
*****************************************************************************/
static
int
Control
(
demux_t
*
p_demux
,
int
i_query
,
va_list
args
)
static
int
Control
(
demux_t
*
p_demux
,
int
i_query
,
va_list
args
)
{
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
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
;
int64_t
*
pi64
;
double
*
pf
,
f
;
double
*
pf
,
f
;
int
i_skp
;
int
i_skp
;
mtime_t
*
i_sk_time
;
vlc_meta_t
**
pp_meta
;
vlc_meta_t
**
pp_meta
;
...
@@ -1133,19 +1202,16 @@ static int Control( demux_t *p_demux, int i_query, va_list args )
...
@@ -1133,19 +1202,16 @@ static int Control( demux_t *p_demux, int i_query, va_list args )
case
DEMUX_GET_LENGTH
:
case
DEMUX_GET_LENGTH
:
pi64
=
(
int64_t
*
)
va_arg
(
args
,
int64_t
*
);
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_SUCCESS
;
}
}
return
VLC_EGENERIC
;
return
VLC_EGENERIC
;
case
DEMUX_GET_POSITION
:
case
DEMUX_GET_POSITION
:
pf
=
(
double
*
)
va_arg
(
args
,
double
*
);
pf
=
(
double
*
)
va_arg
(
args
,
double
*
);
/* if (p_sys->i_pts < p_sys->i_start_pts)
*
pf
=
(
double
)
p_sys
->
i_pts
/
(
1000.0
*
p_segment
->
f_duration
);
*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
);
return
VLC_SUCCESS
;
return
VLC_SUCCESS
;
case
DEMUX_SET_POSITION
:
case
DEMUX_SET_POSITION
:
...
@@ -1155,10 +1221,7 @@ static int Control( demux_t *p_demux, int i_query, va_list args )
...
@@ -1155,10 +1221,7 @@ static int Control( demux_t *p_demux, int i_query, va_list args )
case
DEMUX_GET_TIME
:
case
DEMUX_GET_TIME
:
pi64
=
(
int64_t
*
)
va_arg
(
args
,
int64_t
*
);
pi64
=
(
int64_t
*
)
va_arg
(
args
,
int64_t
*
);
/* if (p_sys->i_pts < p_sys->i_start_pts)
*
pi64
=
p_sys
->
i_pts
;
*pi64 = p_sys->i_start_pts;
else*/
*
pi64
=
p_sys
->
i_pts
;
return
VLC_SUCCESS
;
return
VLC_SUCCESS
;
case
DEMUX_GET_TITLE_INFO
:
case
DEMUX_GET_TITLE_INFO
:
...
@@ -1206,7 +1269,9 @@ static int Control( demux_t *p_demux, int i_query, va_list args )
...
@@ -1206,7 +1269,9 @@ 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
)
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
;
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
;
*
pp_block
=
NULL
;
*
pi_ref1
=
-
1
;
*
pi_ref1
=
-
1
;
...
@@ -1222,14 +1287,14 @@ static int BlockGet( demux_t *p_demux, KaxBlock **pp_block, int64_t *pi_ref1, in
...
@@ -1222,14 +1287,14 @@ static int BlockGet( demux_t *p_demux, KaxBlock **pp_block, int64_t *pi_ref1, in
return
VLC_EGENERIC
;
return
VLC_EGENERIC
;
}
}
el
=
p_s
ys
->
ep
->
Get
();
el
=
p_s
tream
->
ep
->
Get
();
i_level
=
p_s
ys
->
ep
->
GetLevel
();
i_level
=
p_s
tream
->
ep
->
GetLevel
();
if
(
el
==
NULL
&&
*
pp_block
!=
NULL
)
if
(
el
==
NULL
&&
*
pp_block
!=
NULL
)
{
{
/* update the index */
/* update the index */
#define idx p_s
ys->index[p_sys
->i_index - 1]
#define idx p_s
egment->index[p_segment
->i_index - 1]
if
(
p_s
ys
->
i_index
>
0
&&
idx
.
i_time
==
-
1
)
if
(
p_s
egment
->
i_index
>
0
&&
idx
.
i_time
==
-
1
)
{
{
idx
.
i_time
=
(
*
pp_block
)
->
GlobalTimecode
()
/
(
mtime_t
)
1000
;
idx
.
i_time
=
(
*
pp_block
)
->
GlobalTimecode
()
/
(
mtime_t
)
1000
;
idx
.
b_key
=
*
pi_ref1
==
-
1
?
VLC_TRUE
:
VLC_FALSE
;
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
...
@@ -1240,9 +1305,9 @@ static int BlockGet( demux_t *p_demux, KaxBlock **pp_block, int64_t *pi_ref1, in
if
(
el
==
NULL
)
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
;
continue
;
}
}
msg_Warn
(
p_demux
,
"EOF"
);
msg_Warn
(
p_demux
,
"EOF"
);
...
@@ -1254,16 +1319,16 @@ static int BlockGet( demux_t *p_demux, KaxBlock **pp_block, int64_t *pi_ref1, in
...
@@ -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
)
)
if
(
MKV_IS_ID
(
el
,
KaxCluster
)
)
{
{
p_s
ys
->
cluster
=
(
KaxCluster
*
)
el
;
p_s
egment
->
cluster
=
(
KaxCluster
*
)
el
;
/* add it to the index */
/* add it to the index */
if
(
p_s
ys
->
i_index
==
0
||
if
(
p_s
egment
->
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
()
)
)
(
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
)
)
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
...
@@ -1281,12 +1346,12 @@ static int BlockGet( demux_t *p_demux, KaxBlock **pp_block, int64_t *pi_ref1, in
{
{
KaxClusterTimecode
&
ctc
=
*
(
KaxClusterTimecode
*
)
el
;
KaxClusterTimecode
&
ctc
=
*
(
KaxClusterTimecode
*
)
el
;
ctc
.
ReadData
(
p_s
ys
->
es
->
I_O
(),
SCOPE_ALL_DATA
);
ctc
.
ReadData
(
p_s
tream
->
es
->
I_O
(),
SCOPE_ALL_DATA
);
p_s
ys
->
cluster
->
InitTimecode
(
uint64
(
ctc
),
p_sys
->
i_timescale
);
p_s
egment
->
cluster
->
InitTimecode
(
uint64
(
ctc
),
p_segment
->
i_timescale
);
}
}
else
if
(
MKV_IS_ID
(
el
,
KaxBlockGroup
)
)
else
if
(
MKV_IS_ID
(
el
,
KaxBlockGroup
)
)
{
{
p_s
ys
->
ep
->
Down
();
p_s
tream
->
ep
->
Down
();
}
}
}
}
else
if
(
i_level
==
3
)
else
if
(
i_level
==
3
)
...
@@ -1295,23 +1360,23 @@ static int BlockGet( demux_t *p_demux, KaxBlock **pp_block, int64_t *pi_ref1, in
...
@@ -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
=
(
KaxBlock
*
)
el
;
(
*
pp_block
)
->
ReadData
(
p_s
ys
->
es
->
I_O
()
);
(
*
pp_block
)
->
ReadData
(
p_s
tream
->
es
->
I_O
()
);
(
*
pp_block
)
->
SetParent
(
*
p_s
ys
->
cluster
);
(
*
pp_block
)
->
SetParent
(
*
p_s
egment
->
cluster
);
p_s
ys
->
ep
->
Keep
();
p_s
tream
->
ep
->
Keep
();
}
}
else
if
(
MKV_IS_ID
(
el
,
KaxBlockDuration
)
)
else
if
(
MKV_IS_ID
(
el
,
KaxBlockDuration
)
)
{
{
KaxBlockDuration
&
dur
=
*
(
KaxBlockDuration
*
)
el
;
KaxBlockDuration
&
dur
=
*
(
KaxBlockDuration
*
)
el
;
dur
.
ReadData
(
p_s
ys
->
es
->
I_O
()
);
dur
.
ReadData
(
p_s
tream
->
es
->
I_O
()
);
*
pi_duration
=
uint64
(
dur
);
*
pi_duration
=
uint64
(
dur
);
}
}
else
if
(
MKV_IS_ID
(
el
,
KaxReferenceBlock
)
)
else
if
(
MKV_IS_ID
(
el
,
KaxReferenceBlock
)
)
{
{
KaxReferenceBlock
&
ref
=
*
(
KaxReferenceBlock
*
)
el
;
KaxReferenceBlock
&
ref
=
*
(
KaxReferenceBlock
*
)
el
;
ref
.
ReadData
(
p_s
ys
->
es
->
I_O
()
);
ref
.
ReadData
(
p_s
tream
->
es
->
I_O
()
);
if
(
*
pi_ref1
==
-
1
)
if
(
*
pi_ref1
==
-
1
)
{
{
*
pi_ref1
=
int64
(
ref
);
*
pi_ref1
=
int64
(
ref
);
...
@@ -1342,14 +1407,16 @@ static block_t *MemToBlock( demux_t *p_demux, uint8_t *p_mem, int i_mem)
...
@@ -1342,14 +1407,16 @@ static block_t *MemToBlock( demux_t *p_demux, uint8_t *p_mem, int i_mem)
static
void
BlockDecode
(
demux_t
*
p_demux
,
KaxBlock
*
block
,
mtime_t
i_pts
,
static
void
BlockDecode
(
demux_t
*
p_demux
,
KaxBlock
*
block
,
mtime_t
i_pts
,
mtime_t
i_duration
)
mtime_t
i_duration
)
{
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
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
;
int
i_track
;
unsigned
int
i
;
unsigned
int
i
;
vlc_bool_t
b
;
vlc_bool_t
b
;
#define tk p_s
ys
->track[i_track]
#define tk p_s
egment
->track[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
++
)
{
{
if
(
tk
.
i_number
==
block
->
TrackNum
()
)
if
(
tk
.
i_number
==
block
->
TrackNum
()
)
{
{
...
@@ -1357,7 +1424,7 @@ static void BlockDecode( demux_t *p_demux, KaxBlock *block, mtime_t i_pts,
...
@@ -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
()
);
msg_Err
(
p_demux
,
"invalid track number=%d"
,
block
->
TrackNum
()
);
return
;
return
;
...
@@ -1437,43 +1504,47 @@ static void BlockDecode( demux_t *p_demux, KaxBlock *block, mtime_t i_pts,
...
@@ -1437,43 +1504,47 @@ static void BlockDecode( demux_t *p_demux, KaxBlock *block, mtime_t i_pts,
static
void
UpdateCurrentToChapter
(
demux_t
&
demux
)
static
void
UpdateCurrentToChapter
(
demux_t
&
demux
)
{
{
demux_sys_t
&
sys
=
*
demux
.
p_sys
;
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
;
const
chapter_item_t
*
psz_curr_chapter
;
/* update current chapter/seekpoint */
/* update current chapter/seekpoint */
if
(
sys
.
editions
.
size
())
if
(
p_segment
->
editions
.
size
())
{
{
/* 1st, we need to know in which chapter we are */
/* 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 */
/* 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_update
|=
INPUT_UPDATE_SEEKPOINT
;
demux
.
info
.
i_seekpoint
=
psz_curr_chapter
->
i_seekpoint_num
-
1
;
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) */
/* 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
);
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) */
/* 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 */
/* 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;
//
p_segment->
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_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
)
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
;
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
mtime_t
i_time_offset
=
0
;
matroska_stream_t
*
p_stream
=
p_sys
->
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
mtime_t
i_time_offset
=
0
;
KaxBlock
*
block
;
KaxBlock
*
block
;
int64_t
i_block_duration
;
int64_t
i_block_duration
;
...
@@ -1496,38 +1567,38 @@ static void Seek( demux_t *p_demux, mtime_t i_date, double f_percent, const chap
...
@@ -1496,38 +1567,38 @@ static void Seek( demux_t *p_demux, mtime_t i_date, double f_percent, const chap
return
;
return
;
}
}
delete
p_s
ys
->
ep
;
delete
p_s
tream
->
ep
;
p_s
ys
->
ep
=
new
EbmlParser
(
p_sys
->
es
,
p_sys
->
segment
);
p_s
tream
->
ep
=
new
EbmlParser
(
p_stream
->
es
,
p_segment
->
segment
);
p_s
ys
->
cluster
=
NULL
;
p_s
egment
->
cluster
=
NULL
;
/* seek without index or without date */
/* 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
else
{
{
int64_t
i_pos
=
int64_t
(
f_percent
*
stream_Size
(
p_demux
->
s
)
);
int64_t
i_pos
=
int64_t
(
f_percent
*
stream_Size
(
p_demux
->
s
)
);
msg_Dbg
(
p_demux
,
"inacurate way of seeking"
);
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
;
break
;
}
}
}
}
if
(
i_index
==
p_s
ys
->
i_index
)
if
(
i_index
==
p_s
egment
->
i_index
)
{
{
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 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;
EbmlElement *el;
...
@@ -1559,24 +1630,24 @@ static void Seek( demux_t *p_demux, mtime_t i_date, double f_percent, const chap
...
@@ -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
// find the actual time for an ordered edition
if
(
psz_chapter
==
NULL
)
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 */
/* 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
)
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_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_update
|=
INPUT_UPDATE_SEEKPOINT
;
p_demux
->
info
.
i_seekpoint
=
psz_chapter
->
i_seekpoint_num
-
1
;
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
;
break
;
}
}
...
@@ -1588,11 +1659,11 @@ static void Seek( demux_t *p_demux, mtime_t i_date, double f_percent, const chap
...
@@ -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%%)"
,
msg_Dbg
(
p_demux
,
"seek got "
I64Fd
" (%d%%)"
,
p_s
ys
->
index
[
i_index
].
i_time
,
p_s
egment
->
index
[
i_index
].
i_time
,
(
int
)(
100
*
p_s
ys
->
index
[
i_index
].
i_position
/
(
int
)(
100
*
p_s
egment
->
index
[
i_index
].
i_position
/
stream_Size
(
p_demux
->
s
)
)
);
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
);
seek_beginning
);
p_sys
->
i_start_pts
=
i_date
;
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
...
@@ -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
);
es_out_Control
(
p_demux
->
out
,
ES_OUT_RESET_PCR
);
/* now parse until key frame */
/* 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
;
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
)
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
...
@@ -1622,7 +1693,7 @@ static void Seek( demux_t *p_demux, mtime_t i_date, double f_percent, const chap
return
;
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
()
)
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
...
@@ -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
;
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
)
if
(
tk
.
fmt
.
i_cat
==
VIDEO_ES
)
{
{
...
@@ -1660,8 +1731,10 @@ static void Seek( demux_t *p_demux, mtime_t i_date, double f_percent, const chap
...
@@ -1660,8 +1731,10 @@ static void Seek( demux_t *p_demux, mtime_t i_date, double f_percent, const chap
*****************************************************************************/
*****************************************************************************/
static
int
Demux
(
demux_t
*
p_demux
)
static
int
Demux
(
demux_t
*
p_demux
)
{
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
int
i_block_count
=
0
;
matroska_stream_t
*
p_stream
=
p_sys
->
Stream
();
matroska_segment_t
*
p_segment
=
p_stream
->
Segment
();
int
i_block_count
=
0
;
KaxBlock
*
block
;
KaxBlock
*
block
;
int64_t
i_block_duration
;
int64_t
i_block_duration
;
...
@@ -1673,7 +1746,7 @@ static int Demux( demux_t *p_demux)
...
@@ -1673,7 +1746,7 @@ static int Demux( demux_t *p_demux)
if
(
p_sys
->
i_pts
>=
p_sys
->
i_start_pts
)
if
(
p_sys
->
i_pts
>=
p_sys
->
i_start_pts
)
UpdateCurrentToChapter
(
*
p_demux
);
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 */
/* nothing left to read in this ordered edition */
return
0
;
return
0
;
...
@@ -1681,12 +1754,12 @@ static int Demux( demux_t *p_demux)
...
@@ -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
(
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
// 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
;
return
1
;
}
}
...
@@ -1908,27 +1981,29 @@ EbmlElement *EbmlParser::Get( void )
...
@@ -1908,27 +1981,29 @@ EbmlElement *EbmlParser::Get( void )
static
void
LoadCues
(
demux_t
*
p_demux
)
static
void
LoadCues
(
demux_t
*
p_demux
)
{
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
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
;
EbmlParser
*
ep
;
EbmlElement
*
el
,
*
cues
;
EbmlElement
*
el
,
*
cues
;
msg_Dbg
(
p_demux
,
"loading cues"
);
msg_Dbg
(
p_demux
,
"loading cues"
);
p_s
ys
->
in
->
setFilePointer
(
p_sys
->
i_cues_position
,
seek_beginning
);
p_s
tream
->
in
->
setFilePointer
(
p_segment
->
i_cues_position
,
seek_beginning
);
cues
=
p_s
ys
->
es
->
FindNextID
(
KaxCues
::
ClassInfos
,
0xFFFFFFFFL
);
cues
=
p_s
tream
->
es
->
FindNextID
(
KaxCues
::
ClassInfos
,
0xFFFFFFFFL
);
if
(
cues
==
NULL
)
if
(
cues
==
NULL
)
{
{
msg_Err
(
p_demux
,
"cannot load cues (broken seekhead or file)"
);
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
;
return
;
}
}
ep
=
new
EbmlParser
(
p_s
ys
->
es
,
cues
);
ep
=
new
EbmlParser
(
p_s
tream
->
es
,
cues
);
while
(
(
el
=
ep
->
Get
()
)
!=
NULL
)
while
(
(
el
=
ep
->
Get
()
)
!=
NULL
)
{
{
if
(
MKV_IS_ID
(
el
,
KaxCuePoint
)
)
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_track
=
-
1
;
idx
.
i_block_number
=
-
1
;
idx
.
i_block_number
=
-
1
;
...
@@ -1943,9 +2018,9 @@ static void LoadCues( demux_t *p_demux )
...
@@ -1943,9 +2018,9 @@ static void LoadCues( demux_t *p_demux )
{
{
KaxCueTime
&
ctime
=
*
(
KaxCueTime
*
)
el
;
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
)
)
else
if
(
MKV_IS_ID
(
el
,
KaxCueTrackPositions
)
)
{
{
...
@@ -1956,21 +2031,21 @@ static void LoadCues( demux_t *p_demux )
...
@@ -1956,21 +2031,21 @@ static void LoadCues( demux_t *p_demux )
{
{
KaxCueTrack
&
ctrack
=
*
(
KaxCueTrack
*
)
el
;
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
);
idx
.
i_track
=
uint16
(
ctrack
);
}
}
else
if
(
MKV_IS_ID
(
el
,
KaxCueClusterPosition
)
)
else
if
(
MKV_IS_ID
(
el
,
KaxCueClusterPosition
)
)
{
{
KaxCueClusterPosition
&
ccpos
=
*
(
KaxCueClusterPosition
*
)
el
;
KaxCueClusterPosition
&
ccpos
=
*
(
KaxCueClusterPosition
*
)
el
;
ccpos
.
ReadData
(
p_s
ys
->
es
->
I_O
()
);
ccpos
.
ReadData
(
p_s
tream
->
es
->
I_O
()
);
idx
.
i_position
=
p_s
ys
->
segment
->
GetGlobalPosition
(
uint64
(
ccpos
)
);
idx
.
i_position
=
p_s
egment
->
segment
->
GetGlobalPosition
(
uint64
(
ccpos
)
);
}
}
else
if
(
MKV_IS_ID
(
el
,
KaxCueBlockNumber
)
)
else
if
(
MKV_IS_ID
(
el
,
KaxCueBlockNumber
)
)
{
{
KaxCueBlockNumber
&
cbnum
=
*
(
KaxCueBlockNumber
*
)
el
;
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
);
idx
.
i_block_number
=
uint32
(
cbnum
);
}
}
else
else
...
@@ -1993,11 +2068,11 @@ static void LoadCues( demux_t *p_demux )
...
@@ -1993,11 +2068,11 @@ static void LoadCues( demux_t *p_demux )
idx.i_track, idx.i_block_number );
idx.i_track, idx.i_block_number );
#endif
#endif
p_s
ys
->
i_index
++
;
p_s
egment
->
i_index
++
;
if
(
p_s
ys
->
i_index
>=
p_sys
->
i_index_max
)
if
(
p_s
egment
->
i_index
>=
p_segment
->
i_index_max
)
{
{
p_s
ys
->
i_index_max
+=
1024
;
p_s
egment
->
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
->
index
=
(
mkv_index_t
*
)
realloc
(
p_segment
->
index
,
sizeof
(
mkv_index_t
)
*
p_segment
->
i_index_max
);
}
}
#undef idx
#undef idx
}
}
...
@@ -2009,32 +2084,34 @@ static void LoadCues( demux_t *p_demux )
...
@@ -2009,32 +2084,34 @@ static void LoadCues( demux_t *p_demux )
delete
ep
;
delete
ep
;
delete
cues
;
delete
cues
;
p_s
ys
->
b_cues
=
VLC_TRUE
;
p_s
egment
->
b_cues
=
VLC_TRUE
;
msg_Dbg
(
p_demux
,
"loading cues done."
);
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
)
static
void
LoadTags
(
demux_t
*
p_demux
)
{
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
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
;
EbmlParser
*
ep
;
EbmlElement
*
el
,
*
tags
;
EbmlElement
*
el
,
*
tags
;
msg_Dbg
(
p_demux
,
"loading tags"
);
msg_Dbg
(
p_demux
,
"loading tags"
);
p_s
ys
->
in
->
setFilePointer
(
p_sys
->
i_tags_position
,
seek_beginning
);
p_s
tream
->
in
->
setFilePointer
(
p_segment
->
i_tags_position
,
seek_beginning
);
tags
=
p_s
ys
->
es
->
FindNextID
(
KaxTags
::
ClassInfos
,
0xFFFFFFFFL
);
tags
=
p_s
tream
->
es
->
FindNextID
(
KaxTags
::
ClassInfos
,
0xFFFFFFFFL
);
if
(
tags
==
NULL
)
if
(
tags
==
NULL
)
{
{
msg_Err
(
p_demux
,
"cannot load tags (broken seekhead or file)"
);
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
;
return
;
}
}
msg_Dbg
(
p_demux
,
"Tags"
);
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
)
while
(
(
el
=
ep
->
Get
()
)
!=
NULL
)
{
{
if
(
MKV_IS_ID
(
el
,
KaxTag
)
)
if
(
MKV_IS_ID
(
el
,
KaxTag
)
)
...
@@ -2137,7 +2214,7 @@ static void LoadTags( demux_t *p_demux )
...
@@ -2137,7 +2214,7 @@ static void LoadTags( demux_t *p_demux )
delete
tags
;
delete
tags
;
msg_Dbg
(
p_demux
,
"loading tags done."
);
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 )
...
@@ -2146,6 +2223,8 @@ static void LoadTags( demux_t *p_demux )
static
void
ParseSeekHead
(
demux_t
*
p_demux
,
EbmlElement
*
seekhead
)
static
void
ParseSeekHead
(
demux_t
*
p_demux
,
EbmlElement
*
seekhead
)
{
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
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
;
EbmlElement
*
el
;
EbmlMaster
*
m
;
EbmlMaster
*
m
;
unsigned
int
i
;
unsigned
int
i
;
...
@@ -2155,7 +2234,7 @@ static void ParseSeekHead( demux_t *p_demux, EbmlElement *seekhead )
...
@@ -2155,7 +2234,7 @@ static void ParseSeekHead( demux_t *p_demux, EbmlElement *seekhead )
/* Master elements */
/* Master elements */
m
=
static_cast
<
EbmlMaster
*>
(
seekhead
);
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
++
)
for
(
i
=
0
;
i
<
m
->
ListSize
();
i
++
)
{
{
...
@@ -2194,17 +2273,17 @@ static void ParseSeekHead( demux_t *p_demux, EbmlElement *seekhead )
...
@@ -2194,17 +2273,17 @@ static void ParseSeekHead( demux_t *p_demux, EbmlElement *seekhead )
if
(
id
==
KaxCues
::
ClassInfos
.
GlobalId
)
if
(
id
==
KaxCues
::
ClassInfos
.
GlobalId
)
{
{
msg_Dbg
(
p_demux
,
"| | | = cues at "
I64Fd
,
i_pos
);
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
)
else
if
(
id
==
KaxChapters
::
ClassInfos
.
GlobalId
)
{
{
msg_Dbg
(
p_demux
,
"| | | = chapters at "
I64Fd
,
i_pos
);
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
)
else
if
(
id
==
KaxTags
::
ClassInfos
.
GlobalId
)
{
{
msg_Dbg
(
p_demux
,
"| | | = tags at "
I64Fd
,
i_pos
);
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 )
...
@@ -2221,17 +2300,19 @@ static void ParseSeekHead( demux_t *p_demux, EbmlElement *seekhead )
static
void
ParseTrackEntry
(
demux_t
*
p_demux
,
EbmlMaster
*
m
)
static
void
ParseTrackEntry
(
demux_t
*
p_demux
,
EbmlMaster
*
m
)
{
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
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
;
unsigned
int
i
;
mkv_track_t
*
tk
;
mkv_track_t
*
tk
;
msg_Dbg
(
p_demux
,
"| | + Track Entry"
);
msg_Dbg
(
p_demux
,
"| | + Track Entry"
);
p_s
ys
->
i_track
++
;
p_s
egment
->
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
->
track
=
(
mkv_track_t
*
)
realloc
(
p_segment
->
track
,
sizeof
(
mkv_track_t
)
*
(
p_segment
->
i_track
+
1
)
);
/* Init the track */
/* 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
)
);
memset
(
tk
,
0
,
sizeof
(
mkv_track_t
)
);
...
@@ -2241,7 +2322,7 @@ static void ParseTrackEntry( demux_t *p_demux, EbmlMaster *m )
...
@@ -2241,7 +2322,7 @@ static void ParseTrackEntry( demux_t *p_demux, EbmlMaster *m )
tk
->
b_default
=
VLC_TRUE
;
tk
->
b_default
=
VLC_TRUE
;
tk
->
b_enabled
=
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
->
i_extra_data
=
0
;
tk
->
p_extra_data
=
NULL
;
tk
->
p_extra_data
=
NULL
;
tk
->
psz_codec
=
NULL
;
tk
->
psz_codec
=
NULL
;
...
@@ -2625,6 +2706,8 @@ static void ParseTrackEntry( demux_t *p_demux, EbmlMaster *m )
...
@@ -2625,6 +2706,8 @@ static void ParseTrackEntry( demux_t *p_demux, EbmlMaster *m )
static
void
ParseTracks
(
demux_t
*
p_demux
,
EbmlElement
*
tracks
)
static
void
ParseTracks
(
demux_t
*
p_demux
,
EbmlElement
*
tracks
)
{
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
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
;
EbmlElement
*
el
;
EbmlMaster
*
m
;
EbmlMaster
*
m
;
unsigned
int
i
;
unsigned
int
i
;
...
@@ -2634,7 +2717,7 @@ static void ParseTracks( demux_t *p_demux, EbmlElement *tracks )
...
@@ -2634,7 +2717,7 @@ static void ParseTracks( demux_t *p_demux, EbmlElement *tracks )
/* Master elements */
/* Master elements */
m
=
static_cast
<
EbmlMaster
*>
(
tracks
);
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
++
)
for
(
i
=
0
;
i
<
m
->
ListSize
();
i
++
)
{
{
...
@@ -2657,6 +2740,8 @@ static void ParseTracks( demux_t *p_demux, EbmlElement *tracks )
...
@@ -2657,6 +2740,8 @@ static void ParseTracks( demux_t *p_demux, EbmlElement *tracks )
static
void
ParseInfo
(
demux_t
*
p_demux
,
EbmlElement
*
info
)
static
void
ParseInfo
(
demux_t
*
p_demux
,
EbmlElement
*
info
)
{
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
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
;
EbmlElement
*
el
;
EbmlMaster
*
m
;
EbmlMaster
*
m
;
unsigned
int
i
;
unsigned
int
i
;
...
@@ -2666,7 +2751,7 @@ static void ParseInfo( demux_t *p_demux, EbmlElement *info )
...
@@ -2666,7 +2751,7 @@ static void ParseInfo( demux_t *p_demux, EbmlElement *info )
/* Master elements */
/* Master elements */
m
=
static_cast
<
EbmlMaster
*>
(
info
);
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
++
)
for
(
i
=
0
;
i
<
m
->
ListSize
();
i
++
)
{
{
...
@@ -2674,68 +2759,68 @@ static void ParseInfo( demux_t *p_demux, EbmlElement *info )
...
@@ -2674,68 +2759,68 @@ static void ParseInfo( demux_t *p_demux, EbmlElement *info )
if
(
MKV_IS_ID
(
l
,
KaxSegmentUID
)
)
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
)
)
else
if
(
MKV_IS_ID
(
l
,
KaxTimecodeScale
)
)
{
{
KaxTimecodeScale
&
tcs
=
*
(
KaxTimecodeScale
*
)
l
;
KaxTimecodeScale
&
tcs
=
*
(
KaxTimecodeScale
*
)
l
;
p_s
ys
->
i_timescale
=
uint64
(
tcs
);
p_s
egment
->
i_timescale
=
uint64
(
tcs
);
msg_Dbg
(
p_demux
,
"| | + TimecodeScale="
I64Fd
,
msg_Dbg
(
p_demux
,
"| | + TimecodeScale="
I64Fd
,
p_s
ys
->
i_timescale
);
p_s
egment
->
i_timescale
);
}
}
else
if
(
MKV_IS_ID
(
l
,
KaxDuration
)
)
else
if
(
MKV_IS_ID
(
l
,
KaxDuration
)
)
{
{
KaxDuration
&
dur
=
*
(
KaxDuration
*
)
l
;
KaxDuration
&
dur
=
*
(
KaxDuration
*
)
l
;
p_s
ys
->
f_duration
=
float
(
dur
);
p_s
egment
->
f_duration
=
float
(
dur
);
msg_Dbg
(
p_demux
,
"| | + Duration=%f"
,
msg_Dbg
(
p_demux
,
"| | + Duration=%f"
,
p_s
ys
->
f_duration
);
p_s
egment
->
f_duration
);
}
}
else
if
(
MKV_IS_ID
(
l
,
KaxMuxingApp
)
)
else
if
(
MKV_IS_ID
(
l
,
KaxMuxingApp
)
)
{
{
KaxMuxingApp
&
mapp
=
*
(
KaxMuxingApp
*
)
l
;
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"
,
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
)
)
else
if
(
MKV_IS_ID
(
l
,
KaxWritingApp
)
)
{
{
KaxWritingApp
&
wapp
=
*
(
KaxWritingApp
*
)
l
;
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"
,
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
)
)
else
if
(
MKV_IS_ID
(
l
,
KaxSegmentFilename
)
)
{
{
KaxSegmentFilename
&
sfn
=
*
(
KaxSegmentFilename
*
)
l
;
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"
,
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
)
)
else
if
(
MKV_IS_ID
(
l
,
KaxTitle
)
)
{
{
KaxTitle
&
title
=
*
(
KaxTitle
*
)
l
;
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
)
)
if
(
MKV_IS_ID
(
l
,
KaxSegmentFamily
)
)
{
{
KaxSegmentFamily
*
uid
=
static_cast
<
KaxSegmentFamily
*>
(
l
);
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
()
);
msg_Dbg
(
p_demux
,
"| | + family=%d"
,
*
(
uint32
*
)
uid
->
GetBuffer
()
);
}
}
...
@@ -2753,8 +2838,8 @@ static void ParseInfo( demux_t *p_demux, EbmlElement *info )
...
@@ -2753,8 +2838,8 @@ static void ParseInfo( demux_t *p_demux, EbmlElement *info )
asctime_r
(
&
tmres
,
buffer
)
)
asctime_r
(
&
tmres
,
buffer
)
)
{
{
buffer
[
strlen
(
buffer
)
-
1
]
=
'\0'
;
buffer
[
strlen
(
buffer
)
-
1
]
=
'\0'
;
p_s
ys
->
psz_date_utc
=
strdup
(
buffer
);
p_s
egment
->
psz_date_utc
=
strdup
(
buffer
);
msg_Dbg
(
p_demux
,
"| | + Date=%s"
,
p_s
ys
->
psz_date_utc
);
msg_Dbg
(
p_demux
,
"| | + Date=%s"
,
p_s
egment
->
psz_date_utc
);
}
}
}
}
#endif
#endif
...
@@ -2764,7 +2849,7 @@ static void ParseInfo( demux_t *p_demux, EbmlElement *info )
...
@@ -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
...
@@ -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
)
static
void
ParseChapters
(
demux_t
*
p_demux
,
EbmlElement
*
chapters
)
{
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
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
;
EbmlElement
*
el
;
EbmlMaster
*
m
;
EbmlMaster
*
m
;
unsigned
int
i
;
unsigned
int
i
;
...
@@ -2875,7 +2962,7 @@ static void ParseChapters( demux_t *p_demux, EbmlElement *chapters )
...
@@ -2875,7 +2962,7 @@ static void ParseChapters( demux_t *p_demux, EbmlElement *chapters )
/* Master elements */
/* Master elements */
m
=
static_cast
<
EbmlMaster
*>
(
chapters
);
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
++
)
for
(
i
=
0
;
i
<
m
->
ListSize
();
i
++
)
{
{
...
@@ -2909,14 +2996,14 @@ static void ParseChapters( demux_t *p_demux, EbmlElement *chapters )
...
@@ -2909,14 +2996,14 @@ static void ParseChapters( demux_t *p_demux, EbmlElement *chapters )
else
if
(
MKV_IS_ID
(
l
,
KaxEditionFlagDefault
)
)
else
if
(
MKV_IS_ID
(
l
,
KaxEditionFlagDefault
)
)
{
{
if
(
uint8
(
*
static_cast
<
KaxEditionFlagDefault
*>
(
l
))
!=
0
)
if
(
uint8
(
*
static_cast
<
KaxEditionFlagDefault
*>
(
l
))
!=
0
)
i_default_edition
=
p_s
ys
->
editions
.
size
();
i_default_edition
=
p_s
egment
->
editions
.
size
();
}
}
else
else
{
{
msg_Dbg
(
p_demux
,
"| | | + Unknown (%s)"
,
typeid
(
*
l
).
name
()
);
msg_Dbg
(
p_demux
,
"| | | + Unknown (%s)"
,
typeid
(
*
l
).
name
()
);
}
}
}
}
p_s
ys
->
editions
.
push_back
(
edition
);
p_s
egment
->
editions
.
push_back
(
edition
);
}
}
else
else
{
{
...
@@ -2924,19 +3011,19 @@ static void ParseChapters( demux_t *p_demux, EbmlElement *chapters )
...
@@ -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 */
/* 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
)
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 )
...
@@ -2946,34 +3033,36 @@ static void ParseChapters( demux_t *p_demux, EbmlElement *chapters )
static
void
InformationCreate
(
demux_t
*
p_demux
)
static
void
InformationCreate
(
demux_t
*
p_demux
)
{
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
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
;
int
i_track
;
p_sys
->
meta
=
vlc_meta_New
();
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
();
vlc_meta_t
*
mtk
=
vlc_meta_New
();
p_sys
->
meta
->
track
=
(
vlc_meta_t
**
)
realloc
(
p_sys
->
meta
->
track
,
p_sys
->
meta
->
track
=
(
vlc_meta_t
**
)
realloc
(
p_sys
->
meta
->
track
,
...
@@ -3002,7 +3091,7 @@ static void InformationCreate( demux_t *p_demux )
...
@@ -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
;
vlc_bool_t
b_seekable
;
...
@@ -3022,19 +3111,21 @@ static void InformationCreate( demux_t *p_demux )
...
@@ -3022,19 +3111,21 @@ static void InformationCreate( demux_t *p_demux )
static
void
IndexAppendCluster
(
demux_t
*
p_demux
,
KaxCluster
*
cluster
)
static
void
IndexAppendCluster
(
demux_t
*
p_demux
,
KaxCluster
*
cluster
)
{
{
demux_sys_t
*
p_sys
=
p_demux
->
p_sys
;
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_track
=
-
1
;
idx
.
i_block_number
=
-
1
;
idx
.
i_block_number
=
-
1
;
idx
.
i_position
=
cluster
->
GetElementPosition
();
idx
.
i_position
=
cluster
->
GetElementPosition
();
idx
.
i_time
=
-
1
;
idx
.
i_time
=
-
1
;
idx
.
b_key
=
VLC_TRUE
;
idx
.
b_key
=
VLC_TRUE
;
p_s
ys
->
i_index
++
;
p_s
egment
->
i_index
++
;
if
(
p_s
ys
->
i_index
>=
p_sys
->
i_index_max
)
if
(
p_s
egment
->
i_index
>=
p_segment
->
i_index_max
)
{
{
p_s
ys
->
i_index_max
+=
1024
;
p_s
egment
->
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
->
index
=
(
mkv_index_t
*
)
realloc
(
p_segment
->
index
,
sizeof
(
mkv_index_t
)
*
p_segment
->
i_index_max
);
}
}
#undef idx
#undef idx
}
}
...
...
This diff is collapsed.
Click to expand it.
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