mirror of
https://github.com/ZLMediaKit/ZLMediaKit.git
synced 2026-07-03 08:57:32 +08:00
开始修改播放器
This commit is contained in:
@@ -91,8 +91,8 @@ RtpParser::RtpParser(const string& sdp) {
|
||||
RtpParser::~RtpParser() {
|
||||
}
|
||||
|
||||
bool RtpParser::inputRtp(const RtpPacket& rtp) {
|
||||
auto &track = m_mapTracks[rtp.PT];
|
||||
bool RtpParser::inputRtp(const RtpPacket::Ptr & rtp) {
|
||||
auto &track = m_mapTracks[rtp->PT];
|
||||
switch (track.type) {
|
||||
case TrackVideo:
|
||||
if (m_bHaveVideo) {
|
||||
@@ -109,205 +109,19 @@ bool RtpParser::inputRtp(const RtpPacket& rtp) {
|
||||
}
|
||||
}
|
||||
|
||||
inline bool RtpParser::inputVideo(const RtpPacket& rtppack,
|
||||
const RtspTrack& track) {
|
||||
const uint8_t *frame = (uint8_t *) rtppack.payload + rtppack.offset;
|
||||
int length = rtppack.length - rtppack.offset;
|
||||
NALU nal;
|
||||
MakeNalu(*frame, nal);
|
||||
//Type==1:P frame
|
||||
//Type==6:SEI frame
|
||||
//Type==7:SPS frame
|
||||
//Type==8:PPS frame
|
||||
if (nal.type >= 0 && nal.type < 24) {
|
||||
//a full frame
|
||||
m_h264frame.buffer.assign("\x0\x0\x0\x1", 4);
|
||||
m_h264frame.buffer.append((char *)frame, length);
|
||||
m_h264frame.type = nal.type;
|
||||
m_h264frame.timeStamp = rtppack.timeStamp / 90;
|
||||
m_h264frame.sequence = rtppack.sequence;
|
||||
_onGetH264(m_h264frame);
|
||||
m_h264frame.buffer.clear();
|
||||
return (m_h264frame.type == 7);
|
||||
}
|
||||
if (nal.type == 28) {
|
||||
//FU-A
|
||||
FU fu;
|
||||
MakeFU(frame[1], fu);
|
||||
if (fu.S == 1) {
|
||||
//FU-A start
|
||||
char tmp = (nal.forbidden_zero_bit << 7 | nal.nal_ref_idc << 5 | fu.type);
|
||||
m_h264frame.buffer.assign("\x0\x0\x0\x1", 4);
|
||||
m_h264frame.buffer.push_back(tmp);
|
||||
m_h264frame.buffer.append((char *)frame + 2, length - 2);
|
||||
m_h264frame.type = fu.type;
|
||||
m_h264frame.timeStamp = rtppack.timeStamp / 90;
|
||||
m_h264frame.sequence = rtppack.sequence;
|
||||
return (m_h264frame.type == 7); //i frame
|
||||
}
|
||||
inline bool RtpParser::inputVideo(const RtpPacket::Ptr & rtp, const RtspTrack& track) {
|
||||
|
||||
if (rtppack.sequence != (uint16_t)(m_h264frame.sequence + 1)) {
|
||||
m_h264frame.buffer.clear();
|
||||
WarnL << "丢包,帧废弃:" << rtppack.sequence << "," << m_h264frame.sequence;
|
||||
return false;
|
||||
}
|
||||
m_h264frame.sequence = rtppack.sequence;
|
||||
if (fu.E == 1) {
|
||||
//FU-A end
|
||||
m_h264frame.buffer.append((char *)frame + 2, length - 2);
|
||||
m_h264frame.timeStamp = rtppack.timeStamp / 90;
|
||||
_onGetH264(m_h264frame);
|
||||
m_h264frame.buffer.clear();
|
||||
return false;
|
||||
}
|
||||
//FU-A mid
|
||||
m_h264frame.buffer.append((char *)frame + 2, length - 2);
|
||||
return false;
|
||||
}
|
||||
WarnL << nal.type << " " << rtppack.sequence;
|
||||
return false;
|
||||
// 29 FU-B 单NAL单元B模式
|
||||
// 24 STAP-A 单一时间的组合包
|
||||
// 25 STAP-B 单一时间的组合包
|
||||
// 26 MTAP16 多个时间的组合包
|
||||
// 27 MTAP24 多个时间的组合包
|
||||
// 0 udef
|
||||
// 30 udef
|
||||
// 31 udef
|
||||
}
|
||||
|
||||
inline void RtpParser::onGetAudioTrack(const RtspTrack& audio) {
|
||||
for (auto &ch : const_cast<string &>(audio.trackSdp)) {
|
||||
ch = tolower(ch);
|
||||
}
|
||||
if (audio.trackSdp.find("mpeg4-generic") == string::npos) {
|
||||
throw std::runtime_error("只支持aac格式的音频!");
|
||||
}
|
||||
string fConfigStr = FindField(audio.trackSdp.c_str(), "config=", "\r\n");
|
||||
if (fConfigStr.size() != 4) {
|
||||
fConfigStr = FindField(audio.trackSdp.c_str(), "config=", ";");
|
||||
}
|
||||
if (fConfigStr.size() != 4) {
|
||||
throw std::runtime_error("解析aac格式头失败!");
|
||||
}
|
||||
m_strAudioCfg.clear();
|
||||
unsigned int cfg1;
|
||||
sscanf(fConfigStr.substr(0, 2).c_str(), "%02X", &cfg1);
|
||||
cfg1 &= 0x00FF;
|
||||
m_strAudioCfg.push_back(cfg1);
|
||||
unsigned int cfg2;
|
||||
sscanf(fConfigStr.substr(2, 2).c_str(), "%02X", &cfg2);
|
||||
cfg2 &= 0x00FF;
|
||||
m_strAudioCfg.push_back(cfg2);
|
||||
makeAdtsHeader(m_strAudioCfg,m_adts);
|
||||
getAACInfo(m_adts, m_iSampleRate, m_iChannel);
|
||||
if(m_adts.profile >= 3){
|
||||
throw std::runtime_error("不支持该profile的AAC");
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
inline void RtpParser::onGetVideoTrack(const RtspTrack& video) {
|
||||
if (video.trackSdp.find("H264") == string::npos) {
|
||||
throw std::runtime_error("只支持264格式的视频!");
|
||||
}
|
||||
string sps_pps = FindField(video.trackSdp.c_str(), "sprop-parameter-sets=", "\r\n");
|
||||
if(sps_pps.empty()){
|
||||
//SDP里面没SPS_PPS描述,需要在后续rtp中获取
|
||||
m_bParseSpsDelay = true;
|
||||
return;
|
||||
}
|
||||
string base64_SPS = FindField(sps_pps.c_str(), NULL, ",");
|
||||
string base64_PPS = FindField(sps_pps.c_str(), ",", NULL);
|
||||
if(base64_PPS.back() == ';'){
|
||||
base64_PPS.pop_back();
|
||||
}
|
||||
uint8_t SPS_BUF[256], PPS_BUF[256];
|
||||
int SPS_LEN = av_base64_decode(SPS_BUF, base64_SPS.c_str(), sizeof(SPS_BUF));
|
||||
int PPS_LEN = av_base64_decode(PPS_BUF, base64_PPS.c_str(), sizeof(PPS_BUF));
|
||||
|
||||
m_strSPS.assign("\x00\x00\x00\x01", 4);
|
||||
m_strSPS.append((char *) SPS_BUF, SPS_LEN);
|
||||
|
||||
m_strPPS.assign("\x00\x00\x00\x01", 4);
|
||||
m_strPPS.append((char *) PPS_BUF, PPS_LEN);
|
||||
|
||||
string strTmp((char *)SPS_BUF, SPS_LEN);
|
||||
if (!getAVCInfo(strTmp, m_iVideoWidth, m_iVideoHeight, m_fVideoFps)) {
|
||||
throw std::runtime_error("parse sdp failed");
|
||||
}
|
||||
}
|
||||
|
||||
inline bool RtpParser::inputAudio(const RtpPacket& rtppack,
|
||||
const RtspTrack& track) {
|
||||
char *frame = (char *) rtppack.payload + rtppack.offset;
|
||||
int length = rtppack.length - rtppack.offset;
|
||||
|
||||
if (m_adts.aac_frame_length + length - 4 > sizeof(AACFrame::buffer)) {
|
||||
m_adts.aac_frame_length = 7;
|
||||
return false;
|
||||
}
|
||||
memcpy(m_adts.buffer + m_adts.aac_frame_length, frame + 4, length - 4);
|
||||
m_adts.aac_frame_length += (length - 4);
|
||||
if (rtppack.mark == true) {
|
||||
m_adts.sequence = rtppack.sequence;
|
||||
m_adts.timeStamp = rtppack.timeStamp * (1000.0 / m_iSampleRate);
|
||||
writeAdtsHeader(m_adts, m_adts.buffer);
|
||||
onGetAdts(m_adts);
|
||||
m_adts.aac_frame_length = 7;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
inline void RtpParser::_onGetH264(H264Frame& frame) {
|
||||
switch (frame.type) {
|
||||
case 5: { //I
|
||||
H264Frame insertedFrame;
|
||||
insertedFrame.type = 7; //SPS
|
||||
insertedFrame.timeStamp = frame.timeStamp;
|
||||
insertedFrame.buffer = m_strSPS;
|
||||
onGetH264(insertedFrame);
|
||||
|
||||
insertedFrame.type = 8; //PPS
|
||||
insertedFrame.timeStamp = frame.timeStamp;
|
||||
insertedFrame.buffer = m_strPPS;
|
||||
onGetH264(insertedFrame);
|
||||
onGetH264(frame);
|
||||
}
|
||||
break;
|
||||
case 7: {//SPS
|
||||
m_strSPS = frame.buffer;
|
||||
if(m_bParseSpsDelay && !m_strSPS.empty()){
|
||||
m_bParseSpsDelay = false;
|
||||
getAVCInfo(m_strSPS, m_iVideoWidth, m_iVideoHeight, m_fVideoFps);
|
||||
}
|
||||
}
|
||||
break;
|
||||
case 8://PPS
|
||||
m_strPPS=frame.buffer;
|
||||
break;
|
||||
case 1:
|
||||
//B or P
|
||||
onGetH264(frame);
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
inline void RtpParser::onGetH264(H264Frame& frame) {
|
||||
//frame.timeStamp=ticker0.elapsedTime();
|
||||
lock_guard<recursive_mutex> lck(m_mtxCB);
|
||||
if (onVideo) {
|
||||
onVideo(frame);
|
||||
}
|
||||
}
|
||||
|
||||
inline void RtpParser::onGetAdts(AACFrame& frame) {
|
||||
//frame.timeStamp=ticker1.elapsedTime();
|
||||
lock_guard<recursive_mutex> lck(m_mtxCB);
|
||||
if (onAudio) {
|
||||
onAudio(frame);
|
||||
}
|
||||
inline bool RtpParser::inputAudio(const RtpPacket::Ptr &rtppack, const RtspTrack& track) {
|
||||
}
|
||||
|
||||
} /* namespace Rtsp */
|
||||
|
||||
@@ -47,57 +47,12 @@ public:
|
||||
RtpParser(const string &sdp);
|
||||
virtual ~RtpParser();
|
||||
//返回值:true 代表是i帧第一个rtp包
|
||||
bool inputRtp(const RtpPacket &rtp);
|
||||
void setOnVideoCB(const function<void(const H264Frame &frame)> &cb) override{
|
||||
lock_guard<recursive_mutex> lck(m_mtxCB);
|
||||
onVideo = cb;
|
||||
}
|
||||
void setOnAudioCB(const function<void(const AACFrame &frame)> &cb) override{
|
||||
lock_guard<recursive_mutex> lck(m_mtxCB);
|
||||
onAudio = cb;
|
||||
bool inputRtp(const RtpPacket::Ptr &rtp);
|
||||
|
||||
float getDuration() const override {
|
||||
return m_fDuration;
|
||||
}
|
||||
|
||||
int getVideoHeight() const override{
|
||||
return m_iVideoHeight;
|
||||
}
|
||||
|
||||
int getVideoWidth() const override{
|
||||
return m_iVideoWidth;
|
||||
}
|
||||
|
||||
float getVideoFps() const override{
|
||||
return m_fVideoFps;
|
||||
}
|
||||
|
||||
int getAudioSampleRate() const override{
|
||||
return m_iSampleRate;
|
||||
}
|
||||
|
||||
int getAudioSampleBit() const override{
|
||||
return m_iSampleBit;
|
||||
}
|
||||
|
||||
int getAudioChannel() const override{
|
||||
return m_iChannel;
|
||||
}
|
||||
|
||||
const string& getPps() const override{
|
||||
return m_strPPS;
|
||||
}
|
||||
|
||||
const string& getSps() const override{
|
||||
return m_strSPS;
|
||||
}
|
||||
|
||||
const string& getAudioCfg() const override{
|
||||
return m_strAudioCfg;
|
||||
}
|
||||
bool containAudio() const override{
|
||||
return m_bHaveAudio;
|
||||
}
|
||||
bool containVideo() const override{
|
||||
return m_bHaveVideo;
|
||||
}
|
||||
bool isInited() const override{
|
||||
if (m_bHaveAudio && !m_strAudioCfg.size()) {
|
||||
return false;
|
||||
@@ -107,9 +62,6 @@ public:
|
||||
}
|
||||
return true;
|
||||
}
|
||||
float getDuration() const override {
|
||||
return m_fDuration;
|
||||
}
|
||||
private:
|
||||
std::unordered_map<uint8_t, RtspTrack> m_mapTracks;
|
||||
|
||||
@@ -117,33 +69,16 @@ private:
|
||||
inline void onGetVideoTrack(const RtspTrack &video);
|
||||
|
||||
//返回值:true 代表是i帧第一个rtp包
|
||||
inline bool inputVideo(const RtpPacket &rtp, const RtspTrack &track);
|
||||
inline bool inputAudio(const RtpPacket &rtp, const RtspTrack &track);
|
||||
inline void _onGetH264(H264Frame &frame);
|
||||
inline void onGetH264(H264Frame &frame);
|
||||
inline void onGetAdts(AACFrame &frame);
|
||||
//video
|
||||
H264Frame m_h264frame;
|
||||
//aduio
|
||||
AACFrame m_adts;
|
||||
|
||||
int m_iSampleRate = 44100;
|
||||
int m_iSampleBit = 16;
|
||||
int m_iChannel = 1;
|
||||
inline bool inputVideo(const RtpPacket::Ptr &rtp, const RtspTrack &track);
|
||||
inline bool inputAudio(const RtpPacket::Ptr &rtp, const RtspTrack &track);
|
||||
|
||||
string m_strSPS;
|
||||
string m_strPPS;
|
||||
string m_strAudioCfg;
|
||||
int m_iVideoWidth = 0;
|
||||
int m_iVideoHeight = 0;
|
||||
float m_fVideoFps = 0;
|
||||
bool m_bHaveAudio = false;
|
||||
bool m_bHaveVideo= false;
|
||||
float m_fDuration = 0;
|
||||
bool m_bParseSpsDelay = false;
|
||||
|
||||
function<void(const H264Frame &frame)> onVideo;
|
||||
function<void(const AACFrame &frame)> onAudio;
|
||||
recursive_mutex m_mtxCB;
|
||||
};
|
||||
|
||||
|
||||
@@ -108,12 +108,12 @@ public:
|
||||
* @param mtu mtu大小,一般小于1500字节,推荐1400
|
||||
*/
|
||||
virtual void createRtpEncoder(uint32_t ssrc, int mtu) {
|
||||
_encoder = RtpCodec::getRtpCodec (getCodecId(),
|
||||
ssrc,
|
||||
mtu,
|
||||
_sample_rate,
|
||||
_playload_type,
|
||||
getTrackType() * 2);
|
||||
_encoder = RtpCodec::getRtpCodecById(getCodecId(),
|
||||
ssrc,
|
||||
mtu,
|
||||
_sample_rate,
|
||||
_playload_type,
|
||||
getTrackType() * 2);
|
||||
}
|
||||
private:
|
||||
RtpCodec::Ptr _encoder;
|
||||
@@ -275,10 +275,8 @@ public:
|
||||
* 构成函数
|
||||
*/
|
||||
RtspEncoder(){
|
||||
//自适应缓存
|
||||
_rtpRing = std::make_shared<RtpRingInterface::RingType>(0);
|
||||
//禁用缓存
|
||||
_frameRing = std::make_shared<FrameRingInterface::RingType>(1);
|
||||
_rtpRing = std::make_shared<RtpRingInterface::RingType>();
|
||||
_frameRing = std::make_shared<FrameRingInterface::RingType>();
|
||||
}
|
||||
virtual ~RtspEncoder(){}
|
||||
|
||||
|
||||
@@ -442,7 +442,8 @@ bool RtspPlayer::sendPause(bool bPause,float fTime){
|
||||
if (m_aTrackInfo[i].type == TrackVideo) {
|
||||
m_adFistStamp[i] = m_adNowStamp[i] + iTimeInc * 90000.0;
|
||||
}else if (m_aTrackInfo[i].type == TrackAudio){
|
||||
m_adFistStamp[i] = m_adNowStamp[i] + iTimeInc * getAudioSampleRate();
|
||||
//todo(xzl) 修复此处
|
||||
// m_adFistStamp[i] = m_adNowStamp[i] + iTimeInc * getAudioSampleRate();
|
||||
}
|
||||
m_adNowStamp[i] = m_adFistStamp[i];
|
||||
}
|
||||
@@ -718,7 +719,10 @@ float RtspPlayer::getProgressTime() const{
|
||||
if (m_aTrackInfo[i].type == TrackVideo) {
|
||||
iTime[i] = (m_adNowStamp[i] - m_adFistStamp[i]) / 90000.0;
|
||||
}else if (m_aTrackInfo[i].type == TrackAudio){
|
||||
iTime[i] = (m_adNowStamp[i] - m_adFistStamp[i]) / getAudioSampleRate();
|
||||
//todo(xzl) 修复此处
|
||||
#if 0
|
||||
iTime[i] = (m_adNowStamp[i] - m_adFistStamp[i]) / getAudioSampleRate();
|
||||
#endif
|
||||
}
|
||||
}
|
||||
return m_fSeekTo + MAX(iTime[0],iTime[1]);
|
||||
|
||||
@@ -71,8 +71,9 @@ private:
|
||||
}
|
||||
try {
|
||||
m_parser.reset(new RtpParser(sdp));
|
||||
m_parser->setOnVideoCB(m_onGetVideoCB);
|
||||
m_parser->setOnAudioCB(m_onGetAudioCB);
|
||||
//todo(xzl) 修复此处
|
||||
// m_parser->setOnVideoCB(m_onGetVideoCB);
|
||||
// m_parser->setOnAudioCB(m_onGetAudioCB);
|
||||
return true;
|
||||
} catch (std::exception &ex) {
|
||||
WarnL << ex.what();
|
||||
@@ -81,7 +82,7 @@ private:
|
||||
}
|
||||
void onRecvRTP(const RtpPacket::Ptr &rtppt, const RtspTrack &track) override {
|
||||
if(m_parser){
|
||||
m_parser->inputRtp(*rtppt);
|
||||
m_parser->inputRtp(rtppt);
|
||||
}
|
||||
|
||||
if(m_pRtspMediaSrc){
|
||||
|
||||
@@ -63,8 +63,10 @@ void RtspToRtmpMediaSource::makeVideoConfigPkt() {
|
||||
|
||||
////////////sps
|
||||
rtmpPkt->strBuf.push_back(1); // version
|
||||
string m_sps = m_pParser->getSps().substr(4);
|
||||
string m_pps = m_pParser->getPps().substr(4);
|
||||
|
||||
//todo(xzl) 修复此处
|
||||
string m_sps ;//= m_pParser->getSps().substr(4);
|
||||
string m_pps ;//= m_pParser->getPps().substr(4);
|
||||
//DebugL<<hexdump(m_sps.data(), m_sps.size());
|
||||
rtmpPkt->strBuf.push_back(m_sps[1]); // profile
|
||||
rtmpPkt->strBuf.push_back(m_sps[2]); // compat
|
||||
@@ -145,6 +147,8 @@ void RtspToRtmpMediaSource::onGetAdts(const AACFrame& frame) {
|
||||
}
|
||||
|
||||
void RtspToRtmpMediaSource::makeAudioConfigPkt() {
|
||||
//todo(xzl) 修复此处
|
||||
#if 0
|
||||
uint8_t flvStereoOrMono = (m_pParser->getAudioChannel() > 1);
|
||||
uint8_t flvSampleRate;
|
||||
switch (m_pParser->getAudioSampleRate()) {
|
||||
@@ -182,6 +186,8 @@ void RtspToRtmpMediaSource::makeAudioConfigPkt() {
|
||||
rtmpPkt->timeStamp = 0;
|
||||
rtmpPkt->typeId = MSG_AUDIO;
|
||||
m_pRtmpSrc->onGetMedia(rtmpPkt);
|
||||
|
||||
#endif
|
||||
}
|
||||
|
||||
void RtspToRtmpMediaSource::makeMetaData() {
|
||||
@@ -190,6 +196,9 @@ void RtspToRtmpMediaSource::makeMetaData() {
|
||||
AMFValue metaData(AMF_OBJECT);
|
||||
metaData.set("duration", m_pParser->getDuration());
|
||||
metaData.set("fileSize", 0);
|
||||
//todo(xzl) 修复此处
|
||||
#if 0
|
||||
|
||||
if (m_pParser->containVideo()) {
|
||||
metaData.set("width", m_pParser->getVideoWidth());
|
||||
metaData.set("height", m_pParser->getVideoHeight());
|
||||
@@ -208,6 +217,7 @@ void RtspToRtmpMediaSource::makeMetaData() {
|
||||
makeAudioConfigPkt();
|
||||
}
|
||||
|
||||
#endif
|
||||
m_pRtmpSrc->onGetMetaData(metaData);
|
||||
}
|
||||
} /* namespace Rtsp */
|
||||
|
||||
@@ -54,8 +54,9 @@ public:
|
||||
try {
|
||||
m_pParser.reset(new RtpParser(strSdp));
|
||||
m_pRecorder.reset(new MediaRecorder(getVhost(),getApp(),getId(),m_pParser,m_bEnableHls,m_bEnableMp4));
|
||||
m_pParser->setOnAudioCB( std::bind(&RtspToRtmpMediaSource::onGetAdts, this, placeholders::_1));
|
||||
m_pParser->setOnVideoCB( std::bind(&RtspToRtmpMediaSource::onGetH264, this, placeholders::_1));
|
||||
//todo(xzl) 修复此处
|
||||
// m_pParser->setOnAudioCB( std::bind(&RtspToRtmpMediaSource::onGetAdts, this, placeholders::_1));
|
||||
// m_pParser->setOnVideoCB( std::bind(&RtspToRtmpMediaSource::onGetH264, this, placeholders::_1));
|
||||
makeMetaData();
|
||||
} catch (exception &ex) {
|
||||
WarnL << ex.what();
|
||||
@@ -64,7 +65,7 @@ public:
|
||||
}
|
||||
virtual void onGetRTP(const RtpPacket::Ptr &pRtppkt, bool bKeyPos) override{
|
||||
if (m_pParser) {
|
||||
bKeyPos = m_pParser->inputRtp(*pRtppkt);
|
||||
bKeyPos = m_pParser->inputRtp(pRtppkt);
|
||||
}
|
||||
RtspMediaSource::onGetRTP(pRtppkt, bKeyPos);
|
||||
}
|
||||
@@ -77,7 +78,8 @@ public:
|
||||
for (auto &pr : m_mapTracks) {
|
||||
switch (pr.second.type) {
|
||||
case TrackAudio: {
|
||||
pr.second.timeStamp = uiStamp * (m_pParser->getAudioSampleRate() / 1000.0);
|
||||
//todo(xzl) 修复此处
|
||||
// pr.second.timeStamp = uiStamp * (m_pParser->getAudioSampleRate() / 1000.0);
|
||||
}
|
||||
break;
|
||||
case TrackVideo: {
|
||||
@@ -89,6 +91,14 @@ public:
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
protected:
|
||||
void onGetH264(const H264Frame &frame);
|
||||
void onGetAdts(const AACFrame &frame);
|
||||
private:
|
||||
void makeVideoConfigPkt();
|
||||
void makeAudioConfigPkt();
|
||||
void makeMetaData();
|
||||
private:
|
||||
RtpParser::Ptr m_pParser;
|
||||
RtmpMediaSource::Ptr m_pRtmpSrc;
|
||||
@@ -96,11 +106,7 @@ private:
|
||||
MediaRecorder::Ptr m_pRecorder;
|
||||
bool m_bEnableHls;
|
||||
bool m_bEnableMp4;
|
||||
void onGetH264(const H264Frame &frame);
|
||||
void onGetAdts(const AACFrame &frame);
|
||||
void makeVideoConfigPkt();
|
||||
void makeAudioConfigPkt();
|
||||
void makeMetaData();
|
||||
|
||||
};
|
||||
|
||||
} /* namespace Rtsp */
|
||||
|
||||
Reference in New Issue
Block a user