2018-09-20 15:43:49 +08:00
|
|
|
|
//
|
2018-09-18 23:49:48 +08:00
|
|
|
|
// Created by xzl on 2018/9/18.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
#ifndef ZLMEDIAKIT_RTSPMAKER_H
|
|
|
|
|
#define ZLMEDIAKIT_RTSPMAKER_H
|
|
|
|
|
|
2018-10-23 11:09:21 +08:00
|
|
|
|
#include "RTP/H264RtpCodec.h"
|
|
|
|
|
#include "RTP/AACRtpCodec.h"
|
|
|
|
|
#include "Util/base64.h"
|
2018-09-18 23:49:48 +08:00
|
|
|
|
|
|
|
|
|
namespace ZL{
|
|
|
|
|
namespace Rtsp{
|
|
|
|
|
|
2018-09-19 12:34:29 +08:00
|
|
|
|
/**
|
2018-10-23 11:09:21 +08:00
|
|
|
|
* sdp基类
|
|
|
|
|
*/
|
|
|
|
|
class Sdp : public TrackFormat , public RtpRingInterface{
|
2018-09-18 23:49:48 +08:00
|
|
|
|
public:
|
|
|
|
|
typedef std::shared_ptr<Sdp> Ptr;
|
2018-10-23 11:38:56 +08:00
|
|
|
|
Sdp(uint32_t sample_rate, uint8_t playload_type){
|
|
|
|
|
_sample_rate = sample_rate;
|
|
|
|
|
_playload_type = playload_type;
|
|
|
|
|
}
|
2018-09-18 23:49:48 +08:00
|
|
|
|
virtual ~Sdp(){}
|
2018-09-19 12:34:29 +08:00
|
|
|
|
/**
|
|
|
|
|
* 获取sdp字符串
|
|
|
|
|
* @return
|
|
|
|
|
*/
|
2018-10-23 11:09:21 +08:00
|
|
|
|
virtual string getSdp() const = 0;
|
2018-09-19 12:34:29 +08:00
|
|
|
|
|
2018-10-23 11:09:21 +08:00
|
|
|
|
TrackType getTrackType() const override {
|
|
|
|
|
return TrackInvalid;
|
|
|
|
|
}
|
2018-09-19 12:34:29 +08:00
|
|
|
|
|
2018-10-23 11:09:21 +08:00
|
|
|
|
CodecId getCodecId() const override{
|
|
|
|
|
return CodecInvalid;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
FrameRingInterface::RingType::Ptr getFrameRing() const override {
|
|
|
|
|
return _encoder->getFrameRing();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
RtpRingInterface::RingType::Ptr getRtpRing() const override{
|
|
|
|
|
return _encoder->getRtpRing();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void inputFrame(const Frame::Ptr &frame,bool key_pos) override{
|
|
|
|
|
_encoder->inputFrame(frame,key_pos);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void inputRtp(const RtpPacket::Ptr &rtp, bool key_pos) override{
|
|
|
|
|
_encoder->inputRtp(rtp,key_pos);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void setFrameRing(const FrameRingInterface::RingType::Ptr &ring) override{
|
|
|
|
|
if(_encoder){
|
|
|
|
|
_encoder->setFrameRing(ring);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
void setRtpRing(const RtpRingInterface::RingType::Ptr &ring) override{
|
|
|
|
|
if(_encoder){
|
|
|
|
|
_encoder->setRtpRing(ring);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-23 11:38:56 +08:00
|
|
|
|
virtual void createRtpEncoder(uint32_t ssrc, int mtu) {
|
|
|
|
|
_encoder = RtpCodec::getRtpCodec (getCodecId(),
|
|
|
|
|
ssrc,
|
|
|
|
|
mtu,
|
|
|
|
|
_sample_rate,
|
|
|
|
|
_playload_type,
|
|
|
|
|
getTrackType() * 2);
|
|
|
|
|
}
|
|
|
|
|
private:
|
|
|
|
|
RtpCodec::Ptr _encoder;
|
|
|
|
|
uint8_t _playload_type;
|
|
|
|
|
uint32_t _sample_rate;
|
2018-09-18 23:49:48 +08:00
|
|
|
|
};
|
|
|
|
|
|
2018-09-19 12:34:29 +08:00
|
|
|
|
/**
|
2018-10-23 11:09:21 +08:00
|
|
|
|
* sdp中除音视频外的其他描述部分
|
|
|
|
|
*/
|
2018-09-18 23:49:48 +08:00
|
|
|
|
class SdpTitle : public Sdp{
|
|
|
|
|
public:
|
2018-09-19 12:34:29 +08:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 构造title类型sdp
|
|
|
|
|
* @param dur_sec rtsp点播时长,0代表直播,单位秒
|
|
|
|
|
* @param header 自定义sdp描述
|
|
|
|
|
* @param version sdp版本
|
|
|
|
|
*/
|
2018-09-18 23:49:48 +08:00
|
|
|
|
SdpTitle(float dur_sec = 0,
|
|
|
|
|
const map<string,string> &header = map<string,string>(),
|
2018-10-23 11:38:56 +08:00
|
|
|
|
int version = 0) : Sdp(0,0){
|
2018-09-18 23:49:48 +08:00
|
|
|
|
_printer << "v=" << version << "\r\n";
|
|
|
|
|
|
|
|
|
|
if(!header.empty()){
|
|
|
|
|
for (auto &pr : header){
|
|
|
|
|
_printer << pr.first << "=" << pr.second << "\r\n";
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
_printer << "o=- 1383190487994921 1 IN IP4 0.0.0.0\r\n";
|
2018-10-23 11:09:21 +08:00
|
|
|
|
_printer << "s=RTSP Session, streamed by the ZLMediaKit\r\n";
|
|
|
|
|
_printer << "i=ZLMediaKit Live Stream\r\n";
|
2018-09-18 23:49:48 +08:00
|
|
|
|
_printer << "c=IN IP4 0.0.0.0\r\n";
|
|
|
|
|
_printer << "t=0 0\r\n";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(dur_sec <= 0){
|
|
|
|
|
_printer << "a=range:npt=0-\r\n";
|
|
|
|
|
}else{
|
|
|
|
|
_printer << "a=range:npt=0-" << dur_sec << "\r\n";
|
|
|
|
|
}
|
|
|
|
|
_printer << "a=control:*\r\n";
|
|
|
|
|
}
|
2018-09-19 12:34:29 +08:00
|
|
|
|
string getSdp() const override {
|
2018-09-18 23:49:48 +08:00
|
|
|
|
return _printer;
|
|
|
|
|
}
|
|
|
|
|
private:
|
|
|
|
|
_StrPrinter _printer;
|
|
|
|
|
};
|
|
|
|
|
|
2018-09-19 12:34:29 +08:00
|
|
|
|
/**
|
2018-10-23 11:09:21 +08:00
|
|
|
|
* h264类型sdp
|
|
|
|
|
*/
|
2018-09-18 23:49:48 +08:00
|
|
|
|
class SdpH264 : public Sdp {
|
|
|
|
|
public:
|
2018-09-19 12:34:29 +08:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
*
|
|
|
|
|
* @param sps 264 sps,带0x00000001头
|
|
|
|
|
* @param pps 264 pps,带0x00000001头
|
|
|
|
|
* @param sample_rate 时间戳采样率,视频默认90000
|
|
|
|
|
* @param playload_type rtp playload type 默认96
|
|
|
|
|
* @param track_id trackID 默认为TrackVideo
|
|
|
|
|
* @param bitrate 比特率
|
|
|
|
|
*/
|
2018-09-18 23:49:48 +08:00
|
|
|
|
SdpH264(const string &sps,
|
|
|
|
|
const string &pps,
|
|
|
|
|
int sample_rate = 90000,
|
|
|
|
|
int playload_type = 96,
|
2018-10-23 11:38:56 +08:00
|
|
|
|
int bitrate = 4000) : Sdp(sample_rate,playload_type) {
|
2018-09-18 23:49:48 +08:00
|
|
|
|
//视频通道
|
|
|
|
|
_printer << "m=video 0 RTP/AVP " << playload_type << "\r\n";
|
|
|
|
|
_printer << "b=AS:" << bitrate << "\r\n";
|
|
|
|
|
_printer << "a=rtpmap:" << playload_type << " H264/" << sample_rate << "\r\n";
|
|
|
|
|
_printer << "a=fmtp:" << playload_type << " packetization-mode=1;profile-level-id=";
|
|
|
|
|
|
|
|
|
|
char strTemp[100];
|
|
|
|
|
int profile_level_id = 0;
|
|
|
|
|
string strSPS = sps.substr(4);
|
|
|
|
|
string strPPS = pps.substr(4);
|
|
|
|
|
if (strSPS.length() >= 4) { // sanity check
|
|
|
|
|
profile_level_id = (strSPS[1] << 16) | (strSPS[2] << 8) | strSPS[3]; // profile_idc|constraint_setN_flag|level_idc
|
|
|
|
|
}
|
|
|
|
|
memset(strTemp, 0, 100);
|
|
|
|
|
sprintf(strTemp, "%06X", profile_level_id);
|
|
|
|
|
_printer << strTemp;
|
|
|
|
|
_printer << ";sprop-parameter-sets=";
|
|
|
|
|
memset(strTemp, 0, 100);
|
|
|
|
|
av_base64_encode(strTemp, 100, (uint8_t *) strSPS.data(), strSPS.size());
|
|
|
|
|
_printer << strTemp << ",";
|
|
|
|
|
memset(strTemp, 0, 100);
|
|
|
|
|
av_base64_encode(strTemp, 100, (uint8_t *) strPPS.data(), strPPS.size());
|
|
|
|
|
_printer << strTemp << "\r\n";
|
2018-10-23 11:38:56 +08:00
|
|
|
|
_printer << "a=control:trackID=" << getTrackType() << "\r\n";
|
2018-09-18 23:49:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
2018-09-19 12:34:29 +08:00
|
|
|
|
string getSdp() const override {
|
2018-09-18 23:49:48 +08:00
|
|
|
|
return _printer;
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-19 12:34:29 +08:00
|
|
|
|
TrackType getTrackType() const override {
|
2018-09-18 23:49:48 +08:00
|
|
|
|
return TrackVideo;
|
2018-10-23 11:09:21 +08:00
|
|
|
|
}
|
2018-09-18 23:49:48 +08:00
|
|
|
|
|
2018-10-23 11:09:21 +08:00
|
|
|
|
CodecId getCodecId() const override {
|
|
|
|
|
return CodecH264;
|
|
|
|
|
}
|
2018-09-18 23:49:48 +08:00
|
|
|
|
private:
|
|
|
|
|
_StrPrinter _printer;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2018-09-19 12:34:29 +08:00
|
|
|
|
/**
|
2018-10-23 11:09:21 +08:00
|
|
|
|
* aac类型SDP
|
|
|
|
|
*/
|
2018-09-18 23:49:48 +08:00
|
|
|
|
class SdpAAC : public Sdp {
|
|
|
|
|
public:
|
2018-09-19 12:34:29 +08:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 构造aac sdp
|
|
|
|
|
* @param aac_cfg aac两个字节的配置描述
|
|
|
|
|
* @param sample_rate 音频采样率
|
|
|
|
|
* @param playload_type rtp playload type 默认96
|
|
|
|
|
* @param track_id trackID 默认为TrackVideo
|
|
|
|
|
* @param bitrate 比特率
|
|
|
|
|
*/
|
2018-09-18 23:49:48 +08:00
|
|
|
|
SdpAAC(const string &aac_cfg,
|
|
|
|
|
int sample_rate,
|
|
|
|
|
int playload_type = 98,
|
2018-10-23 11:38:56 +08:00
|
|
|
|
int bitrate = 128) : Sdp(sample_rate,playload_type){
|
2018-09-18 23:49:48 +08:00
|
|
|
|
_printer << "m=audio 0 RTP/AVP " << playload_type << "\r\n";
|
|
|
|
|
_printer << "b=AS:" << bitrate << "\r\n";
|
|
|
|
|
_printer << "a=rtpmap:" << playload_type << " MPEG4-GENERIC/" << sample_rate << "\r\n";
|
|
|
|
|
|
|
|
|
|
char configStr[32] = {0};
|
|
|
|
|
snprintf(configStr, sizeof(configStr), "%02X%02x", aac_cfg[0], aac_cfg[1]);
|
|
|
|
|
_printer << "a=fmtp:" << playload_type << " streamtype=5;profile-level-id=1;mode=AAC-hbr;"
|
|
|
|
|
<< "sizelength=13;indexlength=3;indexdeltalength=3;config="
|
|
|
|
|
<< configStr << "\r\n";
|
2018-10-23 11:38:56 +08:00
|
|
|
|
_printer << "a=control:trackID=" << getTrackType() << "\r\n";
|
2018-09-18 23:49:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
2018-09-19 12:34:29 +08:00
|
|
|
|
string getSdp() const override {
|
2018-09-18 23:49:48 +08:00
|
|
|
|
return _printer;
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-19 12:34:29 +08:00
|
|
|
|
TrackType getTrackType() const override {
|
2018-09-18 23:49:48 +08:00
|
|
|
|
return TrackAudio;
|
|
|
|
|
};
|
2018-10-23 11:09:21 +08:00
|
|
|
|
CodecId getCodecId() const override {
|
|
|
|
|
return CodecAAC;
|
|
|
|
|
}
|
2018-09-18 23:49:48 +08:00
|
|
|
|
private:
|
|
|
|
|
_StrPrinter _printer;
|
|
|
|
|
};
|
|
|
|
|
|
2018-09-19 12:34:29 +08:00
|
|
|
|
/**
|
2018-10-23 11:09:21 +08:00
|
|
|
|
* rtsp生成器
|
|
|
|
|
*/
|
|
|
|
|
class RtspEncoder : public FrameRingInterface , public RtpRingInterface{
|
2018-09-18 23:49:48 +08:00
|
|
|
|
public:
|
2018-09-19 12:34:29 +08:00
|
|
|
|
/**
|
|
|
|
|
* 构成函数
|
|
|
|
|
*/
|
2018-10-23 11:09:21 +08:00
|
|
|
|
RtspEncoder(){
|
|
|
|
|
//自适应缓存
|
|
|
|
|
_rtpRing = std::make_shared<RtpRingInterface::RingType>(0);
|
|
|
|
|
//禁用缓存
|
|
|
|
|
_frameRing = std::make_shared<FrameRingInterface::RingType>(1);
|
2018-09-18 23:49:48 +08:00
|
|
|
|
}
|
2018-10-23 11:09:21 +08:00
|
|
|
|
virtual ~RtspEncoder(){}
|
2018-09-19 12:34:29 +08:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 添加音视频track
|
2018-10-23 11:09:21 +08:00
|
|
|
|
* @param sdp 媒体描述对象
|
2018-09-19 12:34:29 +08:00
|
|
|
|
* @param ssrc 媒体rtp ssrc
|
|
|
|
|
* @param mtu 媒体rtp mtu
|
|
|
|
|
*/
|
2018-10-23 11:09:21 +08:00
|
|
|
|
void addTrack(const Sdp::Ptr & sdp,uint32_t ssrc = 0,int mtu = 1400){
|
2018-09-19 12:34:29 +08:00
|
|
|
|
if(ssrc == 0){
|
2018-10-23 11:09:21 +08:00
|
|
|
|
ssrc = ((uint64_t) sdp.get()) & 0xFFFFFFFF;
|
2018-09-18 23:49:48 +08:00
|
|
|
|
}
|
2018-10-23 11:09:21 +08:00
|
|
|
|
sdp->createRtpEncoder(ssrc, mtu);
|
|
|
|
|
sdp->setFrameRing(_frameRing);
|
|
|
|
|
sdp->setRtpRing(_rtpRing);
|
|
|
|
|
_sdp_map[sdp->getTrackType()] = sdp;
|
2018-09-18 23:49:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
2018-09-19 12:34:29 +08:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 获取完整的SDP字符串
|
|
|
|
|
* @return SDP字符串
|
|
|
|
|
*/
|
2018-09-18 23:49:48 +08:00
|
|
|
|
string getSdp() {
|
2018-10-23 11:09:21 +08:00
|
|
|
|
_StrPrinter printer;
|
|
|
|
|
for(auto &pr : _sdp_map){
|
|
|
|
|
printer << pr.second->getSdp() ;
|
|
|
|
|
}
|
|
|
|
|
return printer;
|
2018-09-18 23:49:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
2018-09-19 12:34:29 +08:00
|
|
|
|
|
|
|
|
|
/**
|
2018-10-23 11:09:21 +08:00
|
|
|
|
* 写入帧数据然后打包rtp
|
|
|
|
|
* @param frame 帧数据
|
|
|
|
|
* @param key_pos 是否为关键帧
|
2018-09-19 12:34:29 +08:00
|
|
|
|
*/
|
2018-10-23 11:09:21 +08:00
|
|
|
|
void inputFrame(const Frame::Ptr &frame,bool key_pos = true) override {
|
|
|
|
|
auto it = _sdp_map.find(frame->getTrackType());
|
|
|
|
|
if(it == _sdp_map.end()){
|
|
|
|
|
return ;
|
2018-09-18 23:49:48 +08:00
|
|
|
|
}
|
2018-10-23 11:09:21 +08:00
|
|
|
|
it->second->inputFrame(frame,key_pos);
|
2018-09-18 23:49:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
2018-10-23 11:09:21 +08:00
|
|
|
|
/**
|
|
|
|
|
* 也可以在外部打包好rtp然后再写入
|
|
|
|
|
* @param rtp rtp包
|
|
|
|
|
* @param key_pos 是否为关键帧的第一个rtp包
|
|
|
|
|
*/
|
|
|
|
|
void inputRtp(const RtpPacket::Ptr &rtp, bool key_pos = true) override {
|
2018-10-23 11:38:56 +08:00
|
|
|
|
auto it = _sdp_map.find(rtp->getTrackType());
|
|
|
|
|
if(it == _sdp_map.end()){
|
|
|
|
|
return ;
|
|
|
|
|
}
|
|
|
|
|
it->second->inputRtp(rtp,key_pos);
|
2018-10-23 11:09:21 +08:00
|
|
|
|
}
|
2018-09-18 23:49:48 +08:00
|
|
|
|
|
2018-10-23 11:09:21 +08:00
|
|
|
|
/**
|
|
|
|
|
* 获取rtp环形缓存
|
|
|
|
|
* @return
|
|
|
|
|
*/
|
|
|
|
|
RtpRingInterface::RingType::Ptr getRtpRing() const override{
|
|
|
|
|
return _rtpRing;
|
|
|
|
|
}
|
2018-09-18 23:49:48 +08:00
|
|
|
|
|
2018-10-23 11:09:21 +08:00
|
|
|
|
/**
|
|
|
|
|
* 获取帧环形缓存
|
|
|
|
|
* @return
|
|
|
|
|
*/
|
|
|
|
|
FrameRingInterface::RingType::Ptr getFrameRing() const override{
|
|
|
|
|
return _frameRing;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 设置帧环形缓存
|
|
|
|
|
* @param ring
|
|
|
|
|
*/
|
|
|
|
|
void setFrameRing(const FrameRingInterface::RingType::Ptr &ring) override{
|
|
|
|
|
_frameRing = ring;
|
|
|
|
|
for(auto &pr : _sdp_map){
|
|
|
|
|
pr.second->setFrameRing(ring);
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-09-18 23:49:48 +08:00
|
|
|
|
|
|
|
|
|
|
2018-10-23 11:09:21 +08:00
|
|
|
|
/**
|
|
|
|
|
* 设置rtp环形缓存
|
|
|
|
|
* @param ring
|
|
|
|
|
*/
|
|
|
|
|
void setRtpRing(const RtpRingInterface::RingType::Ptr &ring) override{
|
|
|
|
|
_rtpRing = ring;
|
|
|
|
|
for(auto &pr : _sdp_map){
|
|
|
|
|
pr.second->setRtpRing(ring);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
private:
|
|
|
|
|
map<int,Sdp::Ptr> _sdp_map;
|
|
|
|
|
RtpRingInterface::RingType::Ptr _rtpRing;
|
|
|
|
|
FrameRingInterface::RingType::Ptr _frameRing;
|
|
|
|
|
};
|
2018-09-18 23:49:48 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#endif //ZLMEDIAKIT_RTSPMAKER_H
|