20#include <winpr/config.h>
22#include <winpr/assert.h>
24#include <winpr/pool.h>
25#include <winpr/library.h>
29#define TAG WINPR_TAG("pool")
31#ifdef WINPR_THREAD_POOL
34static INIT_ONCE init_once_module = INIT_ONCE_STATIC_INIT;
35static PTP_WORK(WINAPI* pCreateThreadpoolWork)(PTP_WORK_CALLBACK pfnwk, PVOID pv,
37static VOID(WINAPI* pCloseThreadpoolWork)(PTP_WORK pwk);
38static VOID(WINAPI* pSubmitThreadpoolWork)(PTP_WORK pwk);
39static BOOL(WINAPI* pTrySubmitThreadpoolCallback)(PTP_SIMPLE_CALLBACK pfns, PVOID pv,
41static VOID(WINAPI* pWaitForThreadpoolWorkCallbacks)(PTP_WORK pwk, BOOL fCancelPendingCallbacks);
43static BOOL CALLBACK init_module(
PINIT_ONCE once, PVOID param, PVOID* context)
45 HMODULE kernel32 = LoadLibraryA(
"kernel32.dll");
49 pCreateThreadpoolWork = GetProcAddressAs(kernel32,
"CreateThreadpoolWork",
void*);
50 pCloseThreadpoolWork = GetProcAddressAs(kernel32,
"CloseThreadpoolWork",
void*);
51 pSubmitThreadpoolWork = GetProcAddressAs(kernel32,
"SubmitThreadpoolWork",
void*);
52 pTrySubmitThreadpoolCallback =
53 GetProcAddressAs(kernel32,
"TrySubmitThreadpoolCallback",
void*);
54 pWaitForThreadpoolWorkCallbacks =
55 GetProcAddressAs(kernel32,
"WaitForThreadpoolWorkCallbacks",
void*);
74 PTP_WORK work =
nullptr;
76 if (!InitOnceExecuteOnce(&init_once_module, init_module,
nullptr,
nullptr))
79 if (pCreateThreadpoolWork)
80 return pCreateThreadpoolWork(pfnwk, pv, pcbe);
83 work = (PTP_WORK)calloc(1,
sizeof(TP_WORK));
89 pcbe = &DEFAULT_CALLBACK_ENVIRONMENT;
90 pcbe->Pool = GetDefaultThreadpool();
93 work->CallbackEnvironment = pcbe;
94 work->WorkCallback = pfnwk;
95 work->CallbackParameter = pv;
98 if (pcbe->CleanupGroup)
99 ArrayList_Append(pcbe->CleanupGroup->groups, work);
107VOID winpr_CloseThreadpoolWork(PTP_WORK pwk)
110 if (!InitOnceExecuteOnce(&init_once_module, init_module,
nullptr,
nullptr))
113 if (pCloseThreadpoolWork)
115 pCloseThreadpoolWork(pwk);
122 WINPR_ASSERT(pwk->CallbackEnvironment);
123 if (pwk->CallbackEnvironment->CleanupGroup)
124 ArrayList_Remove(pwk->CallbackEnvironment->CleanupGroup->groups, pwk);
130VOID winpr_SubmitThreadpoolWork(PTP_WORK pwk)
132 PTP_POOL pool =
nullptr;
133 PTP_CALLBACK_INSTANCE callbackInstance =
nullptr;
135 if (!InitOnceExecuteOnce(&init_once_module, init_module,
nullptr,
nullptr))
138 if (pSubmitThreadpoolWork)
140 pSubmitThreadpoolWork(pwk);
147 WINPR_ASSERT(pwk->CallbackEnvironment);
148 pool = pwk->CallbackEnvironment->Pool;
149 callbackInstance = (PTP_CALLBACK_INSTANCE)calloc(1,
sizeof(TP_CALLBACK_INSTANCE));
151 if (callbackInstance)
153 callbackInstance->Work = pwk;
154 CountdownEvent_AddCount(pool->WorkComplete, 1);
155 if (!Queue_Enqueue(pool->PendingQueue, callbackInstance))
156 free(callbackInstance);
161BOOL winpr_TrySubmitThreadpoolCallback(WINPR_ATTR_UNUSED PTP_SIMPLE_CALLBACK pfns,
162 WINPR_ATTR_UNUSED PVOID pv,
166 if (!InitOnceExecuteOnce(&init_once_module, init_module,
nullptr,
nullptr))
169 if (pTrySubmitThreadpoolCallback)
170 return pTrySubmitThreadpoolCallback(pfns, pv, pcbe);
173 WLog_ERR(TAG,
"TrySubmitThreadpoolCallback is not implemented");
177VOID winpr_WaitForThreadpoolWorkCallbacks(PTP_WORK pwk,
178 WINPR_ATTR_UNUSED BOOL fCancelPendingCallbacks)
180 HANDLE
event =
nullptr;
181 PTP_POOL pool =
nullptr;
184 if (!InitOnceExecuteOnce(&init_once_module, init_module,
nullptr,
nullptr))
187 if (pWaitForThreadpoolWorkCallbacks)
189 pWaitForThreadpoolWorkCallbacks(pwk, fCancelPendingCallbacks);
195 WINPR_ASSERT(pwk->CallbackEnvironment);
197 pool = pwk->CallbackEnvironment->Pool;
200 event = CountdownEvent_WaitHandle(pool->WorkComplete);
202 if (WaitForSingleObject(event, INFINITE) != WAIT_OBJECT_0)
203 WLog_ERR(TAG,
"error waiting on work completion");