Factory.cpp 7.77 KB
Newer Older
1 2 3
/*
 * MIT License
 *
xiongziliang committed
4
 * Copyright (c) 2016-2019 xiongziliang <771730766@qq.com>
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
 *
 * This file is part of ZLMediaKit(https://github.com/xiongziliang/ZLMediaKit).
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in all
 * copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 * SOFTWARE.
 */
26 27

#include "Factory.h"
xiongziliang committed
28 29 30 31 32
#include "H264Rtmp.h"
#include "AACRtmp.h"
#include "H264Rtp.h"
#include "AACRtp.h"
#include "H265Rtp.h"
33
#include "Common/Parser.h"
34

xiongziliang committed
35 36
namespace mediakit{

xiongziliang committed
37
Track::Ptr Factory::getTrackBySdp(const SdpTrack::Ptr &track) {
xiongziliang committed
38
    if (strcasecmp(track->_codec.data(), "mpeg4-generic") == 0) {
xiongziliang committed
39
        string aac_cfg_str = FindField(track->_fmtp.data(), "config=", nullptr);
40
        if (aac_cfg_str.empty()) {
xiongziliang committed
41
            aac_cfg_str = FindField(track->_fmtp.data(), "config=", ";");
42
        }
43
        if (aac_cfg_str.empty()) {
44 45
            //如果sdp中获取不到aac config信息,那么在rtp也无法获取,那么忽略该Track
            return nullptr;
46 47 48 49
        }
        string aac_cfg;

        unsigned int cfg1;
xiongziliang committed
50
        sscanf(aac_cfg_str.substr(0, 2).data(), "%02X", &cfg1);
51 52 53 54
        cfg1 &= 0x00FF;
        aac_cfg.push_back(cfg1);

        unsigned int cfg2;
xiongziliang committed
55
        sscanf(aac_cfg_str.substr(2, 2).data(), "%02X", &cfg2);
56 57 58 59 60 61
        cfg2 &= 0x00FF;
        aac_cfg.push_back(cfg2);

        return std::make_shared<AACTrack>(aac_cfg);
    }

xiongziliang committed
62
    if (strcasecmp(track->_codec.data(), "h264") == 0) {
63
        //a=fmtp:96 packetization-mode=1;profile-level-id=42C01F;sprop-parameter-sets=Z0LAH9oBQBboQAAAAwBAAAAPI8YMqA==,aM48gA==
64
        auto map = Parser::parseArgs(FindField(track->_fmtp.data()," ", nullptr),";","=");
65
        auto sps_pps = map["sprop-parameter-sets"];
xiongziliang committed
66 67
        string base64_SPS = FindField(sps_pps.data(), NULL, ",");
        string base64_PPS = FindField(sps_pps.data(), ",", NULL);
68 69
        auto sps = decodeBase64(base64_SPS);
        auto pps = decodeBase64(base64_PPS);
70
        if(sps.empty() || pps.empty()){
71
            //如果sdp里面没有sps/pps,那么可能在后续的rtp里面恢复出sps/pps
72 73 74
            return std::make_shared<H264Track>();
        }

75 76 77
        return std::make_shared<H264Track>(sps,pps,0,0);
    }

xiongziliang committed
78
    if (strcasecmp(track->_codec.data(), "h265") == 0) {
79
        //a=fmtp:96 sprop-sps=QgEBAWAAAAMAsAAAAwAAAwBdoAKAgC0WNrkky/AIAAADAAgAAAMBlQg=; sprop-pps=RAHA8vA8kAA=
80
        auto map = Parser::parseArgs(FindField(track->_fmtp.data()," ", nullptr),";","=");
81 82 83
        auto vps = decodeBase64(map["sprop-vps"]);
        auto sps = decodeBase64(map["sprop-sps"]);
        auto pps = decodeBase64(map["sprop-pps"]);
84 85 86 87
        if(sps.empty() || pps.empty()){
            //如果sdp里面没有sps/pps,那么可能在后续的rtp里面恢复出sps/pps
            return std::make_shared<H265Track>();
        }
88
        return std::make_shared<H265Track>(vps,sps,pps,0,0,0);
89 90
    }

91 92 93

    WarnL << "暂不支持该sdp:" << track->_codec << " " << track->_fmtp;
    return nullptr;
94 95
}

96

97 98 99 100 101
Track::Ptr Factory::getTrackByCodecId(CodecId codecId) {
    switch (codecId){
        case CodecH264:{
            return std::make_shared<H264Track>();
        }
102 103 104
        case CodecH265:{
            return std::make_shared<H265Track>();
        }
105 106 107 108
        case CodecAAC:{
            return std::make_shared<AACTrack>();
        }
        default:
109
            WarnL << "暂不支持该CodecId:" << codecId;
110 111 112 113
            return nullptr;
    }
}

xiongziliang committed
114 115 116
RtpCodec::Ptr Factory::getRtpEncoderBySdp(const Sdp::Ptr &sdp) {
    GET_CONFIG(uint32_t,audio_mtu,Rtp::kAudioMtuSize);
    GET_CONFIG(uint32_t,video_mtu,Rtp::kVideoMtuSize);
xiongziliang committed
117 118 119 120 121 122 123 124 125 126 127 128 129 130
    // 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;
    }
xiongziliang committed
131 132 133 134 135 136
    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){
137
        case CodecH264:
xiongziliang committed
138
            return std::make_shared<H264RtpEncoder>(ssrc,mtu,sample_rate,pt,interleaved);
139
        case CodecH265:
xiongziliang committed
140
            return std::make_shared<H265RtpEncoder>(ssrc,mtu,sample_rate,pt,interleaved);
141
        case CodecAAC:
xiongziliang committed
142
            return std::make_shared<AACRtpEncoder>(ssrc,mtu,sample_rate,pt,interleaved);
143
        default:
xiongziliang committed
144
            WarnL << "暂不支持该CodecId:" << codec_id;
145 146 147 148
            return nullptr;
    }
}

