mirror of
https://github.com/ossrs/srs.git
synced 2025-03-09 15:49:59 +00:00
For #1657: Refine TCP connections arch
This commit is contained in:
parent
4ba66b388b
commit
2a14dc0ebd
10 changed files with 29 additions and 286 deletions
|
@ -330,77 +330,47 @@ ISrsStartableConneciton::~ISrsStartableConneciton()
|
|||
{
|
||||
}
|
||||
|
||||
SrsTcpConnection::SrsTcpConnection(ISrsResourceManager* cm, srs_netfd_t c, string cip, int cport)
|
||||
SrsTcpConnection::SrsTcpConnection(srs_netfd_t c)
|
||||
{
|
||||
manager = cm;
|
||||
stfd = c;
|
||||
ip = cip;
|
||||
port = cport;
|
||||
create_time = srsu2ms(srs_get_system_time());
|
||||
|
||||
skt = new SrsStSocket();
|
||||
clk = new SrsWallClock();
|
||||
kbps = new SrsKbps(clk);
|
||||
kbps->set_io(skt, skt);
|
||||
|
||||
trd = new SrsSTCoroutine("conn", this);
|
||||
}
|
||||
|
||||
SrsTcpConnection::~SrsTcpConnection()
|
||||
{
|
||||
dispose();
|
||||
|
||||
srs_freep(kbps);
|
||||
srs_freep(clk);
|
||||
srs_freep(skt);
|
||||
srs_freep(trd);
|
||||
|
||||
srs_close_stfd(stfd);
|
||||
}
|
||||
|
||||
void SrsTcpConnection::remark(int64_t* in, int64_t* out)
|
||||
{
|
||||
kbps->remark(in, out);
|
||||
}
|
||||
|
||||
void SrsTcpConnection::dispose()
|
||||
{
|
||||
trd->interrupt();
|
||||
}
|
||||
|
||||
srs_error_t SrsTcpConnection::start()
|
||||
srs_error_t SrsTcpConnection::initialize()
|
||||
{
|
||||
srs_error_t err = srs_success;
|
||||
|
||||
|
||||
if ((err = skt->initialize(stfd)) != srs_success) {
|
||||
return srs_error_wrap(err, "init socket");
|
||||
}
|
||||
|
||||
if ((err = trd->start()) != srs_success) {
|
||||
return srs_error_wrap(err, "coroutine");
|
||||
}
|
||||
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
srs_error_t SrsTcpConnection::set_tcp_nodelay(bool v)
|
||||
{
|
||||
srs_error_t err = srs_success;
|
||||
|
||||
|
||||
int r0 = 0;
|
||||
socklen_t nb_v = sizeof(int);
|
||||
int fd = srs_netfd_fileno(stfd);
|
||||
|
||||
|
||||
int ov = 0;
|
||||
if ((r0 = getsockopt(fd, IPPROTO_TCP, TCP_NODELAY, &ov, &nb_v)) != 0) {
|
||||
return srs_error_new(ERROR_SOCKET_NO_NODELAY, "getsockopt fd=%d, r0=%d", fd, r0);
|
||||
}
|
||||
|
||||
|
||||
#ifndef SRS_PERF_TCP_NODELAY
|
||||
srs_warn("ignore TCP_NODELAY, fd=%d, ov=%d", fd, ov);
|
||||
return err;
|
||||
#endif
|
||||
|
||||
|
||||
int iv = (v? 1:0);
|
||||
if ((r0 = setsockopt(fd, IPPROTO_TCP, TCP_NODELAY, &iv, nb_v)) != 0) {
|
||||
return srs_error_new(ERROR_SOCKET_NO_NODELAY, "setsockopt fd=%d, r0=%d", fd, r0);
|
||||
|
@ -408,168 +378,13 @@ srs_error_t SrsTcpConnection::set_tcp_nodelay(bool v)
|
|||
if ((r0 = getsockopt(fd, IPPROTO_TCP, TCP_NODELAY, &iv, &nb_v)) != 0) {
|
||||
return srs_error_new(ERROR_SOCKET_NO_NODELAY, "getsockopt fd=%d, r0=%d", fd, r0);
|
||||
}
|
||||
|
||||
|
||||
srs_trace("set fd=%d TCP_NODELAY %d=>%d", fd, ov, iv);
|
||||
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
srs_error_t SrsTcpConnection::set_socket_buffer(srs_utime_t buffer_v)
|
||||
{
|
||||
srs_error_t err = srs_success;
|
||||
|
||||
int r0 = 0;
|
||||
int fd = srs_netfd_fileno(stfd);
|
||||
socklen_t nb_v = sizeof(int);
|
||||
|
||||
int ov = 0;
|
||||
if ((r0 = getsockopt(fd, SOL_SOCKET, SO_SNDBUF, &ov, &nb_v)) != 0) {
|
||||
return srs_error_new(ERROR_SOCKET_SNDBUF, "getsockopt fd=%d, r0=%d", fd, r0);
|
||||
}
|
||||
|
||||
#ifndef SRS_PERF_MW_SO_SNDBUF
|
||||
srs_warn("ignore SO_SNDBUF, fd=%d, ov=%d", fd, ov);
|
||||
return err;
|
||||
#endif
|
||||
|
||||
// the bytes:
|
||||
// 4KB=4096, 8KB=8192, 16KB=16384, 32KB=32768, 64KB=65536,
|
||||
// 128KB=131072, 256KB=262144, 512KB=524288
|
||||
// the buffer should set to sleep*kbps/8,
|
||||
// for example, your system delivery stream in 1000kbps,
|
||||
// sleep 800ms for small bytes, the buffer should set to:
|
||||
// 800*1000/8=100000B(about 128KB).
|
||||
// other examples:
|
||||
// 2000*3000/8=750000B(about 732KB).
|
||||
// 2000*5000/8=1250000B(about 1220KB).
|
||||
int kbps = 4000;
|
||||
int iv = srsu2ms(buffer_v) * kbps / 8;
|
||||
|
||||
// socket send buffer, system will double it.
|
||||
iv = iv / 2;
|
||||
|
||||
// override the send buffer by macro.
|
||||
#ifdef SRS_PERF_SO_SNDBUF_SIZE
|
||||
iv = SRS_PERF_SO_SNDBUF_SIZE / 2;
|
||||
#endif
|
||||
|
||||
// set the socket send buffer when required larger buffer
|
||||
if (setsockopt(fd, SOL_SOCKET, SO_SNDBUF, &iv, nb_v) < 0) {
|
||||
return srs_error_new(ERROR_SOCKET_SNDBUF, "setsockopt fd=%d, r0=%d", fd, r0);
|
||||
}
|
||||
if ((r0 = getsockopt(fd, SOL_SOCKET, SO_SNDBUF, &iv, &nb_v)) != 0) {
|
||||
return srs_error_new(ERROR_SOCKET_SNDBUF, "getsockopt fd=%d, r0=%d", fd, r0);
|
||||
}
|
||||
|
||||
srs_trace("set fd=%d, SO_SNDBUF=%d=>%d, buffer=%dms", fd, ov, iv, srsu2ms(buffer_v));
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
srs_error_t SrsTcpConnection::cycle()
|
||||
{
|
||||
srs_error_t err = do_cycle();
|
||||
|
||||
// Notify manager to remove it.
|
||||
manager->remove(this);
|
||||
|
||||
// success.
|
||||
if (err == srs_success) {
|
||||
srs_trace("client finished.");
|
||||
return err;
|
||||
}
|
||||
|
||||
// It maybe success with message.
|
||||
if (srs_error_code(err) == ERROR_SUCCESS) {
|
||||
srs_trace("client finished%s.", srs_error_summary(err).c_str());
|
||||
srs_freep(err);
|
||||
return err;
|
||||
}
|
||||
|
||||
// client close peer.
|
||||
// TODO: FIXME: Only reset the error when client closed it.
|
||||
if (srs_is_client_gracefully_close(err)) {
|
||||
srs_warn("client disconnect peer. ret=%d", srs_error_code(err));
|
||||
} else if (srs_is_server_gracefully_close(err)) {
|
||||
srs_warn("server disconnect. ret=%d", srs_error_code(err));
|
||||
} else {
|
||||
srs_error("serve error %s", srs_error_desc(err).c_str());
|
||||
}
|
||||
|
||||
srs_freep(err);
|
||||
return srs_success;
|
||||
}
|
||||
|
||||
string SrsTcpConnection::remote_ip()
|
||||
{
|
||||
return ip;
|
||||
}
|
||||
|
||||
const SrsContextId& SrsTcpConnection::get_id()
|
||||
{
|
||||
return trd->cid();
|
||||
}
|
||||
|
||||
void SrsTcpConnection::expire()
|
||||
{
|
||||
trd->interrupt();
|
||||
}
|
||||
|
||||
SrsTcpConnection2::SrsTcpConnection2(srs_netfd_t c)
|
||||
{
|
||||
stfd = c;
|
||||
skt = new SrsStSocket();
|
||||
}
|
||||
|
||||
SrsTcpConnection2::~SrsTcpConnection2()
|
||||
{
|
||||
srs_freep(skt);
|
||||
srs_close_stfd(stfd);
|
||||
}
|
||||
|
||||
srs_error_t SrsTcpConnection2::initialize()
|
||||
{
|
||||
srs_error_t err = srs_success;
|
||||
|
||||
if ((err = skt->initialize(stfd)) != srs_success) {
|
||||
return srs_error_wrap(err, "init socket");
|
||||
}
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
srs_error_t SrsTcpConnection2::set_tcp_nodelay(bool v)
|
||||
{
|
||||
srs_error_t err = srs_success;
|
||||
|
||||
int r0 = 0;
|
||||
socklen_t nb_v = sizeof(int);
|
||||
int fd = srs_netfd_fileno(stfd);
|
||||
|
||||
int ov = 0;
|
||||
if ((r0 = getsockopt(fd, IPPROTO_TCP, TCP_NODELAY, &ov, &nb_v)) != 0) {
|
||||
return srs_error_new(ERROR_SOCKET_NO_NODELAY, "getsockopt fd=%d, r0=%d", fd, r0);
|
||||
}
|
||||
|
||||
#ifndef SRS_PERF_TCP_NODELAY
|
||||
srs_warn("ignore TCP_NODELAY, fd=%d, ov=%d", fd, ov);
|
||||
return err;
|
||||
#endif
|
||||
|
||||
int iv = (v? 1:0);
|
||||
if ((r0 = setsockopt(fd, IPPROTO_TCP, TCP_NODELAY, &iv, nb_v)) != 0) {
|
||||
return srs_error_new(ERROR_SOCKET_NO_NODELAY, "setsockopt fd=%d, r0=%d", fd, r0);
|
||||
}
|
||||
if ((r0 = getsockopt(fd, IPPROTO_TCP, TCP_NODELAY, &iv, &nb_v)) != 0) {
|
||||
return srs_error_new(ERROR_SOCKET_NO_NODELAY, "getsockopt fd=%d, r0=%d", fd, r0);
|
||||
}
|
||||
|
||||
srs_trace("set fd=%d TCP_NODELAY %d=>%d", fd, ov, iv);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
srs_error_t SrsTcpConnection2::set_socket_buffer(srs_utime_t buffer_v)
|
||||
{
|
||||
srs_error_t err = srs_success;
|
||||
|
||||
|
@ -621,52 +436,52 @@ srs_error_t SrsTcpConnection2::set_socket_buffer(srs_utime_t buffer_v)
|
|||
return err;
|
||||
}
|
||||
|
||||
void SrsTcpConnection2::set_recv_timeout(srs_utime_t tm)
|
||||
void SrsTcpConnection::set_recv_timeout(srs_utime_t tm)
|
||||
{
|
||||
skt->set_recv_timeout(tm);
|
||||
}
|
||||
|
||||
srs_utime_t SrsTcpConnection2::get_recv_timeout()
|
||||
srs_utime_t SrsTcpConnection::get_recv_timeout()
|
||||
{
|
||||
return skt->get_recv_timeout();
|
||||
}
|
||||
|
||||
srs_error_t SrsTcpConnection2::read_fully(void* buf, size_t size, ssize_t* nread)
|
||||
srs_error_t SrsTcpConnection::read_fully(void* buf, size_t size, ssize_t* nread)
|
||||
{
|
||||
return skt->read_fully(buf, size, nread);
|
||||
}
|
||||
|
||||
int64_t SrsTcpConnection2::get_recv_bytes()
|
||||
int64_t SrsTcpConnection::get_recv_bytes()
|
||||
{
|
||||
return skt->get_recv_bytes();
|
||||
}
|
||||
|
||||
int64_t SrsTcpConnection2::get_send_bytes()
|
||||
int64_t SrsTcpConnection::get_send_bytes()
|
||||
{
|
||||
return skt->get_send_bytes();
|
||||
}
|
||||
|
||||
srs_error_t SrsTcpConnection2::read(void* buf, size_t size, ssize_t* nread)
|
||||
srs_error_t SrsTcpConnection::read(void* buf, size_t size, ssize_t* nread)
|
||||
{
|
||||
return skt->read(buf, size, nread);
|
||||
}
|
||||
|
||||
void SrsTcpConnection2::set_send_timeout(srs_utime_t tm)
|
||||
void SrsTcpConnection::set_send_timeout(srs_utime_t tm)
|
||||
{
|
||||
skt->set_send_timeout(tm);
|
||||
}
|
||||
|
||||
srs_utime_t SrsTcpConnection2::get_send_timeout()
|
||||
srs_utime_t SrsTcpConnection::get_send_timeout()
|
||||
{
|
||||
return skt->get_send_timeout();
|
||||
}
|
||||
|
||||
srs_error_t SrsTcpConnection2::write(void* buf, size_t size, ssize_t* nwrite)
|
||||
srs_error_t SrsTcpConnection::write(void* buf, size_t size, ssize_t* nwrite)
|
||||
{
|
||||
return skt->write(buf, size, nwrite);
|
||||
}
|
||||
|
||||
srs_error_t SrsTcpConnection2::writev(const iovec *iov, int iov_size, ssize_t* nwrite)
|
||||
srs_error_t SrsTcpConnection::writev(const iovec *iov, int iov_size, ssize_t* nwrite)
|
||||
{
|
||||
return skt->writev(iov, iov_size, nwrite);
|
||||
}
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue