mirror of
https://github.com/ossrs/srs.git
synced 2025-02-15 04:42:04 +00:00
162 lines
4 KiB
C++
162 lines
4 KiB
C++
//
|
|
// Copyright (c) 2013-2022 The SRS Authors
|
|
//
|
|
// SPDX-License-Identifier: MIT or MulanPSL-2.0
|
|
//
|
|
|
|
#ifndef SRS_APP_THREADS_HPP
|
|
#define SRS_APP_THREADS_HPP
|
|
|
|
#include <srs_core.hpp>
|
|
|
|
#include <srs_app_hourglass.hpp>
|
|
|
|
#include <pthread.h>
|
|
|
|
class SrsThreadPool;
|
|
class SrsProcSelfStat;
|
|
|
|
// Protect server in high load.
|
|
class SrsCircuitBreaker : public ISrsFastTimer
|
|
{
|
|
private:
|
|
// The config for high/critical water level.
|
|
bool enabled_;
|
|
int high_threshold_;
|
|
int high_pulse_;
|
|
int critical_threshold_;
|
|
int critical_pulse_;
|
|
int dying_threshold_;
|
|
int dying_pulse_;
|
|
private:
|
|
// Reset the water-level when CPU is low for N times.
|
|
// @note To avoid the CPU change rapidly.
|
|
int hybrid_high_water_level_;
|
|
int hybrid_critical_water_level_;
|
|
int hybrid_dying_water_level_;
|
|
public:
|
|
SrsCircuitBreaker();
|
|
virtual ~SrsCircuitBreaker();
|
|
public:
|
|
srs_error_t initialize();
|
|
public:
|
|
// Whether hybrid server water-level is high.
|
|
bool hybrid_high_water_level();
|
|
bool hybrid_critical_water_level();
|
|
bool hybrid_dying_water_level();
|
|
// interface ISrsFastTimer
|
|
private:
|
|
srs_error_t on_timer(srs_utime_t interval);
|
|
};
|
|
|
|
extern SrsCircuitBreaker* _srs_circuit_breaker;
|
|
|
|
// Initialize global shared variables cross all threads.
|
|
extern srs_error_t srs_global_initialize();
|
|
|
|
// The thread mutex wrapper, without error.
|
|
class SrsThreadMutex
|
|
{
|
|
private:
|
|
pthread_mutex_t lock_;
|
|
pthread_mutexattr_t attr_;
|
|
public:
|
|
SrsThreadMutex();
|
|
virtual ~SrsThreadMutex();
|
|
public:
|
|
void lock();
|
|
void unlock();
|
|
};
|
|
|
|
// The thread mutex locker.
|
|
// TODO: FIXME: Rename _SRS to _srs
|
|
#define SrsThreadLocker(instance) \
|
|
impl__SrsThreadLocker _SRS_free_##instance(instance)
|
|
|
|
class impl__SrsThreadLocker
|
|
{
|
|
private:
|
|
SrsThreadMutex* lock;
|
|
public:
|
|
impl__SrsThreadLocker(SrsThreadMutex* l) {
|
|
lock = l;
|
|
lock->lock();
|
|
}
|
|
virtual ~impl__SrsThreadLocker() {
|
|
lock->unlock();
|
|
}
|
|
};
|
|
|
|
// The information for a thread.
|
|
class SrsThreadEntry
|
|
{
|
|
public:
|
|
SrsThreadPool* pool;
|
|
std::string label;
|
|
std::string name;
|
|
srs_error_t (*start)(void* arg);
|
|
void* arg;
|
|
int num;
|
|
// @see https://man7.org/linux/man-pages/man2/gettid.2.html
|
|
pid_t tid;
|
|
public:
|
|
// The thread object.
|
|
pthread_t trd;
|
|
// The exit error of thread.
|
|
srs_error_t err;
|
|
public:
|
|
SrsThreadEntry();
|
|
virtual ~SrsThreadEntry();
|
|
};
|
|
|
|
// Allocate a(or almost) fixed thread poll to execute tasks,
|
|
// so that we can take the advantage of multiple CPUs.
|
|
class SrsThreadPool
|
|
{
|
|
private:
|
|
SrsThreadEntry* entry_;
|
|
srs_utime_t interval_;
|
|
private:
|
|
SrsThreadMutex* lock_;
|
|
std::vector<SrsThreadEntry*> threads_;
|
|
private:
|
|
// The hybrid server entry, the cpu percent used for circuit breaker.
|
|
SrsThreadEntry* hybrid_;
|
|
std::vector<SrsThreadEntry*> hybrids_;
|
|
private:
|
|
// The pid file fd, lock the file write when server is running.
|
|
// @remark the init.d script should cleanup the pid file, when stop service,
|
|
// for the server never delete the file; when system startup, the pid in pid file
|
|
// maybe valid but the process is not SRS, the init.d script will never start server.
|
|
int pid_fd;
|
|
public:
|
|
SrsThreadPool();
|
|
virtual ~SrsThreadPool();
|
|
public:
|
|
// Setup the thread-local variables.
|
|
static srs_error_t setup_thread_locals();
|
|
// Initialize the thread pool.
|
|
srs_error_t initialize();
|
|
private:
|
|
// Require the PID file for the whole process.
|
|
virtual srs_error_t acquire_pid_file();
|
|
public:
|
|
// Execute start function with label in thread.
|
|
srs_error_t execute(std::string label, srs_error_t (*start)(void* arg), void* arg);
|
|
// Run in the primordial thread, util stop or quit.
|
|
srs_error_t run();
|
|
// Stop the thread pool and quit the primordial thread.
|
|
void stop();
|
|
public:
|
|
SrsThreadEntry* self();
|
|
SrsThreadEntry* hybrid();
|
|
std::vector<SrsThreadEntry*> hybrids();
|
|
private:
|
|
static void* start(void* arg);
|
|
};
|
|
|
|
// It MUST be thread-safe, global and shared object.
|
|
extern SrsThreadPool* _srs_thread_pool;
|
|
|
|
#endif
|
|
|