149 150
RtpCodec::Ptr Factory::getRtpDecoderByTrack(const Track::Ptr &track) {
    switch (track->getCodecId()){
151 152
        case CodecH264:
            return std::make_shared<H264RtpDecoder>();
153 154
        case CodecH265:
            return std::make_shared<H265RtpDecoder>();
155
        case CodecAAC:
156
            return std::make_shared<AACRtpDecoder>(track->clone());
157
        default:
158
            WarnL << "暂不支持该CodecId:" << track->getCodecId();
159 160 161 162
            return nullptr;
    }
}

163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182
/////////////////////////////rtmp相关///////////////////////////////////////////

Track::Ptr Factory::getTrackByAmf(const AMFValue &amf) {
    CodecId codecId = getCodecIdByAmf(amf);
    if(codecId == CodecInvalid){
        return nullptr;
    }
    return getTrackByCodecId(codecId);
}


CodecId Factory::getCodecIdByAmf(const AMFValue &val){
    if (val.type() == AMF_STRING){
        auto str = val.as_string();
        if(str == "avc1"){
            return CodecH264;
        }
        if(str == "mp4a"){
            return CodecAAC;
        }
xiongziliang committed
183 184 185
        if(str == "hev1" || str == "hvc1"){
            return CodecH265;
        }
186 187 188 189 190 191 192
        WarnL << "暂不支持该Amf:" << str;
        return CodecInvalid;
    }

    if (val.type() != AMF_NULL){
        auto type_id = val.as_integer();
        switch (type_id){
xiongziliang committed
193 194 195
            case 7: return CodecH264;
            case 10: return CodecAAC;
            case 12: return CodecH265;
196 197 198 199
            default:
                WarnL << "暂不支持该Amf:" << type_id;
                return CodecInvalid;
        }
200
    }else{
201
        WarnL << "Metadata不存在相应的Track";
202
    }
203

204 205 206 207
    return CodecInvalid;
}


208 209
RtmpCodec::Ptr Factory::getRtmpCodecByTrack(const Track::Ptr &track) {
    switch (track->getCodecId()){
xiongziliang committed
210
        case CodecH264:
211
            return std::make_shared<H264RtmpEncoder>(track);
xiongziliang committed
212
        case CodecAAC:
213
            return std::make_shared<AACRtmpEncoder>(track);
xiongziliang committed
214
        default:
215
            WarnL << "暂不支持该CodecId:" << track->getCodecId();
xiongziliang committed
216 217 218 219
            return nullptr;
    }
}

220 221
AMFValue Factory::getAmfByCodecId(CodecId codecId) {
    switch (codecId){
xiongziliang committed
222 223 224 225
        case CodecAAC: return AMFValue("mp4a");
        case CodecH264: return AMFValue("avc1");
        case CodecH265: return AMFValue(12);
        default: return AMFValue(AMF_NULL);
226 227
    }
}
228

229

xiongziliang committed
230 231
}//namespace mediakit