mirror of
https://github.com/ossrs/srs.git
synced 2025-03-09 15:49:59 +00:00
for #738, refine code for DVR mp4.
This commit is contained in:
parent
0a054cd6bd
commit
3209ad29e0
6 changed files with 83 additions and 226 deletions
|
@ -44,9 +44,6 @@ using namespace std;
|
||||||
#include <srs_app_utility.hpp>
|
#include <srs_app_utility.hpp>
|
||||||
#include <srs_kernel_mp4.hpp>
|
#include <srs_kernel_mp4.hpp>
|
||||||
|
|
||||||
// update the flv duration and filesize every this interval in ms.
|
|
||||||
#define SRS_DVR_UPDATE_DURATION_INTERVAL 60000
|
|
||||||
|
|
||||||
SrsDvrSegmenter::SrsDvrSegmenter()
|
SrsDvrSegmenter::SrsDvrSegmenter()
|
||||||
{
|
{
|
||||||
req = NULL;
|
req = NULL;
|
||||||
|
@ -87,9 +84,9 @@ string SrsDvrSegmenter::get_path()
|
||||||
return path;
|
return path;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool SrsDvrSegmenter::is_overflow(int64_t max_duration)
|
int64_t SrsDvrSegmenter::get_duration()
|
||||||
{
|
{
|
||||||
return duration >= max_duration;
|
return duration;
|
||||||
}
|
}
|
||||||
|
|
||||||
int SrsDvrSegmenter::open()
|
int SrsDvrSegmenter::open()
|
||||||
|
@ -108,6 +105,8 @@ int SrsDvrSegmenter::open()
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
tmp_dvr_file = path + ".tmp";
|
||||||
|
|
||||||
// create dir first.
|
// create dir first.
|
||||||
std::string dir = srs_path_dirname(path);
|
std::string dir = srs_path_dirname(path);
|
||||||
if ((ret = srs_create_dir_recursively(dir)) != ERROR_SUCCESS) {
|
if ((ret = srs_create_dir_recursively(dir)) != ERROR_SUCCESS) {
|
||||||
|
@ -117,13 +116,9 @@ int SrsDvrSegmenter::open()
|
||||||
srs_info("create dir=%s ok", dir.c_str());
|
srs_info("create dir=%s ok", dir.c_str());
|
||||||
|
|
||||||
// create jitter.
|
// create jitter.
|
||||||
if ((ret = create_jitter()) != ERROR_SUCCESS) {
|
srs_freep(jitter);
|
||||||
srs_error("create jitter failed, path=%s. ret=%d", path.c_str(), ret);
|
jitter = new SrsRtmpJitter();
|
||||||
return ret;
|
duration = 0;
|
||||||
}
|
|
||||||
|
|
||||||
// generate the tmp flv path.
|
|
||||||
tmp_dvr_file = path + ".tmp";
|
|
||||||
|
|
||||||
// open file writer, in append or create mode.
|
// open file writer, in append or create mode.
|
||||||
if ((ret = fs->open(tmp_dvr_file)) != ERROR_SUCCESS) {
|
if ((ret = fs->open(tmp_dvr_file)) != ERROR_SUCCESS) {
|
||||||
|
@ -158,8 +153,7 @@ int SrsDvrSegmenter::write_audio(SrsSharedPtrMessage* shared_audio)
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
int64_t timestamp = plan->filter_timestamp(audio->timestamp);
|
if ((ret = encode_audio(audio)) != ERROR_SUCCESS) {
|
||||||
if ((ret = encode_audio(audio, timestamp)) != ERROR_SUCCESS) {
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -183,15 +177,7 @@ int SrsDvrSegmenter::write_video(SrsSharedPtrMessage* shared_video)
|
||||||
bool is_sequence_header = SrsFlvCodec::video_is_sequence_header(payload, size);
|
bool is_sequence_header = SrsFlvCodec::video_is_sequence_header(payload, size);
|
||||||
bool is_key_frame = SrsFlvCodec::video_is_h264(payload, size)
|
bool is_key_frame = SrsFlvCodec::video_is_h264(payload, size)
|
||||||
&& SrsFlvCodec::video_is_keyframe(payload, size) && !is_sequence_header;
|
&& SrsFlvCodec::video_is_keyframe(payload, size) && !is_sequence_header;
|
||||||
if (is_key_frame) {
|
if ((ret = encode_video(video, is_sequence_header, is_key_frame)) != ERROR_SUCCESS) {
|
||||||
if ((ret = plan->on_video_keyframe()) != ERROR_SUCCESS) {
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
srs_verbose("dvr video is key: %d", is_key_frame);
|
|
||||||
|
|
||||||
int64_t timestamp = plan->filter_timestamp(video->timestamp);
|
|
||||||
if ((ret = encode_video(video, timestamp, is_sequence_header, is_key_frame)) != ERROR_SUCCESS) {
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -253,18 +239,6 @@ string SrsDvrSegmenter::generate_path()
|
||||||
return flv_path;
|
return flv_path;
|
||||||
}
|
}
|
||||||
|
|
||||||
int SrsDvrSegmenter::create_jitter()
|
|
||||||
{
|
|
||||||
int ret = ERROR_SUCCESS;
|
|
||||||
|
|
||||||
srs_freep(jitter);
|
|
||||||
jitter = new SrsRtmpJitter();
|
|
||||||
|
|
||||||
duration = 0;
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
int SrsDvrSegmenter::on_reload_vhost_dvr(std::string vhost)
|
int SrsDvrSegmenter::on_reload_vhost_dvr(std::string vhost)
|
||||||
{
|
{
|
||||||
int ret = ERROR_SUCCESS;
|
int ret = ERROR_SUCCESS;
|
||||||
|
@ -447,13 +421,13 @@ int SrsDvrFlvSegmenter::encode_metadata(SrsSharedPtrMessage* metadata)
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
int SrsDvrFlvSegmenter::encode_audio(SrsSharedPtrMessage* audio, int64_t dts)
|
int SrsDvrFlvSegmenter::encode_audio(SrsSharedPtrMessage* audio)
|
||||||
{
|
{
|
||||||
int ret = ERROR_SUCCESS;
|
int ret = ERROR_SUCCESS;
|
||||||
|
|
||||||
char* payload = audio->payload;
|
char* payload = audio->payload;
|
||||||
int size = audio->size;
|
int size = audio->size;
|
||||||
if ((ret = enc->write_audio(dts, payload, size)) != ERROR_SUCCESS) {
|
if ((ret = enc->write_audio(audio->timestamp, payload, size)) != ERROR_SUCCESS) {
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -464,7 +438,7 @@ int SrsDvrFlvSegmenter::encode_audio(SrsSharedPtrMessage* audio, int64_t dts)
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
int SrsDvrFlvSegmenter::encode_video(SrsSharedPtrMessage* video, int64_t dts, bool sh, bool keyframe)
|
int SrsDvrFlvSegmenter::encode_video(SrsSharedPtrMessage* video, bool sh, bool keyframe)
|
||||||
{
|
{
|
||||||
int ret = ERROR_SUCCESS;
|
int ret = ERROR_SUCCESS;
|
||||||
|
|
||||||
|
@ -489,7 +463,7 @@ int SrsDvrFlvSegmenter::encode_video(SrsSharedPtrMessage* video, int64_t dts, bo
|
||||||
|
|
||||||
char* payload = video->payload;
|
char* payload = video->payload;
|
||||||
int size = video->size;
|
int size = video->size;
|
||||||
if ((ret = enc->write_video(dts, payload, size)) != ERROR_SUCCESS) {
|
if ((ret = enc->write_video(video->timestamp, payload, size)) != ERROR_SUCCESS) {
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -557,13 +531,13 @@ int SrsDvrMp4Segmenter::encode_metadata(SrsSharedPtrMessage* metadata)
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
int SrsDvrMp4Segmenter::encode_audio(SrsSharedPtrMessage* audio, int64_t dts)
|
int SrsDvrMp4Segmenter::encode_audio(SrsSharedPtrMessage* audio)
|
||||||
{
|
{
|
||||||
int ret = ERROR_SUCCESS;
|
int ret = ERROR_SUCCESS;
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
int SrsDvrMp4Segmenter::encode_video(SrsSharedPtrMessage* video, int64_t dts, bool sh, bool keyframe)
|
int SrsDvrMp4Segmenter::encode_video(SrsSharedPtrMessage* video, bool sh, bool keyframe)
|
||||||
{
|
{
|
||||||
int ret = ERROR_SUCCESS;
|
int ret = ERROR_SUCCESS;
|
||||||
return ret;
|
return ret;
|
||||||
|
@ -644,10 +618,11 @@ SrsDvrPlan::~SrsDvrPlan()
|
||||||
srs_freep(async);
|
srs_freep(async);
|
||||||
}
|
}
|
||||||
|
|
||||||
int SrsDvrPlan::initialize(SrsDvrSegmenter* s, SrsRequest* r)
|
int SrsDvrPlan::initialize(SrsOriginHub* h, SrsDvrSegmenter* s, SrsRequest* r)
|
||||||
{
|
{
|
||||||
int ret = ERROR_SUCCESS;
|
int ret = ERROR_SUCCESS;
|
||||||
|
|
||||||
|
hub = h;
|
||||||
req = r;
|
req = r;
|
||||||
segment = s;
|
segment = s;
|
||||||
|
|
||||||
|
@ -662,16 +637,6 @@ int SrsDvrPlan::initialize(SrsDvrSegmenter* s, SrsRequest* r)
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
int SrsDvrPlan::on_video_keyframe()
|
|
||||||
{
|
|
||||||
return ERROR_SUCCESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
int64_t SrsDvrPlan::filter_timestamp(int64_t timestamp)
|
|
||||||
{
|
|
||||||
return timestamp;
|
|
||||||
}
|
|
||||||
|
|
||||||
int SrsDvrPlan::on_meta_data(SrsSharedPtrMessage* shared_metadata)
|
int SrsDvrPlan::on_meta_data(SrsSharedPtrMessage* shared_metadata)
|
||||||
{
|
{
|
||||||
int ret = ERROR_SUCCESS;
|
int ret = ERROR_SUCCESS;
|
||||||
|
@ -795,28 +760,27 @@ void SrsDvrSessionPlan::on_unpublish()
|
||||||
|
|
||||||
SrsDvrSegmentPlan::SrsDvrSegmentPlan()
|
SrsDvrSegmentPlan::SrsDvrSegmentPlan()
|
||||||
{
|
{
|
||||||
segment_duration = -1;
|
cduration = -1;
|
||||||
metadata = sh_video = sh_audio = NULL;
|
wait_keyframe = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
SrsDvrSegmentPlan::~SrsDvrSegmentPlan()
|
SrsDvrSegmentPlan::~SrsDvrSegmentPlan()
|
||||||
{
|
{
|
||||||
srs_freep(sh_video);
|
|
||||||
srs_freep(sh_audio);
|
|
||||||
srs_freep(metadata);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int SrsDvrSegmentPlan::initialize(SrsDvrSegmenter* s, SrsRequest* req)
|
int SrsDvrSegmentPlan::initialize(SrsOriginHub* h, SrsDvrSegmenter* s, SrsRequest* r)
|
||||||
{
|
{
|
||||||
int ret = ERROR_SUCCESS;
|
int ret = ERROR_SUCCESS;
|
||||||
|
|
||||||
if ((ret = SrsDvrPlan::initialize(s, req)) != ERROR_SUCCESS) {
|
if ((ret = SrsDvrPlan::initialize(h, s, r)) != ERROR_SUCCESS) {
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
segment_duration = _srs_config->get_dvr_duration(req->vhost);
|
wait_keyframe = _srs_config->get_dvr_wait_keyframe(req->vhost);
|
||||||
|
|
||||||
|
cduration = _srs_config->get_dvr_duration(req->vhost);
|
||||||
// to ms
|
// to ms
|
||||||
segment_duration *= 1000;
|
cduration *= 1000;
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -851,28 +815,9 @@ void SrsDvrSegmentPlan::on_unpublish()
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
int SrsDvrSegmentPlan::on_meta_data(SrsSharedPtrMessage* shared_metadata)
|
|
||||||
{
|
|
||||||
int ret = ERROR_SUCCESS;
|
|
||||||
|
|
||||||
srs_freep(metadata);
|
|
||||||
metadata = shared_metadata->copy();
|
|
||||||
|
|
||||||
if ((ret = SrsDvrPlan::on_meta_data(shared_metadata)) != ERROR_SUCCESS) {
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
int SrsDvrSegmentPlan::on_audio(SrsSharedPtrMessage* shared_audio)
|
int SrsDvrSegmentPlan::on_audio(SrsSharedPtrMessage* shared_audio)
|
||||||
{
|
{
|
||||||
int ret = ERROR_SUCCESS;
|
int ret = ERROR_SUCCESS;
|
||||||
|
|
||||||
if (SrsFlvCodec::audio_is_sequence_header(shared_audio->payload, shared_audio->size)) {
|
|
||||||
srs_freep(sh_audio);
|
|
||||||
sh_audio = shared_audio->copy();
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((ret = update_duration(shared_audio)) != ERROR_SUCCESS) {
|
if ((ret = update_duration(shared_audio)) != ERROR_SUCCESS) {
|
||||||
return ret;
|
return ret;
|
||||||
|
@ -889,11 +834,6 @@ int SrsDvrSegmentPlan::on_video(SrsSharedPtrMessage* shared_video)
|
||||||
{
|
{
|
||||||
int ret = ERROR_SUCCESS;
|
int ret = ERROR_SUCCESS;
|
||||||
|
|
||||||
if (SrsFlvCodec::video_is_sequence_header(shared_video->payload, shared_video->size)) {
|
|
||||||
srs_freep(sh_video);
|
|
||||||
sh_video = shared_video->copy();
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((ret = update_duration(shared_video)) != ERROR_SUCCESS) {
|
if ((ret = update_duration(shared_video)) != ERROR_SUCCESS) {
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -912,13 +852,13 @@ int SrsDvrSegmentPlan::update_duration(SrsSharedPtrMessage* msg)
|
||||||
srs_assert(segment);
|
srs_assert(segment);
|
||||||
|
|
||||||
// ignore if duration ok.
|
// ignore if duration ok.
|
||||||
if (segment_duration <= 0 || !segment->is_overflow(segment_duration)) {
|
if (cduration <= 0 || segment->get_duration() < cduration) {
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
// when wait keyframe, ignore if no frame arrived.
|
// when wait keyframe, ignore if no frame arrived.
|
||||||
// @see https://github.com/ossrs/srs/issues/177
|
// @see https://github.com/ossrs/srs/issues/177
|
||||||
if (_srs_config->get_dvr_wait_keyframe(req->vhost)) {
|
if (wait_keyframe) {
|
||||||
if (!msg->is_video()) {
|
if (!msg->is_video()) {
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -944,16 +884,27 @@ int SrsDvrSegmentPlan::update_duration(SrsSharedPtrMessage* msg)
|
||||||
}
|
}
|
||||||
|
|
||||||
// update sequence header
|
// update sequence header
|
||||||
if (metadata && (ret = SrsDvrPlan::on_meta_data(metadata)) != ERROR_SUCCESS) {
|
if ((ret = hub->on_dvr_request_sh()) != ERROR_SUCCESS) {
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
if (sh_video && (ret = SrsDvrPlan::on_video(sh_video)) != ERROR_SUCCESS) {
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
if (sh_audio && (ret = SrsDvrPlan::on_audio(sh_audio)) != ERROR_SUCCESS) {
|
|
||||||
|
int SrsDvrSegmentPlan::on_reload_vhost_dvr(string vhost)
|
||||||
|
{
|
||||||
|
int ret = ERROR_SUCCESS;
|
||||||
|
|
||||||
|
if (req->vhost != vhost) {
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
wait_keyframe = _srs_config->get_dvr_wait_keyframe(req->vhost);
|
||||||
|
|
||||||
|
cduration = _srs_config->get_dvr_duration(req->vhost);
|
||||||
|
// to ms
|
||||||
|
cduration *= 1000;
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -997,7 +948,7 @@ int SrsDvr::initialize(SrsOriginHub* h, SrsRequest* r)
|
||||||
segmenter = new SrsDvrFlvSegmenter();
|
segmenter = new SrsDvrFlvSegmenter();
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((ret = plan->initialize(segmenter, r)) != ERROR_SUCCESS) {
|
if ((ret = plan->initialize(hub, segmenter, r)) != ERROR_SUCCESS) {
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1029,8 +980,7 @@ void SrsDvr::on_unpublish()
|
||||||
plan->on_unpublish();
|
plan->on_unpublish();
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: FIXME: source should use shared message instead.
|
int SrsDvr::on_meta_data(SrsSharedPtrMessage* metadata)
|
||||||
int SrsDvr::on_meta_data(SrsOnMetaDataPacket* m)
|
|
||||||
{
|
{
|
||||||
int ret = ERROR_SUCCESS;
|
int ret = ERROR_SUCCESS;
|
||||||
|
|
||||||
|
@ -1039,18 +989,7 @@ int SrsDvr::on_meta_data(SrsOnMetaDataPacket* m)
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
int size = 0;
|
if ((ret = plan->on_meta_data(metadata)) != ERROR_SUCCESS) {
|
||||||
char* payload = NULL;
|
|
||||||
if ((ret = m->encode(size, payload)) != ERROR_SUCCESS) {
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
SrsSharedPtrMessage metadata;
|
|
||||||
if ((ret = metadata.create(NULL, payload, size)) != ERROR_SUCCESS) {
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((ret = plan->on_meta_data(&metadata)) != ERROR_SUCCESS) {
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -39,7 +39,6 @@ class SrsOriginHub;
|
||||||
class SrsRequest;
|
class SrsRequest;
|
||||||
class SrsBuffer;
|
class SrsBuffer;
|
||||||
class SrsRtmpJitter;
|
class SrsRtmpJitter;
|
||||||
class SrsOnMetaDataPacket;
|
|
||||||
class SrsSharedPtrMessage;
|
class SrsSharedPtrMessage;
|
||||||
class SrsFileWriter;
|
class SrsFileWriter;
|
||||||
class SrsFlvEncoder;
|
class SrsFlvEncoder;
|
||||||
|
@ -68,9 +67,11 @@ protected:
|
||||||
private:
|
private:
|
||||||
// The path of current segment flv file path.
|
// The path of current segment flv file path.
|
||||||
std::string path;
|
std::string path;
|
||||||
|
std::string tmp_dvr_file;
|
||||||
|
private:
|
||||||
SrsRequest* req;
|
SrsRequest* req;
|
||||||
SrsDvrPlan* plan;
|
SrsDvrPlan* plan;
|
||||||
std::string tmp_dvr_file;
|
private:
|
||||||
SrsRtmpJitter* jitter;
|
SrsRtmpJitter* jitter;
|
||||||
SrsRtmpJitterAlgorithm jitter_algorithm;
|
SrsRtmpJitterAlgorithm jitter_algorithm;
|
||||||
public:
|
public:
|
||||||
|
@ -81,8 +82,8 @@ public:
|
||||||
virtual int initialize(SrsDvrPlan* p, SrsRequest* r);
|
virtual int initialize(SrsDvrPlan* p, SrsRequest* r);
|
||||||
// Get the current dvr path.
|
// Get the current dvr path.
|
||||||
virtual std::string get_path();
|
virtual std::string get_path();
|
||||||
// Whether segment is overflow.
|
// Get the duration in ms of segment.
|
||||||
virtual bool is_overflow(int64_t max_duration);
|
virtual int64_t get_duration();
|
||||||
// Open new segment file.
|
// Open new segment file.
|
||||||
// @param use_tmp_file Whether use tmp file for DVR, and rename when close.
|
// @param use_tmp_file Whether use tmp file for DVR, and rename when close.
|
||||||
// @remark Ignore when file is already open.
|
// @remark Ignore when file is already open.
|
||||||
|
@ -105,19 +106,12 @@ public:
|
||||||
protected:
|
protected:
|
||||||
virtual int open_encoder() = 0;
|
virtual int open_encoder() = 0;
|
||||||
virtual int encode_metadata(SrsSharedPtrMessage* metadata) = 0;
|
virtual int encode_metadata(SrsSharedPtrMessage* metadata) = 0;
|
||||||
virtual int encode_audio(SrsSharedPtrMessage* audio, int64_t dts) = 0;
|
virtual int encode_audio(SrsSharedPtrMessage* audio) = 0;
|
||||||
virtual int encode_video(SrsSharedPtrMessage* video, int64_t dts, bool sh, bool keyframe) = 0;
|
virtual int encode_video(SrsSharedPtrMessage* video, bool sh, bool keyframe) = 0;
|
||||||
virtual int close_encoder() = 0;
|
virtual int close_encoder() = 0;
|
||||||
private:
|
private:
|
||||||
/**
|
// Generate the flv segment path.
|
||||||
* generate the flv segment path.
|
|
||||||
*/
|
|
||||||
virtual std::string generate_path();
|
virtual std::string generate_path();
|
||||||
/**
|
|
||||||
* create flv jitter. load jitter when flv exists.
|
|
||||||
* @param target_exists whether loads the jitter from exists flv file.
|
|
||||||
*/
|
|
||||||
virtual int create_jitter();
|
|
||||||
// interface ISrsReloadHandler
|
// interface ISrsReloadHandler
|
||||||
public:
|
public:
|
||||||
virtual int on_reload_vhost_dvr(std::string vhost);
|
virtual int on_reload_vhost_dvr(std::string vhost);
|
||||||
|
@ -161,8 +155,8 @@ public:
|
||||||
protected:
|
protected:
|
||||||
virtual int open_encoder();
|
virtual int open_encoder();
|
||||||
virtual int encode_metadata(SrsSharedPtrMessage* metadata);
|
virtual int encode_metadata(SrsSharedPtrMessage* metadata);
|
||||||
virtual int encode_audio(SrsSharedPtrMessage* audio, int64_t dts);
|
virtual int encode_audio(SrsSharedPtrMessage* audio);
|
||||||
virtual int encode_video(SrsSharedPtrMessage* video, int64_t dts, bool sh, bool keyframe);
|
virtual int encode_video(SrsSharedPtrMessage* video, bool sh, bool keyframe);
|
||||||
virtual int close_encoder();
|
virtual int close_encoder();
|
||||||
private:
|
private:
|
||||||
// When update the duration of segment by rtmp msg.
|
// When update the duration of segment by rtmp msg.
|
||||||
|
@ -185,8 +179,8 @@ public:
|
||||||
protected:
|
protected:
|
||||||
virtual int open_encoder();
|
virtual int open_encoder();
|
||||||
virtual int encode_metadata(SrsSharedPtrMessage* metadata);
|
virtual int encode_metadata(SrsSharedPtrMessage* metadata);
|
||||||
virtual int encode_audio(SrsSharedPtrMessage* audio, int64_t dts);
|
virtual int encode_audio(SrsSharedPtrMessage* audio);
|
||||||
virtual int encode_video(SrsSharedPtrMessage* video, int64_t dts, bool sh, bool keyframe);
|
virtual int encode_video(SrsSharedPtrMessage* video, bool sh, bool keyframe);
|
||||||
virtual int close_encoder();
|
virtual int close_encoder();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -208,17 +202,14 @@ public:
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* the plan for dvr.
|
* The DVR plan, when and how to reap segment.
|
||||||
* use to control the following dvr params:
|
|
||||||
* 1. filename: the filename for record file.
|
|
||||||
* 2. reap flv: when to reap the flv and start new piece.
|
|
||||||
*/
|
*/
|
||||||
// TODO: FIXME: the plan is too fat, refine me.
|
class SrsDvrPlan : public ISrsReloadHandler
|
||||||
class SrsDvrPlan
|
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
SrsRequest* req;
|
SrsRequest* req;
|
||||||
protected:
|
protected:
|
||||||
|
SrsOriginHub* hub;
|
||||||
SrsDvrSegmenter* segment;
|
SrsDvrSegmenter* segment;
|
||||||
SrsAsyncCallWorker* async;
|
SrsAsyncCallWorker* async;
|
||||||
bool dvr_enabled;
|
bool dvr_enabled;
|
||||||
|
@ -226,36 +217,23 @@ public:
|
||||||
SrsDvrPlan();
|
SrsDvrPlan();
|
||||||
virtual ~SrsDvrPlan();
|
virtual ~SrsDvrPlan();
|
||||||
public:
|
public:
|
||||||
virtual int initialize(SrsDvrSegmenter* s, SrsRequest* r);
|
virtual int initialize(SrsOriginHub* h, SrsDvrSegmenter* s, SrsRequest* r);
|
||||||
virtual int on_publish() = 0;
|
virtual int on_publish() = 0;
|
||||||
virtual void on_unpublish() = 0;
|
virtual void on_unpublish() = 0;
|
||||||
/**
|
|
||||||
* when got metadata.
|
|
||||||
*/
|
|
||||||
virtual int on_meta_data(SrsSharedPtrMessage* shared_metadata);
|
virtual int on_meta_data(SrsSharedPtrMessage* shared_metadata);
|
||||||
/**
|
|
||||||
* @param shared_audio, directly ptr, copy it if need to save it.
|
|
||||||
*/
|
|
||||||
virtual int on_audio(SrsSharedPtrMessage* shared_audio);
|
virtual int on_audio(SrsSharedPtrMessage* shared_audio);
|
||||||
/**
|
|
||||||
* @param shared_video, directly ptr, copy it if need to save it.
|
|
||||||
*/
|
|
||||||
virtual int on_video(SrsSharedPtrMessage* shared_video);
|
virtual int on_video(SrsSharedPtrMessage* shared_video);
|
||||||
// Internal interface for segmenter.
|
// Internal interface for segmenter.
|
||||||
public:
|
public:
|
||||||
// When segmenter close a segment.
|
// When segmenter close a segment.
|
||||||
virtual int on_reap_segment();
|
virtual int on_reap_segment();
|
||||||
// When segmenter got a keyframe.
|
|
||||||
virtual int on_video_keyframe();
|
|
||||||
// The plan may need to process the timestamp.
|
|
||||||
virtual int64_t filter_timestamp(int64_t timestamp);
|
|
||||||
public:
|
public:
|
||||||
static int create_plan(std::string vhost, SrsDvrPlan** pplan);
|
static int create_plan(std::string vhost, SrsDvrPlan** pplan);
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* session plan: reap flv when session complete(unpublish)
|
* The DVR session plan: reap flv when session complete(unpublish)
|
||||||
*/
|
*/
|
||||||
class SrsDvrSessionPlan : public SrsDvrPlan
|
class SrsDvrSessionPlan : public SrsDvrPlan
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
@ -267,33 +245,32 @@ public:
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* segment plan: reap flv when duration exceed.
|
* The DVR segment plan: reap flv when duration exceed.
|
||||||
*/
|
*/
|
||||||
class SrsDvrSegmentPlan : public SrsDvrPlan
|
class SrsDvrSegmentPlan : public SrsDvrPlan
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
// in config, in ms
|
// in config, in ms
|
||||||
int segment_duration;
|
int cduration;
|
||||||
SrsSharedPtrMessage* sh_audio;
|
bool wait_keyframe;
|
||||||
SrsSharedPtrMessage* sh_video;
|
|
||||||
SrsSharedPtrMessage* metadata;
|
|
||||||
public:
|
public:
|
||||||
SrsDvrSegmentPlan();
|
SrsDvrSegmentPlan();
|
||||||
virtual ~SrsDvrSegmentPlan();
|
virtual ~SrsDvrSegmentPlan();
|
||||||
public:
|
public:
|
||||||
virtual int initialize(SrsDvrSegmenter* s, SrsRequest* req);
|
virtual int initialize(SrsOriginHub* h, SrsDvrSegmenter* s, SrsRequest* r);
|
||||||
virtual int on_publish();
|
virtual int on_publish();
|
||||||
virtual void on_unpublish();
|
virtual void on_unpublish();
|
||||||
virtual int on_meta_data(SrsSharedPtrMessage* shared_metadata);
|
|
||||||
virtual int on_audio(SrsSharedPtrMessage* shared_audio);
|
virtual int on_audio(SrsSharedPtrMessage* shared_audio);
|
||||||
virtual int on_video(SrsSharedPtrMessage* shared_video);
|
virtual int on_video(SrsSharedPtrMessage* shared_video);
|
||||||
private:
|
private:
|
||||||
virtual int update_duration(SrsSharedPtrMessage* msg);
|
virtual int update_duration(SrsSharedPtrMessage* msg);
|
||||||
|
// interface ISrsReloadHandler
|
||||||
|
public:
|
||||||
|
virtual int on_reload_vhost_dvr(std::string vhost);
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* DVR(Digital Video Recorder) to record RTMP stream to flv/mp4 file.
|
* DVR(Digital Video Recorder) to record RTMP stream to flv/mp4 file.
|
||||||
* TODO: FIXME: add utest for it.
|
|
||||||
*/
|
*/
|
||||||
class SrsDvr : public ISrsReloadHandler
|
class SrsDvr : public ISrsReloadHandler
|
||||||
{
|
{
|
||||||
|
@ -330,7 +307,7 @@ public:
|
||||||
/**
|
/**
|
||||||
* get some information from metadata, it's optinal.
|
* get some information from metadata, it's optinal.
|
||||||
*/
|
*/
|
||||||
virtual int on_meta_data(SrsOnMetaDataPacket* m);
|
virtual int on_meta_data(SrsSharedPtrMessage* metadata);
|
||||||
/**
|
/**
|
||||||
* mux the audio packets to dvr.
|
* mux the audio packets to dvr.
|
||||||
* @param shared_audio, directly ptr, copy it if need to save it.
|
* @param shared_audio, directly ptr, copy it if need to save it.
|
||||||
|
|
|
@ -1265,23 +1265,6 @@ void SrsHls::on_unpublish()
|
||||||
hls_enabled = false;
|
hls_enabled = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
int SrsHls::on_meta_data(SrsAmf0Object* metadata)
|
|
||||||
{
|
|
||||||
int ret = ERROR_SUCCESS;
|
|
||||||
|
|
||||||
if (!metadata) {
|
|
||||||
srs_trace("no metadata persent, hls ignored it.");
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (metadata->count() <= 0) {
|
|
||||||
srs_trace("no metadata persent, hls ignored it.");
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
int SrsHls::on_audio(SrsSharedPtrMessage* shared_audio)
|
int SrsHls::on_audio(SrsSharedPtrMessage* shared_audio)
|
||||||
{
|
{
|
||||||
int ret = ERROR_SUCCESS;
|
int ret = ERROR_SUCCESS;
|
||||||
|
|
|
@ -405,10 +405,6 @@ public:
|
||||||
*/
|
*/
|
||||||
virtual void on_unpublish();
|
virtual void on_unpublish();
|
||||||
/**
|
/**
|
||||||
* get some information from metadata, it's optinal.
|
|
||||||
*/
|
|
||||||
virtual int on_meta_data(SrsAmf0Object* metadata);
|
|
||||||
/**
|
|
||||||
* mux the audio packets to ts.
|
* mux the audio packets to ts.
|
||||||
* @param shared_audio, directly ptr, copy it if need to save it.
|
* @param shared_audio, directly ptr, copy it if need to save it.
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -927,27 +927,6 @@ int SrsOriginHub::cycle()
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
int SrsOriginHub::on_original_metadata(SrsOnMetaDataPacket* metadata)
|
|
||||||
{
|
|
||||||
int ret = ERROR_SUCCESS;
|
|
||||||
|
|
||||||
#ifdef SRS_AUTO_HLS
|
|
||||||
if (metadata && (ret = hls->on_meta_data(metadata->metadata)) != ERROR_SUCCESS) {
|
|
||||||
srs_error("hls process onMetaData message failed. ret=%d", ret);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef SRS_AUTO_DVR
|
|
||||||
if (metadata && (ret = dvr->on_meta_data(metadata)) != ERROR_SUCCESS) {
|
|
||||||
srs_error("dvr process onMetaData message failed. ret=%d", ret);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
int SrsOriginHub::on_meta_data(SrsSharedPtrMessage* shared_metadata)
|
int SrsOriginHub::on_meta_data(SrsSharedPtrMessage* shared_metadata)
|
||||||
{
|
{
|
||||||
int ret = ERROR_SUCCESS;
|
int ret = ERROR_SUCCESS;
|
||||||
|
@ -964,6 +943,13 @@ int SrsOriginHub::on_meta_data(SrsSharedPtrMessage* shared_metadata)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef SRS_AUTO_DVR
|
||||||
|
if ((ret = dvr->on_meta_data(shared_metadata)) != ERROR_SUCCESS) {
|
||||||
|
srs_error("dvr process onMetaData message failed. ret=%d", ret);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1245,26 +1231,9 @@ int SrsOriginHub::on_dvr_request_sh()
|
||||||
// feed the dvr the metadata/sequence header,
|
// feed the dvr the metadata/sequence header,
|
||||||
// when reload to start dvr, dvr will never get the sequence header in stream,
|
// 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.
|
// use the SrsSource.on_dvr_request_sh to push the sequence header to DVR.
|
||||||
if (cache_metadata) {
|
if (cache_metadata && (ret = dvr->on_meta_data(cache_metadata)) != ERROR_SUCCESS) {
|
||||||
char* payload = cache_metadata->payload;
|
srs_error("dvr process onMetaData message failed. ret=%d", ret);
|
||||||
int size = cache_metadata->size;
|
return ret;
|
||||||
|
|
||||||
SrsBuffer 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;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (cache_sh_video && (ret = dvr->on_video(cache_sh_video)) != ERROR_SUCCESS) {
|
if (cache_sh_video && (ret = dvr->on_video(cache_sh_video)) != ERROR_SUCCESS) {
|
||||||
|
@ -1976,11 +1945,6 @@ int SrsSource::on_meta_data(SrsCommonMessage* msg, SrsOnMetaDataPacket* metadata
|
||||||
{
|
{
|
||||||
int ret = ERROR_SUCCESS;
|
int ret = ERROR_SUCCESS;
|
||||||
|
|
||||||
// Notify hub about the original metadata.
|
|
||||||
if ((ret = hub->on_original_metadata(metadata)) != ERROR_SUCCESS) {
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
// if allow atc_auto and bravo-atc detected, open atc for vhost.
|
// if allow atc_auto and bravo-atc detected, open atc for vhost.
|
||||||
SrsAmf0Any* prop = NULL;
|
SrsAmf0Any* prop = NULL;
|
||||||
atc = _srs_config->get_atc(req->vhost);
|
atc = _srs_config->get_atc(req->vhost);
|
||||||
|
|
|
@ -456,8 +456,6 @@ public:
|
||||||
// for example, dispose hls in cycle.
|
// for example, dispose hls in cycle.
|
||||||
virtual int cycle();
|
virtual int cycle();
|
||||||
public:
|
public:
|
||||||
// When got a original metadata.
|
|
||||||
virtual int on_original_metadata(SrsOnMetaDataPacket* metadata);
|
|
||||||
// When got a parsed metadata.
|
// When got a parsed metadata.
|
||||||
virtual int on_meta_data(SrsSharedPtrMessage* shared_metadata);
|
virtual int on_meta_data(SrsSharedPtrMessage* shared_metadata);
|
||||||
// When got a parsed audio packet.
|
// When got a parsed audio packet.
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue