Older/MediaServer/Rtmp/RtmpMediaSource.h

199 lines
5.9 KiB
C
Raw Permalink Normal View History

2024-09-28 23:55:00 +08:00
/*
* Copyright (c) 2016-present The ZLMediaKit project authors. All Rights Reserved.
*
* This file is part of ZLMediaKit(https://github.com/ZLMediaKit/ZLMediaKit).
*
* Use of this source code is governed by MIT-like 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_RTMP_RTMPMEDIASOURCE_H_
#define SRC_RTMP_RTMPMEDIASOURCE_H_
#include <mutex>
#include <memory>
#include <string>
#include <functional>
#include <unordered_map>
#include "amf.h"
#include "Rtmp.h"
#include "Common/MediaSource.h"
#include "Common/PacketCache.h"
#include "Util/RingBuffer.h"
#define RTMP_GOP_SIZE 512
namespace mediakit {
/**
* rtmp媒体源的数据抽象
* rtmp有关键的三要素metadataconfig帧
* metadata是非必须的config帧(MP3)
* rtmp推流rtmp服务器就很简单了
* rtmp推拉流协议中metadataconfig帧
* Data abstraction of rtmp media source
* rtmp has three key elements: metadata, config frame, and ordinary frame
* Metadata is optional, and some encoding formats do not have config frames (such as MP3)
* As long as these three elements are generated, it is very simple to implement rtmp push stream and rtmp server
* In the rtmp push and pull stream protocol, metadata is transmitted first, then config frame, and then ordinary frames are continuously transmitted
* [AUTO-TRANSLATED:72d515c8]
*/
class RtmpMediaSource : public MediaSource, public toolkit::RingDelegate<RtmpPacket::Ptr>, private PacketCache<RtmpPacket>{
public:
using Ptr = std::shared_ptr<RtmpMediaSource>;
using RingDataType = std::shared_ptr<toolkit::List<RtmpPacket::Ptr> >;
using RingType = toolkit::RingBuffer<RingDataType>;
/**
*
* @param vhost
* @param app
* @param stream_id id
* @param ring_size 0
* Constructor
* @param vhost Virtual host name
* @param app Application name
* @param stream_id Stream id
* @param ring_size You can set a fixed ring buffer size, 0 is adaptive
* [AUTO-TRANSLATED:5dd23423]
*/
RtmpMediaSource(const MediaTuple& tuple, int ring_size = RTMP_GOP_SIZE): MediaSource(RTMP_SCHEMA, tuple), _ring_size(ring_size) {}
~RtmpMediaSource() override {
try {
flush();
} catch (std::exception &ex) {
WarnL << ex.what();
}
}
/**
*
* Get the ring buffer of the media source
* [AUTO-TRANSLATED:75ac76b6]
*/
const RingType::Ptr &getRing() const {
return _ring;
}
void getPlayerList(const std::function<void(const std::list<toolkit::Any> &info_list)> &cb,
const std::function<toolkit::Any(toolkit::Any &&info)> &on_change) override {
_ring->getInfoList(cb, on_change);
}
/**
*
* @return
* Get the number of players
* @return
* [AUTO-TRANSLATED:0ba31e32]
*/
int readerCount() override {
return _ring ? _ring->readerCount() : 0;
}
/**
* metadata
* Get metadata
* [AUTO-TRANSLATED:270cc022]
*/
template <typename FUNC>
void getMetaData(const FUNC &func) const {
std::lock_guard<std::recursive_mutex> lock(_mtx);
if (_metadata) {
func(_metadata);
}
}
/**
* config帧
* Get all config frames
* [AUTO-TRANSLATED:cc3f5b85]
*/
template <typename FUNC>
void getConfigFrame(const FUNC &func) {
std::lock_guard<std::recursive_mutex> lock(_mtx);
for (auto &pr : _config_frame_map) {
func(pr.second);
}
}
/**
* metadata
* Set metadata
* [AUTO-TRANSLATED:e32234cf]
*/
virtual void setMetaData(const AMFValue &metadata);
/**
* rtmp包
* @param pkt rtmp包
* Input rtmp packet
* @param pkt rtmp packet
* [AUTO-TRANSLATED:ac020e50]
*/
void onWrite(RtmpPacket::Ptr pkt, bool = true) override;
/**
*
* Get the current timestamp
* [AUTO-TRANSLATED:42e38069]
*/
uint32_t getTimeStamp(TrackType trackType) override;
void clearCache() override{
PacketCache<RtmpPacket>::clearCache();
_ring->clearCache();
}
bool haveVideo() const {
return _have_video;
}
bool haveAudio() const {
return _have_audio;
}
private:
/**
* flush rtmp包时触发该函数
* @param rtmp_list rtmp包列表
* @param key_pos
* Trigger this function when batch flushing rtmp packets
* @param rtmp_list rtmp packet list
* @param key_pos Whether it contains key frames
* [AUTO-TRANSLATED:581fe3a4]
*/
void onFlush(std::shared_ptr<toolkit::List<RtmpPacket::Ptr> > rtmp_list, bool key_pos) override {
// 如果不存在视频那么就没有存在GOP缓存的意义所以is_key一直为true确保一直清空GOP缓存 [AUTO-TRANSLATED:5818a8d8]
// If there is no video, then there is no point in having a GOP cache, so is_key is always true to ensure that the GOP cache is always cleared
_ring->write(std::move(rtmp_list), _have_video ? key_pos : true);
}
private:
bool _have_video = false;
bool _have_audio = false;
int _ring_size;
uint32_t _track_stamps[TrackMax] = {0};
AMFValue _metadata;
RingType::Ptr _ring;
mutable std::recursive_mutex _mtx;
std::unordered_map<int, RtmpPacket::Ptr> _config_frame_map;
};
} /* namespace mediakit */
#endif /* SRC_RTMP_RTMPMEDIASOURCE_H_ */