ZLMediaKit/src/Rtmp/RtmpDemuxer.cpp

185 lines
5.7 KiB
C++
Raw Normal View History

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.
*
* 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.
*/
#include "RtmpCodec.h"
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
using namespace std;
2018-10-24 17:17:55 +08:00
namespace mediakit {
2017-04-01 16:35:56 +08:00
2021-11-10 15:25:24 +08:00
size_t RtmpDemuxer::trackCount(const AMFValue &metadata) {
size_t ret = 0;
metadata.object_for_each([&](const string &key, const AMFValue &val) {
if (key == "videocodecid") {
// 找到视频
2021-11-10 15:25:24 +08:00
++ret;
return;
}
if (key == "audiocodecid") {
// 找到音频
2021-11-10 15:25:24 +08:00
++ret;
return;
}
});
return ret;
}
bool RtmpDemuxer::loadMetaData(const AMFValue &val) {
bool ret = false;
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;
val.object_for_each([&](const string &key, const AMFValue &val) {
if (key == "duration") {
_duration = (float)val.as_number();
return;
}
if (key == "audiosamplerate") {
2020-04-18 22:13:11 +08:00
audiosamplerate = val.as_integer();
return;
}
if (key == "audiosamplesize") {
2020-04-18 22:13:11 +08:00
audiosamplesize = val.as_integer();
return;
}
if (key == "stereo") {
2020-04-18 22:13:11 +08:00
audiochannels = val.as_boolean() ? 2 : 1;
return;
}
if (key == "videocodecid") {
// 找到视频
2020-04-18 22:13:11 +08:00
videocodecid = &val;
return;
}
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;
}
});
if (videocodecid) {
// 有视频
ret = true;
makeVideoTrack(*videocodecid, videodatarate * 1024);
2020-04-18 22:13:11 +08:00
}
if (audiocodecid) {
// 有音频
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
}
} catch (std::exception &ex) {
WarnL << ex.what();
}
2021-04-20 17:53:43 +08:00
if (ret) {
// metadata中存在track相关的描述那么我们根据metadata判断有多少个track
2021-04-20 17:53:43 +08:00
addTrackCompleted();
}
return ret;
}
float RtmpDemuxer::getDuration() const {
return _duration;
}
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: {
if (!_try_get_video_track) {
2020-08-30 10:48:34 +08:00
_try_get_video_track = true;
auto codec_id = parseVideoRtmpPacket((uint8_t *)pkt->data(), pkt->size());
makeVideoTrack(Factory::getTrackByCodecId(codec_id), 0);
}
if (_video_rtmp_decoder) {
_video_rtmp_decoder->inputRtmp(pkt);
}
break;
}
case MSG_AUDIO: {
if (!_try_get_audio_track) {
2020-08-30 10:48:34 +08:00
_try_get_audio_track = true;
auto codec = AMFValue(pkt->getRtmpCodecId());
2020-12-05 12:22:17 +08:00
makeAudioTrack(codec, pkt->getAudioSampleRate(), pkt->getAudioChannel(), pkt->getAudioSampleBit(), 0);
}
if (_audio_rtmp_decoder) {
_audio_rtmp_decoder->inputRtmp(pkt);
}
break;
}
default: break;
}
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) {
makeVideoTrack(Factory::getVideoTrackByAmf(videoCodec), bit_rate);
}
void RtmpDemuxer::makeVideoTrack(const Track::Ptr &track, int bit_rate) {
if (_video_rtmp_decoder) {
return;
}
// 生成Track对象
_video_track = dynamic_pointer_cast<VideoTrack>(track);
if (!_video_track) {
return;
}
// 生成rtmpCodec对象以便解码rtmp
_video_rtmp_decoder = Factory::getRtmpCodecByTrack(_video_track, false);
if (!_video_rtmp_decoder) {
// 找不到相应的rtmp解码器该track无效
_video_track.reset();
return;
2018-10-24 22:03:17 +08:00
}
_video_track->setBitRate(bit_rate);
// 设置rtmp解码器代理生成的frame写入该Track
_video_rtmp_decoder->addDelegate(_video_track);
addTrack(_video_track);
_try_get_video_track = true;
2017-04-01 16:35:56 +08:00
}
void RtmpDemuxer::makeAudioTrack(const AMFValue &audioCodec, int sample_rate, int channels, int sample_bit, int bit_rate) {
if (_audio_rtmp_decoder) {
return;
}
// 生成Track对象
_audio_track = dynamic_pointer_cast<AudioTrack>(Factory::getAudioTrackByAmf(audioCodec, sample_rate, channels, sample_bit));
if (!_audio_track) {
return;
}
// 生成rtmpCodec对象以便解码rtmp
_audio_rtmp_decoder = Factory::getRtmpCodecByTrack(_audio_track, false);
if (!_audio_rtmp_decoder) {
// 找不到相应的rtmp解码器该track无效
_audio_track.reset();
return;
2018-10-24 22:03:17 +08:00
}
_audio_track->setBitRate(bit_rate);
// 设置rtmp解码器代理生成的frame写入该Track
_audio_rtmp_decoder->addDelegate(_audio_track);
addTrack(_audio_track);
_try_get_audio_track = true;
2017-04-01 16:35:56 +08:00
}
2018-10-24 22:03:17 +08:00
} /* namespace mediakit */