1
0
Fork 0
mirror of https://github.com/ossrs/srs.git synced 2025-02-15 04:42:04 +00:00
srs/trunk/src/app/srs_app_source.cpp

2385 lines
64 KiB
C++
Raw Normal View History

/*
The MIT License (MIT)
2015-11-11 02:37:50 +00:00
Copyright (c) 2013-2015 SRS(ossrs)
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_source.hpp>
2014-05-29 06:16:34 +00:00
#include <sstream>
#include <algorithm>
2013-12-15 04:04:28 +00:00
using namespace std;
#include <srs_kernel_log.hpp>
2015-01-23 02:07:20 +00:00
#include <srs_rtmp_stack.hpp>
#include <srs_rtmp_amf0.hpp>
2014-05-28 09:37:15 +00:00
#include <srs_kernel_codec.hpp>
#include <srs_app_hls.hpp>
#include <srs_app_forward.hpp>
#include <srs_app_config.hpp>
#include <srs_app_encoder.hpp>
2015-06-13 08:04:59 +00:00
#include <srs_rtmp_stack.hpp>
2014-04-16 01:28:02 +00:00
#include <srs_app_dvr.hpp>
2014-04-17 08:06:49 +00:00
#include <srs_kernel_stream.hpp>
#include <srs_app_edge.hpp>
#include <srs_kernel_utility.hpp>
#include <srs_kernel_codec.hpp>
2015-01-23 02:07:20 +00:00
#include <srs_rtmp_msg_array.hpp>
2015-03-11 05:34:58 +00:00
#include <srs_app_hds.hpp>
2015-03-08 07:33:08 +00:00
#include <srs_app_statistic.hpp>
#include <srs_core_autofree.hpp>
#include <srs_rtmp_utility.hpp>
#define CONST_MAX_JITTER_MS 250
#define CONST_MAX_JITTER_MS_NEG -250
#define DEFAULT_FRAME_TIME_MS 10
// for 26ms per audio packet,
// 115 packets is 3s.
#define SRS_PURE_AUDIO_GUESS_COUNT 115
2015-07-14 03:28:00 +00:00
// when got these videos or audios, pure audio or video, mix ok.
#define SRS_MIX_CORRECT_PURE_AV 10
2016-09-05 06:13:37 +00:00
// the time to cleanup source in ms.
#define SRS_SOURCE_CLEANUP 30000
int _srs_time_jitter_string2int(std::string time_jitter)
{
if (time_jitter == "full") {
return SrsRtmpJitterAlgorithmFULL;
} else if (time_jitter == "zero") {
return SrsRtmpJitterAlgorithmZERO;
} else {
return SrsRtmpJitterAlgorithmOFF;
}
}
SrsRtmpJitter::SrsRtmpJitter()
{
last_pkt_correct_time = -1;
last_pkt_time = 0;
}
SrsRtmpJitter::~SrsRtmpJitter()
{
}
int SrsRtmpJitter::correct(SrsSharedPtrMessage* msg, SrsRtmpJitterAlgorithm ag)
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
// for performance issue
if (ag != SrsRtmpJitterAlgorithmFULL) {
// all jitter correct features is disabled, ignore.
if (ag == SrsRtmpJitterAlgorithmOFF) {
return ret;
}
// start at zero, but donot ensure monotonically increasing.
if (ag == SrsRtmpJitterAlgorithmZERO) {
// for the first time, last_pkt_correct_time is -1.
if (last_pkt_correct_time == -1) {
last_pkt_correct_time = msg->timestamp;
}
msg->timestamp -= last_pkt_correct_time;
return ret;
}
// other algorithm, ignore.
return ret;
}
// full jitter algorithm, do jitter correct.
2014-03-18 03:32:58 +00:00
// set to 0 for metadata.
if (!msg->is_av()) {
msg->timestamp = 0;
2014-03-18 03:32:58 +00:00
return ret;
}
/**
* we use a very simple time jitter detect/correct algorithm:
* 1. delta: ensure the delta is positive and valid,
* we set the delta to DEFAULT_FRAME_TIME_MS,
* if the delta of time is nagative or greater than CONST_MAX_JITTER_MS.
* 2. last_pkt_time: specifies the original packet time,
* is used to detect next jitter.
* 3. last_pkt_correct_time: simply add the positive delta,
* and enforce the time monotonically.
*/
int64_t time = msg->timestamp;
2014-03-18 03:32:58 +00:00
int64_t delta = time - last_pkt_time;
// if jitter detected, reset the delta.
if (delta < CONST_MAX_JITTER_MS_NEG || delta > CONST_MAX_JITTER_MS) {
// use default 10ms to notice the problem of stream.
2015-11-11 02:37:50 +00:00
// @see https://github.com/ossrs/srs/issues/425
delta = DEFAULT_FRAME_TIME_MS;
2014-03-18 03:32:58 +00:00
srs_info("jitter detected, last_pts=%"PRId64", pts=%"PRId64", diff=%"PRId64", last_time=%"PRId64", time=%"PRId64", diff=%"PRId64"",
last_pkt_time, time, time - last_pkt_time, last_pkt_correct_time, last_pkt_correct_time + delta, delta);
} else {
srs_verbose("timestamp no jitter. time=%"PRId64", last_pkt=%"PRId64", correct_to=%"PRId64"",
time, last_pkt_time, last_pkt_correct_time + delta);
}
last_pkt_correct_time = srs_max(0, last_pkt_correct_time + delta);
msg->timestamp = last_pkt_correct_time;
2014-03-18 03:32:58 +00:00
last_pkt_time = time;
return ret;
}
int SrsRtmpJitter::get_time()
{
2014-03-18 03:32:58 +00:00
return (int)last_pkt_correct_time;
}
#ifdef SRS_PERF_QUEUE_FAST_VECTOR
SrsFastVector::SrsFastVector()
{
count = 0;
nb_msgs = SRS_PERF_MW_MSGS * 8;
msgs = new SrsSharedPtrMessage*[nb_msgs];
}
SrsFastVector::~SrsFastVector()
{
free();
srs_freepa(msgs);
}
int SrsFastVector::size()
{
return count;
}
int SrsFastVector::begin()
{
return 0;
}
int SrsFastVector::end()
{
return count;
}
SrsSharedPtrMessage** SrsFastVector::data()
{
return msgs;
}
SrsSharedPtrMessage* SrsFastVector::at(int index)
{
srs_assert(index < count);
return msgs[index];
}
void SrsFastVector::clear()
{
count = 0;
}
void SrsFastVector::erase(int _begin, int _end)
{
srs_assert(_begin < _end);
// move all erased to previous.
for (int i = 0; i < count - _end; i++) {
msgs[_begin + i] = msgs[_end + i];
}
// update the count.
count -= _end - _begin;
}
void SrsFastVector::push_back(SrsSharedPtrMessage* msg)
{
// increase vector.
if (count >= nb_msgs) {
int size = nb_msgs * 2;
SrsSharedPtrMessage** buf = new SrsSharedPtrMessage*[size];
for (int i = 0; i < nb_msgs; i++) {
buf[i] = msgs[i];
}
srs_warn("fast vector incrase %d=>%d", nb_msgs, size);
// use new array.
srs_freepa(msgs);
msgs = buf;
nb_msgs = size;
}
msgs[count++] = msg;
}
void SrsFastVector::free()
{
for (int i = 0; i < count; i++) {
SrsSharedPtrMessage* msg = msgs[i];
srs_freep(msg);
}
count = 0;
}
#endif
SrsMessageQueue::SrsMessageQueue(bool ignore_shrink)
{
_ignore_shrink = ignore_shrink;
2014-03-18 03:32:58 +00:00
queue_size_ms = 0;
av_start_time = av_end_time = -1;
}
2013-12-15 10:25:55 +00:00
SrsMessageQueue::~SrsMessageQueue()
{
2014-03-18 03:32:58 +00:00
clear();
}
int SrsMessageQueue::size()
{
return (int)msgs.size();
}
int SrsMessageQueue::duration()
{
return (int)(av_end_time - av_start_time);
}
2013-12-15 10:25:55 +00:00
void SrsMessageQueue::set_queue_size(double queue_size)
{
2014-03-18 03:32:58 +00:00
queue_size_ms = (int)(queue_size * 1000);
}
int SrsMessageQueue::enqueue(SrsSharedPtrMessage* msg, bool* is_overflow)
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
if (msg->is_av()) {
2014-03-18 03:32:58 +00:00
if (av_start_time == -1) {
av_start_time = msg->timestamp;
2014-03-18 03:32:58 +00:00
}
av_end_time = msg->timestamp;
2014-03-18 03:32:58 +00:00
}
msgs.push_back(msg);
while (av_end_time - av_start_time > queue_size_ms) {
// notice the caller queue already overflow and shrinked.
if (is_overflow) {
*is_overflow = true;
}
2014-03-18 03:32:58 +00:00
shrink();
}
return ret;
}
int SrsMessageQueue::dump_packets(int max_count, SrsSharedPtrMessage** pmsgs, int& count)
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
2014-12-05 14:00:57 +00:00
int nb_msgs = (int)msgs.size();
if (nb_msgs <= 0) {
2014-03-18 03:32:58 +00:00
return ret;
}
srs_assert(max_count > 0);
count = srs_min(max_count, nb_msgs);
2014-12-05 14:00:57 +00:00
SrsSharedPtrMessage** omsgs = msgs.data();
2014-03-18 03:32:58 +00:00
for (int i = 0; i < count; i++) {
2014-12-05 14:00:57 +00:00
pmsgs[i] = omsgs[i];
2014-03-18 03:32:58 +00:00
}
2014-12-05 14:00:57 +00:00
SrsSharedPtrMessage* last = omsgs[count - 1];
av_start_time = last->timestamp;
2014-03-18 03:32:58 +00:00
if (count >= nb_msgs) {
// the pmsgs is big enough and clear msgs at most time.
2014-03-18 03:32:58 +00:00
msgs.clear();
} else {
// erase some vector elements may cause memory copy,
// maybe can use more efficient vector.swap to avoid copy.
// @remark for the pmsgs is big enough, for instance, SRS_PERF_MW_MSGS 128,
// the rtmp play client will get 128msgs once, so this branch rarely execute.
2014-03-18 03:32:58 +00:00
msgs.erase(msgs.begin(), msgs.begin() + count);
}
return ret;
}
int SrsMessageQueue::dump_packets(SrsConsumer* consumer, bool atc, SrsRtmpJitterAlgorithm ag)
{
int ret = ERROR_SUCCESS;
int nb_msgs = (int)msgs.size();
if (nb_msgs <= 0) {
return ret;
}
SrsSharedPtrMessage** omsgs = msgs.data();
for (int i = 0; i < nb_msgs; i++) {
SrsSharedPtrMessage* msg = omsgs[i];
if ((ret = consumer->enqueue(msg, atc, ag)) != ERROR_SUCCESS) {
return ret;
}
}
return ret;
}
2013-12-15 10:25:55 +00:00
void SrsMessageQueue::shrink()
{
SrsSharedPtrMessage* video_sh = NULL;
SrsSharedPtrMessage* audio_sh = NULL;
int msgs_size = (int)msgs.size();
2014-03-18 03:32:58 +00:00
// remove all msg
// igone the sequence header
for (int i = 0; i < (int)msgs.size(); i++) {
SrsSharedPtrMessage* msg = msgs.at(i);
if (msg->is_video() && SrsFlvCodec::video_is_sequence_header(msg->payload, msg->size)) {
srs_freep(video_sh);
video_sh = msg;
continue;
2014-03-18 03:32:58 +00:00
}
else if (msg->is_audio() && SrsFlvCodec::audio_is_sequence_header(msg->payload, msg->size)) {
srs_freep(audio_sh);
audio_sh = msg;
continue;
}
srs_freep(msg);
2014-03-18 03:32:58 +00:00
}
msgs.clear();
// update av_start_time
av_start_time = av_end_time;
//push_back secquence header and update timestamp
if (video_sh) {
video_sh->timestamp = av_end_time;
msgs.push_back(video_sh);
}
if (audio_sh) {
audio_sh->timestamp = av_end_time;
msgs.push_back(audio_sh);
2014-03-18 03:32:58 +00:00
}
if (_ignore_shrink) {
srs_info("shrink the cache queue, size=%d, removed=%d, max=%.2f",
(int)msgs.size(), msgs_size - (int)msgs.size(), queue_size_ms / 1000.0);
} else {
srs_trace("shrink the cache queue, size=%d, removed=%d, max=%.2f",
(int)msgs.size(), msgs_size - (int)msgs.size(), queue_size_ms / 1000.0);
2014-03-18 03:32:58 +00:00
}
}
2013-12-15 10:25:55 +00:00
void SrsMessageQueue::clear()
{
#ifndef SRS_PERF_QUEUE_FAST_VECTOR
std::vector<SrsSharedPtrMessage*>::iterator it;
2013-12-15 10:25:55 +00:00
2014-03-18 03:32:58 +00:00
for (it = msgs.begin(); it != msgs.end(); ++it) {
SrsSharedPtrMessage* msg = *it;
2014-03-18 03:32:58 +00:00
srs_freep(msg);
}
#else
msgs.free();
#endif
2013-12-15 10:25:55 +00:00
2014-03-18 03:32:58 +00:00
msgs.clear();
av_start_time = av_end_time = -1;
2013-12-15 10:25:55 +00:00
}
ISrsWakable::ISrsWakable()
{
}
ISrsWakable::~ISrsWakable()
{
}
SrsConsumer::SrsConsumer(SrsSource* s, SrsConnection* c)
2013-12-15 10:25:55 +00:00
{
source = s;
conn = c;
2014-03-18 03:32:58 +00:00
paused = false;
jitter = new SrsRtmpJitter();
queue = new SrsMessageQueue();
should_update_source_id = false;
#ifdef SRS_PERF_QUEUE_COND_WAIT
mw_wait = st_cond_new();
mw_min_msgs = 0;
mw_duration = 0;
mw_waiting = false;
#endif
2013-12-15 10:25:55 +00:00
}
SrsConsumer::~SrsConsumer()
{
2014-03-18 03:32:58 +00:00
source->on_consumer_destroy(this);
srs_freep(jitter);
srs_freep(queue);
#ifdef SRS_PERF_QUEUE_COND_WAIT
st_cond_destroy(mw_wait);
#endif
2013-12-15 10:25:55 +00:00
}
void SrsConsumer::set_queue_size(double queue_size)
{
2014-03-18 03:32:58 +00:00
queue->set_queue_size(queue_size);
2013-12-15 10:25:55 +00:00
}
void SrsConsumer::update_source_id()
{
should_update_source_id = true;
}
2013-12-15 10:25:55 +00:00
int SrsConsumer::get_time()
{
2014-03-18 03:32:58 +00:00
return jitter->get_time();
2013-12-15 10:25:55 +00:00
}
int SrsConsumer::enqueue(SrsSharedPtrMessage* shared_msg, bool atc, SrsRtmpJitterAlgorithm ag)
2013-12-15 10:25:55 +00:00
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
SrsSharedPtrMessage* msg = shared_msg->copy();
if (!atc) {
if ((ret = jitter->correct(msg, ag)) != ERROR_SUCCESS) {
srs_freep(msg);
return ret;
}
2014-03-18 03:32:58 +00:00
}
if ((ret = queue->enqueue(msg, NULL)) != ERROR_SUCCESS) {
return ret;
}
#ifdef SRS_PERF_QUEUE_COND_WAIT
srs_verbose("enqueue msg, time=%"PRId64", size=%d, duration=%d, waiting=%d, min_msg=%d",
msg->timestamp, msg->size, queue->duration(), mw_waiting, mw_min_msgs);
// fire the mw when msgs is enough.
if (mw_waiting) {
int duration_ms = queue->duration();
bool match_min_msgs = queue->size() > mw_min_msgs;
// For ATC, maybe the SH timestamp bigger than A/V packet,
// when encoder republish or overflow.
// @see https://github.com/ossrs/srs/pull/749
if (atc && duration_ms < 0) {
st_cond_signal(mw_wait);
mw_waiting = false;
return ret;
}
// when duration ok, signal to flush.
if (match_min_msgs && duration_ms > mw_duration) {
st_cond_signal(mw_wait);
mw_waiting = false;
return ret;
}
}
#endif
2014-03-18 03:32:58 +00:00
return ret;
2013-12-15 10:25:55 +00:00
}
int SrsConsumer::dump_packets(SrsMessageArray* msgs, int& count)
2013-12-15 10:25:55 +00:00
{
int ret =ERROR_SUCCESS;
srs_assert(count >= 0);
srs_assert(msgs->max > 0);
// the count used as input to reset the max if positive.
int max = count? srs_min(count, msgs->max) : msgs->max;
// the count specifies the max acceptable count,
// here maybe 1+, and we must set to 0 when got nothing.
count = 0;
if (should_update_source_id) {
srs_trace("update source_id=%d[%d]", source->source_id(), source->source_id());
should_update_source_id = false;
}
2014-03-18 03:32:58 +00:00
// paused, return nothing.
if (paused) {
return ret;
}
// pump msgs from queue.
if ((ret = queue->dump_packets(max, msgs->msgs, count)) != ERROR_SUCCESS) {
return ret;
}
return ret;
}
#ifdef SRS_PERF_QUEUE_COND_WAIT
void SrsConsumer::wait(int nb_msgs, int duration)
{
if (paused) {
st_usleep(SRS_CONSTS_RTMP_PULSE_TIMEOUT_US);
return;
}
mw_min_msgs = nb_msgs;
mw_duration = duration;
int duration_ms = queue->duration();
bool match_min_msgs = queue->size() > mw_min_msgs;
// when duration ok, signal to flush.
if (match_min_msgs && duration_ms > mw_duration) {
return;
}
// the enqueue will notify this cond.
mw_waiting = true;
// use cond block wait for high performance mode.
st_cond_wait(mw_wait);
}
#endif
2013-12-15 10:25:55 +00:00
int SrsConsumer::on_play_client_pause(bool is_pause)
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
srs_trace("stream consumer change pause state %d=>%d", paused, is_pause);
paused = is_pause;
return ret;
}
void SrsConsumer::wakeup()
{
#ifdef SRS_PERF_QUEUE_COND_WAIT
if (mw_waiting) {
st_cond_signal(mw_wait);
mw_waiting = false;
}
#endif
}
SrsGopCache::SrsGopCache()
{
2014-03-18 03:32:58 +00:00
cached_video_count = 0;
enable_gop_cache = true;
audio_after_last_video_count = 0;
}
SrsGopCache::~SrsGopCache()
{
2014-03-18 03:32:58 +00:00
clear();
}
2015-06-07 01:27:47 +00:00
void SrsGopCache::dispose()
{
clear();
}
void SrsGopCache::set(bool enabled)
{
2014-03-18 03:32:58 +00:00
enable_gop_cache = enabled;
if (!enabled) {
srs_info("disable gop cache, clear %d packets.", (int)gop_cache.size());
clear();
return;
}
srs_info("enable gop cache");
}
int SrsGopCache::cache(SrsSharedPtrMessage* shared_msg)
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
if (!enable_gop_cache) {
srs_verbose("gop cache is disabled.");
return ret;
}
// the gop cache know when to gop it.
SrsSharedPtrMessage* msg = shared_msg;
2014-03-18 03:32:58 +00:00
// got video, update the video count if acceptable
if (msg->is_video()) {
// drop video when not h.264
if (!SrsFlvCodec::video_is_h264(msg->payload, msg->size)) {
srs_info("gop cache drop video for none h.264");
return ret;
}
2014-03-18 03:32:58 +00:00
cached_video_count++;
audio_after_last_video_count = 0;
2014-03-18 03:32:58 +00:00
}
// no acceptable video or pure audio, disable the cache.
if (pure_audio()) {
2014-03-18 03:32:58 +00:00
srs_verbose("ignore any frame util got a h264 video frame.");
return ret;
}
// ok, gop cache enabled, and got an audio.
if (msg->is_audio()) {
audio_after_last_video_count++;
}
// clear gop cache when pure audio count overflow
if (audio_after_last_video_count > SRS_PURE_AUDIO_GUESS_COUNT) {
srs_warn("clear gop cache for guess pure audio overflow");
clear();
return ret;
}
2014-03-18 03:32:58 +00:00
// clear gop cache when got key frame
if (msg->is_video() && SrsFlvCodec::video_is_keyframe(msg->payload, msg->size)) {
2014-03-18 03:32:58 +00:00
srs_info("clear gop cache when got keyframe. vcount=%d, count=%d",
cached_video_count, (int)gop_cache.size());
clear();
// curent msg is video frame, so we set to 1.
cached_video_count = 1;
}
// cache the frame.
gop_cache.push_back(msg->copy());
return ret;
}
void SrsGopCache::clear()
{
std::vector<SrsSharedPtrMessage*>::iterator it;
2014-03-18 03:32:58 +00:00
for (it = gop_cache.begin(); it != gop_cache.end(); ++it) {
SrsSharedPtrMessage* msg = *it;
2014-03-18 03:32:58 +00:00
srs_freep(msg);
}
gop_cache.clear();
2014-03-18 03:32:58 +00:00
cached_video_count = 0;
audio_after_last_video_count = 0;
}
2014-03-18 03:32:58 +00:00
int SrsGopCache::dump(SrsConsumer* consumer, bool atc, SrsRtmpJitterAlgorithm jitter_algorithm)
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
std::vector<SrsSharedPtrMessage*>::iterator it;
2014-03-18 03:32:58 +00:00
for (it = gop_cache.begin(); it != gop_cache.end(); ++it) {
SrsSharedPtrMessage* msg = *it;
if ((ret = consumer->enqueue(msg, atc, jitter_algorithm)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("dispatch cached gop failed. ret=%d", ret);
return ret;
}
}
srs_trace("dispatch cached gop success. count=%d, duration=%d", (int)gop_cache.size(), consumer->get_time());
return ret;
}
bool SrsGopCache::empty()
{
return gop_cache.empty();
}
int64_t SrsGopCache::start_time()
{
if (empty()) {
return 0;
}
SrsSharedPtrMessage* msg = gop_cache[0];
srs_assert(msg);
return msg->timestamp;
}
bool SrsGopCache::pure_audio()
{
return cached_video_count == 0;
}
ISrsSourceHandler::ISrsSourceHandler()
{
}
ISrsSourceHandler::~ISrsSourceHandler()
{
}
std::map<std::string, SrsSource*> SrsSource::pool;
int SrsSource::fetch_or_create(SrsRequest* r, ISrsSourceHandler* h, SrsSource** pps)
{
int ret = ERROR_SUCCESS;
SrsSource* source = NULL;
if ((source = fetch(r)) != NULL) {
*pps = source;
return ret;
}
string stream_url = r->get_stream_url();
string vhost = r->vhost;
2014-03-18 03:32:58 +00:00
// should always not exists for create a source.
srs_assert (pool.find(stream_url) == pool.end());
source = new SrsSource();
if ((ret = source->initialize(r, h)) != ERROR_SUCCESS) {
srs_freep(source);
return ret;
2014-03-18 03:32:58 +00:00
}
pool[stream_url] = source;
srs_info("create new source for url=%s, vhost=%s", stream_url.c_str(), vhost.c_str());
2014-03-18 03:32:58 +00:00
*pps = source;
return ret;
}
SrsSource* SrsSource::fetch(SrsRequest* r)
{
SrsSource* source = NULL;
string stream_url = r->get_stream_url();
if (pool.find(stream_url) == pool.end()) {
return NULL;
}
source = pool[stream_url];
// we always update the request of resource,
// for origin auth is on, the token in request maybe invalid,
// and we only need to update the token of request, it's simple.
source->_req->update_auth(r);
return source;
}
void SrsSource::dispose_all()
{
std::map<std::string, SrsSource*>::iterator it;
for (it = pool.begin(); it != pool.end(); ++it) {
SrsSource* source = it->second;
source->dispose();
}
return;
}
int SrsSource::cycle_all()
2016-09-05 06:13:37 +00:00
{
int ret = ERROR_SUCCESS;
2016-09-05 06:13:37 +00:00
int cid = _srs_context->get_id();
ret = do_cycle_all();
2016-09-05 06:13:37 +00:00
_srs_context->set_id(cid);
2016-09-05 06:13:37 +00:00
return ret;
}
int SrsSource::do_cycle_all()
{
int ret = ERROR_SUCCESS;
std::map<std::string, SrsSource*>::iterator it;
2016-09-05 06:13:37 +00:00
for (it = pool.begin(); it != pool.end();) {
SrsSource* source = it->second;
// Do cycle source to cleanup components, such as hls dispose.
if ((ret = source->cycle()) != ERROR_SUCCESS) {
return ret;
}
2016-09-05 06:13:37 +00:00
// TODO: FIXME: support source cleanup.
// @see https://github.com/ossrs/srs/issues/713
// @see https://github.com/ossrs/srs/issues/714
#if 0
// When source expired, remove it.
2016-09-05 06:13:37 +00:00
if (source->expired()) {
int cid = source->source_id();
if (cid == -1 && source->pre_source_id() > 0) {
cid = source->pre_source_id();
}
if (cid > 0) {
_srs_context->set_id(cid);
}
srs_trace("cleanup die source, total=%d", (int)pool.size());
srs_freep(source);
pool.erase(it++);
} else {
++it;
}
#else
++it;
#endif
}
return ret;
}
void SrsSource::destroy()
{
std::map<std::string, SrsSource*>::iterator it;
for (it = pool.begin(); it != pool.end(); ++it) {
SrsSource* source = it->second;
srs_freep(source);
}
pool.clear();
}
SrsMixQueue::SrsMixQueue()
{
nb_videos = 0;
nb_audios = 0;
}
SrsMixQueue::~SrsMixQueue()
{
clear();
}
void SrsMixQueue::clear()
{
std::multimap<int64_t, SrsSharedPtrMessage*>::iterator it;
for (it = msgs.begin(); it != msgs.end(); ++it) {
SrsSharedPtrMessage* msg = it->second;
srs_freep(msg);
}
msgs.clear();
nb_videos = 0;
nb_audios = 0;
}
void SrsMixQueue::push(SrsSharedPtrMessage* msg)
{
msgs.insert(std::make_pair(msg->timestamp, msg));
if (msg->is_video()) {
nb_videos++;
} else {
nb_audios++;
}
}
SrsSharedPtrMessage* SrsMixQueue::pop()
{
2015-07-14 03:28:00 +00:00
bool mix_ok = false;
// pure video
if (nb_videos >= SRS_MIX_CORRECT_PURE_AV && nb_audios == 0) {
mix_ok = true;
}
// pure audio
if (nb_audios >= SRS_MIX_CORRECT_PURE_AV && nb_videos == 0) {
mix_ok = true;
}
// got 1 video and 1 audio, mix ok.
if (nb_videos >= 1 && nb_audios >= 1) {
mix_ok = true;
}
if (!mix_ok) {
return NULL;
}
// pop the first msg.
std::multimap<int64_t, SrsSharedPtrMessage*>::iterator it = msgs.begin();
SrsSharedPtrMessage* msg = it->second;
msgs.erase(it);
if (msg->is_video()) {
nb_videos--;
} else {
nb_audios--;
}
return msg;
}
SrsSource::SrsSource()
{
_req = NULL;
jitter_algorithm = SrsRtmpJitterAlgorithmOFF;
mix_correct = false;
mix_queue = new SrsMixQueue();
2014-03-18 03:32:58 +00:00
#ifdef SRS_AUTO_HLS
hls = new SrsHls();
#endif
2014-04-16 01:28:02 +00:00
#ifdef SRS_AUTO_DVR
dvr = new SrsDvr();
2014-04-16 01:28:02 +00:00
#endif
#ifdef SRS_AUTO_TRANSCODE
2014-03-18 03:32:58 +00:00
encoder = new SrsEncoder();
#endif
2015-03-12 03:15:15 +00:00
#ifdef SRS_AUTO_HDS
2015-03-11 05:34:58 +00:00
hds = new SrsHds(this);
2015-03-12 03:15:15 +00:00
#endif
2014-03-18 03:32:58 +00:00
cache_metadata = cache_sh_video = cache_sh_audio = NULL;
_can_publish = true;
2016-09-05 06:13:37 +00:00
_pre_source_id = _source_id = -1;
die_at = -1;
2014-03-18 03:32:58 +00:00
2014-04-27 01:29:37 +00:00
play_edge = new SrsPlayEdge();
publish_edge = new SrsPublishEdge();
2014-03-18 03:32:58 +00:00
gop_cache = new SrsGopCache();
aggregate_stream = new SrsStream();
2014-03-18 03:32:58 +00:00
is_monotonically_increase = false;
last_packet_time = 0;
2014-03-18 03:32:58 +00:00
_srs_config->subscribe(this);
atc = false;
}
SrsSource::~SrsSource()
{
2014-03-18 03:32:58 +00:00
_srs_config->unsubscribe(this);
// never free the consumers,
// for all consumers are auto free.
consumers.clear();
2014-03-18 03:32:58 +00:00
if (true) {
std::vector<SrsForwarder*>::iterator it;
for (it = forwarders.begin(); it != forwarders.end(); ++it) {
SrsForwarder* forwarder = *it;
srs_freep(forwarder);
}
forwarders.clear();
}
srs_freep(mix_queue);
2014-03-18 03:32:58 +00:00
srs_freep(cache_metadata);
srs_freep(cache_sh_video);
srs_freep(cache_sh_audio);
2014-04-27 01:29:37 +00:00
srs_freep(play_edge);
srs_freep(publish_edge);
2014-03-18 03:32:58 +00:00
srs_freep(gop_cache);
srs_freep(aggregate_stream);
2014-03-18 03:32:58 +00:00
#ifdef SRS_AUTO_HLS
2014-03-18 03:32:58 +00:00
srs_freep(hls);
#endif
2014-04-16 01:28:02 +00:00
#ifdef SRS_AUTO_DVR
srs_freep(dvr);
#endif
#ifdef SRS_AUTO_TRANSCODE
2014-03-18 03:32:58 +00:00
srs_freep(encoder);
#endif
2015-03-12 03:15:15 +00:00
#ifdef SRS_AUTO_HDS
srs_freep(hds);
#endif
2013-12-15 04:34:22 +00:00
2014-04-26 09:16:18 +00:00
srs_freep(_req);
}
void SrsSource::dispose()
{
#ifdef SRS_AUTO_HLS
hls->dispose();
#endif
2015-06-07 01:27:47 +00:00
2015-06-08 01:47:45 +00:00
// cleaup the cached packets.
srs_freep(cache_metadata);
srs_freep(cache_sh_video);
srs_freep(cache_sh_audio);
// cleanup the gop cache.
2015-06-07 01:27:47 +00:00
gop_cache->dispose();
}
int SrsSource::cycle()
{
int ret = ERROR_SUCCESS;
#ifdef SRS_AUTO_HLS
if ((ret = hls->cycle()) != ERROR_SUCCESS) {
return ret;
}
#endif
return ret;
}
2016-09-05 06:13:37 +00:00
bool SrsSource::expired()
{
2016-09-05 06:56:31 +00:00
// unknown state?
if (die_at == -1) {
return false;
}
// still publishing?
2016-11-05 02:46:24 +00:00
if (!_can_publish || !publish_edge->can_publish()) {
2016-09-05 06:56:31 +00:00
return false;
}
// has any consumers?
if (!consumers.empty()) {
2016-09-05 06:13:37 +00:00
return false;
}
int64_t now = srs_get_system_time_ms();
if (now > die_at + SRS_SOURCE_CLEANUP) {
return true;
}
return false;
}
int SrsSource::initialize(SrsRequest* r, ISrsSourceHandler* h)
{
int ret = ERROR_SUCCESS;
srs_assert(h);
srs_assert(!_req);
handler = h;
_req = r->copy();
atc = _srs_config->get_atc(_req->vhost);
#ifdef SRS_AUTO_HLS
if ((ret = hls->initialize(this)) != ERROR_SUCCESS) {
return ret;
}
#endif
#ifdef SRS_AUTO_DVR
if ((ret = dvr->initialize(this, _req)) != ERROR_SUCCESS) {
return ret;
}
#endif
2014-04-26 09:16:18 +00:00
2014-04-27 01:29:37 +00:00
if ((ret = play_edge->initialize(this, _req)) != ERROR_SUCCESS) {
return ret;
}
if ((ret = publish_edge->initialize(this, _req)) != ERROR_SUCCESS) {
2014-04-26 09:16:18 +00:00
return ret;
}
2014-04-28 09:20:35 +00:00
double queue_size = _srs_config->get_queue_length(_req->vhost);
publish_edge->set_queue_size(queue_size);
jitter_algorithm = (SrsRtmpJitterAlgorithm)_srs_config->get_time_jitter(_req->vhost);
mix_correct = _srs_config->get_mix_correct(_req->vhost);
return ret;
}
int SrsSource::on_reload_vhost_atc(string vhost)
2014-04-12 14:00:27 +00:00
{
int ret = ERROR_SUCCESS;
2014-04-26 09:16:18 +00:00
if (_req->vhost != vhost) {
2014-04-12 14:00:27 +00:00
return ret;
}
// atc changed.
bool enabled_atc = _srs_config->get_atc(vhost);
srs_warn("vhost %s atc changed to %d, connected client may corrupt.",
vhost.c_str(), enabled_atc);
gop_cache->clear();
return ret;
}
int SrsSource::on_reload_vhost_gop_cache(string vhost)
2013-12-15 04:04:28 +00:00
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
2014-04-26 09:16:18 +00:00
if (_req->vhost != vhost) {
2014-03-18 03:32:58 +00:00
return ret;
}
// gop cache changed.
bool enabled_cache = _srs_config->get_gop_cache(vhost);
srs_trace("vhost %s gop_cache changed to %d, source url=%s",
2014-04-26 09:16:18 +00:00
vhost.c_str(), enabled_cache, _req->get_stream_url().c_str());
2014-03-18 03:32:58 +00:00
set_cache(enabled_cache);
return ret;
2013-12-15 04:04:28 +00:00
}
int SrsSource::on_reload_vhost_queue_length(string vhost)
2013-12-15 10:25:55 +00:00
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
2014-04-26 09:16:18 +00:00
if (_req->vhost != vhost) {
2014-03-18 03:32:58 +00:00
return ret;
}
2014-04-26 09:16:18 +00:00
double queue_size = _srs_config->get_queue_length(_req->vhost);
2014-03-18 03:32:58 +00:00
if (true) {
std::vector<SrsConsumer*>::iterator it;
for (it = consumers.begin(); it != consumers.end(); ++it) {
SrsConsumer* consumer = *it;
consumer->set_queue_size(queue_size);
}
srs_trace("consumers reload queue size success.");
}
if (true) {
std::vector<SrsForwarder*>::iterator it;
for (it = forwarders.begin(); it != forwarders.end(); ++it) {
SrsForwarder* forwarder = *it;
forwarder->set_queue_size(queue_size);
}
srs_trace("forwarders reload queue size success.");
}
2014-04-28 09:20:35 +00:00
if (true) {
publish_edge->set_queue_size(queue_size);
srs_trace("publish_edge reload queue size success.");
}
2014-03-18 03:32:58 +00:00
return ret;
2013-12-15 10:25:55 +00:00
}
int SrsSource::on_reload_vhost_time_jitter(string vhost)
{
int ret = ERROR_SUCCESS;
if (_req->vhost != vhost) {
return ret;
}
jitter_algorithm = (SrsRtmpJitterAlgorithm)_srs_config->get_time_jitter(_req->vhost);
return ret;
}
int SrsSource::on_reload_vhost_mix_correct(string vhost)
{
int ret = ERROR_SUCCESS;
if (_req->vhost != vhost) {
return ret;
}
bool v = _srs_config->get_mix_correct(_req->vhost);
// when changed, clear the mix queue.
if (v != mix_correct) {
mix_queue->clear();
}
mix_correct = v;
return ret;
}
int SrsSource::on_reload_vhost_forward(string vhost)
2013-12-15 04:34:22 +00:00
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
2014-04-26 09:16:18 +00:00
if (_req->vhost != vhost) {
2014-03-18 03:32:58 +00:00
return ret;
}
2013-12-15 04:34:22 +00:00
2014-03-18 03:32:58 +00:00
// forwarders
destroy_forwarders();
if ((ret = create_forwarders()) != ERROR_SUCCESS) {
srs_error("create forwarders failed. ret=%d", ret);
return ret;
}
2014-03-18 03:32:58 +00:00
srs_trace("vhost %s forwarders reload success", vhost.c_str());
return ret;
2013-12-15 04:34:22 +00:00
}
int SrsSource::on_reload_vhost_hls(string vhost)
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
2014-04-26 09:16:18 +00:00
if (_req->vhost != vhost) {
2014-03-18 03:32:58 +00:00
return ret;
}
#ifdef SRS_AUTO_HLS
2014-03-18 03:32:58 +00:00
hls->on_unpublish();
if ((ret = hls->on_publish(_req, true)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("hls publish failed. ret=%d", ret);
return ret;
}
srs_trace("vhost %s hls reload success", vhost.c_str());
#endif
2014-03-18 03:32:58 +00:00
return ret;
}
2015-03-12 14:38:11 +00:00
int SrsSource::on_reload_vhost_hds(string vhost)
{
int ret = ERROR_SUCCESS;
if (_req->vhost != vhost) {
return ret;
}
#ifdef SRS_AUTO_HDS
hds->on_unpublish();
if ((ret = hds->on_publish(_req)) != ERROR_SUCCESS) {
srs_error("hds publish failed. ret=%d", ret);
return ret;
}
srs_trace("vhost %s hds reload success", vhost.c_str());
#endif
return ret;
}
2014-04-17 08:22:21 +00:00
int SrsSource::on_reload_vhost_dvr(string vhost)
{
int ret = ERROR_SUCCESS;
2014-04-26 09:16:18 +00:00
if (_req->vhost != vhost) {
2014-04-17 08:22:21 +00:00
return ret;
}
#ifdef SRS_AUTO_DVR
// cleanup dvr
2014-04-17 08:22:21 +00:00
dvr->on_unpublish();
// reinitialize the dvr, update plan.
if ((ret = dvr->initialize(this, _req)) != ERROR_SUCCESS) {
return ret;
}
// start to publish by new plan.
2014-04-26 09:16:18 +00:00
if ((ret = dvr->on_publish(_req)) != ERROR_SUCCESS) {
2014-04-17 08:22:21 +00:00
srs_error("dvr publish failed. ret=%d", ret);
return ret;
}
2014-04-17 08:22:21 +00:00
srs_trace("vhost %s dvr reload success", vhost.c_str());
#endif
return ret;
}
int SrsSource::on_reload_vhost_transcode(string vhost)
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
2014-04-26 09:16:18 +00:00
if (_req->vhost != vhost) {
2014-03-18 03:32:58 +00:00
return ret;
}
#ifdef SRS_AUTO_TRANSCODE
2014-03-18 03:32:58 +00:00
encoder->on_unpublish();
2014-04-26 09:16:18 +00:00
if ((ret = encoder->on_publish(_req)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("start encoder failed. ret=%d", ret);
return ret;
}
srs_trace("vhost %s transcode reload success", vhost.c_str());
#endif
2014-03-18 03:32:58 +00:00
return ret;
}
int SrsSource::on_forwarder_start(SrsForwarder* forwarder)
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
// feed the forwarder the metadata/sequence header,
// when reload to enable the forwarder.
if (cache_metadata && (ret = forwarder->on_meta_data(cache_metadata)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("forwarder process onMetaData message failed. ret=%d", ret);
return ret;
}
if (cache_sh_video && (ret = forwarder->on_video(cache_sh_video)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("forwarder process video sequence header message failed. ret=%d", ret);
return ret;
}
if (cache_sh_audio && (ret = forwarder->on_audio(cache_sh_audio)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("forwarder process audio sequence header message failed. ret=%d", ret);
return ret;
}
return ret;
}
int SrsSource::on_hls_start()
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
#ifdef SRS_AUTO_HLS
2014-03-18 03:32:58 +00:00
// feed the hls the metadata/sequence header,
2014-04-16 03:11:53 +00:00
// when reload to start hls, hls will never get the sequence header in stream,
// use the SrsSource.on_hls_start to push the sequence header to HLS.
2014-03-18 03:32:58 +00:00
// TODO: maybe need to decode the metadata?
if (cache_sh_video && (ret = hls->on_video(cache_sh_video, true)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("hls process video sequence header message failed. ret=%d", ret);
return ret;
}
if (cache_sh_audio && (ret = hls->on_audio(cache_sh_audio)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("hls process audio sequence header message failed. ret=%d", ret);
return ret;
}
2014-04-17 08:06:49 +00:00
#endif
return ret;
}
int SrsSource::on_dvr_request_sh()
2014-04-17 08:06:49 +00:00
{
int ret = ERROR_SUCCESS;
#ifdef SRS_AUTO_DVR
// feed the dvr the metadata/sequence header,
// when reload to start dvr, dvr will never get the sequence header in stream,
// use the SrsSource.on_dvr_request_sh to push the sequence header to DVR.
2014-04-17 08:06:49 +00:00
if (cache_metadata) {
char* payload = cache_metadata->payload;
int size = cache_metadata->size;
2014-04-17 08:06:49 +00:00
SrsStream stream;
if ((ret = stream.initialize(payload, size)) != ERROR_SUCCESS) {
srs_error("dvr decode metadata stream failed. ret=%d", ret);
return ret;
}
SrsOnMetaDataPacket pkt;
if ((ret = pkt.decode(&stream)) != ERROR_SUCCESS) {
srs_error("dvr decode metadata packet failed.");
return ret;
}
if ((ret = dvr->on_meta_data(&pkt)) != ERROR_SUCCESS) {
srs_error("dvr process onMetaData message failed. ret=%d", ret);
return ret;
}
}
2014-03-18 03:32:58 +00:00
if (cache_sh_video && (ret = dvr->on_video(cache_sh_video)) != ERROR_SUCCESS) {
2014-04-17 08:06:49 +00:00
srs_error("dvr process video sequence header message failed. ret=%d", ret);
return ret;
}
if (cache_sh_audio && (ret = dvr->on_audio(cache_sh_audio)) != ERROR_SUCCESS) {
2014-04-17 08:06:49 +00:00
srs_error("dvr process audio sequence header message failed. ret=%d", ret);
return ret;
}
#endif
2014-03-18 03:32:58 +00:00
return ret;
}
int SrsSource::on_source_id_changed(int id)
{
int ret = ERROR_SUCCESS;
if (_source_id == id) {
return ret;
}
2016-09-05 06:13:37 +00:00
if (_pre_source_id == -1) {
_pre_source_id = id;
} else if (_pre_source_id != _source_id) {
_pre_source_id = _source_id;
}
_source_id = id;
// notice all consumer
std::vector<SrsConsumer*>::iterator it;
for (it = consumers.begin(); it != consumers.end(); ++it) {
SrsConsumer* consumer = *it;
consumer->update_source_id();
}
return ret;
}
int SrsSource::source_id()
{
return _source_id;
}
2016-09-05 06:13:37 +00:00
int SrsSource::pre_source_id()
{
return _pre_source_id;
}
2015-07-08 09:08:29 +00:00
bool SrsSource::can_publish(bool is_edge)
{
2015-07-08 09:08:29 +00:00
if (is_edge) {
return publish_edge->can_publish();
}
2015-07-08 09:08:29 +00:00
return _can_publish;
}
int SrsSource::on_meta_data(SrsCommonMessage* msg, SrsOnMetaDataPacket* metadata)
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
#ifdef SRS_AUTO_HLS
2014-03-18 03:32:58 +00:00
if (metadata && (ret = hls->on_meta_data(metadata->metadata)) != ERROR_SUCCESS) {
srs_error("hls process onMetaData message failed. ret=%d", ret);
return ret;
}
#endif
2014-03-18 03:32:58 +00:00
2014-04-16 01:28:02 +00:00
#ifdef SRS_AUTO_DVR
2014-04-17 08:06:49 +00:00
if (metadata && (ret = dvr->on_meta_data(metadata)) != ERROR_SUCCESS) {
2014-04-16 01:28:02 +00:00
srs_error("dvr process onMetaData message failed. ret=%d", ret);
return ret;
}
#endif
2014-05-29 06:16:34 +00:00
SrsAmf0Any* prop = NULL;
// when exists the duration, remove it to make ExoPlayer happy.
if (metadata->metadata->get_property("duration") != NULL) {
metadata->metadata->remove("duration");
}
2014-05-29 06:16:34 +00:00
// generate metadata info to print
std::stringstream ss;
if ((prop = metadata->metadata->ensure_property_number("width")) != NULL) {
ss << ", width=" << (int)prop->to_number();
}
if ((prop = metadata->metadata->ensure_property_number("height")) != NULL) {
ss << ", height=" << (int)prop->to_number();
}
if ((prop = metadata->metadata->ensure_property_number("videocodecid")) != NULL) {
ss << ", vcodec=" << (int)prop->to_number();
}
if ((prop = metadata->metadata->ensure_property_number("audiocodecid")) != NULL) {
ss << ", acodec=" << (int)prop->to_number();
}
srs_trace("got metadata%s", ss.str().c_str());
2014-04-16 01:28:02 +00:00
2014-05-29 06:16:34 +00:00
// add server info to metadata
metadata->metadata->set("server", SrsAmf0Any::str(RTMP_SIG_SRS_SERVER));
metadata->metadata->set("srs_primary", SrsAmf0Any::str(RTMP_SIG_SRS_PRIMARY));
metadata->metadata->set("srs_authors", SrsAmf0Any::str(RTMP_SIG_SRS_AUTHROS));
2014-03-18 03:32:58 +00:00
// version, for example, 1.0.0
// add version to metadata, please donot remove it, for debug.
metadata->metadata->set("server_version", SrsAmf0Any::str(RTMP_SIG_SRS_VERSION));
// if allow atc_auto and bravo-atc detected, open atc for vhost.
2014-04-30 02:31:05 +00:00
atc = _srs_config->get_atc(_req->vhost);
if (_srs_config->get_atc_auto(_req->vhost)) {
if ((prop = metadata->metadata->get_property("bravo_atc")) != NULL) {
if (prop->is_string() && prop->to_str() == "true") {
atc = true;
}
2014-04-29 10:27:00 +00:00
}
}
2014-03-18 03:32:58 +00:00
// encode the metadata to payload
int size = 0;
2014-03-18 03:32:58 +00:00
char* payload = NULL;
if ((ret = metadata->encode(size, payload)) != ERROR_SUCCESS) {
srs_error("encode metadata error. ret=%d", ret);
srs_freep(payload);
2014-03-18 03:32:58 +00:00
return ret;
}
srs_verbose("encode metadata success.");
if (size <= 0) {
srs_warn("ignore the invalid metadata. size=%d", size);
return ret;
}
// when already got metadata, drop when reduce sequence header.
bool drop_for_reduce = false;
if (cache_metadata && _srs_config->get_reduce_sequence_header(_req->vhost)) {
drop_for_reduce = true;
srs_warn("drop for reduce sh metadata, size=%d", msg->size);
}
2014-03-18 03:32:58 +00:00
// create a shared ptr message.
srs_freep(cache_metadata);
cache_metadata = new SrsSharedPtrMessage();
2014-03-18 03:32:58 +00:00
// dump message to shared ptr message.
// the payload/size managed by cache_metadata, user should not free it.
if ((ret = cache_metadata->create(&msg->header, payload, size)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("initialize the cache metadata failed. ret=%d", ret);
return ret;
}
srs_verbose("initialize shared ptr metadata success.");
// copy to all consumer
if (!drop_for_reduce) {
2014-03-18 03:32:58 +00:00
std::vector<SrsConsumer*>::iterator it;
for (it = consumers.begin(); it != consumers.end(); ++it) {
SrsConsumer* consumer = *it;
if ((ret = consumer->enqueue(cache_metadata, atc, jitter_algorithm)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("dispatch the metadata failed. ret=%d", ret);
return ret;
}
}
}
// copy to all forwarders
if (true) {
std::vector<SrsForwarder*>::iterator it;
for (it = forwarders.begin(); it != forwarders.end(); ++it) {
SrsForwarder* forwarder = *it;
if ((ret = forwarder->on_meta_data(cache_metadata)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("forwarder process onMetaData message failed. ret=%d", ret);
return ret;
}
}
}
return ret;
}
int SrsSource::on_audio(SrsCommonMessage* shared_audio)
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
// monotically increase detect.
if (!mix_correct && is_monotonically_increase) {
if (last_packet_time > 0 && shared_audio->header.timestamp < last_packet_time) {
is_monotonically_increase = false;
2015-08-05 14:54:29 +00:00
srs_warn("AUDIO: stream not monotonically increase, please open mix_correct.");
}
}
last_packet_time = shared_audio->header.timestamp;
// convert shared_audio to msg, user should not use shared_audio again.
// the payload is transfer to msg, and set to NULL in shared_audio.
SrsSharedPtrMessage msg;
if ((ret = msg.create(shared_audio)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("initialize the audio failed. ret=%d", ret);
return ret;
}
srs_info("Audio dts=%"PRId64", size=%d", msg.timestamp, msg.size);
2014-03-18 03:32:58 +00:00
// directly process the audio message.
if (!mix_correct) {
return on_audio_imp(&msg);
}
// insert msg to the queue.
mix_queue->push(msg.copy());
// fetch someone from mix queue.
SrsSharedPtrMessage* m = mix_queue->pop();
if (!m) {
return ret;
}
// consume the monotonically increase message.
if (m->is_audio()) {
ret = on_audio_imp(m);
} else {
ret = on_video_imp(m);
}
srs_freep(m);
return ret;
}
2015-12-15 08:25:21 +00:00
bool srs_hls_can_continue(int ret, SrsSharedPtrMessage* sh, SrsSharedPtrMessage* msg)
{
// only continue for decode error.
if (ret != ERROR_HLS_DECODE_ERROR) {
return false;
}
// when video size equals to sequence header,
// the video actually maybe a sequence header,
// continue to make ffmpeg happy.
if (sh && sh->size == msg->size) {
srs_warn("the msg is actually a sequence header, ignore this packet.");
return true;
}
return false;
}
int SrsSource::on_audio_imp(SrsSharedPtrMessage* msg)
{
int ret = ERROR_SUCCESS;
srs_info("Audio dts=%"PRId64", size=%d", msg->timestamp, msg->size);
bool is_aac_sequence_header = SrsFlvCodec::audio_is_sequence_header(msg->payload, msg->size);
bool is_sequence_header = is_aac_sequence_header;
// whether consumer should drop for the duplicated sequence header.
bool drop_for_reduce = false;
if (is_sequence_header && cache_sh_audio && _srs_config->get_reduce_sequence_header(_req->vhost)) {
if (cache_sh_audio->size == msg->size) {
drop_for_reduce = srs_bytes_equals(cache_sh_audio->payload, msg->payload, msg->size);
srs_warn("drop for reduce sh audio, size=%d", msg->size);
}
}
// cache the sequence header if aac
// donot cache the sequence header to gop_cache, return here.
if (is_aac_sequence_header) {
// parse detail audio codec
SrsAvcAacCodec codec;
SrsCodecSample sample;
if ((ret = codec.audio_aac_demux(msg->payload, msg->size, &sample)) != ERROR_SUCCESS) {
srs_error("source codec demux audio failed. ret=%d", ret);
return ret;
}
static int flv_sample_sizes[] = {8, 16, 0};
static int flv_sound_types[] = {1, 2, 0};
// when got audio stream info.
SrsStatistic* stat = SrsStatistic::instance();
if ((ret = stat->on_audio_info(_req, SrsCodecAudioAAC, sample.sound_rate, sample.sound_type, codec.aac_object)) != ERROR_SUCCESS) {
return ret;
}
srs_trace("%dB audio sh, codec(%d, profile=%s, %dchannels, %dkbps, %dHZ), "
"flv(%dbits, %dchannels, %dHZ)",
msg->size, codec.audio_codec_id,
srs_codec_aac_object2str(codec.aac_object).c_str(), codec.aac_channels,
codec.audio_data_rate / 1000, aac_sample_rates[codec.aac_sample_rate],
flv_sample_sizes[sample.sound_size], flv_sound_types[sample.sound_type],
flv_sample_rates[sample.sound_rate]);
}
#ifdef SRS_AUTO_HLS
if ((ret = hls->on_audio(msg)) != ERROR_SUCCESS) {
// apply the error strategy for hls.
2015-11-11 02:37:50 +00:00
// @see https://github.com/ossrs/srs/issues/264
std::string hls_error_strategy = _srs_config->get_hls_on_error(_req->vhost);
if (srs_config_hls_is_on_error_ignore(hls_error_strategy)) {
srs_warn("hls process audio message failed, ignore and disable hls. ret=%d", ret);
// unpublish, ignore ret.
hls->on_unpublish();
// ignore.
ret = ERROR_SUCCESS;
} else if (srs_config_hls_is_on_error_continue(hls_error_strategy)) {
2015-12-15 08:25:21 +00:00
if (srs_hls_can_continue(ret, cache_sh_audio, msg)) {
ret = ERROR_SUCCESS;
} else {
srs_warn("hls continue audio failed. ret=%d", ret);
return ret;
}
} else {
srs_warn("hls disconnect publisher for audio error. ret=%d", ret);
return ret;
}
2014-03-18 03:32:58 +00:00
}
#endif
2014-03-18 03:32:58 +00:00
2014-04-16 01:28:02 +00:00
#ifdef SRS_AUTO_DVR
if ((ret = dvr->on_audio(msg)) != ERROR_SUCCESS) {
2014-04-16 01:28:02 +00:00
srs_warn("dvr process audio message failed, ignore and disable dvr. ret=%d", ret);
// unpublish, ignore ret.
dvr->on_unpublish();
// ignore.
ret = ERROR_SUCCESS;
}
#endif
2015-03-11 05:34:58 +00:00
2015-03-12 03:15:15 +00:00
#ifdef SRS_AUTO_HDS
if ((ret = hds->on_audio(msg)) != ERROR_SUCCESS) {
2015-03-12 03:15:15 +00:00
srs_warn("hds process audio message failed, ignore and disable dvr. ret=%d", ret);
2015-03-11 05:34:58 +00:00
// unpublish, ignore ret.
hds->on_unpublish();
// ignore.
ret = ERROR_SUCCESS;
}
2015-03-12 03:15:15 +00:00
#endif
2014-04-16 01:28:02 +00:00
2014-03-18 03:32:58 +00:00
// copy to all consumer
if (!drop_for_reduce) {
for (int i = 0; i < (int)consumers.size(); i++) {
SrsConsumer* consumer = consumers.at(i);
if ((ret = consumer->enqueue(msg, atc, jitter_algorithm)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("dispatch the audio failed. ret=%d", ret);
return ret;
}
}
srs_info("dispatch audio success.");
}
2014-03-18 03:32:58 +00:00
// copy to all forwarders.
if (true) {
std::vector<SrsForwarder*>::iterator it;
for (it = forwarders.begin(); it != forwarders.end(); ++it) {
SrsForwarder* forwarder = *it;
if ((ret = forwarder->on_audio(msg)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("forwarder process audio message failed. ret=%d", ret);
return ret;
}
}
}
// cache the sequence header of aac, or first packet of mp3.
// for example, the mp3 is used for hls to write the "right" audio codec.
2015-03-08 07:33:08 +00:00
// TODO: FIXME: to refine the stream info system.
if (is_aac_sequence_header || !cache_sh_audio) {
2014-03-18 03:32:58 +00:00
srs_freep(cache_sh_audio);
cache_sh_audio = msg->copy();
}
2014-03-18 03:32:58 +00:00
// when sequence header, donot push to gop cache and adjust the timestamp.
if (is_sequence_header) {
return ret;
}
2014-03-18 03:32:58 +00:00
// cache the last gop packets
if ((ret = gop_cache->cache(msg)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("shrink gop cache failed. ret=%d", ret);
return ret;
}
srs_verbose("cache gop success.");
// if atc, update the sequence header to abs time.
if (atc) {
if (cache_sh_audio) {
cache_sh_audio->timestamp = msg->timestamp;
}
if (cache_metadata) {
cache_metadata->timestamp = msg->timestamp;
}
}
2014-03-18 03:32:58 +00:00
return ret;
}
int SrsSource::on_video(SrsCommonMessage* shared_video)
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
// monotically increase detect.
if (!mix_correct && is_monotonically_increase) {
if (last_packet_time > 0 && shared_video->header.timestamp < last_packet_time) {
is_monotonically_increase = false;
srs_warn("VIDEO: stream not monotonically increase, please open mix_correct.");
}
}
last_packet_time = shared_video->header.timestamp;
// drop any unknown header video.
2015-11-11 02:37:50 +00:00
// @see https://github.com/ossrs/srs/issues/421
if (!SrsFlvCodec::video_is_acceptable(shared_video->payload, shared_video->size)) {
char b0 = 0x00;
if (shared_video->size > 0) {
b0 = shared_video->payload[0];
}
srs_warn("drop unknown header video, size=%d, bytes[0]=%#x", shared_video->size, b0);
return ret;
}
// convert shared_video to msg, user should not use shared_video again.
// the payload is transfer to msg, and set to NULL in shared_video.
SrsSharedPtrMessage msg;
if ((ret = msg.create(shared_video)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("initialize the video failed. ret=%d", ret);
return ret;
}
srs_info("Video dts=%"PRId64", size=%d", msg.timestamp, msg.size);
2014-03-18 03:32:58 +00:00
// directly process the audio message.
if (!mix_correct) {
return on_video_imp(&msg);
}
// insert msg to the queue.
mix_queue->push(msg.copy());
// fetch someone from mix queue.
SrsSharedPtrMessage* m = mix_queue->pop();
if (!m) {
return ret;
}
SrsAutoFree(SrsSharedPtrMessage, m);
// consume the monotonically increase message.
if (m->is_audio()) {
ret = on_audio_imp(m);
} else {
ret = on_video_imp(m);
}
srs_freep(m);
return ret;
}
int SrsSource::on_video_imp(SrsSharedPtrMessage* msg)
{
int ret = ERROR_SUCCESS;
srs_info("Video dts=%"PRId64", size=%d", msg->timestamp, msg->size);
bool is_sequence_header = SrsFlvCodec::video_is_sequence_header(msg->payload, msg->size);
// whether consumer should drop for the duplicated sequence header.
bool drop_for_reduce = false;
if (is_sequence_header && cache_sh_video && _srs_config->get_reduce_sequence_header(_req->vhost)) {
if (cache_sh_video->size == msg->size) {
drop_for_reduce = srs_bytes_equals(cache_sh_video->payload, msg->payload, msg->size);
srs_warn("drop for reduce sh video, size=%d", msg->size);
}
}
// cache the sequence header if h264
// donot cache the sequence header to gop_cache, return here.
if (is_sequence_header) {
srs_freep(cache_sh_video);
cache_sh_video = msg->copy();
// parse detail audio codec
SrsAvcAacCodec codec;
// user can disable the sps parse to workaround when parse sps failed.
2015-11-11 02:37:50 +00:00
// @see https://github.com/ossrs/srs/issues/474
codec.avc_parse_sps = _srs_config->get_parse_sps(_req->vhost);
SrsCodecSample sample;
if ((ret = codec.video_avc_demux(msg->payload, msg->size, &sample)) != ERROR_SUCCESS) {
srs_error("source codec demux video failed. ret=%d", ret);
return ret;
}
// when got video stream info.
SrsStatistic* stat = SrsStatistic::instance();
if ((ret = stat->on_video_info(_req, SrsCodecVideoAVC, codec.avc_profile, codec.avc_level)) != ERROR_SUCCESS) {
return ret;
}
srs_trace("%dB video sh, codec(%d, profile=%s, level=%s, %dx%d, %dkbps, %dfps, %ds)",
msg->size, codec.video_codec_id,
srs_codec_avc_profile2str(codec.avc_profile).c_str(),
srs_codec_avc_level2str(codec.avc_level).c_str(), codec.width, codec.height,
codec.video_data_rate / 1000, codec.frame_rate, codec.duration);
}
#ifdef SRS_AUTO_HLS
if ((ret = hls->on_video(msg, is_sequence_header)) != ERROR_SUCCESS) {
// apply the error strategy for hls.
2015-11-11 02:37:50 +00:00
// @see https://github.com/ossrs/srs/issues/264
std::string hls_error_strategy = _srs_config->get_hls_on_error(_req->vhost);
if (srs_config_hls_is_on_error_ignore(hls_error_strategy)) {
srs_warn("hls process video message failed, ignore and disable hls. ret=%d", ret);
// unpublish, ignore ret.
hls->on_unpublish();
// ignore.
ret = ERROR_SUCCESS;
} else if (srs_config_hls_is_on_error_continue(hls_error_strategy)) {
if (srs_hls_can_continue(ret, cache_sh_video, msg)) {
ret = ERROR_SUCCESS;
} else {
srs_warn("hls continue video failed. ret=%d", ret);
return ret;
}
} else {
srs_warn("hls disconnect publisher for video error. ret=%d", ret);
return ret;
}
2014-03-18 03:32:58 +00:00
}
#endif
2014-03-18 03:32:58 +00:00
2014-04-16 01:28:02 +00:00
#ifdef SRS_AUTO_DVR
if ((ret = dvr->on_video(msg)) != ERROR_SUCCESS) {
2014-04-16 01:28:02 +00:00
srs_warn("dvr process video message failed, ignore and disable dvr. ret=%d", ret);
// unpublish, ignore ret.
dvr->on_unpublish();
// ignore.
ret = ERROR_SUCCESS;
}
#endif
2015-03-11 05:34:58 +00:00
2015-03-12 03:15:15 +00:00
#ifdef SRS_AUTO_HDS
if ((ret = hds->on_video(msg)) != ERROR_SUCCESS) {
2015-03-12 03:15:15 +00:00
srs_warn("hds process video message failed, ignore and disable dvr. ret=%d", ret);
2015-03-11 05:34:58 +00:00
// unpublish, ignore ret.
hds->on_unpublish();
// ignore.
ret = ERROR_SUCCESS;
}
2015-03-12 03:15:15 +00:00
#endif
2014-04-16 01:28:02 +00:00
2014-03-18 03:32:58 +00:00
// copy to all consumer
if (!drop_for_reduce) {
for (int i = 0; i < (int)consumers.size(); i++) {
SrsConsumer* consumer = consumers.at(i);
if ((ret = consumer->enqueue(msg, atc, jitter_algorithm)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("dispatch the video failed. ret=%d", ret);
return ret;
}
}
srs_info("dispatch video success.");
}
// copy to all forwarders.
if (!forwarders.empty()) {
2014-03-18 03:32:58 +00:00
std::vector<SrsForwarder*>::iterator it;
for (it = forwarders.begin(); it != forwarders.end(); ++it) {
SrsForwarder* forwarder = *it;
if ((ret = forwarder->on_video(msg)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("forwarder process video message failed. ret=%d", ret);
return ret;
}
}
}
// when sequence header, donot push to gop cache and adjust the timestamp.
if (is_sequence_header) {
return ret;
}
2014-03-18 03:32:58 +00:00
// cache the last gop packets
if ((ret = gop_cache->cache(msg)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("gop cache msg failed. ret=%d", ret);
return ret;
}
srs_verbose("cache gop success.");
// if atc, update the sequence header to abs time.
if (atc) {
if (cache_sh_video) {
cache_sh_video->timestamp = msg->timestamp;
}
if (cache_metadata) {
cache_metadata->timestamp = msg->timestamp;
}
}
2014-03-18 03:32:58 +00:00
return ret;
}
int SrsSource::on_aggregate(SrsCommonMessage* msg)
{
int ret = ERROR_SUCCESS;
SrsStream* stream = aggregate_stream;
if ((ret = stream->initialize(msg->payload, msg->size)) != ERROR_SUCCESS) {
return ret;
}
// the aggregate message always use abs time.
int delta = -1;
while (!stream->empty()) {
if (!stream->require(1)) {
ret = ERROR_RTMP_AGGREGATE;
srs_error("invalid aggregate message type. ret=%d", ret);
return ret;
}
int8_t type = stream->read_1bytes();
if (!stream->require(3)) {
ret = ERROR_RTMP_AGGREGATE;
srs_error("invalid aggregate message size. ret=%d", ret);
return ret;
}
int32_t data_size = stream->read_3bytes();
if (data_size < 0) {
ret = ERROR_RTMP_AGGREGATE;
srs_error("invalid aggregate message size(negative). ret=%d", ret);
return ret;
}
if (!stream->require(3)) {
ret = ERROR_RTMP_AGGREGATE;
srs_error("invalid aggregate message time. ret=%d", ret);
return ret;
}
int32_t timestamp = stream->read_3bytes();
if (!stream->require(1)) {
ret = ERROR_RTMP_AGGREGATE;
srs_error("invalid aggregate message time(high). ret=%d", ret);
return ret;
}
int32_t time_h = stream->read_1bytes();
timestamp |= time_h<<24;
timestamp &= 0x7FFFFFFF;
// adjust abs timestamp in aggregate msg.
// only -1 means uninitialized delta.
if (delta == -1) {
delta = (int)msg->header.timestamp - (int)timestamp;
}
timestamp += delta;
if (!stream->require(3)) {
ret = ERROR_RTMP_AGGREGATE;
srs_error("invalid aggregate message stream_id. ret=%d", ret);
return ret;
}
int32_t stream_id = stream->read_3bytes();
if (data_size > 0 && !stream->require(data_size)) {
ret = ERROR_RTMP_AGGREGATE;
srs_error("invalid aggregate message data. ret=%d", ret);
return ret;
}
// to common message.
SrsCommonMessage o;
o.header.message_type = type;
o.header.payload_length = data_size;
o.header.timestamp_delta = timestamp;
o.header.timestamp = timestamp;
o.header.stream_id = stream_id;
o.header.perfer_cid = msg->header.perfer_cid;
if (data_size > 0) {
o.size = data_size;
o.payload = new char[o.size];
stream->read_bytes(o.payload, o.size);
}
if (!stream->require(4)) {
ret = ERROR_RTMP_AGGREGATE;
srs_error("invalid aggregate message previous tag size. ret=%d", ret);
return ret;
}
stream->read_4bytes();
// process parsed message
if (o.header.is_audio()) {
if ((ret = on_audio(&o)) != ERROR_SUCCESS) {
return ret;
}
} else if (o.header.is_video()) {
if ((ret = on_video(&o)) != ERROR_SUCCESS) {
return ret;
}
}
}
return ret;
}
2014-04-26 09:16:18 +00:00
int SrsSource::on_publish()
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
// update the request object.
2014-04-26 09:16:18 +00:00
srs_assert(_req);
2014-03-18 03:32:58 +00:00
_can_publish = false;
// whatever, the publish thread is the source or edge source,
// save its id to srouce id.
on_source_id_changed(_srs_context->get_id());
// reset the mix queue.
mix_queue->clear();
// detect the monotonically again.
is_monotonically_increase = true;
last_packet_time = 0;
2014-03-18 03:32:58 +00:00
// create forwarders
if ((ret = create_forwarders()) != ERROR_SUCCESS) {
srs_error("create forwarders failed. ret=%d", ret);
return ret;
}
// TODO: FIXME: use initialize to set req.
#ifdef SRS_AUTO_TRANSCODE
2014-04-26 09:16:18 +00:00
if ((ret = encoder->on_publish(_req)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("start encoder failed. ret=%d", ret);
return ret;
}
#endif
2014-03-18 03:32:58 +00:00
// TODO: FIXME: use initialize to set req.
#ifdef SRS_AUTO_HLS
if ((ret = hls->on_publish(_req, false)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("start hls failed. ret=%d", ret);
return ret;
}
#endif
2014-04-16 01:28:02 +00:00
// TODO: FIXME: use initialize to set req.
2014-04-16 01:28:02 +00:00
#ifdef SRS_AUTO_DVR
2014-04-26 09:16:18 +00:00
if ((ret = dvr->on_publish(_req)) != ERROR_SUCCESS) {
2014-04-16 01:28:02 +00:00
srs_error("start dvr failed. ret=%d", ret);
return ret;
}
#endif
2015-03-12 03:15:15 +00:00
#ifdef SRS_AUTO_HDS
2015-03-11 05:34:58 +00:00
if ((ret = hds->on_publish(_req)) != ERROR_SUCCESS) {
srs_error("start hds failed. ret=%d", ret);
return ret;
}
2015-03-12 03:15:15 +00:00
#endif
2015-03-11 05:34:58 +00:00
// notify the handler.
srs_assert(handler);
if ((ret = handler->on_publish(this, _req)) != ERROR_SUCCESS) {
srs_error("handle on publish failed. ret=%d", ret);
return ret;
}
2015-05-08 08:45:25 +00:00
SrsStatistic* stat = SrsStatistic::instance();
stat->on_stream_publish(_req, _source_id);
2014-03-18 03:32:58 +00:00
return ret;
}
void SrsSource::on_unpublish()
{
2016-09-05 06:13:37 +00:00
// ignore when already unpublished.
if (_can_publish) {
return;
}
2014-03-18 03:32:58 +00:00
// destroy all forwarders
destroy_forwarders();
#ifdef SRS_AUTO_TRANSCODE
2014-03-18 03:32:58 +00:00
encoder->on_unpublish();
#endif
#ifdef SRS_AUTO_HLS
2014-03-18 03:32:58 +00:00
hls->on_unpublish();
#endif
2014-04-16 01:28:02 +00:00
#ifdef SRS_AUTO_DVR
dvr->on_unpublish();
#endif
2015-03-12 03:15:15 +00:00
#ifdef SRS_AUTO_HDS
2015-03-11 05:34:58 +00:00
hds->on_unpublish();
2015-03-12 03:15:15 +00:00
#endif
2015-03-11 05:34:58 +00:00
// only clear the gop cache,
// donot clear the sequence header, for it maybe not changed,
// when drop dup sequence header, drop the metadata also.
2014-03-18 03:32:58 +00:00
gop_cache->clear();
srs_info("clear cache/metadata when unpublish.");
2014-05-29 06:16:34 +00:00
srs_trace("cleanup when unpublish");
2014-03-18 03:32:58 +00:00
_can_publish = true;
_source_id = -1;
// notify the handler.
srs_assert(handler);
2015-05-08 08:45:25 +00:00
SrsStatistic* stat = SrsStatistic::instance();
stat->on_stream_close(_req);
handler->on_unpublish(this, _req);
2016-09-05 06:13:37 +00:00
// no consumer, stream is die.
if (consumers.empty()) {
die_at = srs_get_system_time_ms();
}
}
int SrsSource::create_consumer(SrsConnection* conn, SrsConsumer*& consumer, bool ds, bool dm, bool dg)
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
consumer = new SrsConsumer(this, conn);
2014-03-18 03:32:58 +00:00
consumers.push_back(consumer);
2014-04-26 09:16:18 +00:00
double queue_size = _srs_config->get_queue_length(_req->vhost);
2014-03-18 03:32:58 +00:00
consumer->set_queue_size(queue_size);
// if atc, update the sequence header to gop cache time.
if (atc && !gop_cache->empty()) {
if (cache_metadata) {
cache_metadata->timestamp = gop_cache->start_time();
}
if (cache_sh_video) {
cache_sh_video->timestamp = gop_cache->start_time();
}
if (cache_sh_audio) {
cache_sh_audio->timestamp = gop_cache->start_time();
}
}
// copy metadata.
if (dm && cache_metadata && (ret = consumer->enqueue(cache_metadata, atc, jitter_algorithm)) != ERROR_SUCCESS) {
srs_error("dispatch metadata failed. ret=%d", ret);
return ret;
}
srs_info("dispatch metadata success");
// copy sequence header
// copy audio sequence first, for hls to fast parse the "right" audio codec.
2015-11-11 02:37:50 +00:00
// @see https://github.com/ossrs/srs/issues/301
if (ds && cache_sh_audio && (ret = consumer->enqueue(cache_sh_audio, atc, jitter_algorithm)) != ERROR_SUCCESS) {
srs_error("dispatch audio sequence header failed. ret=%d", ret);
return ret;
}
srs_info("dispatch audio sequence header success");
if (ds && cache_sh_video && (ret = consumer->enqueue(cache_sh_video, atc, jitter_algorithm)) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("dispatch video sequence header failed. ret=%d", ret);
return ret;
}
srs_info("dispatch video sequence header success");
// copy gop cache to client.
if (dg && (ret = gop_cache->dump(consumer, atc, jitter_algorithm)) != ERROR_SUCCESS) {
return ret;
}
// print status.
if (dg) {
srs_trace("create consumer, queue_size=%.2f, jitter=%d", queue_size, jitter_algorithm);
} else {
srs_trace("create consumer, ignore gop cache, jitter=%d", jitter_algorithm);
2014-03-18 03:32:58 +00:00
}
// for edge, when play edge stream, check the state
if (_srs_config->get_vhost_is_edge(_req->vhost)) {
// notice edge to start for the first client.
if ((ret = play_edge->on_client_play()) != ERROR_SUCCESS) {
srs_error("notice edge start play stream failed. ret=%d", ret);
return ret;
}
}
2014-03-18 03:32:58 +00:00
return ret;
}
void SrsSource::on_consumer_destroy(SrsConsumer* consumer)
{
2014-03-18 03:32:58 +00:00
std::vector<SrsConsumer*>::iterator it;
it = std::find(consumers.begin(), consumers.end(), consumer);
if (it != consumers.end()) {
consumers.erase(it);
}
srs_info("handle consumer destroy success.");
if (consumers.empty()) {
2014-04-27 01:29:37 +00:00
play_edge->on_all_client_stop();
2016-09-05 06:13:37 +00:00
die_at = srs_get_system_time_ms();
}
}
void SrsSource::set_cache(bool enabled)
{
2014-03-18 03:32:58 +00:00
gop_cache->set(enabled);
}
SrsRtmpJitterAlgorithm SrsSource::jitter()
{
return jitter_algorithm;
}
2014-04-27 01:29:37 +00:00
int SrsSource::on_edge_start_publish()
{
return publish_edge->on_client_publish();
2014-04-25 08:35:03 +00:00
}
int SrsSource::on_edge_proxy_publish(SrsCommonMessage* msg)
{
2014-04-27 06:57:28 +00:00
return publish_edge->on_proxy_publish(msg);
}
void SrsSource::on_edge_proxy_unpublish()
{
publish_edge->on_proxy_unpublish();
}
2013-12-15 04:34:22 +00:00
int SrsSource::create_forwarders()
{
2014-03-18 03:32:58 +00:00
int ret = ERROR_SUCCESS;
2014-04-26 09:16:18 +00:00
SrsConfDirective* conf = _srs_config->get_forward(_req->vhost);
2014-03-18 03:32:58 +00:00
for (int i = 0; conf && i < (int)conf->args.size(); i++) {
std::string forward_server = conf->args.at(i);
SrsForwarder* forwarder = new SrsForwarder(this);
forwarders.push_back(forwarder);
// initialize the forwarder with request.
if ((ret = forwarder->initialize(_req, forward_server)) != ERROR_SUCCESS) {
return ret;
}
2014-03-18 03:32:58 +00:00
2014-04-26 09:16:18 +00:00
double queue_size = _srs_config->get_queue_length(_req->vhost);
2014-03-18 03:32:58 +00:00
forwarder->set_queue_size(queue_size);
if ((ret = forwarder->on_publish()) != ERROR_SUCCESS) {
2014-03-18 03:32:58 +00:00
srs_error("start forwarder failed. "
"vhost=%s, app=%s, stream=%s, forward-to=%s",
2014-04-26 09:16:18 +00:00
_req->vhost.c_str(), _req->app.c_str(), _req->stream.c_str(),
2014-03-18 03:32:58 +00:00
forward_server.c_str());
return ret;
}
}
return ret;
2013-12-15 04:34:22 +00:00
}
void SrsSource::destroy_forwarders()
{
2014-03-18 03:32:58 +00:00
std::vector<SrsForwarder*>::iterator it;
for (it = forwarders.begin(); it != forwarders.end(); ++it) {
SrsForwarder* forwarder = *it;
forwarder->on_unpublish();
srs_freep(forwarder);
}
forwarders.clear();
2013-12-15 04:34:22 +00:00
}