/** * The MIT License (MIT) * * Copyright (c) 2013-2020 Runner365 * * 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. */ #include "srt_handle.hpp" #include "time_help.h" #include #include #include #include #include #include #include #include #include #include #include static bool MONITOR_STATICS_ENABLE = false; static long long MONITOR_TIMEOUT = 5000; const unsigned int DEF_DATA_SIZE = 188*7; const long long CHECK_ALIVE_INTERVAL = 5*1000; const long long CHECK_ALIVE_TIMEOUT = 5*1000; long long srt_now_ms = 0; srt_handle::srt_handle(int pollid):_handle_pollid(pollid) ,_last_timestamp(0) ,_last_check_alive_ts(0) { } srt_handle::~srt_handle() { } void srt_handle::debug_statics(SRTSOCKET srtsocket, const std::string& streamid) { SRT_TRACEBSTATS mon; srt_bstats(srtsocket, &mon, 1); std::ostringstream output; long long now_ul = now_ms(); if (!MONITOR_STATICS_ENABLE) { return; } if (_last_timestamp == 0) { _last_timestamp = now_ul; return; } if ((now_ul - _last_timestamp) < MONITOR_TIMEOUT) { return; } _last_timestamp = now_ul; output << "======= SRT STATS: sid=" << streamid << std::endl; output << "PACKETS SENT: " << std::setw(11) << mon.pktSent << " RECEIVED: " << std::setw(11) << mon.pktRecv << std::endl; output << "LOST PKT SENT: " << std::setw(11) << mon.pktSndLoss << " RECEIVED: " << std::setw(11) << mon.pktRcvLoss << std::endl; output << "REXMIT SENT: " << std::setw(11) << mon.pktRetrans << " RECEIVED: " << std::setw(11) << mon.pktRcvRetrans << std::endl; output << "DROP PKT SENT: " << std::setw(11) << mon.pktSndDrop << " RECEIVED: " << std::setw(11) << mon.pktRcvDrop << std::endl; output << "RATE SENDING: " << std::setw(11) << mon.mbpsSendRate << " RECEIVING: " << std::setw(11) << mon.mbpsRecvRate << std::endl; output << "BELATED RECEIVED: " << std::setw(11) << mon.pktRcvBelated << " AVG TIME: " << std::setw(11) << mon.pktRcvAvgBelatedTime << std::endl; output << "REORDER DISTANCE: " << std::setw(11) << mon.pktReorderDistance << std::endl; output << "WINDOW FLOW: " << std::setw(11) << mon.pktFlowWindow << " CONGESTION: " << std::setw(11) << mon.pktCongestionWindow << " FLIGHT: " << std::setw(11) << mon.pktFlightSize << std::endl; output << "LINK RTT: " << std::setw(9) << mon.msRTT << "ms BANDWIDTH: " << std::setw(7) << mon.mbpsBandwidth << "Mb/s " << std::endl; output << "BUFFERLEFT: SND: " << std::setw(11) << mon.byteAvailSndBuf << " RCV: " << std::setw(11) << mon.byteAvailRcvBuf << std::endl; srs_trace("\r\n%s", output.str().c_str()); return; } void srt_handle::add_new_puller(SRT_CONN_PTR conn_ptr, std::string stream_id) { _conn_map.insert(std::make_pair(conn_ptr->get_conn(), conn_ptr)); auto iter = _streamid_map.find(stream_id); if (iter == _streamid_map.end()) { std::unordered_map srtsocket_map; srtsocket_map.insert(std::make_pair(conn_ptr->get_conn(), conn_ptr)); _streamid_map.insert(std::make_pair(stream_id, srtsocket_map)); srs_trace("add new puller fd:%d, streamid:%s", conn_ptr->get_conn(), stream_id.c_str()); } else { iter->second.insert(std::make_pair(conn_ptr->get_conn(), conn_ptr)); srs_trace("add new puller fd:%d, streamid:%s, size:%d", conn_ptr->get_conn(), stream_id.c_str(), iter->second.size()); } return; } void srt_handle::close_pull_conn(SRTSOCKET srtsocket, std::string stream_id) { srs_warn("close_pull_conn read_fd=%d, streamid=%s", srtsocket, stream_id.c_str()); srt_epoll_remove_usock(_handle_pollid, srtsocket); auto streamid_iter = _streamid_map.find(stream_id); if (streamid_iter != _streamid_map.end()) { auto srtsocket_map = streamid_iter->second; if (srtsocket_map.size() == 0) { _streamid_map.erase(stream_id); } else if (srtsocket_map.size() == 1) { srtsocket_map.erase(srtsocket); _streamid_map.erase(stream_id); } else { srtsocket_map.erase(srtsocket); } } else { assert(0); } auto conn_iter = _conn_map.find(srtsocket); if (conn_iter != _conn_map.end()) { _conn_map.erase(conn_iter); return; } else { assert(0); } return; } SRT_CONN_PTR srt_handle::get_srt_conn(SRTSOCKET conn_srt_socket) { SRT_CONN_PTR ret_conn; auto iter = _conn_map.find(conn_srt_socket); if (iter == _conn_map.end()) { return ret_conn; } ret_conn = iter->second; return ret_conn; } void srt_handle::add_newconn(SRT_CONN_PTR conn_ptr, int events) { int val_i; int opt_len = sizeof(int); int64_t val_i64; int opt64_len = sizeof(int64_t); srt_getsockopt(conn_ptr->get_conn(), 0, SRTO_LATENCY, &val_i, &opt_len); srs_trace("srto SRTO_LATENCY=%d", val_i); srt_getsockopt(conn_ptr->get_conn(), 0, SRTO_PEERLATENCY, &val_i, &opt_len); srs_trace("srto SRTO_PEERLATENCY=%d", val_i); srt_getsockopt(conn_ptr->get_conn(), 0, SRTO_RCVLATENCY, &val_i, &opt_len); srs_trace("srto SRTO_RCVLATENCY=%d", val_i); srt_getsockopt(conn_ptr->get_conn(), 0, SRTO_SNDBUF, &val_i, &opt_len); srs_trace("srto SRTO_SNDBUF=%d", val_i); srt_getsockopt(conn_ptr->get_conn(), 0, SRTO_RCVBUF, &val_i, &opt_len); srs_trace("srto SRTO_RCVBUF=%d", val_i); srt_getsockopt(conn_ptr->get_conn(), 0, SRTO_MAXBW, &val_i64, &opt64_len); srs_trace("srto SRTO_MAXBW=%d", val_i64); srs_trace("srt mix_correct is %s.", _srs_config->get_srt_mix_correct() ? "enable" : "disable"); srs_trace("srt h264 sei filter is %s.", _srs_config->get_srt_sei_filter() ? "enable" : "disable"); if (conn_ptr->get_mode() == PULL_SRT_MODE) { add_new_puller(conn_ptr, conn_ptr->get_subpath()); } else { if(add_new_pusher(conn_ptr) == false) { srs_trace("push connection is repeated and rejected, fd:%d, streamid:%s", conn_ptr->get_conn(), conn_ptr->get_streamid().c_str()); conn_ptr->close(); return; } } srs_trace("new conn added fd:%d, event:0x%08x", conn_ptr->get_conn(), events); int ret = srt_epoll_add_usock(_handle_pollid, conn_ptr->get_conn(), &events); if (ret < 0) { srs_error("srt handle run add epoll error:%d", ret); return; } return; } void srt_handle::handle_push_data(SRT_SOCKSTATUS status, const std::string& subpath, SRTSOCKET conn_fd) { SRT_CONN_PTR srt_conn_ptr; unsigned char data[DEF_DATA_SIZE]; int ret; srt_conn_ptr = get_srt_conn(conn_fd); if (!srt_conn_ptr) { srs_error("handle_push_data fd:%d fail to find srt connection.", conn_fd); return; } if (status != SRTS_CONNECTED) { srs_error("handle_push_data error status:%d fd:%d", status, conn_fd); close_push_conn(conn_fd); return; } ret = srt_conn_ptr->read(data, DEF_DATA_SIZE); if (ret <= 0) { srs_error("handle_push_data srt connect read error:%d, fd:%d", ret, conn_fd); close_push_conn(conn_fd); return; } srt_conn_ptr->update_timestamp(srt_now_ms); srt2rtmp::get_instance()->insert_data_message(data, ret, subpath); //send data to subscriber(players) //streamid, play map auto streamid_iter = _streamid_map.find(subpath); if (streamid_iter == _streamid_map.end()) {//no puler srs_info("receive data size(%d) from pusher(%d) but no puller", ret, conn_fd); return; } srs_info("receive data size(%d) from pusher(%d) to pullers, count:%d", ret, conn_fd, streamid_iter->second.size()); for (auto puller_iter = streamid_iter->second.begin(); puller_iter != streamid_iter->second.end(); puller_iter++) { auto player_conn = puller_iter->second; if (!player_conn) { srs_error("handle_push_data get srt connect error from fd:%d", puller_iter->first); continue; } int write_ret = player_conn->write(data, ret); srs_info("send data size(%d) to puller fd:%d", write_ret, puller_iter->first); if (write_ret > 0) { puller_iter->second->update_timestamp(srt_now_ms); } } return; } void srt_handle::check_alive() { long long diff_t; std::list conn_list; if (_last_check_alive_ts == 0) { _last_check_alive_ts = srt_now_ms; return; } diff_t = srt_now_ms - _last_check_alive_ts; if (diff_t < CHECK_ALIVE_INTERVAL) { return; } for (auto conn_iter = _conn_map.begin(); conn_iter != _conn_map.end(); conn_iter++) { long long timeout = srt_now_ms - conn_iter->second->get_last_ts(); if (timeout > CHECK_ALIVE_TIMEOUT) { conn_list.push_back(conn_iter->second); } } for (auto del_iter = conn_list.begin(); del_iter != conn_list.end(); del_iter++) { SRT_CONN_PTR conn_ptr = *del_iter; if (conn_ptr->get_mode() == PUSH_SRT_MODE) { srs_warn("check alive close pull connection fd:%d, streamid:%s", conn_ptr->get_conn(), conn_ptr->get_subpath().c_str()); close_push_conn(conn_ptr->get_conn()); } else if (conn_ptr->get_mode() == PULL_SRT_MODE) { srs_warn("check alive close pull connection fd:%d, streamid:%s", conn_ptr->get_conn(), conn_ptr->get_subpath().c_str()); close_pull_conn(conn_ptr->get_conn(), conn_ptr->get_subpath()); } else { srs_error("check_alive get unkown srt mode:%d, fd:%d", conn_ptr->get_mode(), conn_ptr->get_conn()); assert(0); } } } void srt_handle::close_push_conn(SRTSOCKET srtsocket) { auto iter = _conn_map.find(srtsocket); if (iter != _conn_map.end()) { SRT_CONN_PTR conn_ptr = iter->second; auto push_iter = _push_conn_map.find(conn_ptr->get_subpath()); if (push_iter != _push_conn_map.end()) { _push_conn_map.erase(push_iter); } _conn_map.erase(iter); srt2rtmp::get_instance()->insert_ctrl_message(SRT_MSG_CLOSE_TYPE, conn_ptr->get_subpath()); conn_ptr->close(); } srt_epoll_remove_usock(_handle_pollid, srtsocket); return; } bool srt_handle::add_new_pusher(SRT_CONN_PTR conn_ptr) { auto push_iter = _push_conn_map.find(conn_ptr->get_subpath()); if (push_iter != _push_conn_map.end()) { return false; } _push_conn_map.insert(std::make_pair(conn_ptr->get_subpath(), conn_ptr)); _conn_map.insert(std::make_pair(conn_ptr->get_conn(), conn_ptr)); srs_trace("srt_handle add new pusher streamid:%s, subpath:%s", conn_ptr->get_streamid().c_str(), conn_ptr->get_subpath().c_str()); return true; } void srt_handle::handle_pull_data(SRT_SOCKSTATUS status, const std::string& subpath, SRTSOCKET conn_fd) { srs_info("handle_pull_data status:%d, subpath:%s, fd:%d", status, subpath.c_str(), conn_fd); auto conn_ptr = get_srt_conn(conn_fd); if (!conn_ptr) { srs_error("handle_pull_data fail to find fd(%d)", conn_fd); assert(0); return; } conn_ptr->update_timestamp(srt_now_ms); } void srt_handle::handle_srt_socket(SRT_SOCKSTATUS status, SRTSOCKET conn_fd) { std::string subpath; int mode; auto conn_ptr = get_srt_conn(conn_fd); if (!conn_ptr) { if (status != SRTS_CLOSED) { srs_error("handle_srt_socket find srt connection error, fd:%d, status:%d", conn_fd, status); } return; } bool ret = get_streamid_info(conn_ptr->get_streamid(), mode, subpath); if (!ret) { conn_ptr->close(); conn_ptr = nullptr; return; } if (mode == PUSH_SRT_MODE) { switch (status) { case SRTS_CONNECTED: { handle_push_data(status, subpath, conn_fd); break; } case SRTS_BROKEN: { srs_warn("srt push disconnected event fd:%d, streamid:%s", conn_fd, conn_ptr->get_streamid().c_str()); close_push_conn(conn_fd); break; } default: srs_error("push mode unkown status:%d, fd:%d", status, conn_fd); break; } } else if (mode == PULL_SRT_MODE) { switch (status) { case SRTS_CONNECTED: { handle_pull_data(status, subpath, conn_fd); break; } case SRTS_BROKEN: { srs_warn("srt pull disconnected fd:%d, streamid:%s", conn_fd, conn_ptr->get_streamid().c_str()); close_pull_conn(conn_fd, subpath); break; } default: srs_error("pull mode unkown status:%d, fd:%d", status, conn_fd); break; } } else { assert(0); } return; }