2014-03-01 02:14:25 +00:00
|
|
|
/*
|
|
|
|
The MIT License (MIT)
|
|
|
|
|
2015-12-23 03:35:40 +00:00
|
|
|
Copyright (c) 2013-2016 SRS(ossrs)
|
2014-03-01 02:14:25 +00:00
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2014-03-02 13:49:09 +00:00
|
|
|
#include <srs_app_st.hpp>
|
2014-03-01 02:14:25 +00:00
|
|
|
|
2015-10-13 09:37:59 +00:00
|
|
|
#include <string>
|
|
|
|
using namespace std;
|
|
|
|
|
2014-10-08 05:27:48 +00:00
|
|
|
#include <srs_kernel_error.hpp>
|
|
|
|
#include <srs_kernel_log.hpp>
|
2015-10-13 09:59:51 +00:00
|
|
|
#include <srs_app_utility.hpp>
|
2014-10-08 05:27:48 +00:00
|
|
|
|
2015-09-19 05:46:55 +00:00
|
|
|
namespace internal
|
|
|
|
{
|
|
|
|
ISrsThreadHandler::ISrsThreadHandler()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
ISrsThreadHandler::~ISrsThreadHandler()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void ISrsThreadHandler::on_thread_start()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
int ISrsThreadHandler::on_before_cycle()
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ISrsThreadHandler::on_end_cycle()
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ISrsThreadHandler::on_thread_stop()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsThread::SrsThread(const char* name, ISrsThreadHandler* thread_handler, int64_t interval_us, bool joinable)
|
|
|
|
{
|
|
|
|
_name = name;
|
|
|
|
handler = thread_handler;
|
|
|
|
cycle_interval_us = interval_us;
|
|
|
|
|
|
|
|
tid = NULL;
|
|
|
|
loop = false;
|
|
|
|
really_terminated = true;
|
|
|
|
_cid = -1;
|
|
|
|
_joinable = joinable;
|
|
|
|
disposed = false;
|
|
|
|
|
|
|
|
// in start(), the thread cycle method maybe stop and remove the thread itself,
|
|
|
|
// and the thread start() is waiting for the _cid, and segment fault then.
|
2015-11-11 02:45:45 +00:00
|
|
|
// @see https://github.com/ossrs/srs/issues/110
|
2015-09-19 05:46:55 +00:00
|
|
|
// thread will set _cid, callback on_thread_start(), then wait for the can_run signal.
|
|
|
|
can_run = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsThread::~SrsThread()
|
|
|
|
{
|
|
|
|
stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsThread::cid()
|
|
|
|
{
|
|
|
|
return _cid;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsThread::start()
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
if(tid) {
|
|
|
|
srs_info("thread %s already running.", _name);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if((tid = st_thread_create(thread_fun, this, (_joinable? 1:0), 0)) == NULL){
|
|
|
|
ret = ERROR_ST_CREATE_CYCLE_THREAD;
|
|
|
|
srs_error("st_thread_create failed. ret=%d", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-11-05 06:48:25 +00:00
|
|
|
disposed = false;
|
2015-09-19 05:46:55 +00:00
|
|
|
// we set to loop to true for thread to run.
|
|
|
|
loop = true;
|
|
|
|
|
|
|
|
// wait for cid to ready, for parent thread to get the cid.
|
2015-11-05 06:48:25 +00:00
|
|
|
while (_cid < 0) {
|
2015-09-19 05:46:55 +00:00
|
|
|
st_usleep(10 * 1000);
|
|
|
|
}
|
|
|
|
|
|
|
|
// now, cycle thread can run.
|
|
|
|
can_run = true;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SrsThread::stop()
|
|
|
|
{
|
|
|
|
if (!tid) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
loop = false;
|
|
|
|
|
|
|
|
dispose();
|
|
|
|
|
2015-11-05 06:48:25 +00:00
|
|
|
_cid = -1;
|
|
|
|
can_run = false;
|
2015-09-19 05:46:55 +00:00
|
|
|
tid = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsThread::can_loop()
|
|
|
|
{
|
|
|
|
return loop;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SrsThread::stop_loop()
|
|
|
|
{
|
|
|
|
loop = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SrsThread::dispose()
|
|
|
|
{
|
|
|
|
if (disposed) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// the interrupt will cause the socket to read/write error,
|
|
|
|
// which will terminate the cycle thread.
|
|
|
|
st_thread_interrupt(tid);
|
|
|
|
|
|
|
|
// when joinable, wait util quit.
|
|
|
|
if (_joinable) {
|
|
|
|
// wait the thread to exit.
|
|
|
|
int ret = st_thread_join(tid, NULL);
|
|
|
|
if (ret) {
|
|
|
|
srs_warn("core: ignore join thread failed.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// wait the thread actually terminated.
|
|
|
|
// sometimes the thread join return -1, for example,
|
|
|
|
// when thread use st_recvfrom, the thread join return -1.
|
|
|
|
// so here, we use a variable to ensure the thread stopped.
|
|
|
|
// @remark even the thread not joinable, we must ensure the thread stopped when stop.
|
|
|
|
while (!really_terminated) {
|
|
|
|
st_usleep(10 * 1000);
|
|
|
|
|
|
|
|
if (really_terminated) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
srs_warn("core: wait thread to actually terminated");
|
|
|
|
}
|
|
|
|
|
|
|
|
disposed = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SrsThread::thread_cycle()
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
_srs_context->generate_id();
|
|
|
|
srs_info("thread %s cycle start", _name);
|
|
|
|
|
|
|
|
_cid = _srs_context->get_id();
|
|
|
|
|
|
|
|
srs_assert(handler);
|
|
|
|
handler->on_thread_start();
|
|
|
|
|
|
|
|
// thread is running now.
|
|
|
|
really_terminated = false;
|
|
|
|
|
|
|
|
// wait for cid to ready, for parent thread to get the cid.
|
|
|
|
while (!can_run && loop) {
|
|
|
|
st_usleep(10 * 1000);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (loop) {
|
|
|
|
if ((ret = handler->on_before_cycle()) != ERROR_SUCCESS) {
|
|
|
|
srs_warn("thread %s on before cycle failed, ignored and retry, ret=%d", _name, ret);
|
|
|
|
goto failed;
|
|
|
|
}
|
2016-08-08 06:09:10 +00:00
|
|
|
srs_info("thread %s on before cycle success", _name);
|
2015-09-19 05:46:55 +00:00
|
|
|
|
|
|
|
if ((ret = handler->cycle()) != ERROR_SUCCESS) {
|
|
|
|
if (!srs_is_client_gracefully_close(ret) && !srs_is_system_control_error(ret)) {
|
|
|
|
srs_warn("thread %s cycle failed, ignored and retry, ret=%d", _name, ret);
|
|
|
|
}
|
|
|
|
goto failed;
|
|
|
|
}
|
|
|
|
srs_info("thread %s cycle success", _name);
|
|
|
|
|
|
|
|
if ((ret = handler->on_end_cycle()) != ERROR_SUCCESS) {
|
|
|
|
srs_warn("thread %s on end cycle failed, ignored and retry, ret=%d", _name, ret);
|
|
|
|
goto failed;
|
|
|
|
}
|
|
|
|
srs_info("thread %s on end cycle success", _name);
|
|
|
|
|
|
|
|
failed:
|
|
|
|
if (!loop) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// to improve performance, donot sleep when interval is zero.
|
2015-11-11 02:45:45 +00:00
|
|
|
// @see: https://github.com/ossrs/srs/issues/237
|
2015-09-19 05:46:55 +00:00
|
|
|
if (cycle_interval_us != 0) {
|
|
|
|
st_usleep(cycle_interval_us);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// readly terminated now.
|
|
|
|
really_terminated = true;
|
|
|
|
|
2015-12-22 08:18:09 +00:00
|
|
|
handler->on_thread_stop();
|
|
|
|
srs_info("thread %s cycle finished", _name);
|
2015-09-19 05:46:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void* SrsThread::thread_fun(void* arg)
|
|
|
|
{
|
|
|
|
SrsThread* obj = (SrsThread*)arg;
|
|
|
|
srs_assert(obj);
|
|
|
|
|
|
|
|
obj->thread_cycle();
|
|
|
|
|
|
|
|
st_thread_exit(NULL);
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-14 00:43:38 +00:00
|
|
|
SrsStSocket::SrsStSocket(st_netfd_t client_stfd)
|
|
|
|
{
|
|
|
|
stfd = client_stfd;
|
|
|
|
send_timeout = recv_timeout = ST_UTIME_NO_TIMEOUT;
|
|
|
|
recv_bytes = send_bytes = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsStSocket::~SrsStSocket()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsStSocket::is_never_timeout(int64_t timeout_us)
|
|
|
|
{
|
|
|
|
return timeout_us == (int64_t)ST_UTIME_NO_TIMEOUT;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SrsStSocket::set_recv_timeout(int64_t timeout_us)
|
|
|
|
{
|
|
|
|
recv_timeout = timeout_us;
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t SrsStSocket::get_recv_timeout()
|
|
|
|
{
|
|
|
|
return recv_timeout;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SrsStSocket::set_send_timeout(int64_t timeout_us)
|
|
|
|
{
|
|
|
|
send_timeout = timeout_us;
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t SrsStSocket::get_send_timeout()
|
|
|
|
{
|
|
|
|
return send_timeout;
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t SrsStSocket::get_recv_bytes()
|
|
|
|
{
|
|
|
|
return recv_bytes;
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t SrsStSocket::get_send_bytes()
|
|
|
|
{
|
|
|
|
return send_bytes;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsStSocket::read(void* buf, size_t size, ssize_t* nread)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
ssize_t nb_read = st_read(stfd, buf, size, recv_timeout);
|
|
|
|
if (nread) {
|
|
|
|
*nread = nb_read;
|
|
|
|
}
|
|
|
|
|
|
|
|
// On success a non-negative integer indicating the number of bytes actually read is returned
|
|
|
|
// (a value of 0 means the network connection is closed or end of file is reached).
|
|
|
|
// Otherwise, a value of -1 is returned and errno is set to indicate the error.
|
|
|
|
if (nb_read <= 0) {
|
2015-11-11 02:37:50 +00:00
|
|
|
// @see https://github.com/ossrs/srs/issues/200
|
2015-06-14 00:43:38 +00:00
|
|
|
if (nb_read < 0 && errno == ETIME) {
|
|
|
|
return ERROR_SOCKET_TIMEOUT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nb_read == 0) {
|
|
|
|
errno = ECONNRESET;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ERROR_SOCKET_READ;
|
|
|
|
}
|
|
|
|
|
|
|
|
recv_bytes += nb_read;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsStSocket::read_fully(void* buf, size_t size, ssize_t* nread)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
ssize_t nb_read = st_read_fully(stfd, buf, size, recv_timeout);
|
|
|
|
if (nread) {
|
|
|
|
*nread = nb_read;
|
|
|
|
}
|
|
|
|
|
|
|
|
// On success a non-negative integer indicating the number of bytes actually read is returned
|
|
|
|
// (a value less than nbyte means the network connection is closed or end of file is reached)
|
|
|
|
// Otherwise, a value of -1 is returned and errno is set to indicate the error.
|
|
|
|
if (nb_read != (ssize_t)size) {
|
2015-11-11 02:37:50 +00:00
|
|
|
// @see https://github.com/ossrs/srs/issues/200
|
2015-06-14 00:43:38 +00:00
|
|
|
if (nb_read < 0 && errno == ETIME) {
|
|
|
|
return ERROR_SOCKET_TIMEOUT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nb_read >= 0) {
|
|
|
|
errno = ECONNRESET;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ERROR_SOCKET_READ_FULLY;
|
|
|
|
}
|
|
|
|
|
|
|
|
recv_bytes += nb_read;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsStSocket::write(void* buf, size_t size, ssize_t* nwrite)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
ssize_t nb_write = st_write(stfd, buf, size, send_timeout);
|
|
|
|
if (nwrite) {
|
|
|
|
*nwrite = nb_write;
|
|
|
|
}
|
|
|
|
|
|
|
|
// On success a non-negative integer equal to nbyte is returned.
|
|
|
|
// Otherwise, a value of -1 is returned and errno is set to indicate the error.
|
|
|
|
if (nb_write <= 0) {
|
2015-11-11 02:37:50 +00:00
|
|
|
// @see https://github.com/ossrs/srs/issues/200
|
2015-06-14 00:43:38 +00:00
|
|
|
if (nb_write < 0 && errno == ETIME) {
|
|
|
|
return ERROR_SOCKET_TIMEOUT;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ERROR_SOCKET_WRITE;
|
|
|
|
}
|
|
|
|
|
|
|
|
send_bytes += nb_write;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsStSocket::writev(const iovec *iov, int iov_size, ssize_t* nwrite)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
ssize_t nb_write = st_writev(stfd, iov, iov_size, send_timeout);
|
|
|
|
if (nwrite) {
|
|
|
|
*nwrite = nb_write;
|
|
|
|
}
|
|
|
|
|
|
|
|
// On success a non-negative integer equal to nbyte is returned.
|
|
|
|
// Otherwise, a value of -1 is returned and errno is set to indicate the error.
|
|
|
|
if (nb_write <= 0) {
|
2015-11-11 02:37:50 +00:00
|
|
|
// @see https://github.com/ossrs/srs/issues/200
|
2015-06-14 00:43:38 +00:00
|
|
|
if (nb_write < 0 && errno == ETIME) {
|
|
|
|
return ERROR_SOCKET_TIMEOUT;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ERROR_SOCKET_WRITE;
|
|
|
|
}
|
|
|
|
|
|
|
|
send_bytes += nb_write;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-10-13 09:37:59 +00:00
|
|
|
SrsTcpClient::SrsTcpClient()
|
|
|
|
{
|
2015-10-13 09:59:51 +00:00
|
|
|
io = NULL;
|
|
|
|
stfd = NULL;
|
2015-10-13 09:37:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SrsTcpClient::~SrsTcpClient()
|
|
|
|
{
|
2015-10-13 09:59:51 +00:00
|
|
|
close();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsTcpClient::connected()
|
|
|
|
{
|
|
|
|
return io;
|
2015-10-13 09:37:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int SrsTcpClient::connect(string host, int port, int64_t timeout)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
2015-10-13 09:59:51 +00:00
|
|
|
|
|
|
|
// when connected, ignore.
|
|
|
|
if (io) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// connect host.
|
|
|
|
if ((ret = srs_socket_connect(host, port, timeout, &stfd)) != ERROR_SUCCESS) {
|
2015-12-08 10:32:37 +00:00
|
|
|
srs_error("connect server %s:%d failed. ret=%d", host.c_str(), port, ret);
|
2015-10-13 09:59:51 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
io = new SrsStSocket(stfd);
|
|
|
|
|
2015-10-13 09:37:59 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-10-13 09:59:51 +00:00
|
|
|
void SrsTcpClient::close()
|
|
|
|
{
|
|
|
|
// when closed, ignore.
|
|
|
|
if (!io) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
srs_freep(io);
|
|
|
|
srs_close_stfd(stfd);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsTcpClient::is_never_timeout(int64_t timeout_us)
|
|
|
|
{
|
|
|
|
return io->is_never_timeout(timeout_us);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SrsTcpClient::set_recv_timeout(int64_t timeout_us)
|
|
|
|
{
|
|
|
|
io->set_recv_timeout(timeout_us);
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t SrsTcpClient::get_recv_timeout()
|
|
|
|
{
|
|
|
|
return io->get_recv_timeout();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SrsTcpClient::set_send_timeout(int64_t timeout_us)
|
|
|
|
{
|
|
|
|
io->set_send_timeout(timeout_us);
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t SrsTcpClient::get_send_timeout()
|
|
|
|
{
|
|
|
|
return io->get_send_timeout();
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t SrsTcpClient::get_recv_bytes()
|
|
|
|
{
|
|
|
|
return io->get_recv_bytes();
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t SrsTcpClient::get_send_bytes()
|
|
|
|
{
|
|
|
|
return io->get_send_bytes();
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsTcpClient::read(void* buf, size_t size, ssize_t* nread)
|
|
|
|
{
|
|
|
|
return io->read(buf, size, nread);
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsTcpClient::read_fully(void* buf, size_t size, ssize_t* nread)
|
|
|
|
{
|
|
|
|
return io->read_fully(buf, size, nread);
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsTcpClient::write(void* buf, size_t size, ssize_t* nwrite)
|
|
|
|
{
|
|
|
|
return io->write(buf, size, nwrite);
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsTcpClient::writev(const iovec *iov, int iov_size, ssize_t* nwrite)
|
|
|
|
{
|
|
|
|
return io->writev(iov, iov_size, nwrite);
|
|
|
|
}
|
|
|
|
|
2015-03-09 05:21:04 +00:00
|
|
|
#ifdef __linux__
|
2014-10-08 05:27:48 +00:00
|
|
|
#include <sys/epoll.h>
|
2015-03-09 05:21:04 +00:00
|
|
|
|
2014-10-08 05:27:48 +00:00
|
|
|
bool srs_st_epoll_is_supported(void)
|
|
|
|
{
|
|
|
|
struct epoll_event ev;
|
|
|
|
|
|
|
|
ev.events = EPOLLIN;
|
|
|
|
ev.data.ptr = NULL;
|
|
|
|
/* Guaranteed to fail */
|
|
|
|
epoll_ctl(-1, EPOLL_CTL_ADD, -1, &ev);
|
|
|
|
|
|
|
|
return (errno != ENOSYS);
|
|
|
|
}
|
2015-03-10 04:50:27 +00:00
|
|
|
#endif
|
2014-10-08 05:27:48 +00:00
|
|
|
|
2015-05-27 23:03:38 +00:00
|
|
|
int srs_st_init()
|
2014-10-08 05:27:48 +00:00
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
2015-03-09 05:21:04 +00:00
|
|
|
#ifdef __linux__
|
2014-10-08 05:27:48 +00:00
|
|
|
// check epoll, some old linux donot support epoll.
|
2015-11-11 02:37:50 +00:00
|
|
|
// @see https://github.com/ossrs/srs/issues/162
|
2014-10-08 05:27:48 +00:00
|
|
|
if (!srs_st_epoll_is_supported()) {
|
|
|
|
ret = ERROR_ST_SET_EPOLL;
|
2015-03-09 05:21:04 +00:00
|
|
|
srs_error("epoll required on Linux. ret=%d", ret);
|
2014-10-08 05:27:48 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2015-03-09 05:21:04 +00:00
|
|
|
#endif
|
2014-10-08 05:27:48 +00:00
|
|
|
|
2015-03-09 05:21:04 +00:00
|
|
|
// Select the best event system available on the OS. In Linux this is
|
|
|
|
// epoll(). On BSD it will be kqueue.
|
2014-10-08 05:27:48 +00:00
|
|
|
if (st_set_eventsys(ST_EVENTSYS_ALT) == -1) {
|
|
|
|
ret = ERROR_ST_SET_EPOLL;
|
2015-03-09 05:21:04 +00:00
|
|
|
srs_error("st_set_eventsys use %s failed. ret=%d", st_get_eventsys_name(), ret);
|
2014-10-08 05:27:48 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2015-09-24 04:15:12 +00:00
|
|
|
srs_info("st_set_eventsys to %s", st_get_eventsys_name());
|
2015-03-09 05:21:04 +00:00
|
|
|
|
2014-10-08 05:27:48 +00:00
|
|
|
if(st_init() != 0){
|
|
|
|
ret = ERROR_ST_INITIALIZE;
|
|
|
|
srs_error("st_init failed. ret=%d", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
2015-03-09 05:21:04 +00:00
|
|
|
srs_trace("st_init success, use %s", st_get_eventsys_name());
|
2014-10-08 05:27:48 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-03-01 02:14:25 +00:00
|
|
|
void srs_close_stfd(st_netfd_t& stfd)
|
|
|
|
{
|
2014-03-18 03:32:58 +00:00
|
|
|
if (stfd) {
|
2016-09-01 08:51:09 +00:00
|
|
|
// we must ensure the close is ok.
|
|
|
|
int err = st_netfd_close(stfd);
|
|
|
|
srs_assert(err != -1);
|
2016-09-02 07:10:43 +00:00
|
|
|
stfd = NULL;
|
2014-03-18 03:32:58 +00:00
|
|
|
}
|
2014-03-01 02:14:25 +00:00
|
|
|
}
|
2014-08-02 14:18:39 +00:00
|
|
|
|