일단 iocp는 완성한 듯 윈도우에서만 되면 될 듯
This commit is contained in:
4
.vscode/settings.json
vendored
4
.vscode/settings.json
vendored
@@ -79,6 +79,8 @@
|
|||||||
"set": "cpp",
|
"set": "cpp",
|
||||||
"util": "cpp",
|
"util": "cpp",
|
||||||
"expected": "cpp",
|
"expected": "cpp",
|
||||||
"complex": "cpp"
|
"complex": "cpp",
|
||||||
|
"__config": "cpp",
|
||||||
|
"ranges": "cpp"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
cmake_minimum_required(VERSION 3.5)
|
cmake_minimum_required(VERSION 3.5)
|
||||||
|
|
||||||
set(PROJECT_NAME "Chattring")
|
set(PROJECT_NAME "Chattring")
|
||||||
|
set(CMAKE_BUILD_TYPE Debug)
|
||||||
if(NOT CMAKE_BUILD_TYPE)
|
if(NOT CMAKE_BUILD_TYPE)
|
||||||
set(CMAKE_BUILD_TYPE Release CACHE STRING "Choose the type of build." FORCE)
|
set(CMAKE_BUILD_TYPE Release CACHE STRING "Choose the type of build." FORCE)
|
||||||
endif()
|
endif()
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ int main() {
|
|||||||
Chattr::TCPSocket sock;
|
Chattr::TCPSocket sock;
|
||||||
sock.init(AF_INET6);
|
sock.init(AF_INET6);
|
||||||
|
|
||||||
Chattr::Address serveraddr(AF_INET6, "fd8a:5f:3adb:774d:dfae:983f:2e7a:ffba", 9010);
|
Chattr::Address serveraddr(AF_INET6, "::1", 9011);
|
||||||
|
|
||||||
sock.connect(serveraddr);
|
sock.connect(serveraddr);
|
||||||
spdlog::info("Connection established from {}", (std::string)serveraddr);
|
spdlog::info("Connection established from {}", (std::string)serveraddr);
|
||||||
@@ -27,5 +27,5 @@ int main() {
|
|||||||
sock.recv(&packet.serialized, 1500, 0);
|
sock.recv(&packet.serialized, 1500, 0);
|
||||||
packet.convToH();
|
packet.convToH();
|
||||||
|
|
||||||
// sleep(1000000);
|
sleep(1000000);
|
||||||
}
|
}
|
||||||
@@ -23,14 +23,11 @@ int main() {
|
|||||||
Chattr::ThreadPool threadPool(0);
|
Chattr::ThreadPool threadPool(0);
|
||||||
Chattr::IOCP iocp;
|
Chattr::IOCP iocp;
|
||||||
iocp.init(&threadPool, _IOCPClient);
|
iocp.init(&threadPool, _IOCPClient);
|
||||||
// struct Chattr::WSAManager wsaManager;
|
|
||||||
|
|
||||||
Chattr::TCPSocket sock;
|
Chattr::TCPSocket sock;
|
||||||
struct Chattr::Address serveraddr;
|
struct Chattr::Address serveraddr;
|
||||||
Chattr::TCPSocket clientSock;
|
Chattr::TCPSocket clientSock;
|
||||||
struct Chattr::Address clientAddr;
|
struct Chattr::Address clientAddr;
|
||||||
bool enable = true;
|
|
||||||
sock.setsockopt(SOL_SOCKET, SO_KEEPALIVE, (const char*)&enable, sizeof(enable));
|
|
||||||
if (config.ipVersion == 4) {
|
if (config.ipVersion == 4) {
|
||||||
sock.init(AF_INET);
|
sock.init(AF_INET);
|
||||||
serveraddr.set(AF_INET, INADDR_ANY, config.listenPort);
|
serveraddr.set(AF_INET, INADDR_ANY, config.listenPort);
|
||||||
@@ -64,7 +61,7 @@ int main() {
|
|||||||
ptr->wsabuf.len = 1500;
|
ptr->wsabuf.len = 1500;
|
||||||
ptr->IOCPInstance = &iocp;
|
ptr->IOCPInstance = &iocp;
|
||||||
|
|
||||||
iocp.registerSocket(ptr->socket.sock);
|
iocp.registerSocket(ptr);
|
||||||
|
|
||||||
int returnData = iocp.recv(ptr);
|
int returnData = iocp.recv(ptr);
|
||||||
}
|
}
|
||||||
@@ -76,17 +73,8 @@ void _IOCPClient(Chattr::ThreadPool* thread, Chattr::IOCPPASSINDATA* data) {
|
|||||||
int totalRedSize = 0;
|
int totalRedSize = 0;
|
||||||
int redSize = 0;
|
int redSize = 0;
|
||||||
|
|
||||||
// spdlog::info("Receving from: [{}]", (std::string)sock.remoteAddr);
|
if (data->wsabuf.len == data->sendbytes)
|
||||||
|
return;
|
||||||
//while (totalRedSize < packetSize) {
|
|
||||||
// redSize = data->socket.recv(&pack.serialized, 1500 - totalRedSize, 0);
|
|
||||||
//
|
|
||||||
// if (redSize <= 0) {
|
|
||||||
// // spdlog::info("Client disconnected. [{}]", (std::string)sock.remoteAddr);
|
|
||||||
// return;
|
|
||||||
// }
|
|
||||||
// totalRedSize += redSize;
|
|
||||||
//}
|
|
||||||
|
|
||||||
memcpy(pack.serialized, data->wsabuf.buf, data->wsabuf.len);
|
memcpy(pack.serialized, data->wsabuf.buf, data->wsabuf.len);
|
||||||
|
|
||||||
@@ -94,7 +82,6 @@ void _IOCPClient(Chattr::ThreadPool* thread, Chattr::IOCPPASSINDATA* data) {
|
|||||||
bool packetError = false;
|
bool packetError = false;
|
||||||
Chattr::DataPostPacket dataPostPacket;
|
Chattr::DataPostPacket dataPostPacket;
|
||||||
|
|
||||||
|
|
||||||
Chattr::ResponsePacket packet;
|
Chattr::ResponsePacket packet;
|
||||||
std::queue<Chattr::ResponsePacket> responsePackets;
|
std::queue<Chattr::ResponsePacket> responsePackets;
|
||||||
|
|
||||||
@@ -166,7 +153,18 @@ void _IOCPClient(Chattr::ThreadPool* thread, Chattr::IOCPPASSINDATA* data) {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
int returnData = data->IOCPInstance->recv(data);
|
packet.__data.packetType = Chattr::PacketType::PACKET_RESPONSE;
|
||||||
|
packet.__data.requestType = Chattr::RequestType::DATA;
|
||||||
|
packet.__data.dataType = Chattr::DataType::TEXT;
|
||||||
|
packet.__data.packetLength = sizeof(Chattr::ResponseStatusCode);
|
||||||
|
packet.__data.responseStatusCode = Chattr::ResponseStatusCode::BAD_REQUEST;
|
||||||
|
packet.convToN();
|
||||||
|
|
||||||
|
memcpy(data->wsabuf.buf, packet.serialized, data->wsabuf.len);
|
||||||
|
|
||||||
|
int returnData = data->IOCPInstance->send(data, 0);
|
||||||
|
|
||||||
|
// int returnData = data->IOCPInstance->recv(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
void _TCPSendClient(Chattr::ThreadPool* thread, Chattr::TCPSocket sock, std::queue<Chattr::ResponsePacket> packets) {
|
void _TCPSendClient(Chattr::ThreadPool* thread, Chattr::TCPSocket sock, std::queue<Chattr::ResponsePacket> packets) {
|
||||||
|
|||||||
@@ -5,13 +5,22 @@
|
|||||||
|
|
||||||
namespace Chattr {
|
namespace Chattr {
|
||||||
|
|
||||||
void IOCP::registerSocket(SOCKET sock) {
|
void IOCP::registerSocket(Chattr::IOCPPASSINDATA* data) {
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
HANDLE returnData = ::CreateIoCompletionPort((HANDLE)sock, completionPort_, sock, 0);
|
HANDLE returnData = ::CreateIoCompletionPort((HANDLE)data->socket.sock, completionPort_, data->socket.sock, 0);
|
||||||
if (returnData == 0)
|
if (returnData == 0)
|
||||||
completionPort_ = returnData;
|
completionPort_ = returnData;
|
||||||
#elif __linux__
|
#elif __linux__
|
||||||
|
// int flags = ::fcntl(data->socket.sock, F_GETFL);
|
||||||
|
// flags |= O_NONBLOCK;
|
||||||
|
// fcntl(data->socket.sock, F_SETFL, flags);
|
||||||
|
|
||||||
|
struct epoll_event ev;
|
||||||
|
ev.events = EPOLLIN | EPOLLONESHOT;
|
||||||
|
ev.data.ptr = data;
|
||||||
|
int rc = epoll_ctl(epollfd_, EPOLL_CTL_ADD, data->socket.sock, &ev);
|
||||||
|
if (rc < 0)
|
||||||
|
log::critical("epoll_ctl()");
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -20,7 +29,10 @@ int IOCP::recv(Chattr::IOCPPASSINDATA* data) {
|
|||||||
DWORD recvbytes = 0, flags = 0;
|
DWORD recvbytes = 0, flags = 0;
|
||||||
return ::WSARecv(data->socket.sock, &data->wsabuf, 1, &recvbytes, &flags, &data->overlapped, NULL);
|
return ::WSARecv(data->socket.sock, &data->wsabuf, 1, &recvbytes, &flags, &data->overlapped, NULL);
|
||||||
#elif __linux__
|
#elif __linux__
|
||||||
return -1;
|
struct epoll_event ev;
|
||||||
|
ev.events = EPOLLIN | EPOLLONESHOT;
|
||||||
|
ev.data.ptr = data;
|
||||||
|
return ::epoll_ctl(epollfd_, EPOLL_CTL_MOD, data->socket.sock, &ev);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -29,7 +41,10 @@ int IOCP::send(Chattr::IOCPPASSINDATA* data, int __flags) {
|
|||||||
DWORD sendbytes = 0;
|
DWORD sendbytes = 0;
|
||||||
return ::WSASend(data->socket.sock, &data->wsabuf, 1, &sendbytes, __flags, &data->overlapped, NULL);
|
return ::WSASend(data->socket.sock, &data->wsabuf, 1, &sendbytes, __flags, &data->overlapped, NULL);
|
||||||
#elif __linux__
|
#elif __linux__
|
||||||
return -1;
|
struct epoll_event ev;
|
||||||
|
ev.events = EPOLLOUT | EPOLLONESHOT;
|
||||||
|
ev.data.ptr = data;
|
||||||
|
return ::epoll_ctl(epollfd_, EPOLL_CTL_MOD, data->socket.sock, &ev);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -4,10 +4,12 @@ namespace Chattr {
|
|||||||
|
|
||||||
Thread::Thread(Thread&& other) noexcept {
|
Thread::Thread(Thread&& other) noexcept {
|
||||||
other.detach();
|
other.detach();
|
||||||
|
memcpy(this, &other, sizeof(Thread));
|
||||||
}
|
}
|
||||||
|
|
||||||
Thread& Thread::operator=(Thread&& other) noexcept {
|
Thread& Thread::operator=(Thread&& other) noexcept {
|
||||||
other.detach();
|
other.detach();
|
||||||
|
memcpy(this, &other, sizeof(Thread));
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -16,8 +18,6 @@ Thread::~Thread() {
|
|||||||
spdlog::critical("There is not joined thread");
|
spdlog::critical("There is not joined thread");
|
||||||
std::exit(EXIT_FAILURE);
|
std::exit(EXIT_FAILURE);
|
||||||
}
|
}
|
||||||
if (returnValuePtr != nullptr)
|
|
||||||
delete returnValuePtr;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Thread::detach() {
|
void Thread::detach() {
|
||||||
|
|||||||
@@ -63,11 +63,16 @@ void* ThreadPool::Worker() {
|
|||||||
#elif __linux__
|
#elif __linux__
|
||||||
pthread_t pid = pthread_self();
|
pthread_t pid = pthread_self();
|
||||||
#endif
|
#endif
|
||||||
|
spdlog::debug("ThreadPool Worker : {} up", pid);
|
||||||
while (!terminate_) {
|
while (!terminate_) {
|
||||||
std::unique_lock<std::mutex> lock(jobQueueMutex);
|
std::unique_lock<std::mutex> lock(jobQueueMutex);
|
||||||
jobQueueCV_.wait(lock, [this]() { return !this->jobs_.empty() || terminate_; });
|
jobQueueCV_.wait(lock, [this]() { return !this->jobs_.empty() || terminate_; });
|
||||||
if (this->jobs_.empty())
|
if (this->jobs_.empty() || terminate_) {
|
||||||
|
jobs_ = std::queue<std::packaged_task<void()>>();
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
|
if (this->jobs_.empty())
|
||||||
|
continue;
|
||||||
|
|
||||||
auto job = std::move(jobs_.front());
|
auto job = std::move(jobs_.front());
|
||||||
jobs_.pop();
|
jobs_.pop();
|
||||||
@@ -76,6 +81,7 @@ void* ThreadPool::Worker() {
|
|||||||
spdlog::debug("ThreadPool Worker : {} Executing a job", pid);
|
spdlog::debug("ThreadPool Worker : {} Executing a job", pid);
|
||||||
job();
|
job();
|
||||||
}
|
}
|
||||||
|
spdlog::debug("ThreadPool Worker : {} down", pid);
|
||||||
|
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,6 +4,7 @@
|
|||||||
#include "Socket/TCPSocket.hpp"
|
#include "Socket/TCPSocket.hpp"
|
||||||
#include "Socket/Log.hpp"
|
#include "Socket/Log.hpp"
|
||||||
#include <functional>
|
#include <functional>
|
||||||
|
#include "precomp.hpp"
|
||||||
|
|
||||||
#ifndef _WIN32
|
#ifndef _WIN32
|
||||||
|
|
||||||
@@ -52,21 +53,81 @@ public:
|
|||||||
threadPool->enqueueJob(iocpWather, completionPort_, callback);
|
threadPool->enqueueJob(iocpWather, completionPort_, callback);
|
||||||
};
|
};
|
||||||
#elif __linux__
|
#elif __linux__
|
||||||
static void iocpWather(ThreadPool* threadPool, std::function<void(ThreadPool*, IOCPPASSINDATA*)> callback) {
|
static void socketReader(ThreadPool* threadPool, epoll_event event, int epollfd, std::function<void(ThreadPool*, IOCPPASSINDATA*)> callback) {
|
||||||
// DWORD tid = GetCurrentThreadId();
|
IOCPPASSINDATA* iocpData = (IOCPPASSINDATA*)event.data.ptr;
|
||||||
// spdlog::debug("Waiting IO to complete on TID: {}.", tid);
|
|
||||||
// IOCPPASSINDATA* data;
|
pthread_t tid = pthread_self();
|
||||||
// SOCKET sock;
|
|
||||||
// DWORD cbTransfrred;
|
if (iocpData == nullptr) {
|
||||||
// int retVal = GetQueuedCompletionStatus(completionPort_, &cbTransfrred, (PULONG_PTR)&sock, (LPOVERLAPPED*)&data, INFINITE);
|
spdlog::error("invalid call on {}", tid);
|
||||||
// if (retVal == 0 || cbTransfrred == 0) {
|
return;
|
||||||
// spdlog::info("Client disconnected. [{}]", (std::string)(data->socket.remoteAddr));
|
}
|
||||||
// delete data;
|
|
||||||
// threadPool->enqueueJob(iocpWather, completionPort_, callback);
|
spdlog::info("reading on tid: {} [{}]", tid, (std::string)iocpData->socket.remoteAddr);
|
||||||
// return;
|
|
||||||
// }
|
int redSize = 0;
|
||||||
// threadPool->enqueueJob(callback, data);
|
int packetSize = iocpData->wsabuf.len;
|
||||||
// threadPool->enqueueJob(iocpWather, completionPort_, callback);
|
int totalRedSize = 0;
|
||||||
|
|
||||||
|
while (totalRedSize < packetSize) {
|
||||||
|
redSize = iocpData->socket.recv(iocpData->buf, packetSize - totalRedSize, 0);
|
||||||
|
|
||||||
|
if (redSize <= 0) {
|
||||||
|
spdlog::info("Client disconnected. [{}]", (std::string)iocpData->socket.remoteAddr);
|
||||||
|
::epoll_ctl(epollfd, EPOLL_CTL_DEL, iocpData->socket.sock, NULL);
|
||||||
|
delete iocpData;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
totalRedSize += redSize;
|
||||||
|
}
|
||||||
|
iocpData->sendbytes = packetSize - totalRedSize;
|
||||||
|
iocpData->recvbytes = totalRedSize;
|
||||||
|
threadPool->enqueueJob(callback, iocpData);
|
||||||
|
};
|
||||||
|
static void socketWriter(ThreadPool* threadPool, epoll_event event, int epollfd, std::function<void(ThreadPool*, IOCPPASSINDATA*)> callback) {
|
||||||
|
IOCPPASSINDATA* iocpData = (IOCPPASSINDATA*)event.data.ptr;
|
||||||
|
|
||||||
|
int packetSize = iocpData->wsabuf.len;
|
||||||
|
int totalSentSize = 0;
|
||||||
|
int sentSize = 0;
|
||||||
|
|
||||||
|
spdlog::info("Sending to: [{}]", (std::string)iocpData->socket.remoteAddr);
|
||||||
|
|
||||||
|
while (totalSentSize < packetSize) {
|
||||||
|
sentSize = iocpData->socket.send(iocpData->buf, packetSize - totalSentSize, 0);
|
||||||
|
|
||||||
|
if (sentSize <= 0) {
|
||||||
|
spdlog::info("Client disconnected. [{}]", (std::string)iocpData->socket.remoteAddr);
|
||||||
|
::epoll_ctl(epollfd, EPOLL_CTL_DEL, iocpData->socket.sock, NULL);
|
||||||
|
delete iocpData;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
totalSentSize += sentSize;
|
||||||
|
}
|
||||||
|
iocpData->recvbytes = packetSize - totalSentSize;
|
||||||
|
iocpData->sendbytes = totalSentSize;
|
||||||
|
threadPool->enqueueJob(callback, iocpData);
|
||||||
|
};
|
||||||
|
static void iocpWatcher(ThreadPool* threadPool, int epollfd, std::function<void(ThreadPool*, IOCPPASSINDATA*)> callback) {
|
||||||
|
struct epoll_event events[FD_SETSIZE];
|
||||||
|
pthread_t tid = pthread_self();
|
||||||
|
|
||||||
|
spdlog::debug("epoll waiting on {}", tid);
|
||||||
|
int nready = ::epoll_wait(epollfd, events, FD_SETSIZE, -1);
|
||||||
|
|
||||||
|
for (int i=0; i<nready; i++) {
|
||||||
|
struct epoll_event current_event = events[i];
|
||||||
|
|
||||||
|
if (current_event.events & EPOLLIN) {
|
||||||
|
std::function<void(ThreadPool*, IOCPPASSINDATA*)> task(callback);
|
||||||
|
threadPool->enqueueJob(socketReader, current_event, epollfd, task);
|
||||||
|
}
|
||||||
|
else if (current_event.events & EPOLLOUT) {
|
||||||
|
std::function<void(ThreadPool*, IOCPPASSINDATA*)> task(callback);
|
||||||
|
threadPool->enqueueJob(socketWriter, current_event, epollfd, task);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
threadPool->enqueueJob(iocpWatcher, epollfd, callback);
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@@ -78,26 +139,30 @@ public:
|
|||||||
if (completionPort_ == NULL)
|
if (completionPort_ == NULL)
|
||||||
log::critical("CreateIoCompletionPort()");
|
log::critical("CreateIoCompletionPort()");
|
||||||
#elif __linux__
|
#elif __linux__
|
||||||
|
epollfd_ = ::epoll_create(1);
|
||||||
#endif
|
#endif
|
||||||
auto boundFunc = [callback = std::move(callback)](ThreadPool* __IOCPThread, IOCPPASSINDATA* data) mutable {
|
auto boundFunc = [callback = std::move(callback)](ThreadPool* __IOCPThread, IOCPPASSINDATA* data) mutable {
|
||||||
callback(__IOCPThread, data);
|
callback(__IOCPThread, data);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#ifdef _WIN32
|
||||||
int tCount = __IOCPThread->threadCount;
|
int tCount = __IOCPThread->threadCount;
|
||||||
|
|
||||||
spdlog::info("Resizing threadpool size to: {}", tCount * 2);
|
spdlog::info("Resizing threadpool size to: {}", tCount * 2);
|
||||||
|
|
||||||
__IOCPThread->respawnWorker(tCount * 2);
|
__IOCPThread->respawnWorker(tCount * 2);
|
||||||
|
|
||||||
spdlog::info("Set IOCP Worker count to: {}", tCount);
|
|
||||||
for (int i = 0; i < tCount; i++) {
|
for (int i = 0; i < tCount; i++) {
|
||||||
std::function<void(ThreadPool*, IOCPPASSINDATA*)> task(boundFunc);
|
std::function<void(ThreadPool*, IOCPPASSINDATA*)> task(boundFunc);
|
||||||
__IOCPThread->enqueueJob(iocpWather, task);
|
__IOCPThread->enqueueJob(iocpWather, completionPort_, task);
|
||||||
}
|
}
|
||||||
|
#elif __linux__
|
||||||
|
spdlog::info("Spawning 1 Epoll Waiter...");
|
||||||
|
__IOCPThread->enqueueJob(iocpWatcher, epollfd_, boundFunc);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
void registerSocket(SOCKET sock);
|
void registerSocket(Chattr::IOCPPASSINDATA* data);
|
||||||
|
|
||||||
int recv(Chattr::IOCPPASSINDATA* data);
|
int recv(Chattr::IOCPPASSINDATA* data);
|
||||||
int send(Chattr::IOCPPASSINDATA* data, int __flags);
|
int send(Chattr::IOCPPASSINDATA* data, int __flags);
|
||||||
@@ -109,7 +174,7 @@ private:
|
|||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
HANDLE completionPort_ = INVALID_HANDLE_VALUE;
|
HANDLE completionPort_ = INVALID_HANDLE_VALUE;
|
||||||
#elif __linux__
|
#elif __linux__
|
||||||
int epollfd = -1;
|
int epollfd_ = -1;
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -9,6 +9,9 @@
|
|||||||
#endif
|
#endif
|
||||||
#include <functional>
|
#include <functional>
|
||||||
#include <future>
|
#include <future>
|
||||||
|
#include <typeinfo>
|
||||||
|
|
||||||
|
#include "Socket/Log.hpp"
|
||||||
|
|
||||||
namespace Chattr {
|
namespace Chattr {
|
||||||
|
|
||||||
@@ -40,7 +43,7 @@ public:
|
|||||||
(!std::is_void_v<std::invoke_result_t<_Callable, _Args...>>)
|
(!std::is_void_v<std::invoke_result_t<_Callable, _Args...>>)
|
||||||
Thread(_Callable&& __f, _Args&&... __args) {
|
Thread(_Callable&& __f, _Args&&... __args) {
|
||||||
auto boundFunc = [this, __f, ... __args = std::move(__args)]() mutable {
|
auto boundFunc = [this, __f, ... __args = std::move(__args)]() mutable {
|
||||||
returnValuePtr = new std::invoke_result_t<_Callable, _Args...>(__f(std::move(__args)...));
|
returnValuePtr = std::make_shared<std::invoke_result_t<_Callable, _Args...>>(__f(std::move(__args)...));
|
||||||
};
|
};
|
||||||
std::packaged_task<void()>* funcPtr = new std::packaged_task<void()>(std::move(boundFunc));
|
std::packaged_task<void()>* funcPtr = new std::packaged_task<void()>(std::move(boundFunc));
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
@@ -60,8 +63,10 @@ public:
|
|||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
handle_ = (HANDLE)_beginthreadex(nullptr, 0, thread_func, funcPtr, 0, nullptr);
|
handle_ = (HANDLE)_beginthreadex(nullptr, 0, thread_func, funcPtr, 0, nullptr);
|
||||||
#elif __linux__
|
#elif __linux__
|
||||||
pthread_create(&handle_, NULL, thread_func, funcPtr);
|
int rc = pthread_create(&handle_, NULL, thread_func, funcPtr);
|
||||||
#endif
|
#endif
|
||||||
|
if (handle_ <= 0 || rc != 0)
|
||||||
|
log::critical("pthread_create()");
|
||||||
}
|
}
|
||||||
~Thread();
|
~Thread();
|
||||||
|
|
||||||
@@ -69,7 +74,7 @@ public:
|
|||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
WaitForSingleObject(handle_, INFINITE);
|
WaitForSingleObject(handle_, INFINITE);
|
||||||
#elif __linux__
|
#elif __linux__
|
||||||
pthread_join(handle_, &returnValuePtr);
|
pthread_join(handle_, nullptr);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
template<typename _RetType>
|
template<typename _RetType>
|
||||||
@@ -78,7 +83,7 @@ public:
|
|||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
WaitForSingleObject(handle_, INFINITE);
|
WaitForSingleObject(handle_, INFINITE);
|
||||||
#elif __linux__
|
#elif __linux__
|
||||||
pthread_join(handle_, &returnValuePtr);
|
pthread_join(handle_, nullptr);
|
||||||
#endif
|
#endif
|
||||||
return *static_cast<_RetType *>(returnValuePtr);
|
return *static_cast<_RetType *>(returnValuePtr);
|
||||||
}
|
}
|
||||||
@@ -90,7 +95,7 @@ private:
|
|||||||
#elif __linux__
|
#elif __linux__
|
||||||
pthread_t handle_;
|
pthread_t handle_;
|
||||||
#endif
|
#endif
|
||||||
void* returnValuePtr = nullptr;
|
std::shared_ptr<void> returnValuePtr;
|
||||||
bool detached = false;
|
bool detached = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -10,7 +10,9 @@
|
|||||||
#elif __linux__
|
#elif __linux__
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
#include <sys/socket.h>
|
#include <sys/socket.h>
|
||||||
|
#include <sys/epoll.h>
|
||||||
#include <arpa/inet.h>
|
#include <arpa/inet.h>
|
||||||
|
#include <fcntl.h>
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <netinet/in.h>
|
#include <netinet/in.h>
|
||||||
#define SOCKET int
|
#define SOCKET int
|
||||||
|
|||||||
Reference in New Issue
Block a user