2014-11-22 09:58:02 +00:00
|
|
|
/*
|
|
|
|
The MIT License (MIT)
|
|
|
|
|
|
|
|
Copyright (c) 2013-2014 winlin
|
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
|
|
|
this software and associated documentation files (the "Software"), to deal in
|
|
|
|
the Software without restriction, including without limitation the rights to
|
|
|
|
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
|
|
|
the Software, and to permit persons to whom the Software is furnished to do so,
|
|
|
|
subject to the following conditions:
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be included in all
|
|
|
|
copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
|
|
|
FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
|
|
|
COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
|
|
|
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
|
|
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <srs_app_recv_thread.hpp>
|
|
|
|
|
|
|
|
#include <srs_protocol_rtmp.hpp>
|
|
|
|
#include <srs_protocol_stack.hpp>
|
2014-12-01 15:38:51 +00:00
|
|
|
#include <srs_app_rtmp_conn.hpp>
|
2014-12-03 11:27:27 +00:00
|
|
|
#include <srs_protocol_buffer.hpp>
|
2014-12-03 14:39:25 +00:00
|
|
|
#include <srs_kernel_utility.hpp>
|
2014-12-04 05:43:55 +00:00
|
|
|
#include <srs_core_performance.hpp>
|
2014-12-04 10:21:04 +00:00
|
|
|
#include <srs_app_config.hpp>
|
|
|
|
|
|
|
|
using namespace std;
|
2014-12-03 11:27:27 +00:00
|
|
|
|
2014-12-04 10:35:50 +00:00
|
|
|
// the max small bytes to group
|
|
|
|
#define SRS_MR_SMALL_BYTES 4096
|
|
|
|
|
2014-12-01 14:39:22 +00:00
|
|
|
ISrsMessageHandler::ISrsMessageHandler()
|
2014-11-22 09:58:02 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2014-12-01 14:39:22 +00:00
|
|
|
ISrsMessageHandler::~ISrsMessageHandler()
|
2014-11-22 09:58:02 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2014-12-01 14:45:45 +00:00
|
|
|
SrsRecvThread::SrsRecvThread(ISrsMessageHandler* msg_handler, SrsRtmpServer* rtmp_sdk, int timeout_ms)
|
2014-11-22 09:58:02 +00:00
|
|
|
{
|
2014-12-01 14:45:45 +00:00
|
|
|
timeout = timeout_ms;
|
2014-12-01 14:39:22 +00:00
|
|
|
handler = msg_handler;
|
|
|
|
rtmp = rtmp_sdk;
|
2014-12-03 06:05:15 +00:00
|
|
|
trd = new SrsThread("recv", this, 0, true);
|
2014-11-22 09:58:02 +00:00
|
|
|
}
|
|
|
|
|
2014-12-01 14:39:22 +00:00
|
|
|
SrsRecvThread::~SrsRecvThread()
|
2014-11-22 11:15:40 +00:00
|
|
|
{
|
2014-12-01 14:39:22 +00:00
|
|
|
// stop recv thread.
|
|
|
|
stop();
|
2014-11-22 11:15:40 +00:00
|
|
|
|
2014-12-01 14:39:22 +00:00
|
|
|
// destroy the thread.
|
|
|
|
srs_freep(trd);
|
2014-11-22 09:58:02 +00:00
|
|
|
}
|
|
|
|
|
2014-12-01 14:39:22 +00:00
|
|
|
int SrsRecvThread::start()
|
2014-11-22 09:58:02 +00:00
|
|
|
{
|
|
|
|
return trd->start();
|
|
|
|
}
|
|
|
|
|
2014-12-01 14:39:22 +00:00
|
|
|
void SrsRecvThread::stop()
|
2014-11-22 09:58:02 +00:00
|
|
|
{
|
|
|
|
trd->stop();
|
|
|
|
}
|
|
|
|
|
2014-12-01 14:39:22 +00:00
|
|
|
int SrsRecvThread::cycle()
|
2014-11-22 09:58:02 +00:00
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
2014-12-01 14:39:22 +00:00
|
|
|
|
2014-12-02 11:28:36 +00:00
|
|
|
while (trd->can_loop()) {
|
|
|
|
if (!handler->can_handle()) {
|
|
|
|
st_usleep(timeout * 1000);
|
|
|
|
continue;
|
2014-11-22 09:58:02 +00:00
|
|
|
}
|
2014-12-02 11:28:36 +00:00
|
|
|
|
|
|
|
SrsMessage* msg = NULL;
|
2014-12-02 07:21:08 +00:00
|
|
|
|
2014-12-02 11:28:36 +00:00
|
|
|
// recv and handle message
|
|
|
|
ret = rtmp->recv_message(&msg);
|
|
|
|
if (ret == ERROR_SUCCESS) {
|
|
|
|
ret = handler->handle(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ret != ERROR_SUCCESS) {
|
|
|
|
if (!srs_is_client_gracefully_close(ret)) {
|
|
|
|
srs_error("thread process message failed. ret=%d", ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
// we use no timeout to recv, should never got any error.
|
|
|
|
trd->stop_loop();
|
|
|
|
|
|
|
|
// notice the handler got a recv error.
|
|
|
|
handler->on_recv_error(ret);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
srs_verbose("thread loop recv message. ret=%d", ret);
|
2014-11-22 09:58:02 +00:00
|
|
|
}
|
2014-12-01 14:39:22 +00:00
|
|
|
|
2014-11-22 09:58:02 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-12-01 15:38:51 +00:00
|
|
|
void SrsRecvThread::stop_loop()
|
|
|
|
{
|
|
|
|
trd->stop_loop();
|
|
|
|
}
|
|
|
|
|
2014-12-01 14:39:22 +00:00
|
|
|
void SrsRecvThread::on_thread_start()
|
2014-11-22 10:08:45 +00:00
|
|
|
{
|
|
|
|
// the multiple messages writev improve performance large,
|
|
|
|
// but the timeout recv will cause 33% sys call performance,
|
|
|
|
// to use isolate thread to recv, can improve about 33% performance.
|
|
|
|
// @see https://github.com/winlinvip/simple-rtmp-server/issues/194
|
|
|
|
// @see: https://github.com/winlinvip/simple-rtmp-server/issues/217
|
|
|
|
rtmp->set_recv_timeout(ST_UTIME_NO_TIMEOUT);
|
2014-12-02 09:16:20 +00:00
|
|
|
|
|
|
|
handler->on_thread_start();
|
2014-11-22 10:08:45 +00:00
|
|
|
}
|
|
|
|
|
2014-12-01 14:39:22 +00:00
|
|
|
void SrsRecvThread::on_thread_stop()
|
2014-11-22 10:08:45 +00:00
|
|
|
{
|
|
|
|
// reset the timeout to pulse mode.
|
2014-12-01 14:45:45 +00:00
|
|
|
rtmp->set_recv_timeout(timeout * 1000);
|
2014-12-02 09:16:20 +00:00
|
|
|
|
|
|
|
handler->on_thread_stop();
|
2014-11-22 10:08:45 +00:00
|
|
|
}
|
|
|
|
|
2014-12-01 14:45:45 +00:00
|
|
|
SrsQueueRecvThread::SrsQueueRecvThread(SrsRtmpServer* rtmp_sdk, int timeout_ms)
|
2014-12-01 14:53:03 +00:00
|
|
|
: trd(this, rtmp_sdk, timeout_ms)
|
2014-12-01 14:39:22 +00:00
|
|
|
{
|
2014-12-02 09:16:20 +00:00
|
|
|
rtmp = rtmp_sdk;
|
2014-12-02 07:21:08 +00:00
|
|
|
recv_error_code = ERROR_SUCCESS;
|
2014-12-01 14:39:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SrsQueueRecvThread::~SrsQueueRecvThread()
|
|
|
|
{
|
2014-12-01 14:53:03 +00:00
|
|
|
trd.stop();
|
2014-12-01 14:39:22 +00:00
|
|
|
|
|
|
|
// clear all messages.
|
|
|
|
std::vector<SrsMessage*>::iterator it;
|
|
|
|
for (it = queue.begin(); it != queue.end(); ++it) {
|
|
|
|
SrsMessage* msg = *it;
|
|
|
|
srs_freep(msg);
|
|
|
|
}
|
|
|
|
queue.clear();
|
|
|
|
}
|
|
|
|
|
2014-12-01 14:53:03 +00:00
|
|
|
int SrsQueueRecvThread::start()
|
|
|
|
{
|
|
|
|
return trd.start();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SrsQueueRecvThread::stop()
|
|
|
|
{
|
|
|
|
trd.stop();
|
|
|
|
}
|
|
|
|
|
2014-12-01 14:39:22 +00:00
|
|
|
bool SrsQueueRecvThread::empty()
|
|
|
|
{
|
|
|
|
return queue.empty();
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsQueueRecvThread::size()
|
|
|
|
{
|
|
|
|
return (int)queue.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsMessage* SrsQueueRecvThread::pump()
|
|
|
|
{
|
|
|
|
srs_assert(!queue.empty());
|
|
|
|
|
|
|
|
SrsMessage* msg = *queue.begin();
|
|
|
|
|
|
|
|
queue.erase(queue.begin());
|
|
|
|
|
|
|
|
return msg;
|
|
|
|
}
|
|
|
|
|
2014-12-02 07:21:08 +00:00
|
|
|
int SrsQueueRecvThread::error_code()
|
|
|
|
{
|
|
|
|
return recv_error_code;
|
|
|
|
}
|
|
|
|
|
2014-12-01 14:39:22 +00:00
|
|
|
bool SrsQueueRecvThread::can_handle()
|
|
|
|
{
|
|
|
|
// we only recv one message and then process it,
|
|
|
|
// for the message may cause the thread to stop,
|
|
|
|
// when stop, the thread is freed, so the messages
|
|
|
|
// are dropped.
|
|
|
|
return empty();
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsQueueRecvThread::handle(SrsMessage* msg)
|
|
|
|
{
|
|
|
|
// put into queue, the send thread will get and process it,
|
|
|
|
// @see SrsRtmpConn::process_play_control_msg
|
|
|
|
queue.push_back(msg);
|
|
|
|
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
}
|
2014-12-01 15:38:51 +00:00
|
|
|
|
2014-12-02 07:21:08 +00:00
|
|
|
void SrsQueueRecvThread::on_recv_error(int ret)
|
|
|
|
{
|
|
|
|
recv_error_code = ret;
|
|
|
|
}
|
|
|
|
|
2014-12-02 09:16:20 +00:00
|
|
|
void SrsQueueRecvThread::on_thread_start()
|
|
|
|
{
|
|
|
|
// disable the protocol auto response,
|
|
|
|
// for the isolate recv thread should never send any messages.
|
|
|
|
rtmp->set_auto_response(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SrsQueueRecvThread::on_thread_stop()
|
|
|
|
{
|
|
|
|
// enable the protocol auto response,
|
|
|
|
// for the isolate recv thread terminated.
|
|
|
|
rtmp->set_auto_response(true);
|
|
|
|
}
|
|
|
|
|
2014-12-01 15:38:51 +00:00
|
|
|
SrsPublishRecvThread::SrsPublishRecvThread(
|
2014-12-04 10:21:04 +00:00
|
|
|
SrsRtmpServer* rtmp_sdk,
|
|
|
|
SrsRequest* _req, int mr_sock_fd, int timeout_ms,
|
2014-12-01 15:38:51 +00:00
|
|
|
SrsRtmpConn* conn, SrsSource* source, bool is_fmle, bool is_edge
|
|
|
|
): trd(this, rtmp_sdk, timeout_ms)
|
|
|
|
{
|
2014-12-02 09:16:20 +00:00
|
|
|
rtmp = rtmp_sdk;
|
2014-12-04 10:21:04 +00:00
|
|
|
|
2014-12-01 15:38:51 +00:00
|
|
|
_conn = conn;
|
|
|
|
_source = source;
|
|
|
|
_is_fmle = is_fmle;
|
|
|
|
_is_edge = is_edge;
|
|
|
|
|
|
|
|
recv_error_code = ERROR_SUCCESS;
|
|
|
|
_nb_msgs = 0;
|
2014-12-03 04:08:29 +00:00
|
|
|
error = st_cond_new();
|
2014-12-04 10:21:04 +00:00
|
|
|
|
|
|
|
req = _req;
|
|
|
|
mr_fd = mr_sock_fd;
|
2014-12-03 14:39:25 +00:00
|
|
|
|
2014-12-04 10:21:04 +00:00
|
|
|
// the mr settings,
|
|
|
|
// @see https://github.com/winlinvip/simple-rtmp-server/issues/241
|
|
|
|
mr = _srs_config->get_mr_enabled(req->vhost);
|
|
|
|
mr_sleep = _srs_config->get_mr_sleep_ms(req->vhost);
|
|
|
|
|
|
|
|
_srs_config->subscribe(this);
|
2014-12-01 15:38:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SrsPublishRecvThread::~SrsPublishRecvThread()
|
|
|
|
{
|
2014-12-04 10:21:04 +00:00
|
|
|
_srs_config->unsubscribe(this);
|
|
|
|
|
2014-12-01 15:38:51 +00:00
|
|
|
trd.stop();
|
2014-12-03 04:08:29 +00:00
|
|
|
st_cond_destroy(error);
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsPublishRecvThread::wait(int timeout_ms)
|
|
|
|
{
|
|
|
|
if (recv_error_code != ERROR_SUCCESS) {
|
|
|
|
return recv_error_code;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ignore any return of cond wait.
|
|
|
|
st_cond_timedwait(error, timeout_ms * 1000);
|
|
|
|
|
|
|
|
return ERROR_SUCCESS;
|
2014-12-01 15:38:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int64_t SrsPublishRecvThread::nb_msgs()
|
|
|
|
{
|
|
|
|
return _nb_msgs;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsPublishRecvThread::error_code()
|
|
|
|
{
|
|
|
|
return recv_error_code;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsPublishRecvThread::start()
|
|
|
|
{
|
|
|
|
return trd.start();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SrsPublishRecvThread::stop()
|
|
|
|
{
|
|
|
|
trd.stop();
|
|
|
|
}
|
|
|
|
|
2014-12-03 11:27:27 +00:00
|
|
|
void SrsPublishRecvThread::on_thread_start()
|
|
|
|
{
|
|
|
|
// we donot set the auto response to false,
|
|
|
|
// for the main thread never send message.
|
|
|
|
|
2014-12-04 05:43:55 +00:00
|
|
|
#ifdef SRS_PERF_MERGED_READ
|
2014-12-04 10:21:04 +00:00
|
|
|
// for mr.
|
|
|
|
update_buffer(mr, mr_sleep);
|
2014-12-04 05:43:55 +00:00
|
|
|
#endif
|
2014-12-03 11:27:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SrsPublishRecvThread::on_thread_stop()
|
|
|
|
{
|
|
|
|
// we donot set the auto response to true,
|
|
|
|
// for we donot set to false yet.
|
|
|
|
|
|
|
|
// when thread stop, signal the conn thread which wait.
|
|
|
|
// @see https://github.com/winlinvip/simple-rtmp-server/issues/244
|
|
|
|
st_cond_signal(error);
|
|
|
|
|
2014-12-04 05:43:55 +00:00
|
|
|
#ifdef SRS_PERF_MERGED_READ
|
2014-12-03 11:27:27 +00:00
|
|
|
// disable the merge read
|
|
|
|
// @see https://github.com/winlinvip/simple-rtmp-server/issues/241
|
2014-12-04 07:43:37 +00:00
|
|
|
rtmp->set_merge_read(false, NULL);
|
2014-12-04 05:43:55 +00:00
|
|
|
#endif
|
2014-12-03 11:27:27 +00:00
|
|
|
}
|
|
|
|
|
2014-12-01 15:38:51 +00:00
|
|
|
bool SrsPublishRecvThread::can_handle()
|
|
|
|
{
|
|
|
|
// publish thread always can handle message.
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsPublishRecvThread::handle(SrsMessage* msg)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
_nb_msgs++;
|
|
|
|
|
2014-12-02 11:28:36 +00:00
|
|
|
// the rtmp connection will handle this message
|
|
|
|
ret = _conn->handle_publish_message(_source, msg, _is_fmle, _is_edge);
|
2014-12-01 15:38:51 +00:00
|
|
|
|
|
|
|
// must always free it,
|
|
|
|
// the source will copy it if need to use.
|
|
|
|
srs_freep(msg);
|
2014-12-02 11:28:36 +00:00
|
|
|
|
2014-12-01 15:38:51 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2014-12-02 07:21:08 +00:00
|
|
|
|
|
|
|
void SrsPublishRecvThread::on_recv_error(int ret)
|
|
|
|
{
|
|
|
|
recv_error_code = ret;
|
2014-12-03 04:08:29 +00:00
|
|
|
|
|
|
|
// when recv thread error, signal the conn thread to process it.
|
|
|
|
// @see https://github.com/winlinvip/simple-rtmp-server/issues/244
|
|
|
|
st_cond_signal(error);
|
2014-12-02 07:21:08 +00:00
|
|
|
}
|
2014-12-02 09:16:20 +00:00
|
|
|
|
2014-12-04 05:43:55 +00:00
|
|
|
#ifdef SRS_PERF_MERGED_READ
|
2014-12-03 14:39:25 +00:00
|
|
|
void SrsPublishRecvThread::on_read(ssize_t nread)
|
2014-12-02 09:16:20 +00:00
|
|
|
{
|
2014-12-04 10:21:04 +00:00
|
|
|
if (!mr) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nread < 0 || mr_sleep <= 0) {
|
2014-12-03 11:27:27 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-12-03 04:08:29 +00:00
|
|
|
|
2014-12-03 11:27:27 +00:00
|
|
|
/**
|
|
|
|
* to improve read performance, merge some packets then read,
|
|
|
|
* when it on and read small bytes, we sleep to wait more data.,
|
|
|
|
* that is, we merge some data to read together.
|
|
|
|
* @see https://github.com/winlinvip/simple-rtmp-server/issues/241
|
|
|
|
*/
|
2014-12-04 07:33:17 +00:00
|
|
|
if (nread < SRS_MR_SMALL_BYTES) {
|
2014-12-04 10:21:04 +00:00
|
|
|
st_usleep(mr_sleep * 1000);
|
2014-12-03 11:27:27 +00:00
|
|
|
}
|
2014-12-02 09:16:20 +00:00
|
|
|
}
|
2014-12-04 05:43:55 +00:00
|
|
|
#endif
|
2014-12-04 10:21:04 +00:00
|
|
|
|
|
|
|
int SrsPublishRecvThread::on_reload_vhost_mr(string vhost)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
// the mr settings,
|
|
|
|
// @see https://github.com/winlinvip/simple-rtmp-server/issues/241
|
|
|
|
bool mr_enabled = _srs_config->get_mr_enabled(req->vhost);
|
|
|
|
int sleep_ms = _srs_config->get_mr_sleep_ms(req->vhost);
|
|
|
|
update_buffer(mr_enabled, sleep_ms);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SrsPublishRecvThread::update_buffer(bool mr_enabled, int sleep_ms)
|
|
|
|
{
|
|
|
|
// TODO: FIXME: refine it.
|
|
|
|
|
|
|
|
#ifdef SRS_PERF_MERGED_READ
|
2014-12-04 10:35:50 +00:00
|
|
|
// update the buffer.
|
|
|
|
if (true) {
|
2014-12-04 10:21:04 +00:00
|
|
|
// the underlayer api will set to SRS_MR_SOCKET_BUFFER bytes.
|
|
|
|
// 4KB=4096, 8KB=8192, 16KB=16384, 32KB=32768, 64KB=65536,
|
|
|
|
// 128KB=131072, 256KB=262144, 512KB=524288
|
|
|
|
// the buffer should set to SRS_MR_MAX_SLEEP_MS*kbps/8,
|
|
|
|
// for example, your system delivery stream in 1000kbps,
|
|
|
|
// sleep 800ms for small bytes, the buffer should set to:
|
|
|
|
// 800*1000/8=100000B(about 128KB).
|
|
|
|
// 2000*3000/8=750000B(about 732KB).
|
|
|
|
int kbps = 3000;
|
|
|
|
int socket_buffer_size = mr_sleep * kbps / 8;
|
|
|
|
|
|
|
|
// socket recv buffer, system will double it.
|
|
|
|
int nb_rbuf = socket_buffer_size / 2;
|
|
|
|
socklen_t sock_buf_size = sizeof(int);
|
|
|
|
if (setsockopt(mr_fd, SOL_SOCKET, SO_RCVBUF, &nb_rbuf, sock_buf_size) < 0) {
|
|
|
|
srs_warn("set sock SO_RCVBUF=%d failed.", nb_rbuf);
|
|
|
|
}
|
|
|
|
getsockopt(mr_fd, SOL_SOCKET, SO_RCVBUF, &nb_rbuf, &sock_buf_size);
|
|
|
|
|
|
|
|
srs_trace("merged read sockbuf=%d, actual=%d, sleep %d when nread<=%d",
|
|
|
|
socket_buffer_size, nb_rbuf, mr_sleep, SRS_MR_SMALL_BYTES);
|
|
|
|
|
|
|
|
rtmp->set_recv_buffer(nb_rbuf);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// update to new state
|
|
|
|
mr = mr_enabled;
|
|
|
|
mr_sleep = sleep_ms;
|
|
|
|
|
|
|
|
#ifdef SRS_PERF_MERGED_READ
|
|
|
|
// apply new state.
|
|
|
|
if (mr) {
|
|
|
|
// enable the merge read
|
|
|
|
// @see https://github.com/winlinvip/simple-rtmp-server/issues/241
|
|
|
|
rtmp->set_merge_read(true, this);
|
|
|
|
} else {
|
|
|
|
// disable the merge read
|
|
|
|
// @see https://github.com/winlinvip/simple-rtmp-server/issues/241
|
|
|
|
rtmp->set_merge_read(false, NULL);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|