Skip to content
项目
群组
代码片段
帮助
当前项目
正在载入...
登录 / 注册
切换导航面板
Z
ZLMediaKit
概览
Overview
Details
Activity
Cycle Analytics
版本库
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
问题
0
Issues
0
列表
Board
标记
里程碑
合并请求
0
Merge Requests
0
CI / CD
CI / CD
流水线
作业
日程表
图表
维基
Wiki
代码片段
Snippets
成员
Collapse sidebar
Close sidebar
活动
图像
聊天
创建新问题
作业
提交
Issue Boards
Open sidebar
张翔宇
ZLMediaKit
Commits
af2b1246
Commit
af2b1246
authored
Apr 26, 2021
by
xia-chu
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
统一帧合并逻辑
parent
d005ccf3
隐藏空白字符变更
内嵌
并排
正在显示
12 个修改的文件
包含
174 行增加
和
186 行删除
+174
-186
src/Extension/Frame.cpp
+98
-0
src/Extension/Frame.h
+28
-0
src/Record/MP4.cpp
+0
-8
src/Record/MP4Muxer.cpp
+16
-40
src/Record/MP4Muxer.h
+1
-1
src/Record/TsMuxer.cpp
+12
-28
src/Record/TsMuxer.h
+3
-3
src/Rtp/Decoder.cpp
+2
-30
src/Rtp/Decoder.h
+1
-13
src/Rtp/PSEncoder.cpp
+11
-26
src/Rtp/PSEncoder.h
+1
-1
tests/test_player.cpp
+1
-36
没有找到文件。
src/Extension/Frame.cpp
查看文件 @
af2b1246
...
...
@@ -140,4 +140,102 @@ const char *CodecInfo::getCodecName() {
TrackType
CodecInfo
::
getTrackType
()
{
return
mediakit
::
getTrackType
(
getCodecId
());
}
static
size_t
constexpr
kMaxFrameCacheSize
=
100
;
bool
FrameMerger
::
willFlush
(
const
Frame
::
Ptr
&
frame
)
const
{
if
(
_frameCached
.
empty
())
{
return
false
;
}
switch
(
_type
)
{
case
none
:
{
//frame不是完整的帧,我们合并为一帧
bool
new_frame
=
false
;
switch
(
frame
->
getCodecId
())
{
case
CodecH264
:
case
CodecH265
:
{
//如果是新的一帧,前面的缓存需要输出
new_frame
=
frame
->
prefixSize
();
break
;
}
default
:
break
;
}
return
new_frame
||
_frameCached
.
back
()
->
dts
()
!=
frame
->
dts
()
||
_frameCached
.
size
()
>
kMaxFrameCacheSize
;
}
case
mp4_nal_size
:
case
h264_prefix
:
{
if
(
_frameCached
.
back
()
->
dts
()
!=
frame
->
dts
())
{
//时间戳变化了
return
true
;
}
if
(
frame
->
getCodecId
()
==
CodecH264
&&
H264_TYPE
(
frame
->
data
()[
frame
->
prefixSize
()])
==
H264Frame
::
NAL_B_P
)
{
//如果是264的b/p帧,那么也刷新输出
return
true
;
}
return
_frameCached
.
size
()
>
kMaxFrameCacheSize
;
}
default
:
/*不可达*/
assert
(
0
);
return
true
;
}
}
void
FrameMerger
::
doMerge
(
BufferLikeString
&
merged
,
const
Frame
::
Ptr
&
frame
)
const
{
switch
(
_type
)
{
case
none
:
{
merged
.
append
(
frame
->
data
(),
frame
->
size
());
break
;
}
case
h264_prefix
:
{
if
(
frame
->
prefixSize
())
{
merged
.
append
(
frame
->
data
(),
frame
->
size
());
}
else
{
merged
.
append
(
"
\x00\x00\x00\x01
"
,
4
);
merged
.
append
(
frame
->
data
(),
frame
->
size
());
}
break
;
}
case
mp4_nal_size
:
{
uint32_t
nalu_size
=
(
uint32_t
)
(
frame
->
size
()
-
frame
->
prefixSize
());
nalu_size
=
htonl
(
nalu_size
);
merged
.
append
((
char
*
)
&
nalu_size
,
4
);
merged
.
append
(
frame
->
data
()
+
frame
->
prefixSize
(),
frame
->
size
()
-
frame
->
prefixSize
());
break
;
}
default
:
/*不可达*/
assert
(
0
);
break
;
}
}
void
FrameMerger
::
inputFrame
(
const
Frame
::
Ptr
&
frame
,
const
onOutput
&
cb
)
{
if
(
willFlush
(
frame
))
{
Frame
::
Ptr
back
=
_frameCached
.
back
();
Buffer
::
Ptr
merged_frame
=
back
;
bool
have_idr
=
back
->
keyFrame
();
if
(
_frameCached
.
size
()
!=
1
||
_type
==
mp4_nal_size
)
{
//在MP4模式下,一帧数据也需要在前添加nalu_size
BufferLikeString
merged
;
merged
.
reserve
(
back
->
size
()
+
1024
);
_frameCached
.
for_each
([
&
](
const
Frame
::
Ptr
&
frame
)
{
doMerge
(
merged
,
frame
);
if
(
frame
->
keyFrame
())
{
have_idr
=
true
;
}
});
merged_frame
=
std
::
make_shared
<
BufferOffset
<
BufferLikeString
>
>
(
std
::
move
(
merged
));
}
cb
(
back
->
dts
(),
back
->
pts
(),
merged_frame
,
have_idr
);
_frameCached
.
clear
();
}
_frameCached
.
emplace_back
(
Frame
::
getCacheAbleFrame
(
frame
));
}
FrameMerger
::
FrameMerger
(
int
type
)
{
_type
=
type
;
}
void
FrameMerger
::
clear
()
{
_frameCached
.
clear
();
}
}
//namespace mediakit
src/Extension/Frame.h
查看文件 @
af2b1246
...
...
@@ -428,5 +428,32 @@ private:
Buffer
::
Ptr
_buf
;
};
/**
* 合并一些时间戳相同的frame
*/
class
FrameMerger
{
public
:
using
onOutput
=
function
<
void
(
uint32_t
dts
,
uint32_t
pts
,
const
Buffer
::
Ptr
&
buffer
,
bool
have_idr
)
>
;
enum
{
none
=
0
,
h264_prefix
,
mp4_nal_size
,
};
FrameMerger
(
int
type
);
~
FrameMerger
()
=
default
;
void
clear
();
void
inputFrame
(
const
Frame
::
Ptr
&
frame
,
const
onOutput
&
cb
);
private
:
bool
willFlush
(
const
Frame
::
Ptr
&
frame
)
const
;
void
doMerge
(
BufferLikeString
&
buffer
,
const
Frame
::
Ptr
&
frame
)
const
;
private
:
int
_type
;
List
<
Frame
::
Ptr
>
_frameCached
;
};
}
//namespace mediakit
#endif //ZLMEDIAKIT_FRAME_H
\ No newline at end of file
src/Record/MP4.cpp
查看文件 @
af2b1246
...
...
@@ -80,14 +80,6 @@ int mp4_writer_write(mp4_writer_t* mp4, int track, const void* data, size_t byte
}
}
int
mp4_writer_write_l
(
mp4_writer_t
*
mp4
,
int
track
,
const
void
*
data
,
size_t
bytes
,
int64_t
pts
,
int64_t
dts
,
int
flags
,
int
add_nalu_size
){
if
(
mp4
->
is_fmp4
)
{
return
fmp4_writer_write_l
(
mp4
->
u
.
fmp4
,
track
,
data
,
bytes
,
pts
,
dts
,
flags
,
add_nalu_size
);
}
else
{
return
mov_writer_write_l
(
mp4
->
u
.
mov
,
track
,
data
,
bytes
,
pts
,
dts
,
flags
,
add_nalu_size
);
}
}
int
mp4_writer_save_segment
(
mp4_writer_t
*
mp4
){
if
(
mp4
->
is_fmp4
)
{
return
fmp4_writer_save_segment
(
mp4
->
u
.
fmp4
);
...
...
src/Record/MP4Muxer.cpp
查看文件 @
af2b1246
...
...
@@ -60,7 +60,7 @@ void MP4MuxerInterface::resetTracks() {
_started
=
false
;
_have_video
=
false
;
_mov_writter
=
nullptr
;
_frame
Cached
.
clear
();
_frame
_merger
.
clear
();
_codec_to_trackid
.
clear
();
}
...
...
@@ -92,47 +92,22 @@ void MP4MuxerInterface::inputFrame(const Frame::Ptr &frame) {
break
;
}
}
case
CodecH265
:
{
//这里的代码逻辑是让SPS、PPS、IDR这些时间戳相同的帧打包到一起当做一个帧处理,
if
(
!
_frameCached
.
empty
()
&&
_frameCached
.
back
()
->
dts
()
!=
frame
->
dts
())
{
Frame
::
Ptr
back
=
_frameCached
.
back
();
//求相对时间戳
track_info
.
stamp
.
revise
(
back
->
dts
(),
back
->
pts
(),
dts_out
,
pts_out
);
if
(
_frameCached
.
size
()
!=
1
)
{
//缓存中有多帧,需要按照mp4格式合并一起
BufferLikeString
merged
;
merged
.
reserve
(
back
->
size
()
+
1024
);
_frameCached
.
for_each
([
&
](
const
Frame
::
Ptr
&
frame
)
{
uint32_t
nalu_size
=
(
uint32_t
)(
frame
->
size
()
-
frame
->
prefixSize
());
nalu_size
=
htonl
(
nalu_size
);
merged
.
append
((
char
*
)
&
nalu_size
,
4
);
merged
.
append
(
frame
->
data
()
+
frame
->
prefixSize
(),
frame
->
size
()
-
frame
->
prefixSize
());
});
mp4_writer_write
(
_mov_writter
.
get
(),
track_info
.
track_id
,
merged
.
data
(),
merged
.
size
(),
pts_out
,
dts_out
,
back
->
keyFrame
()
?
MOV_AV_FLAG_KEYFREAME
:
0
);
}
else
{
//缓存中只有一帧视频
mp4_writer_write_l
(
_mov_writter
.
get
(),
track_info
.
track_id
,
back
->
data
()
+
back
->
prefixSize
(),
back
->
size
()
-
back
->
prefixSize
(),
pts_out
,
dts_out
,
back
->
keyFrame
()
?
MOV_AV_FLAG_KEYFREAME
:
0
,
1
/*需要生成头4个字节的MP4格式start code*/
);
}
_frameCached
.
clear
();
}
//缓存帧,时间戳相同的帧合并一起写入mp4
_frameCached
.
emplace_back
(
Frame
::
getCacheAbleFrame
(
frame
));
}
_frame_merger
.
inputFrame
(
frame
,
[
&
](
uint32_t
dts
,
uint32_t
pts
,
const
Buffer
::
Ptr
&
buffer
,
bool
have_idr
)
{
track_info
.
stamp
.
revise
(
dts
,
pts
,
dts_out
,
pts_out
);
mp4_writer_write
(
_mov_writter
.
get
(),
track_info
.
track_id
,
buffer
->
data
(),
buffer
->
size
(),
pts_out
,
dts_out
,
have_idr
?
MOV_AV_FLAG_KEYFREAME
:
0
);
});
break
;
}
default
:
{
track_info
.
stamp
.
revise
(
frame
->
dts
(),
frame
->
pts
(),
dts_out
,
pts_out
);
mp4_writer_write
(
_mov_writter
.
get
(),
...
...
@@ -142,8 +117,9 @@ void MP4MuxerInterface::inputFrame(const Frame::Ptr &frame) {
pts_out
,
dts_out
,
frame
->
keyFrame
()
?
MOV_AV_FLAG_KEYFREAME
:
0
);
}
break
;
}
}
}
...
...
src/Record/MP4Muxer.h
查看文件 @
af2b1246
...
...
@@ -72,8 +72,8 @@ private:
int
track_id
=
-
1
;
Stamp
stamp
;
};
List
<
Frame
::
Ptr
>
_frameCached
;
unordered_map
<
int
,
track_info
>
_codec_to_trackid
;
FrameMerger
_frame_merger
{
FrameMerger
::
mp4_nal_size
};
};
class
MP4Muxer
:
public
MP4MuxerInterface
{
...
...
src/Record/TsMuxer.cpp
查看文件 @
af2b1246
...
...
@@ -103,32 +103,14 @@ void TsMuxer::inputFrame(const Frame::Ptr &frame) {
case
CodecH265
:
{
//这里的代码逻辑是让SPS、PPS、IDR这些时间戳相同的帧打包到一起当做一个帧处理,
if
(
!
_frameCached
.
empty
()
&&
_frameCached
.
back
()
->
dts
()
!=
frame
->
dts
())
{
Frame
::
Ptr
back
=
_frameCached
.
back
();
Buffer
::
Ptr
merged_frame
=
back
;
if
(
_frameCached
.
size
()
!=
1
)
{
BufferLikeString
merged
;
merged
.
reserve
(
back
->
size
()
+
1024
);
_frameCached
.
for_each
([
&
](
const
Frame
::
Ptr
&
frame
)
{
if
(
frame
->
prefixSize
())
{
merged
.
append
(
frame
->
data
(),
frame
->
size
());
}
else
{
merged
.
append
(
"
\x00\x00\x00\x01
"
,
4
);
merged
.
append
(
frame
->
data
(),
frame
->
size
());
}
if
(
frame
->
keyFrame
())
{
_is_idr_fast_packet
=
true
;
}
});
merged_frame
=
std
::
make_shared
<
BufferOffset
<
BufferLikeString
>
>
(
std
::
move
(
merged
));
}
track_info
.
stamp
.
revise
(
back
->
dts
(),
back
->
pts
(),
dts_out
,
pts_out
);
_frame_merger
.
inputFrame
(
frame
,
[
&
](
uint32_t
dts
,
uint32_t
pts
,
const
Buffer
::
Ptr
&
buffer
,
bool
have_idr
){
track_info
.
stamp
.
revise
(
dts
,
pts
,
dts_out
,
pts_out
);
//取视频时间戳为TS的时间戳
_timestamp
=
(
uint32_t
)
dts_out
;
mpeg_ts_write
(
_context
,
track_info
.
track_id
,
back
->
keyFrame
()
?
0x0001
:
0
,
pts_out
*
90LL
,
dts_out
*
90LL
,
merged_frame
->
data
(),
merged_frame
->
size
())
;
_frameCached
.
clear
();
}
_frameCached
.
emplace_back
(
Frame
::
getCacheAbleFrame
(
frame
)
);
_timestamp
=
(
uint32_t
)
dts_out
;
_is_idr_fast_packet
=
have_idr
;
mpeg_ts_write
(
_context
,
track_info
.
track_id
,
have_idr
?
0x0001
:
0
,
pts_out
*
90LL
,
dts_out
*
90LL
,
buffer
->
data
(),
buffer
->
size
());
}
);
break
;
}
...
...
@@ -141,11 +123,12 @@ void TsMuxer::inputFrame(const Frame::Ptr &frame) {
default
:
{
track_info
.
stamp
.
revise
(
frame
->
dts
(),
frame
->
pts
(),
dts_out
,
pts_out
);
if
(
!
_have_video
)
{
if
(
!
_have_video
)
{
//没有视频时,才以音频时间戳为TS的时间戳
_timestamp
=
(
uint32_t
)
dts_out
;
_timestamp
=
(
uint32_t
)
dts_out
;
}
mpeg_ts_write
(
_context
,
track_info
.
track_id
,
frame
->
keyFrame
()
?
0x0001
:
0
,
pts_out
*
90LL
,
dts_out
*
90LL
,
frame
->
data
(),
frame
->
size
());
mpeg_ts_write
(
_context
,
track_info
.
track_id
,
frame
->
keyFrame
()
?
0x0001
:
0
,
pts_out
*
90LL
,
dts_out
*
90LL
,
frame
->
data
(),
frame
->
size
());
break
;
}
}
...
...
@@ -187,6 +170,7 @@ void TsMuxer::uninit() {
_context
=
nullptr
;
}
_codec_to_trackid
.
clear
();
_frame_merger
.
clear
();
}
}
//namespace mediakit
...
...
src/Record/TsMuxer.h
查看文件 @
af2b1246
...
...
@@ -59,6 +59,8 @@ private:
void
stampSync
();
private
:
bool
_have_video
=
false
;
bool
_is_idr_fast_packet
=
false
;
void
*
_context
=
nullptr
;
char
_tsbuf
[
188
];
uint32_t
_timestamp
=
0
;
...
...
@@ -67,9 +69,7 @@ private:
Stamp
stamp
;
};
unordered_map
<
int
,
track_info
>
_codec_to_trackid
;
List
<
Frame
::
Ptr
>
_frameCached
;
bool
_is_idr_fast_packet
=
false
;
bool
_have_video
=
false
;
FrameMerger
_frame_merger
{
FrameMerger
::
h264_prefix
};
};
}
//namespace mediakit
...
...
src/Rtp/Decoder.cpp
查看文件 @
af2b1246
...
...
@@ -100,34 +100,6 @@ static const char *getCodecName(int codec_id) {
}
}
void
FrameMerger
::
inputFrame
(
const
Frame
::
Ptr
&
frame
,
const
function
<
void
(
uint32_t
dts
,
uint32_t
pts
,
const
Buffer
::
Ptr
&
buffer
)
>
&
cb
){
bool
flush
=
false
;
switch
(
frame
->
getCodecId
())
{
case
CodecH264
:
case
CodecH265
:{
//如果是新的一帧,前面的缓存需要输出
flush
=
frame
->
prefixSize
();
break
;
}
default
:
break
;
}
if
(
!
_frameCached
.
empty
()
&&
(
flush
||
_frameCached
.
back
()
->
dts
()
!=
frame
->
dts
()))
{
Frame
::
Ptr
back
=
_frameCached
.
back
();
Buffer
::
Ptr
merged_frame
=
back
;
if
(
_frameCached
.
size
()
!=
1
){
BufferLikeString
merged
;
merged
.
reserve
(
back
->
size
()
+
1024
);
_frameCached
.
for_each
([
&
](
const
Frame
::
Ptr
&
frame
){
merged
.
append
(
frame
->
data
(),
frame
->
size
());
});
merged_frame
=
std
::
make_shared
<
BufferOffset
<
BufferLikeString
>
>
(
std
::
move
(
merged
));
}
cb
(
back
->
dts
(),
back
->
pts
(),
merged_frame
);
_frameCached
.
clear
();
}
_frameCached
.
emplace_back
(
Frame
::
getCacheAbleFrame
(
frame
));
}
void
DecoderImp
::
onStream
(
int
stream
,
int
codecid
,
const
void
*
extra
,
size_t
bytes
,
int
finish
){
switch
(
codecid
)
{
case
PSI_STREAM_H264
:
{
...
...
@@ -188,7 +160,7 @@ void DecoderImp::onDecode(int stream,int codecid,int flags,int64_t pts,int64_t d
switch
(
codecid
)
{
case
PSI_STREAM_H264
:
{
auto
frame
=
std
::
make_shared
<
H264FrameNoCacheAble
>
((
char
*
)
data
,
bytes
,
(
uint32_t
)
dts
,
(
uint32_t
)
pts
,
prefixSize
((
char
*
)
data
,
bytes
));
_merger
.
inputFrame
(
frame
,[
this
](
uint32_t
dts
,
uint32_t
pts
,
const
Buffer
::
Ptr
&
buffer
)
{
_merger
.
inputFrame
(
frame
,[
this
](
uint32_t
dts
,
uint32_t
pts
,
const
Buffer
::
Ptr
&
buffer
,
bool
)
{
onFrame
(
std
::
make_shared
<
FrameWrapper
<
H264FrameNoCacheAble
>
>
(
buffer
,
dts
,
pts
,
prefixSize
(
buffer
->
data
(),
buffer
->
size
()),
0
));
});
break
;
...
...
@@ -196,7 +168,7 @@ void DecoderImp::onDecode(int stream,int codecid,int flags,int64_t pts,int64_t d
case
PSI_STREAM_H265
:
{
auto
frame
=
std
::
make_shared
<
H265FrameNoCacheAble
>
((
char
*
)
data
,
bytes
,
(
uint32_t
)
dts
,
(
uint32_t
)
pts
,
prefixSize
((
char
*
)
data
,
bytes
));
_merger
.
inputFrame
(
frame
,[
this
](
uint32_t
dts
,
uint32_t
pts
,
const
Buffer
::
Ptr
&
buffer
)
{
_merger
.
inputFrame
(
frame
,[
this
](
uint32_t
dts
,
uint32_t
pts
,
const
Buffer
::
Ptr
&
buffer
,
bool
)
{
onFrame
(
std
::
make_shared
<
FrameWrapper
<
H265FrameNoCacheAble
>
>
(
buffer
,
dts
,
pts
,
prefixSize
(
buffer
->
data
(),
buffer
->
size
()),
0
));
});
break
;
...
...
src/Rtp/Decoder.h
查看文件 @
af2b1246
...
...
@@ -34,18 +34,6 @@ protected:
virtual
~
Decoder
()
=
default
;
};
/**
* 合并一些时间戳相同的frame
*/
class
FrameMerger
{
public
:
FrameMerger
()
=
default
;
~
FrameMerger
()
=
default
;
void
inputFrame
(
const
Frame
::
Ptr
&
frame
,
const
function
<
void
(
uint32_t
dts
,
uint32_t
pts
,
const
Buffer
::
Ptr
&
buffer
)
>
&
cb
);
private
:
List
<
Frame
::
Ptr
>
_frameCached
;
};
class
DecoderImp
{
public
:
typedef
enum
{
...
...
@@ -71,7 +59,7 @@ private:
private
:
Decoder
::
Ptr
_decoder
;
MediaSinkInterface
*
_sink
;
FrameMerger
_merger
;
FrameMerger
_merger
{
FrameMerger
::
none
}
;
Ticker
_last_unsported_print
;
};
...
...
src/Rtp/PSEncoder.cpp
查看文件 @
af2b1246
...
...
@@ -123,33 +123,18 @@ void PSEncoder::inputFrame(const Frame::Ptr &frame) {
break
;
}
}
case
CodecH265
:
{
//这里的代码逻辑是让SPS、PPS、IDR这些时间戳相同的帧打包到一起当做一个帧处理,
if
(
!
_frameCached
.
empty
()
&&
_frameCached
.
back
()
->
dts
()
!=
frame
->
dts
())
{
Frame
::
Ptr
back
=
_frameCached
.
back
();
Buffer
::
Ptr
merged_frame
=
back
;
if
(
_frameCached
.
size
()
!=
1
)
{
BufferLikeString
merged
;
merged
.
reserve
(
back
->
size
()
+
1024
);
_frameCached
.
for_each
([
&
](
const
Frame
::
Ptr
&
frame
)
{
if
(
frame
->
prefixSize
())
{
merged
.
append
(
frame
->
data
(),
frame
->
size
());
}
else
{
merged
.
append
(
"
\x00\x00\x00\x01
"
,
4
);
merged
.
append
(
frame
->
data
(),
frame
->
size
());
}
});
merged_frame
=
std
::
make_shared
<
BufferOffset
<
BufferLikeString
>
>
(
std
::
move
(
merged
));
}
track_info
.
stamp
.
revise
(
back
->
dts
(),
back
->
pts
(),
dts_out
,
pts_out
);
_timestamp
=
(
uint32_t
)
dts_out
;
ps_muxer_input
(
_muxer
.
get
(),
track_info
.
track_id
,
back
->
keyFrame
()
?
0x0001
:
0
,
pts_out
*
90LL
,
dts_out
*
90LL
,
merged_frame
->
data
(),
merged_frame
->
size
());
_frameCached
.
clear
();
}
_frameCached
.
emplace_back
(
Frame
::
getCacheAbleFrame
(
frame
));
}
_frame_merger
.
inputFrame
(
frame
,
[
&
](
uint32_t
dts
,
uint32_t
pts
,
const
Buffer
::
Ptr
&
buffer
,
bool
have_idr
)
{
track_info
.
stamp
.
revise
(
dts
,
pts
,
dts_out
,
pts_out
);
//取视频时间戳为TS的时间戳
_timestamp
=
(
uint32_t
)
dts_out
;
ps_muxer_input
(
_muxer
.
get
(),
track_info
.
track_id
,
have_idr
?
0x0001
:
0
,
pts_out
*
90LL
,
dts_out
*
90LL
,
buffer
->
data
(),
buffer
->
size
());
});
break
;
}
case
CodecAAC
:
{
if
(
frame
->
prefixSize
()
==
0
)
{
...
...
@@ -160,11 +145,11 @@ void PSEncoder::inputFrame(const Frame::Ptr &frame) {
default
:
{
track_info
.
stamp
.
revise
(
frame
->
dts
(),
frame
->
pts
(),
dts_out
,
pts_out
);
_timestamp
=
(
uint32_t
)
dts_out
;
_timestamp
=
(
uint32_t
)
dts_out
;
ps_muxer_input
(
_muxer
.
get
(),
track_info
.
track_id
,
frame
->
keyFrame
()
?
0x0001
:
0
,
pts_out
*
90LL
,
dts_out
*
90LL
,
frame
->
data
(),
frame
->
size
());
}
break
;
}
}
}
...
...
src/Rtp/PSEncoder.h
查看文件 @
af2b1246
...
...
@@ -61,9 +61,9 @@ private:
private
:
uint32_t
_timestamp
=
0
;
BufferRaw
::
Ptr
_buffer
;
List
<
Frame
::
Ptr
>
_frameCached
;
std
::
shared_ptr
<
struct
ps_muxer_t
>
_muxer
;
unordered_map
<
int
,
track_info
>
_codec_to_trackid
;
FrameMerger
_frame_merger
{
FrameMerger
::
h264_prefix
};
};
class
PSEncoderImp
:
public
PSEncoder
{
...
...
tests/test_player.cpp
查看文件 @
af2b1246
...
...
@@ -23,41 +23,6 @@ using namespace std;
using
namespace
toolkit
;
using
namespace
mediakit
;
/**
* 合并一些时间戳相同的frame
*/
class
FrameMerger
{
public
:
FrameMerger
()
=
default
;
virtual
~
FrameMerger
()
=
default
;
void
inputFrame
(
const
Frame
::
Ptr
&
frame
,
const
function
<
void
(
uint32_t
dts
,
uint32_t
pts
,
const
Buffer
::
Ptr
&
buffer
)
>
&
cb
){
if
(
!
_frameCached
.
empty
()
&&
_frameCached
.
back
()
->
dts
()
!=
frame
->
dts
())
{
Frame
::
Ptr
back
=
_frameCached
.
back
();
Buffer
::
Ptr
merged_frame
=
back
;
if
(
_frameCached
.
size
()
!=
1
){
string
merged
;
_frameCached
.
for_each
([
&
](
const
Frame
::
Ptr
&
frame
){
if
(
frame
->
prefixSize
()){
merged
.
append
(
frame
->
data
(),
frame
->
size
());
}
else
{
merged
.
append
(
"
\x00\x00\x00\x01
"
,
4
);
merged
.
append
(
frame
->
data
(),
frame
->
size
());
}
});
merged_frame
=
std
::
make_shared
<
BufferString
>
(
std
::
move
(
merged
));
}
cb
(
back
->
dts
(),
back
->
pts
(),
merged_frame
);
_frameCached
.
clear
();
}
_frameCached
.
emplace_back
(
Frame
::
getCacheAbleFrame
(
frame
));
}
private
:
List
<
Frame
::
Ptr
>
_frameCached
;
};
#ifdef WIN32
#include <TCHAR.h>
...
...
@@ -129,7 +94,7 @@ int main(int argc, char *argv[]) {
displayer
.
set
<
YuvDisplayer
>
(
nullptr
,
url
);
}
if
(
!
merger
){
merger
.
set
<
FrameMerger
>
();
merger
.
set
<
FrameMerger
>
(
FrameMerger
::
h264_prefix
);
}
merger
.
get
<
FrameMerger
>
().
inputFrame
(
frame
,[
&
](
uint32_t
dts
,
uint32_t
pts
,
const
Buffer
::
Ptr
&
buffer
){
AVFrame
*
pFrame
=
nullptr
;
...
...
编写
预览
Markdown
格式
0%
重试
或
添加新文件
添加附件
取消
您添加了
0
人
到此讨论。请谨慎行事。
请先完成此评论的编辑!
取消
请
注册
或者
登录
后发表评论