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_st.cpp

557 lines
13 KiB
C++
Raw Normal View History

2017-03-25 09:21:39 +00:00
/**
* The MIT License (MIT)
*
* Copyright (c) 2013-2017 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.
*/
2014-03-01 02:14:25 +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>
2016-09-05 06:16:24 +00:00
#include <srs_app_log.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()
{
}
2017-03-01 04:23:44 +00:00
SrsThread::SrsThread(const char* n, ISrsThreadHandler* h, int64_t ims, bool j)
2015-09-19 05:46:55 +00:00
{
2017-03-01 04:23:44 +00:00
name = n;
handler = h;
cims = ims;
2015-09-19 05:46:55 +00:00
2017-03-01 04:23:44 +00:00
trd = NULL;
2015-09-19 05:46:55 +00:00
loop = false;
2017-03-01 04:23:44 +00:00
context_id = -1;
joinable = j;
2015-09-19 05:46:55 +00:00
}
SrsThread::~SrsThread()
{
stop();
}
int SrsThread::cid()
{
2017-03-01 04:23:44 +00:00
return context_id;
2015-09-19 05:46:55 +00:00
}
int SrsThread::start()
{
int ret = ERROR_SUCCESS;
2017-03-01 04:23:44 +00:00
if(trd) {
srs_info("thread %s already running.", name);
2015-09-19 05:46:55 +00:00
return ret;
}
2017-03-01 04:23:44 +00:00
loop = true;
if((trd = st_thread_create(pfn, this, (joinable? 1:0), 0)) == NULL){
2015-09-19 05:46:55 +00:00
ret = ERROR_ST_CREATE_CYCLE_THREAD;
srs_error("st_thread_create failed. ret=%d", ret);
return ret;
}
return ret;
}
void SrsThread::stop()
{
2017-03-01 04:23:44 +00:00
if (!trd) {
2015-09-19 05:46:55 +00:00
return;
}
2017-03-01 04:23:44 +00:00
// notify the cycle to stop loop.
loop = false;
// the interrupt will cause the socket to read/write error,
// which will terminate the cycle thread.
st_thread_interrupt(trd);
// when joinable, wait util quit.
if (joinable) {
// wait the thread to exit.
int ret = st_thread_join(trd, NULL);
srs_assert(ret == ERROR_SUCCESS);
}
2015-09-19 05:46:55 +00:00
2017-03-01 04:23:44 +00:00
trd = NULL;
2015-09-19 05:46:55 +00:00
}
bool SrsThread::can_loop()
{
return loop;
}
void SrsThread::stop_loop()
{
loop = false;
}
2017-03-01 04:23:44 +00:00
void SrsThread::cycle()
2015-09-19 05:46:55 +00:00
{
int ret = ERROR_SUCCESS;
2016-12-14 10:21:50 +00:00
// TODO: FIXME: it's better for user to specifies the cid,
// because sometimes we need to merge cid, for example,
// the publish thread should use the same cid of connection.
2015-09-19 05:46:55 +00:00
_srs_context->generate_id();
2017-03-01 04:23:44 +00:00
srs_info("thread %s cycle start", name);
context_id = _srs_context->get_id();
2015-09-19 05:46:55 +00:00
srs_assert(handler);
handler->on_thread_start();
while (loop) {
if ((ret = handler->on_before_cycle()) != ERROR_SUCCESS) {
2017-03-01 04:23:44 +00:00
srs_warn("thread %s on before cycle failed, ignored and retry, ret=%d", name, ret);
2015-09-19 05:46:55 +00:00
goto failed;
}
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)) {
2017-03-01 04:23:44 +00:00
srs_warn("thread %s cycle failed, ignored and retry, ret=%d", name, ret);
2015-09-19 05:46:55 +00:00
}
goto failed;
}
srs_info("thread %s cycle success", name);
2015-09-19 05:46:55 +00:00
if ((ret = handler->on_end_cycle()) != ERROR_SUCCESS) {
2017-03-01 04:23:44 +00:00
srs_warn("thread %s on end cycle failed, ignored and retry, ret=%d", name, ret);
2015-09-19 05:46:55 +00:00
goto failed;
}
srs_info("thread %s on end cycle success", name);
2015-09-19 05:46:55 +00:00
failed:
if (!loop) {
break;
}
// Should never use no timeout, just ignore it.
2015-09-19 05:46:55 +00:00
// 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
if (cims != 0 && cims != SRS_CONSTS_NO_TMMS) {
st_usleep(cims * 1000);
2015-09-19 05:46:55 +00:00
}
}
2017-03-01 04:23:44 +00:00
srs_info("thread %s cycle finished", name);
2016-12-14 10:21:50 +00:00
// @remark in this callback, user may delete this, so never use this->xxx anymore.
handler->on_thread_stop();
2015-09-19 05:46:55 +00:00
}
2017-03-01 04:23:44 +00:00
void* SrsThread::pfn(void* arg)
2015-09-19 05:46:55 +00:00
{
SrsThread* obj = (SrsThread*)arg;
srs_assert(obj);
2017-03-01 04:23:44 +00:00
obj->cycle();
2015-09-19 05:46:55 +00:00
2016-12-14 10:21:50 +00:00
// delete cid for valgrind to detect memory leak.
2016-09-05 06:16:24 +00:00
SrsThreadContext* ctx = dynamic_cast<SrsThreadContext*>(_srs_context);
if (ctx) {
ctx->clear_cid();
}
2015-09-19 05:46:55 +00:00
st_thread_exit(NULL);
return NULL;
}
}
SrsStSocket::SrsStSocket()
2015-06-14 00:43:38 +00:00
{
stfd = NULL;
stm = rtm = SRS_CONSTS_NO_TMMS;
rbytes = sbytes = 0;
2015-06-14 00:43:38 +00:00
}
SrsStSocket::~SrsStSocket()
{
}
int SrsStSocket::initialize(st_netfd_t fd)
{
stfd = fd;
return ERROR_SUCCESS;
}
bool SrsStSocket::is_never_timeout(int64_t tm)
2015-06-14 00:43:38 +00:00
{
return tm == SRS_CONSTS_NO_TMMS;
2015-06-14 00:43:38 +00:00
}
void SrsStSocket::set_recv_timeout(int64_t tm)
2015-06-14 00:43:38 +00:00
{
rtm = tm;
2015-06-14 00:43:38 +00:00
}
int64_t SrsStSocket::get_recv_timeout()
{
return rtm;
2015-06-14 00:43:38 +00:00
}
void SrsStSocket::set_send_timeout(int64_t tm)
2015-06-14 00:43:38 +00:00
{
stm = tm;
2015-06-14 00:43:38 +00:00
}
int64_t SrsStSocket::get_send_timeout()
{
return stm;
2015-06-14 00:43:38 +00:00
}
int64_t SrsStSocket::get_recv_bytes()
{
return rbytes;
2015-06-14 00:43:38 +00:00
}
int64_t SrsStSocket::get_send_bytes()
{
return sbytes;
2015-06-14 00:43:38 +00:00
}
int SrsStSocket::read(void* buf, size_t size, ssize_t* nread)
{
int ret = ERROR_SUCCESS;
ssize_t nb_read;
if (rtm == SRS_CONSTS_NO_TMMS) {
nb_read = st_read(stfd, buf, size, ST_UTIME_NO_TIMEOUT);
} else {
2017-01-19 07:51:55 +00:00
nb_read = st_read(stfd, buf, size, rtm * 1000);
}
2015-06-14 00:43:38 +00:00
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;
}
rbytes += nb_read;
2015-06-14 00:43:38 +00:00
return ret;
}
int SrsStSocket::read_fully(void* buf, size_t size, ssize_t* nread)
{
int ret = ERROR_SUCCESS;
ssize_t nb_read;
if (rtm == SRS_CONSTS_NO_TMMS) {
nb_read = st_read_fully(stfd, buf, size, ST_UTIME_NO_TIMEOUT);
} else {
2017-01-19 07:51:55 +00:00
nb_read = st_read_fully(stfd, buf, size, rtm * 1000);
}
2015-06-14 00:43:38 +00:00
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;
}
rbytes += nb_read;
2015-06-14 00:43:38 +00:00
return ret;
}
int SrsStSocket::write(void* buf, size_t size, ssize_t* nwrite)
{
int ret = ERROR_SUCCESS;
ssize_t nb_write;
if (stm == SRS_CONSTS_NO_TMMS) {
nb_write = st_write(stfd, buf, size, ST_UTIME_NO_TIMEOUT);
} else {
2017-01-19 07:51:55 +00:00
nb_write = st_write(stfd, buf, size, stm * 1000);
}
2015-06-14 00:43:38 +00:00
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;
}
sbytes += nb_write;
2015-06-14 00:43:38 +00:00
return ret;
}
int SrsStSocket::writev(const iovec *iov, int iov_size, ssize_t* nwrite)
{
int ret = ERROR_SUCCESS;
ssize_t nb_write;
if (stm == SRS_CONSTS_NO_TMMS) {
nb_write = st_writev(stfd, iov, iov_size, ST_UTIME_NO_TIMEOUT);
} else {
2017-01-19 07:51:55 +00:00
nb_write = st_writev(stfd, iov, iov_size, stm * 1000);
}
2015-06-14 00:43:38 +00:00
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;
}
sbytes += nb_write;
2015-06-14 00:43:38 +00:00
return ret;
}
SrsTcpClient::SrsTcpClient(string h, int p, int64_t tm)
2015-10-13 09:37:59 +00:00
{
2015-10-13 09:59:51 +00:00
stfd = NULL;
io = new SrsStSocket();
host = h;
port = p;
timeout = tm;
2015-10-13 09:37:59 +00:00
}
SrsTcpClient::~SrsTcpClient()
{
2015-10-13 09:59:51 +00:00
close();
srs_freep(io);
2015-10-13 09:59:51 +00:00
}
int SrsTcpClient::connect()
2015-10-13 09:37:59 +00:00
{
int ret = ERROR_SUCCESS;
2015-10-13 09:59:51 +00:00
close();
2015-10-13 09:59:51 +00:00
srs_assert(stfd == NULL);
if ((ret = srs_socket_connect(host, port, timeout, &stfd)) != ERROR_SUCCESS) {
srs_error("connect tcp://%s:%d failed, to=%"PRId64"ms. ret=%d", host.c_str(), port, timeout, ret);
2015-10-13 09:59:51 +00:00
return ret;
}
if ((ret = io->initialize(stfd)) != ERROR_SUCCESS) {
return ret;
}
2015-10-13 09:59:51 +00:00
2015-10-13 09:37:59 +00:00
return ret;
}
2015-10-13 09:59:51 +00:00
void SrsTcpClient::close()
{
// Ignore when already closed.
2015-10-13 09:59:51 +00:00
if (!io) {
return;
}
srs_close_stfd(stfd);
}
bool SrsTcpClient::is_never_timeout(int64_t tm)
2015-10-13 09:59:51 +00:00
{
return io->is_never_timeout(tm);
2015-10-13 09:59:51 +00:00
}
void SrsTcpClient::set_recv_timeout(int64_t tm)
2015-10-13 09:59:51 +00:00
{
io->set_recv_timeout(tm);
2015-10-13 09:59:51 +00:00
}
int64_t SrsTcpClient::get_recv_timeout()
{
return io->get_recv_timeout();
}
void SrsTcpClient::set_send_timeout(int64_t tm)
2015-10-13 09:59:51 +00:00
{
io->set_send_timeout(tm);
2015-10-13 09:59:51 +00:00
}
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);
}
#ifdef __linux__
2014-10-08 05:27:48 +00:00
#include <sys/epoll.h>
2014-10-08 05:27:48 +00:00
bool srs_st_epoll_is_supported(void)
{
struct epoll_event ev;
2017-03-25 09:21:39 +00:00
2014-10-08 05:27:48 +00:00
ev.events = EPOLLIN;
ev.data.ptr = NULL;
/* Guaranteed to fail */
epoll_ctl(-1, EPOLL_CTL_ADD, -1, &ev);
2017-03-25 09:21:39 +00:00
2014-10-08 05:27:48 +00:00
return (errno != ENOSYS);
}
#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;
#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;
srs_error("epoll required on Linux. ret=%d", ret);
2014-10-08 05:27:48 +00:00
return ret;
}
#endif
2014-10-08 05:27:48 +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;
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());
2017-03-25 09:21:39 +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;
}
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) {
// 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