/* * Copyright (c) 2016 The ZLMediaKit project authors. All Rights Reserved. * * This file is part of ZLMediaKit(https://github.com/xia-chu/ZLMediaKit). * * 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. */ #ifndef SRC_RTSP_RTSPTORTMPMEDIASOURCE_H_ #define SRC_RTSP_RTSPTORTMPMEDIASOURCE_H_ #include "Rtmp/amf.h" #include "RtspMediaSource.h" #include "RtspDemuxer.h" #include "Common/MultiMediaSourceMuxer.h" namespace mediakit { class RtspMediaSourceImp : public RtspMediaSource, private TrackListener, public MultiMediaSourceMuxer::Listener { public: typedef std::shared_ptr Ptr; /** * 构造函数 * @param vhost 虚拟主机 * @param app 应用名 * @param id 流id * @param ringSize 环形缓存大小 */ RtspMediaSourceImp(const std::string &vhost, const std::string &app, const std::string &id, int ringSize = RTP_GOP_SIZE) : RtspMediaSource(vhost, app, id,ringSize) { _demuxer = std::make_shared(); _demuxer->setTrackListener(this); } ~RtspMediaSourceImp() = default; /** * 设置sdp */ void setSdp(const std::string &strSdp) override { if (!getSdp().empty()) { return; } _demuxer->loadSdp(strSdp); RtspMediaSource::setSdp(strSdp); } /** * 输入rtp并解析 */ void onWrite(RtpPacket::Ptr rtp, bool key_pos) override { if (_all_track_ready && !_muxer->isEnabled()) { //获取到所有Track后,并且未开启转协议,那么不需要解复用rtp //在关闭rtp解复用后,无法知道是否为关键帧,这样会导致无法秒开,或者开播花屏 key_pos = rtp->type == TrackVideo; } else { //需要解复用rtp key_pos = _demuxer->inputRtp(rtp); } GET_CONFIG(bool, directProxy, Rtsp::kDirectProxy); if (directProxy) { //直接代理模式才直接使用原始rtp RtspMediaSource::onWrite(std::move(rtp), key_pos); } } /** * 获取观看总人数,包括(hls/rtsp/rtmp) */ int totalReaderCount() override{ return readerCount() + (_muxer ? _muxer->totalReaderCount() : 0); } /** * 设置协议转换选项 */ void setProtocolOption(const ProtocolOption &option) { GET_CONFIG(bool, direct_proxy, Rtsp::kDirectProxy); //开启直接代理模式时,rtsp直接代理,不重复产生;但是有些rtsp推流端,由于sdp中已有sps pps,rtp中就不再包括sps pps, //导致rtc无法播放,所以在rtsp推流rtc播放时,建议关闭直接代理模式 _option = option; _option.enable_rtsp = !direct_proxy; _muxer = std::make_shared(getVhost(), getApp(), getId(), _demuxer->getDuration(), _option); _muxer->setMediaListener(getListener()); _muxer->setTrackListener(std::static_pointer_cast(shared_from_this())); //让_muxer对象拦截一部分事件(比如说录像相关事件) MediaSource::setListener(_muxer); for (auto &track : _demuxer->getTracks(false)) { _muxer->addTrack(track); track->addDelegate(_muxer); } } const ProtocolOption &getProtocolOption() const { return _option; } /** * _demuxer触发的添加Track事件 */ bool addTrack(const Track::Ptr &track) override { if (_muxer) { if (_muxer->addTrack(track)) { track->addDelegate(_muxer); return true; } } return false; } /** * _demuxer触发的Track添加完毕事件 */ void addTrackCompleted() override { if (_muxer) { _muxer->addTrackCompleted(); } } void resetTracks() override { if (_muxer) { _muxer->resetTracks(); } } /** * _muxer触发的所有Track就绪的事件 */ void onAllTrackReady() override{ _all_track_ready = true; } /** * 设置事件监听器 * @param listener 监听器 */ void setListener(const std::weak_ptr &listener) override{ if (_muxer) { //_muxer对象不能处理的事件再给listener处理 _muxer->setMediaListener(listener); } else { //未创建_muxer对象,事件全部给listener处理 MediaSource::setListener(listener); } } private: bool _all_track_ready = false; ProtocolOption _option; RtspDemuxer::Ptr _demuxer; MultiMediaSourceMuxer::Ptr _muxer; }; } /* namespace mediakit */ #endif /* SRC_RTSP_RTSPTORTMPMEDIASOURCE_H_ */