2018-10-25 10:00:17 +08:00
|
|
|
|
/*
|
2020-04-04 20:30:09 +08:00
|
|
|
|
* Copyright (c) 2016 The ZLMediaKit project authors. All Rights Reserved.
|
2018-10-25 10:00:17 +08:00
|
|
|
|
*
|
|
|
|
|
* This file is part of ZLMediaKit(https://github.com/xiongziliang/ZLMediaKit).
|
|
|
|
|
*
|
2020-04-04 20:30:09 +08:00
|
|
|
|
* Use of this source code is governed by MIT license that can be found in the
|
|
|
|
|
* LICENSE file in the root of the source tree. All contributing project authors
|
|
|
|
|
* may be found in the AUTHORS file in the root of the source tree.
|
2018-10-25 10:00:17 +08:00
|
|
|
|
*/
|
2018-10-24 09:23:57 +08:00
|
|
|
|
|
|
|
|
|
#include "Factory.h"
|
2020-04-04 22:54:49 +08:00
|
|
|
|
#include "Rtmp/Rtmp.h"
|
2019-06-28 17:37:11 +08:00
|
|
|
|
#include "H264Rtmp.h"
|
2020-04-04 22:54:49 +08:00
|
|
|
|
#include "H265Rtmp.h"
|
2019-06-28 17:37:11 +08:00
|
|
|
|
#include "AACRtmp.h"
|
2020-04-08 15:42:52 +08:00
|
|
|
|
#include "G711Rtmp.h"
|
2019-06-28 17:37:11 +08:00
|
|
|
|
#include "H264Rtp.h"
|
|
|
|
|
#include "AACRtp.h"
|
2020-04-08 15:42:52 +08:00
|
|
|
|
#include "G711Rtp.h"
|
2019-06-28 17:37:11 +08:00
|
|
|
|
#include "H265Rtp.h"
|
2019-09-05 14:56:33 +08:00
|
|
|
|
#include "Common/Parser.h"
|
2018-10-24 09:23:57 +08:00
|
|
|
|
|
2018-10-24 17:17:55 +08:00
|
|
|
|
namespace mediakit{
|
|
|
|
|
|
2018-10-26 09:56:29 +08:00
|
|
|
|
Track::Ptr Factory::getTrackBySdp(const SdpTrack::Ptr &track) {
|
2018-11-27 11:05:44 +08:00
|
|
|
|
if (strcasecmp(track->_codec.data(), "mpeg4-generic") == 0) {
|
2019-03-27 18:41:52 +08:00
|
|
|
|
string aac_cfg_str = FindField(track->_fmtp.data(), "config=", nullptr);
|
2019-08-06 16:23:20 +08:00
|
|
|
|
if (aac_cfg_str.empty()) {
|
2019-03-27 18:41:52 +08:00
|
|
|
|
aac_cfg_str = FindField(track->_fmtp.data(), "config=", ";");
|
2018-10-24 09:23:57 +08:00
|
|
|
|
}
|
2019-08-06 16:23:20 +08:00
|
|
|
|
if (aac_cfg_str.empty()) {
|
2019-10-14 16:52:15 +08:00
|
|
|
|
//如果sdp中获取不到aac config信息,那么在rtp也无法获取,那么忽略该Track
|
|
|
|
|
return nullptr;
|
2018-10-24 09:23:57 +08:00
|
|
|
|
}
|
|
|
|
|
string aac_cfg;
|
|
|
|
|
|
|
|
|
|
unsigned int cfg1;
|
2019-03-27 18:41:52 +08:00
|
|
|
|
sscanf(aac_cfg_str.substr(0, 2).data(), "%02X", &cfg1);
|
2018-10-24 09:23:57 +08:00
|
|
|
|
cfg1 &= 0x00FF;
|
|
|
|
|
aac_cfg.push_back(cfg1);
|
|
|
|
|
|
|
|
|
|
unsigned int cfg2;
|
2019-03-27 18:41:52 +08:00
|
|
|
|
sscanf(aac_cfg_str.substr(2, 2).data(), "%02X", &cfg2);
|
2018-10-24 09:23:57 +08:00
|
|
|
|
cfg2 &= 0x00FF;
|
|
|
|
|
aac_cfg.push_back(cfg2);
|
|
|
|
|
|
|
|
|
|
return std::make_shared<AACTrack>(aac_cfg);
|
|
|
|
|
}
|
|
|
|
|
|
2020-04-08 15:42:52 +08:00
|
|
|
|
if (strcasecmp(track->_codec.data(), "PCMA") == 0) {
|
|
|
|
|
return std::make_shared<G711Track>(CodecG711A);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (strcasecmp(track->_codec.data(), "PCMU") == 0) {
|
|
|
|
|
return std::make_shared<G711Track>(CodecG711U);
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-27 11:05:44 +08:00
|
|
|
|
if (strcasecmp(track->_codec.data(), "h264") == 0) {
|
2019-10-08 09:39:54 +08:00
|
|
|
|
//a=fmtp:96 packetization-mode=1;profile-level-id=42C01F;sprop-parameter-sets=Z0LAH9oBQBboQAAAAwBAAAAPI8YMqA==,aM48gA==
|
2019-10-08 09:44:08 +08:00
|
|
|
|
auto map = Parser::parseArgs(FindField(track->_fmtp.data()," ", nullptr),";","=");
|
2019-09-25 22:16:35 +08:00
|
|
|
|
auto sps_pps = map["sprop-parameter-sets"];
|
2019-03-27 18:41:52 +08:00
|
|
|
|
string base64_SPS = FindField(sps_pps.data(), NULL, ",");
|
|
|
|
|
string base64_PPS = FindField(sps_pps.data(), ",", NULL);
|
2018-10-24 09:23:57 +08:00
|
|
|
|
auto sps = decodeBase64(base64_SPS);
|
|
|
|
|
auto pps = decodeBase64(base64_PPS);
|
2019-11-21 15:11:43 +08:00
|
|
|
|
if(sps.empty() || pps.empty()){
|
2019-11-21 16:31:50 +08:00
|
|
|
|
//如果sdp里面没有sps/pps,那么可能在后续的rtp里面恢复出sps/pps
|
2019-11-21 15:11:43 +08:00
|
|
|
|
return std::make_shared<H264Track>();
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-24 09:23:57 +08:00
|
|
|
|
return std::make_shared<H264Track>(sps,pps,0,0);
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-27 11:05:44 +08:00
|
|
|
|
if (strcasecmp(track->_codec.data(), "h265") == 0) {
|
2018-10-30 17:11:36 +08:00
|
|
|
|
//a=fmtp:96 sprop-sps=QgEBAWAAAAMAsAAAAwAAAwBdoAKAgC0WNrkky/AIAAADAAgAAAMBlQg=; sprop-pps=RAHA8vA8kAA=
|
2019-10-08 09:44:08 +08:00
|
|
|
|
auto map = Parser::parseArgs(FindField(track->_fmtp.data()," ", nullptr),";","=");
|
2019-09-05 14:56:33 +08:00
|
|
|
|
auto vps = decodeBase64(map["sprop-vps"]);
|
|
|
|
|
auto sps = decodeBase64(map["sprop-sps"]);
|
|
|
|
|
auto pps = decodeBase64(map["sprop-pps"]);
|
2019-11-21 16:31:50 +08:00
|
|
|
|
if(sps.empty() || pps.empty()){
|
|
|
|
|
//如果sdp里面没有sps/pps,那么可能在后续的rtp里面恢复出sps/pps
|
|
|
|
|
return std::make_shared<H265Track>();
|
|
|
|
|
}
|
2019-09-05 14:56:33 +08:00
|
|
|
|
return std::make_shared<H265Track>(vps,sps,pps,0,0,0);
|
2018-10-24 18:41:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-03-26 17:12:21 +08:00
|
|
|
|
WarnL << "暂不支持该sdp:" << track->getName();
|
2018-10-30 17:11:36 +08:00
|
|
|
|
return nullptr;
|
2018-10-24 18:41:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Track::Ptr Factory::getTrackByCodecId(CodecId codecId) {
|
|
|
|
|
switch (codecId){
|
|
|
|
|
case CodecH264:{
|
|
|
|
|
return std::make_shared<H264Track>();
|
|
|
|
|
}
|
2018-10-30 17:11:36 +08:00
|
|
|
|
case CodecH265:{
|
|
|
|
|
return std::make_shared<H265Track>();
|
|
|
|
|
}
|
2018-10-24 18:41:37 +08:00
|
|
|
|
case CodecAAC:{
|
|
|
|
|
return std::make_shared<AACTrack>();
|
|
|
|
|
}
|
2020-04-08 15:42:52 +08:00
|
|
|
|
case CodecG711A: {
|
|
|
|
|
return std::make_shared<G711Track>(CodecG711A);
|
|
|
|
|
}
|
|
|
|
|
case CodecG711U: {
|
|
|
|
|
return std::make_shared<G711Track>(CodecG711U);
|
|
|
|
|
}
|
2018-10-24 18:41:37 +08:00
|
|
|
|
default:
|
2018-10-25 09:26:11 +08:00
|
|
|
|
WarnL << "暂不支持该CodecId:" << codecId;
|
2018-10-24 18:41:37 +08:00
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-06-28 16:12:39 +08:00
|
|
|
|
RtpCodec::Ptr Factory::getRtpEncoderBySdp(const Sdp::Ptr &sdp) {
|
|
|
|
|
GET_CONFIG(uint32_t,audio_mtu,Rtp::kAudioMtuSize);
|
|
|
|
|
GET_CONFIG(uint32_t,video_mtu,Rtp::kVideoMtuSize);
|
2019-07-19 09:42:48 +08:00
|
|
|
|
// ssrc不冲突即可,可以为任意的32位整形
|
|
|
|
|
static atomic<uint32_t> s_ssrc(0);
|
|
|
|
|
uint32_t ssrc = s_ssrc++;
|
|
|
|
|
if(!ssrc){
|
|
|
|
|
//ssrc不能为0
|
|
|
|
|
ssrc = 1;
|
|
|
|
|
}
|
|
|
|
|
if(sdp->getTrackType() == TrackVideo){
|
|
|
|
|
//视频的ssrc是偶数,方便调试
|
|
|
|
|
ssrc = 2 * ssrc;
|
|
|
|
|
}else{
|
|
|
|
|
//音频ssrc是奇数
|
|
|
|
|
ssrc = 2 * ssrc + 1;
|
|
|
|
|
}
|
2019-06-28 16:12:39 +08:00
|
|
|
|
auto mtu = (sdp->getTrackType() == TrackVideo ? video_mtu : audio_mtu);
|
|
|
|
|
auto sample_rate = sdp->getSampleRate();
|
|
|
|
|
auto pt = sdp->getPlayloadType();
|
|
|
|
|
auto interleaved = sdp->getTrackType() * 2;
|
|
|
|
|
auto codec_id = sdp->getCodecId();
|
|
|
|
|
switch (codec_id){
|
2018-10-24 09:23:57 +08:00
|
|
|
|
case CodecH264:
|
2019-06-28 16:12:39 +08:00
|
|
|
|
return std::make_shared<H264RtpEncoder>(ssrc,mtu,sample_rate,pt,interleaved);
|
2018-10-30 17:11:36 +08:00
|
|
|
|
case CodecH265:
|
2019-06-28 16:12:39 +08:00
|
|
|
|
return std::make_shared<H265RtpEncoder>(ssrc,mtu,sample_rate,pt,interleaved);
|
2018-10-24 09:23:57 +08:00
|
|
|
|
case CodecAAC:
|
2019-06-28 16:12:39 +08:00
|
|
|
|
return std::make_shared<AACRtpEncoder>(ssrc,mtu,sample_rate,pt,interleaved);
|
2020-04-08 15:42:52 +08:00
|
|
|
|
case CodecG711A:
|
|
|
|
|
case CodecG711U:
|
|
|
|
|
return std::make_shared<G711RtpEncoder>(ssrc, mtu, sample_rate, pt, interleaved);
|
2018-10-24 09:23:57 +08:00
|
|
|
|
default:
|
2019-06-28 16:12:39 +08:00
|
|
|
|
WarnL << "暂不支持该CodecId:" << codec_id;
|
2018-10-24 09:23:57 +08:00
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-24 12:21:29 +08:00
|
|
|
|
RtpCodec::Ptr Factory::getRtpDecoderByTrack(const Track::Ptr &track) {
|
|
|
|
|
switch (track->getCodecId()){
|
2018-10-24 09:23:57 +08:00
|
|
|
|
case CodecH264:
|
|
|
|
|
return std::make_shared<H264RtpDecoder>();
|
2018-10-30 17:11:36 +08:00
|
|
|
|
case CodecH265:
|
|
|
|
|
return std::make_shared<H265RtpDecoder>();
|
2018-10-24 09:23:57 +08:00
|
|
|
|
case CodecAAC:
|
2019-01-24 12:21:29 +08:00
|
|
|
|
return std::make_shared<AACRtpDecoder>(track->clone());
|
2020-04-08 15:42:52 +08:00
|
|
|
|
case CodecG711A:
|
|
|
|
|
case CodecG711U:
|
|
|
|
|
return std::make_shared<G711RtpDecoder>(track->clone());
|
2018-10-24 09:23:57 +08:00
|
|
|
|
default:
|
2020-03-08 21:19:20 +08:00
|
|
|
|
WarnL << "暂不支持该CodecId:" << track->getCodecName();
|
2018-10-24 09:23:57 +08:00
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-30 17:11:36 +08:00
|
|
|
|
/////////////////////////////rtmp相关///////////////////////////////////////////
|
|
|
|
|
|
2020-04-18 18:46:20 +08:00
|
|
|
|
static CodecId getVideoCodecIdByAmf(const AMFValue &val){
|
2018-10-30 17:11:36 +08:00
|
|
|
|
if (val.type() == AMF_STRING){
|
|
|
|
|
auto str = val.as_string();
|
|
|
|
|
if(str == "avc1"){
|
|
|
|
|
return CodecH264;
|
|
|
|
|
}
|
|
|
|
|
if(str == "mp4a"){
|
|
|
|
|
return CodecAAC;
|
|
|
|
|
}
|
2019-12-12 22:25:55 +08:00
|
|
|
|
if(str == "hev1" || str == "hvc1"){
|
|
|
|
|
return CodecH265;
|
|
|
|
|
}
|
2018-10-30 17:11:36 +08:00
|
|
|
|
WarnL << "暂不支持该Amf:" << str;
|
|
|
|
|
return CodecInvalid;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (val.type() != AMF_NULL){
|
|
|
|
|
auto type_id = val.as_integer();
|
|
|
|
|
switch (type_id){
|
2020-04-04 22:54:49 +08:00
|
|
|
|
case FLV_CODEC_H264: return CodecH264;
|
|
|
|
|
case FLV_CODEC_AAC: return CodecAAC;
|
|
|
|
|
case FLV_CODEC_H265: return CodecH265;
|
2018-10-30 17:11:36 +08:00
|
|
|
|
default:
|
|
|
|
|
WarnL << "暂不支持该Amf:" << type_id;
|
|
|
|
|
return CodecInvalid;
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-11-15 15:14:05 +08:00
|
|
|
|
|
2018-10-30 17:11:36 +08:00
|
|
|
|
return CodecInvalid;
|
|
|
|
|
}
|
|
|
|
|
|
2020-04-18 18:46:20 +08:00
|
|
|
|
Track::Ptr Factory::getVideoTrackByAmf(const AMFValue &amf) {
|
|
|
|
|
CodecId codecId = getVideoCodecIdByAmf(amf);
|
|
|
|
|
if(codecId == CodecInvalid){
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
return getTrackByCodecId(codecId);
|
|
|
|
|
}
|
2018-10-30 17:11:36 +08:00
|
|
|
|
|
2020-04-18 18:46:20 +08:00
|
|
|
|
static CodecId getAudioCodecIdByAmf(const AMFValue &val) {
|
2020-04-17 17:47:10 +08:00
|
|
|
|
if (val.type() == AMF_STRING) {
|
|
|
|
|
auto str = val.as_string();
|
|
|
|
|
if (str == "mp4a") {
|
|
|
|
|
return CodecAAC;
|
|
|
|
|
}
|
|
|
|
|
WarnL << "暂不支持该Amf:" << str;
|
|
|
|
|
return CodecInvalid;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (val.type() != AMF_NULL) {
|
|
|
|
|
auto type_id = val.as_integer();
|
|
|
|
|
switch (type_id) {
|
2020-04-18 18:46:20 +08:00
|
|
|
|
case FLV_CODEC_AAC : return CodecAAC;
|
|
|
|
|
case FLV_CODEC_G711A : return CodecG711A;
|
|
|
|
|
case FLV_CODEC_G711U : return CodecG711U;
|
|
|
|
|
default : WarnL << "暂不支持该Amf:" << type_id; return CodecInvalid;
|
2020-04-17 17:47:10 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return CodecInvalid;
|
|
|
|
|
}
|
|
|
|
|
|
2020-04-18 18:46:20 +08:00
|
|
|
|
Track::Ptr Factory::getAudioTrackByAmf(const AMFValue& amf){
|
|
|
|
|
CodecId codecId = getAudioCodecIdByAmf(amf);
|
|
|
|
|
if (codecId == CodecInvalid) {
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
return getTrackByCodecId(codecId);
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-25 14:16:40 +08:00
|
|
|
|
RtmpCodec::Ptr Factory::getRtmpCodecByTrack(const Track::Ptr &track) {
|
|
|
|
|
switch (track->getCodecId()){
|
2018-10-24 22:03:17 +08:00
|
|
|
|
case CodecH264:
|
2018-10-25 14:16:40 +08:00
|
|
|
|
return std::make_shared<H264RtmpEncoder>(track);
|
2018-10-24 22:03:17 +08:00
|
|
|
|
case CodecAAC:
|
2018-10-25 14:16:40 +08:00
|
|
|
|
return std::make_shared<AACRtmpEncoder>(track);
|
2020-04-04 22:54:49 +08:00
|
|
|
|
case CodecH265:
|
|
|
|
|
return std::make_shared<H265RtmpEncoder>(track);
|
2020-04-08 15:42:52 +08:00
|
|
|
|
case CodecG711A:
|
|
|
|
|
case CodecG711U:
|
|
|
|
|
return std::make_shared<G711RtmpEncoder>(track);
|
2018-10-24 22:03:17 +08:00
|
|
|
|
default:
|
2020-03-08 21:19:20 +08:00
|
|
|
|
WarnL << "暂不支持该CodecId:" << track->getCodecName();
|
2018-10-24 22:03:17 +08:00
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-25 14:49:47 +08:00
|
|
|
|
AMFValue Factory::getAmfByCodecId(CodecId codecId) {
|
|
|
|
|
switch (codecId){
|
2020-04-18 18:46:20 +08:00
|
|
|
|
//此处用string标明rtmp编码类型目的是为了兼容某些android系统
|
2019-12-12 22:25:55 +08:00
|
|
|
|
case CodecAAC: return AMFValue("mp4a");
|
|
|
|
|
case CodecH264: return AMFValue("avc1");
|
2020-04-04 22:54:49 +08:00
|
|
|
|
case CodecH265: return AMFValue(FLV_CODEC_H265);
|
2020-04-17 17:47:10 +08:00
|
|
|
|
case CodecG711A: return AMFValue(FLV_CODEC_G711A);
|
|
|
|
|
case CodecG711U: return AMFValue(FLV_CODEC_G711U);
|
2019-12-12 22:25:55 +08:00
|
|
|
|
default: return AMFValue(AMF_NULL);
|
2018-10-25 14:49:47 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2018-10-24 18:41:37 +08:00
|
|
|
|
|
2018-10-24 17:17:55 +08:00
|
|
|
|
}//namespace mediakit
|
|
|
|
|
|