임시저장(빌드 안됨) 졸려서 자러갈거야
This commit is contained in:
		| @@ -1,5 +1,6 @@ | ||||
| #pragma once | ||||
| #include "Socket/TCPSocket.hpp" | ||||
| #include "Utils/Snowflake.hpp" | ||||
| #include <vector> | ||||
|  | ||||
| namespace Chattr { | ||||
| @@ -17,5 +18,6 @@ public: | ||||
| private: | ||||
| 	struct std::vector<Chattr::TCPSocket> tcpSock_; | ||||
| 	struct std::vector<Chattr::Socket> udpSock_; | ||||
| 	struct Snowflake sessId_; | ||||
| }; | ||||
| } | ||||
| @@ -12,39 +12,53 @@ | ||||
|  | ||||
| namespace Chattr { | ||||
|  | ||||
| template<typename F, typename... Args> | ||||
| concept ReturnsVoidPtr = std::is_same_v< | ||||
|     std::invoke_result_t<F, Args...>, | ||||
|     void* | ||||
| >; | ||||
|  | ||||
| class Thread { | ||||
| public: | ||||
| #ifdef _WIN32 | ||||
|     static unsigned __stdcall thread_func(LPVOID param) { | ||||
|         auto task(static_cast<std::packaged_task<void()>*>(param)); | ||||
|         // auto task = static_cast<std::function<void *()>*>(param); | ||||
|         std::unique_ptr<std::function<void*()>> task(static_cast<std::function<void* ()>*>(param)); | ||||
|         (*task)(); | ||||
|         delete task; | ||||
|         return 0; | ||||
|     } | ||||
| #elif __linux__ | ||||
|     static void* thread_func(void *param) { | ||||
|         auto task(static_cast<std::packaged_task<void()>*>(param)); | ||||
|         auto task(static_cast<std::packaged_task<void*()>*>(param)); | ||||
|         (*task)(); | ||||
|         delete task; | ||||
|         return 0; | ||||
|     } | ||||
| #endif | ||||
|     Thread(Thread&&) noexcept; | ||||
|     Thread& operator=(Thread&&) noexcept; | ||||
|  | ||||
|     Thread(const Thread&) = delete; | ||||
|     Thread& operator=(Thread&) = delete; | ||||
|  | ||||
|     template<typename _Callable, typename... _Args> | ||||
|     requires (!std::is_same_v<std::decay_t<_Callable>, Thread>) //복사 생성하면 안 되므로 | ||||
|         requires ReturnsVoidPtr<_Callable, _Args...> && (!std::is_same_v<std::decay_t<_Callable>, Thread>) | ||||
|     Thread(_Callable&& __f, _Args&&... __args) { | ||||
|         auto boundFunc = [__f = std::move(__f), ... __args = std::move(__args)]() mutable { | ||||
|             __f(std::move(__args)...); | ||||
|         auto boundFunc = [this, __f = std::move(__f), ... __args = std::move(__args)]() mutable -> void* { | ||||
|             void* ret = __f(std::move(__args)...); | ||||
|             returnValue = ret; | ||||
|             return ret; | ||||
|         }; | ||||
|         auto funcPtr = new std::packaged_task<void()>(std::move(boundFunc)); | ||||
|         std::function<void* ()> funcPtr = std::move(boundFunc); | ||||
| #ifdef _WIN32 | ||||
|         handle_ = (HANDLE)_beginthreadex(nullptr, 0, thread_func, funcPtr, 0, nullptr); | ||||
|         handle_ = (HANDLE)_beginthreadex(nullptr, 0, thread_func, new std::function<void *()>(std::move(funcPtr)), 0, nullptr); | ||||
| #elif __linux__ | ||||
|         pthread_create(&handle_, NULL, thread_func, funcPtr); | ||||
| #endif | ||||
|     } | ||||
|     ~Thread(); | ||||
|  | ||||
|     void join(); | ||||
|     void* join(); | ||||
|     void detach(); | ||||
|  | ||||
| private: | ||||
| @@ -53,6 +67,7 @@ private: | ||||
| #elif __linux__ | ||||
|     pthread_t handle_; | ||||
| #endif | ||||
|     void* returnValue = nullptr; | ||||
|     bool detached = false; | ||||
| }; | ||||
|  | ||||
|   | ||||
							
								
								
									
										42
									
								
								include/Utils/ThreadPool.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								include/Utils/ThreadPool.hpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | ||||
| #pragma once | ||||
| #include "Thread.hpp" | ||||
| #include <vector> | ||||
| #include <queue> | ||||
| #include <functional> | ||||
| #include <condition_variable> | ||||
| #include <mutex> | ||||
| #include <memory> | ||||
|  | ||||
| namespace Chattr { | ||||
|  | ||||
| class ThreadPool { | ||||
| public: | ||||
| 	ThreadPool(std::uint32_t numThreads); | ||||
| 	~ThreadPool(); | ||||
|  | ||||
| 	template<typename _Callable, typename... _Args> | ||||
| 		requires ReturnsVoidPtr<_Callable, _Args...> && (!std::is_same_v<std::decay_t<_Callable>, Thread>) | ||||
| 	int enqueueJob(_Callable&& __job, void* retVal, _Args&&... __args) { | ||||
| 		if (terminate_) { | ||||
| 			spdlog::error("Cannot run jobs on threads that terminating..."); | ||||
| 			return -1; | ||||
| 		} | ||||
|  | ||||
| 		std::lock_guard<std::mutex> lock(jobQueueMutex); | ||||
| 		auto boundFunc = std::bind(std::forward<_Callable>(__job), std::forward<_Args>(__args)...); | ||||
| 		std::function<void*()> job(std::move(boundFunc)); | ||||
| 		jobs_.push(job, retVal); | ||||
| 		jobQueueCV_.notify_one(); | ||||
| 	} | ||||
|  | ||||
| private: | ||||
| 	void* Worker(); | ||||
|  | ||||
| 	std::condition_variable jobQueueCV_; | ||||
| 	std::mutex jobQueueMutex; | ||||
| 	std::queue<std::pair<std::function<void*()>, void*>> jobs_; | ||||
| 	std::vector<Chattr::Thread> workers_; | ||||
| 	bool terminate_ = false; | ||||
| }; | ||||
|  | ||||
| } | ||||
		Reference in New Issue
	
	Block a user