mirror of
https://github.com/ossrs/srs.git
synced 2025-03-09 15:49:59 +00:00
371 lines
8 KiB
C++
371 lines
8 KiB
C++
//
|
|
// Copyright (c) 2013-2024 The SRS Authors
|
|
//
|
|
// SPDX-License-Identifier: MIT
|
|
//
|
|
|
|
#include <srs_app_pithy_print.hpp>
|
|
|
|
#include <stdlib.h>
|
|
using namespace std;
|
|
|
|
#include <srs_kernel_log.hpp>
|
|
#include <srs_app_config.hpp>
|
|
#include <srs_kernel_error.hpp>
|
|
#include <srs_kernel_utility.hpp>
|
|
|
|
SrsStageInfo::SrsStageInfo(int _stage_id, double ratio)
|
|
{
|
|
stage_id = _stage_id;
|
|
nb_clients = 0;
|
|
age = 0;
|
|
nn_count = 0;
|
|
interval_ratio = ratio;
|
|
|
|
update_print_time();
|
|
|
|
_srs_config->subscribe(this);
|
|
}
|
|
|
|
SrsStageInfo::~SrsStageInfo()
|
|
{
|
|
_srs_config->unsubscribe(this);
|
|
}
|
|
|
|
void SrsStageInfo::update_print_time()
|
|
{
|
|
interval = _srs_config->get_pithy_print();
|
|
}
|
|
|
|
void SrsStageInfo::elapse(srs_utime_t diff)
|
|
{
|
|
age += diff;
|
|
}
|
|
|
|
bool SrsStageInfo::can_print()
|
|
{
|
|
srs_utime_t can_print_age = nb_clients * (srs_utime_t)(interval_ratio * interval);
|
|
|
|
bool can_print = age >= can_print_age;
|
|
if (can_print) {
|
|
age = 0;
|
|
}
|
|
|
|
return can_print;
|
|
}
|
|
|
|
srs_error_t SrsStageInfo::on_reload_pithy_print()
|
|
{
|
|
update_print_time();
|
|
return srs_success;
|
|
}
|
|
|
|
SrsStageManager::SrsStageManager()
|
|
{
|
|
}
|
|
|
|
SrsStageManager::~SrsStageManager()
|
|
{
|
|
map<int, SrsStageInfo*>::iterator it;
|
|
for (it = stages.begin(); it != stages.end(); ++it) {
|
|
SrsStageInfo* stage = it->second;
|
|
srs_freep(stage);
|
|
}
|
|
}
|
|
|
|
SrsStageInfo* SrsStageManager::fetch_or_create(int stage_id, bool* pnew)
|
|
{
|
|
std::map<int, SrsStageInfo*>::iterator it = stages.find(stage_id);
|
|
|
|
// Create one if not exists.
|
|
if (it == stages.end()) {
|
|
SrsStageInfo* stage = new SrsStageInfo(stage_id);
|
|
stages[stage_id] = stage;
|
|
|
|
if (pnew) {
|
|
*pnew = true;
|
|
}
|
|
|
|
return stage;
|
|
}
|
|
|
|
// Exists, fetch it.
|
|
SrsStageInfo* stage = it->second;
|
|
|
|
if (pnew) {
|
|
*pnew = false;
|
|
}
|
|
|
|
return stage;
|
|
}
|
|
|
|
SrsErrorPithyPrint::SrsErrorPithyPrint(double ratio)
|
|
{
|
|
nn_count = 0;
|
|
ratio_ = ratio;
|
|
}
|
|
|
|
SrsErrorPithyPrint::~SrsErrorPithyPrint()
|
|
{
|
|
}
|
|
|
|
bool SrsErrorPithyPrint::can_print(srs_error_t err, uint32_t* pnn)
|
|
{
|
|
int error_code = srs_error_code(err);
|
|
return can_print(error_code, pnn);
|
|
}
|
|
|
|
bool SrsErrorPithyPrint::can_print(int error_code, uint32_t* pnn)
|
|
{
|
|
bool new_stage = false;
|
|
SrsStageInfo* stage = stages.fetch_or_create(error_code, &new_stage);
|
|
|
|
// Increase the count.
|
|
stage->nn_count++;
|
|
nn_count++;
|
|
|
|
if (pnn) {
|
|
*pnn = stage->nn_count;
|
|
}
|
|
|
|
// Always and only one client.
|
|
if (new_stage) {
|
|
stage->nb_clients = 1;
|
|
stage->interval_ratio = ratio_;
|
|
}
|
|
|
|
srs_utime_t tick = ticks[error_code];
|
|
if (!tick) {
|
|
ticks[error_code] = tick = srs_get_system_time();
|
|
}
|
|
|
|
srs_utime_t diff = srs_get_system_time() - tick;
|
|
diff = srs_max(0, diff);
|
|
|
|
stage->elapse(diff);
|
|
ticks[error_code] = srs_get_system_time();
|
|
|
|
return new_stage || stage->can_print();
|
|
}
|
|
|
|
SrsAlonePithyPrint::SrsAlonePithyPrint() : info_(0)
|
|
{
|
|
//stage work for one print
|
|
info_.nb_clients = 1;
|
|
|
|
previous_tick_ = srs_get_system_time();
|
|
}
|
|
|
|
SrsAlonePithyPrint::~SrsAlonePithyPrint()
|
|
{
|
|
}
|
|
|
|
void SrsAlonePithyPrint::elapse()
|
|
{
|
|
srs_utime_t diff = srs_get_system_time() - previous_tick_;
|
|
previous_tick_ = srs_get_system_time();
|
|
|
|
diff = srs_max(0, diff);
|
|
|
|
info_.elapse(diff);
|
|
}
|
|
|
|
bool SrsAlonePithyPrint::can_print()
|
|
{
|
|
return info_.can_print();
|
|
}
|
|
|
|
// The global stage manager for pithy print, multiple stages.
|
|
SrsStageManager* _srs_stages = NULL;
|
|
|
|
SrsPithyPrint::SrsPithyPrint(int _stage_id)
|
|
{
|
|
stage_id = _stage_id;
|
|
cache_ = NULL;
|
|
client_id = enter_stage();
|
|
previous_tick = srs_get_system_time();
|
|
_age = 0;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////
|
|
// pithy-print consts values
|
|
///////////////////////////////////////////////////////////
|
|
// the pithy stage for all play clients.
|
|
#define SRS_CONSTS_STAGE_PLAY_USER 1
|
|
// the pithy stage for all publish clients.
|
|
#define SRS_CONSTS_STAGE_PUBLISH_USER 2
|
|
// the pithy stage for all forward clients.
|
|
#define SRS_CONSTS_STAGE_FORWARDER 3
|
|
// the pithy stage for all encoders.
|
|
#define SRS_CONSTS_STAGE_ENCODER 4
|
|
// the pithy stage for all hls.
|
|
#define SRS_CONSTS_STAGE_HLS 5
|
|
// the pithy stage for all ingesters.
|
|
#define SRS_CONSTS_STAGE_INGESTER 6
|
|
// the pithy stage for all edge.
|
|
#define SRS_CONSTS_STAGE_EDGE 7
|
|
// the pithy stage for all stream caster.
|
|
#define SRS_CONSTS_STAGE_CASTER 8
|
|
// the pithy stage for all http stream.
|
|
#define SRS_CONSTS_STAGE_HTTP_STREAM 9
|
|
// the pithy stage for all http stream cache.
|
|
#define SRS_CONSTS_STAGE_HTTP_STREAM_CACHE 10
|
|
// for the ng-exec stage.
|
|
#define SRS_CONSTS_STAGE_EXEC 11
|
|
// for the rtc play
|
|
#define SRS_CONSTS_STAGE_RTC_PLAY 12
|
|
// for the rtc send
|
|
#define SRS_CONSTS_STAGE_RTC_SEND 13
|
|
// for the rtc recv
|
|
#define SRS_CONSTS_STAGE_RTC_RECV 14
|
|
|
|
#ifdef SRS_SRT
|
|
// the pithy stage for srt play clients.
|
|
#define SRS_CONSTS_STAGE_SRT_PLAY 15
|
|
// the pithy stage for srt publish clients.
|
|
#define SRS_CONSTS_STAGE_SRT_PUBLISH 16
|
|
#endif
|
|
|
|
SrsPithyPrint* SrsPithyPrint::create_rtmp_play()
|
|
{
|
|
return new SrsPithyPrint(SRS_CONSTS_STAGE_PLAY_USER);
|
|
}
|
|
|
|
SrsPithyPrint* SrsPithyPrint::create_rtmp_publish()
|
|
{
|
|
return new SrsPithyPrint(SRS_CONSTS_STAGE_PUBLISH_USER);
|
|
}
|
|
|
|
SrsPithyPrint* SrsPithyPrint::create_hls()
|
|
{
|
|
return new SrsPithyPrint(SRS_CONSTS_STAGE_HLS);
|
|
}
|
|
|
|
SrsPithyPrint* SrsPithyPrint::create_forwarder()
|
|
{
|
|
return new SrsPithyPrint(SRS_CONSTS_STAGE_FORWARDER);
|
|
}
|
|
|
|
SrsPithyPrint* SrsPithyPrint::create_encoder()
|
|
{
|
|
return new SrsPithyPrint(SRS_CONSTS_STAGE_ENCODER);
|
|
}
|
|
|
|
SrsPithyPrint* SrsPithyPrint::create_exec()
|
|
{
|
|
return new SrsPithyPrint(SRS_CONSTS_STAGE_EXEC);
|
|
}
|
|
|
|
SrsPithyPrint* SrsPithyPrint::create_ingester()
|
|
{
|
|
return new SrsPithyPrint(SRS_CONSTS_STAGE_INGESTER);
|
|
}
|
|
|
|
SrsPithyPrint* SrsPithyPrint::create_edge()
|
|
{
|
|
return new SrsPithyPrint(SRS_CONSTS_STAGE_EDGE);
|
|
}
|
|
|
|
SrsPithyPrint* SrsPithyPrint::create_caster()
|
|
{
|
|
return new SrsPithyPrint(SRS_CONSTS_STAGE_CASTER);
|
|
}
|
|
|
|
SrsPithyPrint* SrsPithyPrint::create_http_stream()
|
|
{
|
|
return new SrsPithyPrint(SRS_CONSTS_STAGE_HTTP_STREAM);
|
|
}
|
|
|
|
SrsPithyPrint* SrsPithyPrint::create_http_stream_cache()
|
|
{
|
|
return new SrsPithyPrint(SRS_CONSTS_STAGE_HTTP_STREAM_CACHE);
|
|
}
|
|
|
|
SrsPithyPrint* SrsPithyPrint::create_rtc_play()
|
|
{
|
|
return new SrsPithyPrint(SRS_CONSTS_STAGE_RTC_PLAY);
|
|
}
|
|
|
|
SrsPithyPrint* SrsPithyPrint::create_rtc_send(int fd)
|
|
{
|
|
return new SrsPithyPrint(fd<<16 | SRS_CONSTS_STAGE_RTC_SEND);
|
|
}
|
|
|
|
SrsPithyPrint* SrsPithyPrint::create_rtc_recv(int fd)
|
|
{
|
|
return new SrsPithyPrint(fd<<16 | SRS_CONSTS_STAGE_RTC_RECV);
|
|
}
|
|
|
|
#ifdef SRS_SRT
|
|
SrsPithyPrint* SrsPithyPrint::create_srt_play()
|
|
{
|
|
return new SrsPithyPrint(SRS_CONSTS_STAGE_SRT_PLAY);
|
|
}
|
|
|
|
SrsPithyPrint* SrsPithyPrint::create_srt_publish()
|
|
{
|
|
return new SrsPithyPrint(SRS_CONSTS_STAGE_SRT_PUBLISH);
|
|
}
|
|
#endif
|
|
|
|
SrsPithyPrint::~SrsPithyPrint()
|
|
{
|
|
leave_stage();
|
|
}
|
|
|
|
int SrsPithyPrint::enter_stage()
|
|
{
|
|
SrsStageInfo* stage = _srs_stages->fetch_or_create(stage_id);
|
|
srs_assert(stage != NULL);
|
|
client_id = stage->nb_clients++;
|
|
|
|
srs_verbose("enter stage, stage_id=%d, client_id=%d, nb_clients=%d",
|
|
stage->stage_id, client_id, stage->nb_clients);
|
|
|
|
return client_id;
|
|
}
|
|
|
|
void SrsPithyPrint::leave_stage()
|
|
{
|
|
SrsStageInfo* stage = _srs_stages->fetch_or_create(stage_id);
|
|
srs_assert(stage != NULL);
|
|
|
|
stage->nb_clients--;
|
|
|
|
srs_verbose("leave stage, stage_id=%d, client_id=%d, nb_clients=%d",
|
|
stage->stage_id, client_id, stage->nb_clients);
|
|
}
|
|
|
|
void SrsPithyPrint::elapse()
|
|
{
|
|
SrsStageInfo* stage = cache_;
|
|
if (!stage) {
|
|
stage = cache_ = _srs_stages->fetch_or_create(stage_id);
|
|
}
|
|
srs_assert(stage != NULL);
|
|
|
|
srs_utime_t diff = srs_get_system_time() - previous_tick;
|
|
diff = srs_max(0, diff);
|
|
|
|
stage->elapse(diff);
|
|
_age += diff;
|
|
previous_tick = srs_get_system_time();
|
|
}
|
|
|
|
bool SrsPithyPrint::can_print()
|
|
{
|
|
SrsStageInfo* stage = cache_;
|
|
if (!stage) {
|
|
stage = cache_ = _srs_stages->fetch_or_create(stage_id);
|
|
}
|
|
srs_assert(stage != NULL);
|
|
|
|
return stage->can_print();
|
|
}
|
|
|
|
srs_utime_t SrsPithyPrint::age()
|
|
{
|
|
return _age;
|
|
}
|
|
|
|
|