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
85f05adc
Commit
85f05adc
authored
Jun 24, 2019
by
xiongziliang
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
优化代码,提升可读性
parent
fec7fe5b
隐藏空白字符变更
内嵌
并排
正在显示
3 个修改的文件
包含
117 行增加
和
97 行删除
+117
-97
3rdpart/ZLToolKit
+1
-1
src/Rtsp/RtpReceiver.cpp
+95
-83
src/Rtsp/RtpReceiver.h
+21
-13
没有找到文件。
ZLToolKit
@
dd8928dd
Subproject commit
f922d96f98108186e8d9bb9b468c83c0b37d9558
Subproject commit
dd8928ddaecb71cbb5da447dbaf5af29b24b2e7d
src/Rtsp/RtpReceiver.cpp
查看文件 @
85f05adc
...
@@ -28,152 +28,164 @@
...
@@ -28,152 +28,164 @@
#include "RtpReceiver.h"
#include "RtpReceiver.h"
#define POP_HEAD(trackidx) \
#define POP_HEAD(trackidx) \
auto it = _
amapRtpSort
[trackidx].begin(); \
auto it = _
rtp_sort_cache_map
[trackidx].begin(); \
onRtpSorted(it->second, trackidx); \
onRtpSorted(it->second, trackidx); \
_
amapRtpSort
[trackidx].erase(it);
_
rtp_sort_cache_map
[trackidx].erase(it);
#
define AV_RB16(x) \
#define AV_RB16(x) \
((((const uint8_t*)(x))[0] << 8) | \
((((const uint8_t*)(x))[0] << 8) | \
((const uint8_t*)(x))[1])
((const uint8_t*)(x))[1])
#define RTP_MAX_SIZE (10 * 1024)
namespace
mediakit
{
namespace
mediakit
{
RtpReceiver
::
RtpReceiver
()
{}
RtpReceiver
::
RtpReceiver
()
{}
RtpReceiver
::~
RtpReceiver
()
{}
RtpReceiver
::~
RtpReceiver
()
{}
bool
RtpReceiver
::
handleOneRtp
(
int
iTrackidx
,
SdpTrack
::
Ptr
&
track
,
unsigned
char
*
pucData
,
unsigned
int
uiL
en
)
{
bool
RtpReceiver
::
handleOneRtp
(
int
track_index
,
SdpTrack
::
Ptr
&
track
,
unsigned
char
*
rtp_raw_ptr
,
unsigned
int
rtp_raw_l
en
)
{
auto
pt_ptr
=
_pktP
ool
.
obtain
();
auto
rtp_ptr
=
_rtp_p
ool
.
obtain
();
auto
&
rtp
pt
=*
pt
_ptr
;
auto
&
rtp
=
*
rtp
_ptr
;
auto
length
=
uiL
en
+
4
;
auto
length
=
rtp_raw_l
en
+
4
;
rtp
pt
.
interleaved
=
track
->
_interleaved
;
rtp
.
interleaved
=
track
->
_interleaved
;
rtp
pt
.
mark
=
pucData
[
1
]
>>
7
;
rtp
.
mark
=
rtp_raw_ptr
[
1
]
>>
7
;
rtp
pt
.
PT
=
pucData
[
1
]
&
0x7F
;
rtp
.
PT
=
rtp_raw_ptr
[
1
]
&
0x7F
;
//序列号
//序列号
memcpy
(
&
rtp
pt
.
sequence
,
pucData
+
2
,
2
);
//内存对齐
memcpy
(
&
rtp
.
sequence
,
rtp_raw_ptr
+
2
,
2
);
//内存对齐
rtp
pt
.
sequence
=
ntohs
(
rtppt
.
sequence
);
rtp
.
sequence
=
ntohs
(
rtp
.
sequence
);
//时间戳
//时间戳
memcpy
(
&
rtp
pt
.
timeStamp
,
pucData
+
4
,
4
);
//内存对齐
memcpy
(
&
rtp
.
timeStamp
,
rtp_raw_ptr
+
4
,
4
);
//内存对齐
if
(
!
track
->
_samplerate
){
if
(
!
track
->
_samplerate
){
//无法把时间戳转换成毫秒
return
false
;
return
false
;
}
}
//时间戳转换成毫秒
//时间戳转换成毫秒
rtp
pt
.
timeStamp
=
ntohl
(
rtppt
.
timeStamp
)
*
1000LL
/
track
->
_samplerate
;
rtp
.
timeStamp
=
ntohl
(
rtp
.
timeStamp
)
*
1000LL
/
track
->
_samplerate
;
//ssrc
//ssrc
memcpy
(
&
rtppt
.
ssrc
,
pucData
+
8
,
4
);
//内存对齐
memcpy
(
&
rtp
.
ssrc
,
rtp_raw_ptr
+
8
,
4
);
//内存对齐
rtppt
.
ssrc
=
ntohl
(
rtppt
.
ssrc
);
rtp
.
ssrc
=
ntohl
(
rtp
.
ssrc
);
rtppt
.
type
=
track
->
_type
;
rtp
.
type
=
track
->
_type
;
if
(
track
->
_ssrc
==
0
)
{
track
->
_ssrc
=
rtppt
.
ssrc
;
if
(
track
->
_ssrc
!=
rtp
.
ssrc
)
{
//保存SSRC
if
(
track
->
_ssrc
==
0
)
{
}
else
if
(
track
->
_ssrc
!=
rtppt
.
ssrc
)
{
//保存SSRC至track对象
//ssrc错误
track
->
_ssrc
=
rtp
.
ssrc
;
WarnL
<<
"ssrc错误:"
<<
rtppt
.
ssrc
<<
" != "
<<
track
->
_ssrc
;
}
else
{
if
(
_aui32SsrcErrorCnt
[
iTrackidx
]
++
>
10
)
{
//ssrc错误
//ssrc切换后清除老数据
WarnL
<<
"ssrc错误:"
<<
rtp
.
ssrc
<<
" != "
<<
track
->
_ssrc
;
WarnL
<<
"ssrc更换:"
<<
track
->
_ssrc
<<
" -> "
<<
rtppt
.
ssrc
;
if
(
_ssrc_err_count
[
track_index
]
++
>
10
)
{
_amapRtpSort
[
iTrackidx
].
clear
();
//ssrc切换后清除老数据
track
->
_ssrc
=
rtppt
.
ssrc
;
WarnL
<<
"ssrc更换:"
<<
track
->
_ssrc
<<
" -> "
<<
rtp
.
ssrc
;
_rtp_sort_cache_map
[
track_index
].
clear
();
track
->
_ssrc
=
rtp
.
ssrc
;
}
return
false
;
}
}
return
false
;
}
}
_aui32SsrcErrorCnt
[
iTrackidx
]
=
0
;
//ssrc匹配正确,不匹配计数清零
_ssrc_err_count
[
track_index
]
=
0
;
//获取rtp中媒体数据偏移量
//获取rtp中媒体数据偏移量
rtppt
.
offset
=
12
+
4
;
rtp
.
offset
=
12
+
4
;
int
csrc
=
pucData
[
0
]
&
0x0f
;
int
csrc
=
rtp_raw_ptr
[
0
]
&
0x0f
;
int
ext
=
pucData
[
0
]
&
0x10
;
int
ext
=
rtp_raw_ptr
[
0
]
&
0x10
;
rtppt
.
offset
+=
4
*
csrc
;
rtp
.
offset
+=
4
*
csrc
;
if
(
ext
)
{
if
(
ext
&&
rtp_raw_len
>=
rtp
.
offset
)
{
if
(
uiLen
<
rtppt
.
offset
){
return
false
;
}
/* calculate the header extension length (stored as number of 32-bit words) */
/* calculate the header extension length (stored as number of 32-bit words) */
ext
=
(
AV_RB16
(
pucData
+
rtppt
.
offset
-
2
)
+
1
)
<<
2
;
ext
=
(
AV_RB16
(
rtp_raw_ptr
+
rtp
.
offset
-
2
)
+
1
)
<<
2
;
rtp
pt
.
offset
+=
ext
;
rtp
.
offset
+=
ext
;
}
}
if
(
length
-
rtppt
.
offset
<=
0
){
if
(
length
<=
rtp
.
offset
){
WarnL
<<
"无有效负载的rtp包:"
<<
length
<<
"<="
<<
(
int
)
rtp
pt
.
offset
;
WarnL
<<
"无有效负载的rtp包:"
<<
length
<<
"<="
<<
(
int
)
rtp
.
offset
;
return
false
;
return
false
;
}
}
rtppt
.
setCapacity
(
length
);
if
(
length
>
RTP_MAX_SIZE
){
rtppt
.
setSize
(
length
);
WarnL
<<
"超大的rtp包:"
<<
length
<<
">"
<<
RTP_MAX_SIZE
;
uint8_t
*
payload_ptr
=
(
uint8_t
*
)
rtppt
.
data
();
return
false
;
}
//设置rtp负载长度
rtp
.
setCapacity
(
length
);
rtp
.
setSize
(
length
);
uint8_t
*
payload_ptr
=
(
uint8_t
*
)
rtp
.
data
();
payload_ptr
[
0
]
=
'$'
;
payload_ptr
[
0
]
=
'$'
;
payload_ptr
[
1
]
=
rtp
pt
.
interleaved
;
payload_ptr
[
1
]
=
rtp
.
interleaved
;
payload_ptr
[
2
]
=
uiL
en
>>
8
;
payload_ptr
[
2
]
=
rtp_raw_l
en
>>
8
;
payload_ptr
[
3
]
=
(
uiL
en
&
0x00FF
);
payload_ptr
[
3
]
=
(
rtp_raw_l
en
&
0x00FF
);
//拷贝rtp负载
//拷贝rtp负载
memcpy
(
payload_ptr
+
4
,
pucData
,
uiLen
);
memcpy
(
payload_ptr
+
4
,
rtp_raw_ptr
,
rtp_raw_len
);
//排序rtp
sortRtp
(
rtp_ptr
,
track_index
);
return
true
;
}
/////////////////////////////////RTP排序逻辑///////////////////////////////////
void
RtpReceiver
::
sortRtp
(
const
RtpPacket
::
Ptr
&
rtp
,
int
track_index
){
if
(
rtp
pt
.
sequence
!=
_aui16LastSeq
[
iTrackidx
]
+
1
&&
_aui16LastSeq
[
iTrackid
x
]
!=
0
){
if
(
rtp
->
sequence
!=
_last_seq
[
track_index
]
+
1
&&
_last_seq
[
track_inde
x
]
!=
0
){
//包乱序或丢包
//包乱序或丢包
_aui32SeqOkCnt
[
iTrackidx
]
=
0
;
_seq_ok_count
[
track_index
]
=
0
;
_abSortStarted
[
iTrackidx
]
=
true
;
_sort_started
[
track_index
]
=
true
;
// WarnL << "包乱序或丢包:" << iTrackidx <<" " << rtppt.sequence << " " << _aui16LastSeq[iTrackidx];
if
(
_last_seq
[
track_index
]
>
rtp
->
sequence
&&
_last_seq
[
track_index
]
-
rtp
->
sequence
>
0xFF
){
if
(
_aui16LastSeq
[
iTrackidx
]
>
rtppt
.
sequence
&&
_aui16LastSeq
[
iTrackidx
]
-
rtppt
.
sequence
>
0xFF
){
//sequence回环,清空所有排序缓存
//sequence回环,清空所有排序缓存
while
(
_
amapRtpSort
[
iTrackid
x
].
size
())
{
while
(
_
rtp_sort_cache_map
[
track_inde
x
].
size
())
{
POP_HEAD
(
iTrackid
x
)
POP_HEAD
(
track_inde
x
)
}
}
++
_
clcyeCount
[
iTrackid
x
];
++
_
seq_cycle_count
[
track_inde
x
];
}
}
}
else
{
}
else
{
//正确序列的包
//正确序列的包
_
aui32SeqOkCnt
[
iTrackid
x
]
++
;
_
seq_ok_count
[
track_inde
x
]
++
;
}
}
_aui16LastSeq
[
iTrackidx
]
=
rtppt
.
sequence
;
_last_seq
[
track_index
]
=
rtp
->
sequence
;
//开始排序缓存
//开始排序缓存
if
(
_
abSortStarted
[
iTrackid
x
])
{
if
(
_
sort_started
[
track_inde
x
])
{
_
amapRtpSort
[
iTrackidx
].
emplace
(
rtppt
.
sequence
,
pt_ptr
);
_
rtp_sort_cache_map
[
track_index
].
emplace
(
rtp
->
sequence
,
rtp
);
GET_CONFIG
(
uint32_t
,
clearCount
,
Rtp
::
kClearCount
);
GET_CONFIG
(
uint32_t
,
clearCount
,
Rtp
::
kClearCount
);
GET_CONFIG
(
uint32_t
,
maxRtpCount
,
Rtp
::
kMaxRtpCount
);
GET_CONFIG
(
uint32_t
,
maxRtpCount
,
Rtp
::
kMaxRtpCount
);
if
(
_
aui32SeqOkCnt
[
iTrackid
x
]
>=
clearCount
)
{
if
(
_
seq_ok_count
[
track_inde
x
]
>=
clearCount
)
{
//网络环境改善,需要清空排序缓存
//网络环境改善,需要清空排序缓存
_
aui32SeqOkCnt
[
iTrackid
x
]
=
0
;
_
seq_ok_count
[
track_inde
x
]
=
0
;
_
abSortStarted
[
iTrackid
x
]
=
false
;
_
sort_started
[
track_inde
x
]
=
false
;
while
(
_
amapRtpSort
[
iTrackid
x
].
size
())
{
while
(
_
rtp_sort_cache_map
[
track_inde
x
].
size
())
{
POP_HEAD
(
iTrackid
x
)
POP_HEAD
(
track_inde
x
)
}
}
}
else
if
(
_
amapRtpSort
[
iTrackid
x
].
size
()
>=
maxRtpCount
)
{
}
else
if
(
_
rtp_sort_cache_map
[
track_inde
x
].
size
()
>=
maxRtpCount
)
{
//排序缓存溢出
//排序缓存溢出
POP_HEAD
(
iTrackid
x
)
POP_HEAD
(
track_inde
x
)
}
}
}
else
{
}
else
{
//正确序列
//正确序列
onRtpSorted
(
pt_ptr
,
iTrackid
x
);
onRtpSorted
(
rtp
,
track_inde
x
);
}
}
//////////////////////////////////////////////////////////////////////////////////
return
true
;
}
}
void
RtpReceiver
::
clear
()
{
void
RtpReceiver
::
clear
()
{
CLEAR_ARR
(
_
aui16LastS
eq
)
CLEAR_ARR
(
_
last_s
eq
)
CLEAR_ARR
(
_
aui32SsrcErrorC
nt
)
CLEAR_ARR
(
_
ssrc_err_cou
nt
)
CLEAR_ARR
(
_
aui32SeqOkC
nt
)
CLEAR_ARR
(
_
seq_ok_cou
nt
)
CLEAR_ARR
(
_
abSortS
tarted
)
CLEAR_ARR
(
_
sort_s
tarted
)
CLEAR_ARR
(
_
clcyeC
ount
)
CLEAR_ARR
(
_
seq_cycle_c
ount
)
_
amapRtpSort
[
0
].
clear
();
_
rtp_sort_cache_map
[
0
].
clear
();
_
amapRtpSort
[
1
].
clear
();
_
rtp_sort_cache_map
[
1
].
clear
();
}
}
void
RtpReceiver
::
setPoolSize
(
int
size
)
{
void
RtpReceiver
::
setPoolSize
(
int
size
)
{
_
pktP
ool
.
setSize
(
size
);
_
rtp_p
ool
.
setSize
(
size
);
}
}
int
RtpReceiver
::
getJitterSize
(
int
iTrackidx
){
int
RtpReceiver
::
getJitterSize
(
int
iTrackidx
){
return
_
amapRtpSort
[
iTrackidx
].
size
();
return
_
rtp_sort_cache_map
[
iTrackidx
].
size
();
}
}
int
RtpReceiver
::
getCycleCount
(
int
iTrackidx
){
int
RtpReceiver
::
getCycleCount
(
int
iTrackidx
){
return
_
clcyeC
ount
[
iTrackidx
];
return
_
seq_cycle_c
ount
[
iTrackidx
];
}
}
...
...
src/Rtsp/RtpReceiver.h
查看文件 @
85f05adc
...
@@ -48,33 +48,41 @@ protected:
...
@@ -48,33 +48,41 @@ protected:
/**
/**
* 输入数据指针生成并排序rtp包
* 输入数据指针生成并排序rtp包
* @param
iTrackid
x track下标索引
* @param
track_inde
x track下标索引
* @param track sdp track相关信息
* @param track sdp track相关信息
* @param
pucData
rtp数据指针
* @param
rtp_raw_ptr
rtp数据指针
* @param
uiL
en rtp数据指针长度
* @param
rtp_raw_l
en rtp数据指针长度
* @return 解析成功返回true
* @return 解析成功返回true
*/
*/
bool
handleOneRtp
(
int
iTrackidx
,
SdpTrack
::
Ptr
&
track
,
unsigned
char
*
pucData
,
unsigned
int
uiL
en
);
bool
handleOneRtp
(
int
track_index
,
SdpTrack
::
Ptr
&
track
,
unsigned
char
*
rtp_raw_ptr
,
unsigned
int
rtp_raw_l
en
);
/**
/**
* rtp数据包排序后输出
* rtp数据包排序后输出
* @param rtppt rtp数据包
* @param rtppt rtp数据包
* @param trackidx track索引
* @param trackidx track索引
*/
*/
virtual
void
onRtpSorted
(
const
RtpPacket
::
Ptr
&
rtp
pt
,
int
trackidx
){}
virtual
void
onRtpSorted
(
const
RtpPacket
::
Ptr
&
rtp
,
int
trackidx
){}
void
clear
();
void
clear
();
void
setPoolSize
(
int
size
);
void
setPoolSize
(
int
size
);
int
getJitterSize
(
int
iTrackidx
);
int
getJitterSize
(
int
iTrackidx
);
int
getCycleCount
(
int
iTrackidx
);
int
getCycleCount
(
int
iTrackidx
);
private
:
private
:
uint32_t
_aui32SsrcErrorCnt
[
2
]
=
{
0
,
0
};
void
sortRtp
(
const
RtpPacket
::
Ptr
&
rtp
,
int
track_index
);
/* RTP包排序所用参数 */
private
:
uint16_t
_aui16LastSeq
[
2
]
=
{
0
,
0
};
//ssrc不匹配计数
uint32_t
_aui32SeqOkCnt
[
2
]
=
{
0
,
0
};
uint32_t
_ssrc_err_count
[
2
]
=
{
0
,
0
};
uint32_t
_clcyeCount
[
2
]
=
{
0
,
0
};
//上次seq
bool
_abSortStarted
[
2
]
=
{
0
,
0
};
uint16_t
_last_seq
[
2
]
=
{
0
,
0
};
map
<
uint16_t
,
RtpPacket
::
Ptr
>
_amapRtpSort
[
2
];
//seq连续次数计数
RtspMediaSource
::
PoolType
_pktPool
;
uint32_t
_seq_ok_count
[
2
]
=
{
0
,
0
};
//seq回环次数计数
uint32_t
_seq_cycle_count
[
2
]
=
{
0
,
0
};
//是否开始seq排序
bool
_sort_started
[
2
]
=
{
0
,
0
};
//rtp排序缓存,根据seq排序
map
<
uint16_t
,
RtpPacket
::
Ptr
>
_rtp_sort_cache_map
[
2
];
//rtp循环池
RtspMediaSource
::
PoolType
_rtp_pool
;
};
};
}
//namespace mediakit
}
//namespace mediakit
...
...
编写
预览
Markdown
格式
0%
重试
或
添加新文件
添加附件
取消
您添加了
0
人
到此讨论。请谨慎行事。
请先完成此评论的编辑!
取消
请
注册
或者
登录
后发表评论