![]() System : Linux absol.cf 5.4.0-198-generic #218-Ubuntu SMP Fri Sep 27 20:18:53 UTC 2024 x86_64 User : www-data ( 33) PHP Version : 7.4.33 Disable Function : pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wifcontinued,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_get_handler,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,pcntl_async_signals,pcntl_unshare, Directory : /usr/local/lib/node_modules/mediasoup/worker/src/handles/ |
Upload File : |
#define MS_CLASS "TcpServer" // #define MS_LOG_DEV_LEVEL 3 #include "handles/TcpServer.hpp" #include "Logger.hpp" #include "MediaSoupErrors.hpp" #include "Utils.hpp" /* Static methods for UV callbacks. */ inline static void onConnection(uv_stream_t* handle, int status) { auto* server = static_cast<TcpServer*>(handle->data); if (server) server->OnUvConnection(status); } inline static void onClose(uv_handle_t* handle) { delete handle; } /* Instance methods. */ // NOLINTNEXTLINE(cppcoreguidelines-pro-type-member-init) TcpServer::TcpServer(uv_tcp_t* uvHandle, int backlog) : uvHandle(uvHandle) { MS_TRACE(); int err; this->uvHandle->data = static_cast<void*>(this); err = uv_listen( reinterpret_cast<uv_stream_t*>(this->uvHandle), backlog, static_cast<uv_connection_cb>(onConnection)); if (err != 0) { uv_close(reinterpret_cast<uv_handle_t*>(this->uvHandle), static_cast<uv_close_cb>(onClose)); MS_THROW_ERROR("uv_listen() failed: %s", uv_strerror(err)); } // Set local address. if (!SetLocalAddress()) { uv_close(reinterpret_cast<uv_handle_t*>(this->uvHandle), static_cast<uv_close_cb>(onClose)); MS_THROW_ERROR("error setting local IP and port"); } } TcpServer::~TcpServer() { MS_TRACE(); if (!this->closed) Close(); } void TcpServer::Close() { MS_TRACE(); if (this->closed) return; this->closed = true; // Tell the UV handle that the TcpServer has been closed. this->uvHandle->data = nullptr; MS_DEBUG_DEV("closing %zu active connections", this->connections.size()); for (auto* connection : this->connections) { delete connection; } uv_close(reinterpret_cast<uv_handle_t*>(this->uvHandle), static_cast<uv_close_cb>(onClose)); } void TcpServer::Dump() const { MS_DUMP("<TcpServer>"); MS_DUMP( " [TCP, local:%s :%" PRIu16 ", status:%s, connections:%zu]", this->localIp.c_str(), static_cast<uint16_t>(this->localPort), (!this->closed) ? "open" : "closed", this->connections.size()); MS_DUMP("</TcpServer>"); } void TcpServer::AcceptTcpConnection(TcpConnection* connection) { MS_TRACE(); MS_ASSERT(connection != nullptr, "TcpConnection pointer was not allocated by the user"); try { connection->Setup(this, &(this->localAddr), this->localIp, this->localPort); } catch (const MediaSoupError& error) { delete connection; return; } // Accept the connection. int err = uv_accept( reinterpret_cast<uv_stream_t*>(this->uvHandle), reinterpret_cast<uv_stream_t*>(connection->GetUvHandle())); if (err != 0) MS_ABORT("uv_accept() failed: %s", uv_strerror(err)); // Start receiving data. try { // NOTE: This may throw. connection->Start(); } catch (const MediaSoupError& error) { delete connection; return; } // Store it. this->connections.insert(connection); } bool TcpServer::SetLocalAddress() { MS_TRACE(); int err; int len = sizeof(this->localAddr); err = uv_tcp_getsockname(this->uvHandle, reinterpret_cast<struct sockaddr*>(&this->localAddr), &len); if (err != 0) { MS_ERROR("uv_tcp_getsockname() failed: %s", uv_strerror(err)); return false; } int family; Utils::IP::GetAddressInfo( reinterpret_cast<const struct sockaddr*>(&this->localAddr), family, this->localIp, this->localPort); return true; } inline void TcpServer::OnUvConnection(int status) { MS_TRACE(); if (this->closed) return; if (status != 0) { MS_ERROR("error while receiving a new TCP connection: %s", uv_strerror(status)); return; } // Notify the subclass about a new TCP connection attempt. UserOnTcpConnectionAlloc(); } inline void TcpServer::OnTcpConnectionClosed(TcpConnection* connection) { MS_TRACE(); MS_DEBUG_DEV("TCP connection closed"); // Remove the TcpConnection from the set. this->connections.erase(connection); // Notify the subclass. UserOnTcpConnectionClosed(connection); // Delete it. delete connection; }