Avoid blocking the poller thread

This commit is contained in:
xiongziliang 2023-12-31 22:10:13 +08:00
parent 7c42ac4d79
commit a106f8dfc0

View File

@ -10,11 +10,8 @@
#include <map> #include <map>
#include <iostream> #include <iostream>
#include "Util/MD5.h"
#include "Util/File.h"
#include "Util/logger.h"
#include "Util/SSLBox.h"
#include "Util/util.h" #include "Util/util.h"
#include "Util/logger.h"
#include "Network/TcpServer.h" #include "Network/TcpServer.h"
#include "Common/config.h" #include "Common/config.h"
#include "Rtsp/RtspSession.h" #include "Rtsp/RtspSession.h"
@ -29,102 +26,102 @@ using namespace mediakit;
static semaphore sem; static semaphore sem;
#if defined(ENABLE_RTPPROXY) #if defined(ENABLE_RTPPROXY)
static bool loadFile(const char *path, const EventPoller::Ptr &poller){ static bool loadFile(const char *path, const EventPoller::Ptr &poller) {
FILE *fp = fopen(path, "rb"); std::shared_ptr<FILE> fp(fopen(path, "rb"), [](FILE *fp) {
sem.post();
if (fp) {
fclose(fp);
}
});
if (!fp) { if (!fp) {
WarnL << "open file failed:" << path; WarnL << "open file failed:" << path;
return false; return false;
} }
uint64_t timeStamp_last = 0;
uint16_t len;
char rtp[0xFFFF];
struct sockaddr_storage addr; struct sockaddr_storage addr;
memset(&addr, 0, sizeof(addr)); memset(&addr, 0, sizeof(addr));
addr.ss_family = AF_INET; addr.ss_family = AF_INET;
auto sock = Socket::createSocket(poller); auto sock = Socket::createSocket(poller);
size_t total_size = 0; auto process = RtpSelector::Instance().getProcess("test", true);
RtpProcess::Ptr process;
uint32_t ssrc = 0;
while (true) {
if (2 != fread(&len, 1, 2, fp)) {
WarnL;
break;
}
len = ntohs(len);
if (len < 12 || len > sizeof(rtp)) {
WarnL << len;
break;
}
if (len != fread(rtp, 1, len, fp)) { uint64_t stamp_last = 0;
WarnL; auto total_size = std::make_shared<size_t>(0);
break; auto do_read = [fp, total_size, sock, addr, process, stamp_last]() mutable -> int {
} uint16_t len;
total_size += len; char rtp[0xFFFF];
uint64_t timeStamp = 0; while (true) {
if (2 != fread(&len, 1, 2, fp.get())) {
if (!process) { WarnL << "Read rtp size failed";
if (!RtpSelector::getSSRC(rtp, len, ssrc)) { // 重新播放
WarnL << "get ssrc from rtp failed:" << len; fseek(fp.get(), 0, SEEK_SET);
return false; return 1;
} }
process = RtpSelector::Instance().getProcess(printSSRC(ssrc), true); len = ntohs(len);
} if (len < 12 || len > sizeof(rtp)) {
if (process) { WarnL << "Invalid rtp size: " << len;
return 0;
}
if (len != fread(rtp, 1, len, fp.get())) {
WarnL << "Read rtp data failed";
return 0;
}
(*total_size) += len;
uint64_t stamp = 0;
try { try {
process->inputRtp(true, sock, rtp, len, (struct sockaddr *)&addr, &timeStamp); process->inputRtp(true, sock, rtp, len, (struct sockaddr *)&addr, &stamp);
} catch (...) { } catch (std::exception &ex) {
RtpSelector::Instance().delProcess(printSSRC(ssrc), process.get()); WarnL << "Input rtp failed: " << ex.what();
throw; return 0;
}
auto diff = stamp - stamp_last;
if (diff < 0 || diff > 500) {
diff = 1;
}
if (diff) {
stamp_last = stamp;
return diff;
} }
} }
};
auto diff = timeStamp - timeStamp_last; poller->doDelayTask(1, [do_read, total_size, process]() mutable {
if (diff > 0 && diff < 500) { auto ret = do_read();
usleep(diff * 1000); if (!ret) {
} else { WarnL << *total_size / 1024 << "KB";
usleep(1 * 1000); RtpSelector::Instance().delProcess("test", process.get());
} }
timeStamp_last = timeStamp; return ret;
} });
WarnL << total_size / 1024 << "KB";
fclose(fp);
return true; return true;
} }
#endif//#if defined(ENABLE_RTPPROXY) #endif // #if defined(ENABLE_RTPPROXY)
int main(int argc,char *argv[]) { int main(int argc, char *argv[]) {
//设置日志 // 设置日志
Logger::Instance().add(std::make_shared<ConsoleChannel>("ConsoleChannel")); Logger::Instance().add(std::make_shared<ConsoleChannel>("ConsoleChannel"));
#if defined(ENABLE_RTPPROXY) #if defined(ENABLE_RTPPROXY)
//启动异步日志线程 // 启动异步日志线程
Logger::Instance().setWriter(std::make_shared<AsyncLogWriter>()); Logger::Instance().setWriter(std::make_shared<AsyncLogWriter>());
loadIniConfig((exeDir() + "config.ini").data()); loadIniConfig((exeDir() + "config.ini").data());
TcpServer::Ptr rtspSrv(new TcpServer()); TcpServer::Ptr rtspSrv(new TcpServer());
TcpServer::Ptr rtmpSrv(new TcpServer()); TcpServer::Ptr rtmpSrv(new TcpServer());
TcpServer::Ptr httpSrv(new TcpServer()); TcpServer::Ptr httpSrv(new TcpServer());
rtspSrv->start<RtspSession>(554);//默认554 rtspSrv->start<RtspSession>(554); // 默认554
rtmpSrv->start<RtmpSession>(1935);//默认1935 rtmpSrv->start<RtmpSession>(1935); // 默认1935
httpSrv->start<HttpSession>(80);//默认80 httpSrv->start<HttpSession>(80); // 默认80
//此处选择是否导出调试文件 // 此处选择是否导出调试文件
// mINI::Instance()[RtpProxy::kDumpDir] = "/Users/xzl/Desktop/"; // mINI::Instance()[RtpProxy::kDumpDir] = "/Users/xzl/Desktop/";
if (argc == 2){ if (argc == 2) {
auto poller = EventPollerPool::Instance().getPoller(); loadFile(argv[1], EventPollerPool::Instance().getPoller());
poller->async_first([poller,argv](){
loadFile(argv[1],poller);
sem.post();
});
sem.wait(); sem.wait();
sleep(1); } else {
ErrorL << "parameter error.";
} }
else
ErrorL << "parameter error.";
#else #else
ErrorL << "please ENABLE_RTPPROXY and then test"; ErrorL << "please ENABLE_RTPPROXY and then test";
#endif//#if defined(ENABLE_RTPPROXY) #endif // #if defined(ENABLE_RTPPROXY)
return 0; return 0;
} }