2017-10-09 22:11:01 +08:00
|
|
|
|
/*
|
2020-04-04 20:30:09 +08:00
|
|
|
|
* Copyright (c) 2016 The ZLMediaKit project authors. All Rights Reserved.
|
|
|
|
|
|
*
|
2021-01-17 18:31:50 +08:00
|
|
|
|
* This file is part of ZLMediaKit(https://github.com/xia-chu/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.
|
|
|
|
|
|
*/
|
2017-04-01 16:35:56 +08:00
|
|
|
|
|
2018-10-24 18:09:54 +08:00
|
|
|
|
#include "RtmpDemuxer.h"
|
2018-10-30 14:59:42 +08:00
|
|
|
|
#include "Extension/Factory.h"
|
2017-04-01 16:35:56 +08:00
|
|
|
|
|
2018-10-24 17:17:55 +08:00
|
|
|
|
namespace mediakit {
|
2017-04-01 16:35:56 +08:00
|
|
|
|
|
2020-05-28 17:03:12 +08:00
|
|
|
|
bool RtmpDemuxer::loadMetaData(const AMFValue &val){
|
|
|
|
|
|
bool ret = false;
|
2019-12-26 12:10:54 +08:00
|
|
|
|
try {
|
2020-04-18 22:13:11 +08:00
|
|
|
|
int audiosamplerate = 0;
|
|
|
|
|
|
int audiochannels = 0;
|
|
|
|
|
|
int audiosamplesize = 0;
|
2020-12-05 12:22:17 +08:00
|
|
|
|
int videodatarate = 0;
|
|
|
|
|
|
int audiodatarate = 0;
|
2020-04-18 22:13:11 +08:00
|
|
|
|
const AMFValue *audiocodecid = nullptr;
|
|
|
|
|
|
const AMFValue *videocodecid = nullptr;
|
2019-12-26 12:10:54 +08:00
|
|
|
|
val.object_for_each([&](const string &key, const AMFValue &val) {
|
|
|
|
|
|
if (key == "duration") {
|
2021-01-17 18:31:50 +08:00
|
|
|
|
_fDuration = (float)val.as_number();
|
2019-12-26 12:10:54 +08:00
|
|
|
|
return;
|
|
|
|
|
|
}
|
2020-05-28 17:03:12 +08:00
|
|
|
|
if (key == "audiosamplerate") {
|
2020-04-18 22:13:11 +08:00
|
|
|
|
audiosamplerate = val.as_integer();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
2020-05-28 17:03:12 +08:00
|
|
|
|
if (key == "audiosamplesize") {
|
2020-04-18 22:13:11 +08:00
|
|
|
|
audiosamplesize = val.as_integer();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
2020-05-28 17:03:12 +08:00
|
|
|
|
if (key == "stereo") {
|
2020-04-18 22:13:11 +08:00
|
|
|
|
audiochannels = val.as_boolean() ? 2 : 1;
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
2020-05-28 17:03:12 +08:00
|
|
|
|
if (key == "videocodecid") {
|
2020-04-18 22:13:11 +08:00
|
|
|
|
//找到视频
|
|
|
|
|
|
videocodecid = &val;
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
2020-05-28 17:03:12 +08:00
|
|
|
|
if (key == "audiocodecid") {
|
2020-04-18 22:13:11 +08:00
|
|
|
|
//找到音频
|
|
|
|
|
|
audiocodecid = &val;
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
2020-12-05 12:22:17 +08:00
|
|
|
|
if (key == "audiodatarate") {
|
|
|
|
|
|
audiodatarate = val.as_integer();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (key == "videodatarate") {
|
|
|
|
|
|
videodatarate = val.as_integer();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
2019-12-26 12:10:54 +08:00
|
|
|
|
});
|
2020-05-28 17:03:12 +08:00
|
|
|
|
if (videocodecid) {
|
2020-04-18 22:13:11 +08:00
|
|
|
|
//有视频
|
2020-05-28 17:03:12 +08:00
|
|
|
|
ret = true;
|
2020-12-05 12:22:17 +08:00
|
|
|
|
makeVideoTrack(*videocodecid, videodatarate * 1024);
|
2020-04-18 22:13:11 +08:00
|
|
|
|
}
|
2020-05-28 17:03:12 +08:00
|
|
|
|
if (audiocodecid) {
|
2020-04-18 22:13:11 +08:00
|
|
|
|
//有音频
|
2020-05-28 17:03:12 +08:00
|
|
|
|
ret = true;
|
2020-12-05 12:22:17 +08:00
|
|
|
|
makeAudioTrack(*audiocodecid, audiosamplerate, audiochannels, audiosamplesize, audiodatarate * 1024);
|
2020-04-18 22:13:11 +08:00
|
|
|
|
}
|
2020-05-28 17:03:12 +08:00
|
|
|
|
} catch (std::exception &ex) {
|
2019-12-26 12:10:54 +08:00
|
|
|
|
WarnL << ex.what();
|
|
|
|
|
|
}
|
2020-05-28 17:03:12 +08:00
|
|
|
|
return ret;
|
2019-12-26 12:10:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2020-09-06 18:22:04 +08:00
|
|
|
|
void RtmpDemuxer::inputRtmp(const RtmpPacket::Ptr &pkt) {
|
2020-08-30 10:48:34 +08:00
|
|
|
|
switch (pkt->type_id) {
|
2018-10-24 22:03:17 +08:00
|
|
|
|
case MSG_VIDEO: {
|
2020-09-06 18:22:04 +08:00
|
|
|
|
if (!_try_get_video_track) {
|
2020-08-30 10:48:34 +08:00
|
|
|
|
_try_get_video_track = true;
|
2018-10-24 22:03:17 +08:00
|
|
|
|
auto codec = AMFValue(pkt->getMediaType());
|
2020-12-05 12:22:17 +08:00
|
|
|
|
makeVideoTrack(codec, 0);
|
2018-03-01 14:32:11 +08:00
|
|
|
|
}
|
2020-09-06 18:22:04 +08:00
|
|
|
|
if (_video_rtmp_decoder) {
|
|
|
|
|
|
_video_rtmp_decoder->inputRtmp(pkt);
|
2019-03-01 18:47:58 +08:00
|
|
|
|
}
|
2020-09-06 18:22:04 +08:00
|
|
|
|
break;
|
2018-03-01 14:32:11 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
case MSG_AUDIO: {
|
2020-09-06 18:22:04 +08:00
|
|
|
|
if (!_try_get_audio_track) {
|
2020-08-30 10:48:34 +08:00
|
|
|
|
_try_get_audio_track = true;
|
2018-10-24 22:03:17 +08:00
|
|
|
|
auto codec = AMFValue(pkt->getMediaType());
|
2020-12-05 12:22:17 +08:00
|
|
|
|
makeAudioTrack(codec, pkt->getAudioSampleRate(), pkt->getAudioChannel(), pkt->getAudioSampleBit(), 0);
|
2018-03-01 14:32:11 +08:00
|
|
|
|
}
|
2020-09-06 18:22:04 +08:00
|
|
|
|
if (_audio_rtmp_decoder) {
|
|
|
|
|
|
_audio_rtmp_decoder->inputRtmp(pkt);
|
2019-03-01 18:47:58 +08:00
|
|
|
|
}
|
2020-09-06 18:22:04 +08:00
|
|
|
|
break;
|
2018-03-01 14:32:11 +08:00
|
|
|
|
}
|
2020-09-06 18:22:04 +08:00
|
|
|
|
default : break;
|
2018-03-01 14:32:11 +08:00
|
|
|
|
}
|
2017-04-01 16:35:56 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2020-12-05 12:22:17 +08:00
|
|
|
|
void RtmpDemuxer::makeVideoTrack(const AMFValue &videoCodec, int bit_rate) {
|
2018-10-24 22:03:17 +08:00
|
|
|
|
//生成Track对象
|
2020-04-17 17:47:10 +08:00
|
|
|
|
_videoTrack = dynamic_pointer_cast<VideoTrack>(Factory::getVideoTrackByAmf(videoCodec));
|
2018-10-24 22:03:17 +08:00
|
|
|
|
if (_videoTrack) {
|
2020-12-05 12:22:17 +08:00
|
|
|
|
_videoTrack->setBitRate(bit_rate);
|
2018-10-24 22:03:17 +08:00
|
|
|
|
//生成rtmpCodec对象以便解码rtmp
|
2020-08-30 10:48:34 +08:00
|
|
|
|
_video_rtmp_decoder = Factory::getRtmpCodecByTrack(_videoTrack, false);
|
|
|
|
|
|
if (_video_rtmp_decoder) {
|
2018-10-24 22:03:17 +08:00
|
|
|
|
//设置rtmp解码器代理,生成的frame写入该Track
|
2020-08-30 10:48:34 +08:00
|
|
|
|
_video_rtmp_decoder->addDelegate(_videoTrack);
|
2019-12-26 11:53:19 +08:00
|
|
|
|
onAddTrack(_videoTrack);
|
2020-08-30 10:48:34 +08:00
|
|
|
|
_try_get_video_track = true;
|
2018-10-24 22:03:17 +08:00
|
|
|
|
} else {
|
|
|
|
|
|
//找不到相应的rtmp解码器,该track无效
|
|
|
|
|
|
_videoTrack.reset();
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2017-04-01 16:35:56 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2020-12-05 12:22:17 +08:00
|
|
|
|
void RtmpDemuxer::makeAudioTrack(const AMFValue &audioCodec,int sample_rate, int channels, int sample_bit, int bit_rate) {
|
2018-10-24 22:03:17 +08:00
|
|
|
|
//生成Track对象
|
2020-04-18 22:13:11 +08:00
|
|
|
|
_audioTrack = dynamic_pointer_cast<AudioTrack>(Factory::getAudioTrackByAmf(audioCodec, sample_rate, channels, sample_bit));
|
2018-10-24 22:03:17 +08:00
|
|
|
|
if (_audioTrack) {
|
2020-12-05 12:22:17 +08:00
|
|
|
|
_audioTrack->setBitRate(bit_rate);
|
2018-10-24 22:03:17 +08:00
|
|
|
|
//生成rtmpCodec对象以便解码rtmp
|
2020-08-30 10:48:34 +08:00
|
|
|
|
_audio_rtmp_decoder = Factory::getRtmpCodecByTrack(_audioTrack, false);
|
|
|
|
|
|
if (_audio_rtmp_decoder) {
|
2018-10-24 22:03:17 +08:00
|
|
|
|
//设置rtmp解码器代理,生成的frame写入该Track
|
2020-08-30 10:48:34 +08:00
|
|
|
|
_audio_rtmp_decoder->addDelegate(_audioTrack);
|
2019-12-26 11:53:19 +08:00
|
|
|
|
onAddTrack(_audioTrack);
|
2020-08-30 10:48:34 +08:00
|
|
|
|
_try_get_audio_track = true;
|
2018-10-24 22:03:17 +08:00
|
|
|
|
} else {
|
|
|
|
|
|
//找不到相应的rtmp解码器,该track无效
|
|
|
|
|
|
_audioTrack.reset();
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2017-04-01 16:35:56 +08:00
|
|
|
|
}
|
2018-10-24 22:03:17 +08:00
|
|
|
|
|
2017-04-01 16:35:56 +08:00
|
|
|
|
|
2018-10-24 17:17:55 +08:00
|
|
|
|
} /* namespace mediakit */
|