mirror of
https://github.com/ossrs/srs.git
synced 2025-03-09 15:49:59 +00:00
RTC: Cache the large buffer allocation
This commit is contained in:
parent
6e9cb059b3
commit
4d0863468a
9 changed files with 135 additions and 133 deletions
|
@ -500,6 +500,9 @@ rtc_server {
|
||||||
# Whether enable the RTP packet cache.
|
# Whether enable the RTP packet cache.
|
||||||
# default: off
|
# default: off
|
||||||
rtp_cache off;
|
rtp_cache off;
|
||||||
|
#Whether enable the RTP message(a large buffer) cache.
|
||||||
|
# default: off
|
||||||
|
rtp_msg_cache off;
|
||||||
# The black-hole to copy packet to, for debugging.
|
# The black-hole to copy packet to, for debugging.
|
||||||
# For example, when debugging Chrome publish stream, the received packets are encrypted cipher,
|
# For example, when debugging Chrome publish stream, the received packets are encrypted cipher,
|
||||||
# we can set the publisher black-hole, SRS will copy the plaintext packets to black-hole, and
|
# we can set the publisher black-hole, SRS will copy the plaintext packets to black-hole, and
|
||||||
|
|
|
@ -3651,7 +3651,7 @@ srs_error_t SrsConfig::check_normal_config()
|
||||||
string n = conf->at(i)->name;
|
string n = conf->at(i)->name;
|
||||||
if (n != "enabled" && n != "listen" && n != "dir" && n != "candidate" && n != "ecdsa"
|
if (n != "enabled" && n != "listen" && n != "dir" && n != "candidate" && n != "ecdsa"
|
||||||
&& n != "encrypt" && n != "reuseport" && n != "merge_nalus" && n != "perf_stat" && n != "black_hole"
|
&& n != "encrypt" && n != "reuseport" && n != "merge_nalus" && n != "perf_stat" && n != "black_hole"
|
||||||
&& n != "ip_family" && n != "rtp_cache") {
|
&& n != "ip_family" && n != "rtp_cache" && n != "rtp_msg_cache") {
|
||||||
return srs_error_new(ERROR_SYSTEM_CONFIG_INVALID, "illegal rtc_server.%s", n.c_str());
|
return srs_error_new(ERROR_SYSTEM_CONFIG_INVALID, "illegal rtc_server.%s", n.c_str());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -4919,6 +4919,23 @@ bool SrsConfig::get_rtc_server_rtp_cache()
|
||||||
return SRS_CONF_PERFER_FALSE(conf->arg0());
|
return SRS_CONF_PERFER_FALSE(conf->arg0());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool SrsConfig::get_rtc_server_rtp_msg_cache()
|
||||||
|
{
|
||||||
|
static bool DEFAULT = false;
|
||||||
|
|
||||||
|
SrsConfDirective* conf = root->get("rtc_server");
|
||||||
|
if (!conf) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
|
conf = conf->get("rtp_msg_cache");
|
||||||
|
if (!conf || conf->arg0().empty()) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
|
return SRS_CONF_PERFER_FALSE(conf->arg0());
|
||||||
|
}
|
||||||
|
|
||||||
bool SrsConfig::get_rtc_server_black_hole()
|
bool SrsConfig::get_rtc_server_black_hole()
|
||||||
{
|
{
|
||||||
static bool DEFAULT = false;
|
static bool DEFAULT = false;
|
||||||
|
|
|
@ -535,6 +535,7 @@ public:
|
||||||
virtual bool get_rtc_server_merge_nalus();
|
virtual bool get_rtc_server_merge_nalus();
|
||||||
virtual bool get_rtc_server_perf_stat();
|
virtual bool get_rtc_server_perf_stat();
|
||||||
virtual bool get_rtc_server_rtp_cache();
|
virtual bool get_rtc_server_rtp_cache();
|
||||||
|
virtual bool get_rtc_server_rtp_msg_cache();
|
||||||
virtual bool get_rtc_server_black_hole();
|
virtual bool get_rtc_server_black_hole();
|
||||||
virtual std::string get_rtc_server_black_hole_addr();
|
virtual std::string get_rtc_server_black_hole_addr();
|
||||||
private:
|
private:
|
||||||
|
|
|
@ -1193,11 +1193,11 @@ srs_error_t SrsRtcPublishStream::on_rtp_plaintext(char* plaintext, int nb_plaint
|
||||||
SrsRtpPacket2* pkt = _srs_rtp_cache->allocate();
|
SrsRtpPacket2* pkt = _srs_rtp_cache->allocate();
|
||||||
|
|
||||||
// Copy the packet body.
|
// Copy the packet body.
|
||||||
pkt->wrap(plaintext, nb_plaintext);
|
char* p = pkt->wrap(plaintext, nb_plaintext);
|
||||||
srs_assert(pkt->cache_buffer()->pos() == 0);
|
|
||||||
|
|
||||||
// Handle the packet.
|
// Handle the packet.
|
||||||
err = do_on_rtp_plaintext(pkt);
|
SrsBuffer buf(p, nb_plaintext);
|
||||||
|
err = do_on_rtp_plaintext(pkt, &buf);
|
||||||
|
|
||||||
// Release the packet to cache.
|
// Release the packet to cache.
|
||||||
_srs_rtp_cache->recycle(pkt);
|
_srs_rtp_cache->recycle(pkt);
|
||||||
|
@ -1205,14 +1205,14 @@ srs_error_t SrsRtcPublishStream::on_rtp_plaintext(char* plaintext, int nb_plaint
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
srs_error_t SrsRtcPublishStream::do_on_rtp_plaintext(SrsRtpPacket2* pkt)
|
srs_error_t SrsRtcPublishStream::do_on_rtp_plaintext(SrsRtpPacket2* pkt, SrsBuffer* buf)
|
||||||
{
|
{
|
||||||
srs_error_t err = srs_success;
|
srs_error_t err = srs_success;
|
||||||
|
|
||||||
pkt->set_decode_handler(this);
|
pkt->set_decode_handler(this);
|
||||||
pkt->set_extension_types(&extension_types_);
|
pkt->set_extension_types(&extension_types_);
|
||||||
|
|
||||||
if ((err = pkt->decode(pkt->cache_buffer())) != srs_success) {
|
if ((err = pkt->decode(buf)) != srs_success) {
|
||||||
return srs_error_wrap(err, "decode rtp packet");
|
return srs_error_wrap(err, "decode rtp packet");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -357,7 +357,7 @@ private:
|
||||||
// @remark We copy the plaintext, user should free it.
|
// @remark We copy the plaintext, user should free it.
|
||||||
srs_error_t on_rtp_plaintext(char* plaintext, int nb_plaintext);
|
srs_error_t on_rtp_plaintext(char* plaintext, int nb_plaintext);
|
||||||
private:
|
private:
|
||||||
srs_error_t do_on_rtp_plaintext(SrsRtpPacket2* pkt);
|
srs_error_t do_on_rtp_plaintext(SrsRtpPacket2* pkt, SrsBuffer* buf);
|
||||||
public:
|
public:
|
||||||
srs_error_t check_send_nacks();
|
srs_error_t check_send_nacks();
|
||||||
public:
|
public:
|
||||||
|
|
|
@ -287,7 +287,10 @@ srs_error_t SrsRtcServer::initialize()
|
||||||
bool rtp_cache = _srs_config->get_rtc_server_rtp_cache();
|
bool rtp_cache = _srs_config->get_rtc_server_rtp_cache();
|
||||||
_srs_rtp_cache->set_enabled(rtp_cache);
|
_srs_rtp_cache->set_enabled(rtp_cache);
|
||||||
|
|
||||||
srs_trace("RTC server init ok, rtp_cache=%d", rtp_cache);
|
bool rtp_msg_cache = _srs_config->get_rtc_server_rtp_msg_cache();
|
||||||
|
_srs_rtp_msg_cache->set_enabled(rtp_msg_cache);
|
||||||
|
|
||||||
|
srs_trace("RTC server init ok, rc=%d, rmc=%d", rtp_cache, rtp_msg_cache);
|
||||||
|
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
|
@ -312,6 +312,8 @@ public:
|
||||||
SrsSharedPtrMessage();
|
SrsSharedPtrMessage();
|
||||||
virtual ~SrsSharedPtrMessage();
|
virtual ~SrsSharedPtrMessage();
|
||||||
public:
|
public:
|
||||||
|
// For object cache to reset and reuse it.
|
||||||
|
bool reset() { return true; }
|
||||||
// Create shared ptr message,
|
// Create shared ptr message,
|
||||||
// copy header, manage the payload of msg,
|
// copy header, manage the payload of msg,
|
||||||
// set the payload to NULL to prevent double free.
|
// set the payload to NULL to prevent double free.
|
||||||
|
|
|
@ -811,7 +811,6 @@ SrsRtpPacket2::SrsRtpPacket2()
|
||||||
{
|
{
|
||||||
payload = NULL;
|
payload = NULL;
|
||||||
shared_msg = NULL;
|
shared_msg = NULL;
|
||||||
cache_buffer_ = NULL;
|
|
||||||
|
|
||||||
reset();
|
reset();
|
||||||
|
|
||||||
|
@ -821,11 +820,17 @@ SrsRtpPacket2::SrsRtpPacket2()
|
||||||
SrsRtpPacket2::~SrsRtpPacket2()
|
SrsRtpPacket2::~SrsRtpPacket2()
|
||||||
{
|
{
|
||||||
srs_freep(payload);
|
srs_freep(payload);
|
||||||
srs_freep(shared_msg);
|
|
||||||
srs_freep(cache_buffer_);
|
// Recyle the real owner of message, no other reference object.
|
||||||
|
if (shared_msg && shared_msg->count() == 0) {
|
||||||
|
_srs_rtp_msg_cache->recycle(shared_msg);
|
||||||
|
shared_msg = NULL;
|
||||||
|
} else {
|
||||||
|
srs_freep(shared_msg);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void SrsRtpPacket2::reset()
|
bool SrsRtpPacket2::reset()
|
||||||
{
|
{
|
||||||
nalu_type = SrsAvcNaluTypeReserved;
|
nalu_type = SrsAvcNaluTypeReserved;
|
||||||
frame_type = SrsFrameTypeReserved;
|
frame_type = SrsFrameTypeReserved;
|
||||||
|
@ -839,38 +844,49 @@ void SrsRtpPacket2::reset()
|
||||||
// and it's different for each packet.
|
// and it's different for each packet.
|
||||||
srs_freep(payload);
|
srs_freep(payload);
|
||||||
|
|
||||||
// We should reset the cached buffer.
|
// Recyle the real owner of message, no other reference object.
|
||||||
if (cache_buffer_) {
|
if (shared_msg && shared_msg->count() == 0) {
|
||||||
cache_buffer_->skip(-1 * cache_buffer_->pos());
|
_srs_rtp_msg_cache->recycle(shared_msg);
|
||||||
|
shared_msg = NULL;
|
||||||
|
} else {
|
||||||
|
srs_freep(shared_msg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
char* SrsRtpPacket2::wrap(int size)
|
char* SrsRtpPacket2::wrap(int size)
|
||||||
{
|
{
|
||||||
// If the buffer is large enough, reuse it.
|
// If the buffer is large enough, reuse it.
|
||||||
if (shared_msg && shared_msg->size >= size) {
|
if (shared_msg && shared_msg->size >= size) {
|
||||||
// The size maybe changed, so we MUST reset it.
|
|
||||||
cache_buffer_->set_size(size);
|
|
||||||
|
|
||||||
return shared_msg->payload;
|
return shared_msg->payload;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create buffer if empty or not large enough.
|
// Create a large enough message, with under-layer buffer.
|
||||||
srs_freep(shared_msg);
|
while (true) {
|
||||||
shared_msg = new SrsSharedPtrMessage();
|
srs_freep(shared_msg);
|
||||||
|
shared_msg = _srs_rtp_msg_cache->allocate();
|
||||||
|
|
||||||
// For RTC, we use larger under-layer buffer for each packet.
|
// If got a cached message(which has payload), but it's too small,
|
||||||
int nb_buffer = srs_max(size, kRtpPacketSize);
|
// we free it and allocate a larger one.
|
||||||
char* buf = new char[nb_buffer];
|
if (shared_msg->payload && shared_msg->size < size) {
|
||||||
shared_msg->wrap(buf, nb_buffer);
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
// The size of buffer must equal to the actual size.
|
// Create under-layer buffer for new message
|
||||||
srs_freep(cache_buffer_);
|
if (!shared_msg->payload) {
|
||||||
cache_buffer_ = new SrsBuffer(buf, size);
|
// For RTC, we use larger under-layer buffer for each packet.
|
||||||
|
int nb_buffer = srs_max(size, kRtpPacketSize);
|
||||||
|
char* buf = new char[nb_buffer];
|
||||||
|
shared_msg->wrap(buf, nb_buffer);
|
||||||
|
|
||||||
++_srs_pps_objs_rbuf->sugar;
|
++_srs_pps_objs_rbuf->sugar;
|
||||||
|
}
|
||||||
|
|
||||||
return buf;
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return shared_msg->payload;
|
||||||
}
|
}
|
||||||
|
|
||||||
char* SrsRtpPacket2::wrap(char* data, int size)
|
char* SrsRtpPacket2::wrap(char* data, int size)
|
||||||
|
@ -885,29 +901,9 @@ char* SrsRtpPacket2::wrap(SrsSharedPtrMessage* msg)
|
||||||
srs_freep(shared_msg);
|
srs_freep(shared_msg);
|
||||||
shared_msg = msg->copy();
|
shared_msg = msg->copy();
|
||||||
|
|
||||||
srs_freep(cache_buffer_);
|
|
||||||
cache_buffer_ = new SrsBuffer(msg->payload, msg->size);
|
|
||||||
|
|
||||||
return msg->payload;
|
return msg->payload;
|
||||||
}
|
}
|
||||||
|
|
||||||
SrsBuffer* SrsRtpPacket2::cache_buffer() const
|
|
||||||
{
|
|
||||||
return cache_buffer_;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool SrsRtpPacket2::try_recycle()
|
|
||||||
{
|
|
||||||
// When recycling, and there is references about he shared buffer, we must free
|
|
||||||
// the shared message(may not free the buffer) to stop reuse the shared message.
|
|
||||||
if (shared_msg && shared_msg->count() > 0) {
|
|
||||||
srs_freep(shared_msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
// OK, allow to recycle this object.
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void SrsRtpPacket2::set_padding(int size)
|
void SrsRtpPacket2::set_padding(int size)
|
||||||
{
|
{
|
||||||
header.set_padding(size);
|
header.set_padding(size);
|
||||||
|
@ -1026,70 +1022,9 @@ srs_error_t SrsRtpPacket2::decode(SrsBuffer* buf)
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
SrsRtpPacketCacheManager::SrsRtpPacketCacheManager()
|
SrsRtpObjectCacheManager<SrsRtpPacket2>* _srs_rtp_cache = new SrsRtpObjectCacheManager<SrsRtpPacket2>();
|
||||||
{
|
|
||||||
enabled_ = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
SrsRtpPacketCacheManager::~SrsRtpPacketCacheManager()
|
SrsRtpObjectCacheManager<SrsSharedPtrMessage>* _srs_rtp_msg_cache = new SrsRtpObjectCacheManager<SrsSharedPtrMessage>();
|
||||||
{
|
|
||||||
list<SrsRtpPacket2*>::iterator it;
|
|
||||||
for (it = cache_pkts_.begin(); it != cache_pkts_.end(); ++it) {
|
|
||||||
SrsRtpPacket2* pkt = *it;
|
|
||||||
srs_freep(pkt);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void SrsRtpPacketCacheManager::set_enabled(bool v)
|
|
||||||
{
|
|
||||||
enabled_ = v;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool SrsRtpPacketCacheManager::enabled()
|
|
||||||
{
|
|
||||||
return enabled_;
|
|
||||||
}
|
|
||||||
|
|
||||||
SrsRtpPacket2* SrsRtpPacketCacheManager::allocate()
|
|
||||||
{
|
|
||||||
if (!enabled_ || cache_pkts_.empty()) {
|
|
||||||
return new SrsRtpPacket2();
|
|
||||||
}
|
|
||||||
|
|
||||||
SrsRtpPacket2* pkt = cache_pkts_.back();
|
|
||||||
cache_pkts_.pop_back();
|
|
||||||
|
|
||||||
// We MUST reset it to reuse it.
|
|
||||||
pkt->reset();
|
|
||||||
|
|
||||||
return pkt;
|
|
||||||
}
|
|
||||||
|
|
||||||
void SrsRtpPacketCacheManager::recycle(SrsRtpPacket2* p)
|
|
||||||
{
|
|
||||||
// The p may be NULL, because srs_freep(NULL) is ok.
|
|
||||||
if (!p) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: FIXME: Directly free to keep low memory?
|
|
||||||
if (!enabled_) {
|
|
||||||
srs_freep(p);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// If there is any reference about the message, we should free the
|
|
||||||
// shared message then recycle it(or free it).
|
|
||||||
if (!p->try_recycle()) {
|
|
||||||
srs_freep(p);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Recycle it.
|
|
||||||
cache_pkts_.push_back(p);
|
|
||||||
}
|
|
||||||
|
|
||||||
SrsRtpPacketCacheManager* _srs_rtp_cache = new SrsRtpPacketCacheManager();
|
|
||||||
|
|
||||||
SrsRtpRawPayload::SrsRtpRawPayload()
|
SrsRtpRawPayload::SrsRtpRawPayload()
|
||||||
{
|
{
|
||||||
|
|
|
@ -285,8 +285,6 @@ public:
|
||||||
SrsRtpHeader header;
|
SrsRtpHeader header;
|
||||||
ISrsRtpPayloader* payload;
|
ISrsRtpPayloader* payload;
|
||||||
private:
|
private:
|
||||||
// The buffer bind to the shared message.
|
|
||||||
SrsBuffer* cache_buffer_;
|
|
||||||
// The original shared message, all RTP packets can refer to its data.
|
// The original shared message, all RTP packets can refer to its data.
|
||||||
SrsSharedPtrMessage* shared_msg;
|
SrsSharedPtrMessage* shared_msg;
|
||||||
// Helper fields.
|
// Helper fields.
|
||||||
|
@ -306,17 +304,12 @@ public:
|
||||||
virtual ~SrsRtpPacket2();
|
virtual ~SrsRtpPacket2();
|
||||||
public:
|
public:
|
||||||
// Reset the object to reuse it.
|
// Reset the object to reuse it.
|
||||||
void reset();
|
virtual bool reset();
|
||||||
// Wrap buffer to shared_message, which is managed by us.
|
// Wrap buffer to shared_message, which is managed by us.
|
||||||
char* wrap(int size);
|
char* wrap(int size);
|
||||||
char* wrap(char* data, int size);
|
char* wrap(char* data, int size);
|
||||||
// Wrap the shared message, we copy it.
|
// Wrap the shared message, we copy it.
|
||||||
char* wrap(SrsSharedPtrMessage* msg);
|
char* wrap(SrsSharedPtrMessage* msg);
|
||||||
// Get the cache buffer which binds to the shared message.
|
|
||||||
SrsBuffer* cache_buffer() const;
|
|
||||||
// Try to start recycle, return whether it's reusable.
|
|
||||||
// @remark If not reusable, user should free it directly.
|
|
||||||
bool try_recycle();
|
|
||||||
public:
|
public:
|
||||||
// Set the padding of RTP packet.
|
// Set the padding of RTP packet.
|
||||||
void set_padding(int size);
|
void set_padding(int size);
|
||||||
|
@ -337,27 +330,75 @@ public:
|
||||||
virtual srs_error_t decode(SrsBuffer* buf);
|
virtual srs_error_t decode(SrsBuffer* buf);
|
||||||
};
|
};
|
||||||
|
|
||||||
// The RTP packet cache manager.
|
// The RTP packet or message cache manager.
|
||||||
class SrsRtpPacketCacheManager
|
template<typename T>
|
||||||
|
class SrsRtpObjectCacheManager
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
bool enabled_;
|
bool enabled_;
|
||||||
std::list<SrsRtpPacket2*> cache_pkts_;
|
std::list<T*> cache_objs_;
|
||||||
public:
|
public:
|
||||||
SrsRtpPacketCacheManager();
|
SrsRtpObjectCacheManager() {
|
||||||
virtual ~SrsRtpPacketCacheManager();
|
enabled_ = false;
|
||||||
|
}
|
||||||
|
virtual ~SrsRtpObjectCacheManager() {
|
||||||
|
typedef typename std::list<T*>::iterator iterator;
|
||||||
|
for (iterator it = cache_objs_.begin(); it != cache_objs_.end(); ++it) {
|
||||||
|
T* obj = *it;
|
||||||
|
srs_freep(obj);
|
||||||
|
}
|
||||||
|
}
|
||||||
public:
|
public:
|
||||||
// Enable or disable cache.
|
// Enable or disable cache.
|
||||||
void set_enabled(bool v);
|
void set_enabled(bool v) {
|
||||||
bool enabled();
|
enabled_ = v;
|
||||||
// Try to allocate from cache, create new packet if no cache.
|
}
|
||||||
SrsRtpPacket2* allocate();
|
bool enabled() {
|
||||||
// Recycle the packet to cache.
|
return enabled_;
|
||||||
|
}
|
||||||
|
// Try to allocate from cache, create new object if no cache.
|
||||||
|
T* allocate() {
|
||||||
|
while (true) {
|
||||||
|
if (!enabled_ || cache_objs_.empty()) {
|
||||||
|
return new T();
|
||||||
|
}
|
||||||
|
|
||||||
|
T* obj = cache_objs_.back();
|
||||||
|
cache_objs_.pop_back();
|
||||||
|
|
||||||
|
// If reset the object fail, drop the cached object.
|
||||||
|
if (!obj->reset()) {
|
||||||
|
srs_freep(obj);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
return obj;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Recycle the object to cache.
|
||||||
// @remark User can directly free the packet.
|
// @remark User can directly free the packet.
|
||||||
void recycle(SrsRtpPacket2* p);
|
void recycle(T* p) {
|
||||||
|
// The p may be NULL, because srs_freep(NULL) is ok.
|
||||||
|
if (!p) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: FIXME: Directly free to keep low memory?
|
||||||
|
if (!enabled_) {
|
||||||
|
srs_freep(p);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Recycle it.
|
||||||
|
cache_objs_.push_back(p);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
extern SrsRtpPacketCacheManager* _srs_rtp_cache;
|
// For RTP packets cache.
|
||||||
|
extern SrsRtpObjectCacheManager<SrsRtpPacket2>* _srs_rtp_cache;
|
||||||
|
|
||||||
|
// For RTP packet shared messages cache.
|
||||||
|
extern SrsRtpObjectCacheManager<SrsSharedPtrMessage>* _srs_rtp_msg_cache;
|
||||||
|
|
||||||
// Single payload data.
|
// Single payload data.
|
||||||
class SrsRtpRawPayload : public ISrsRtpPayloader
|
class SrsRtpRawPayload : public ISrsRtpPayloader
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue