|
Packit |
1fb8d4 |
/**
|
|
Packit |
1fb8d4 |
* WinPR: Windows Portable Runtime
|
|
Packit |
1fb8d4 |
* Thread Pool API (Work)
|
|
Packit |
1fb8d4 |
*
|
|
Packit |
1fb8d4 |
* Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
|
|
Packit |
1fb8d4 |
*
|
|
Packit |
1fb8d4 |
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
Packit |
1fb8d4 |
* you may not use this file except in compliance with the License.
|
|
Packit |
1fb8d4 |
* You may obtain a copy of the License at
|
|
Packit |
1fb8d4 |
*
|
|
Packit |
1fb8d4 |
* http://www.apache.org/licenses/LICENSE-2.0
|
|
Packit |
1fb8d4 |
*
|
|
Packit |
1fb8d4 |
* Unless required by applicable law or agreed to in writing, software
|
|
Packit |
1fb8d4 |
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
Packit |
1fb8d4 |
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
Packit |
1fb8d4 |
* See the License for the specific language governing permissions and
|
|
Packit |
1fb8d4 |
* limitations under the License.
|
|
Packit |
1fb8d4 |
*/
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
#ifdef HAVE_CONFIG_H
|
|
Packit |
1fb8d4 |
#include "config.h"
|
|
Packit |
1fb8d4 |
#endif
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
#include <winpr/crt.h>
|
|
Packit |
1fb8d4 |
#include <winpr/pool.h>
|
|
Packit |
1fb8d4 |
#include <winpr/library.h>
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
#include "pool.h"
|
|
Packit |
1fb8d4 |
#include "../log.h"
|
|
Packit |
1fb8d4 |
#define TAG WINPR_TAG("pool")
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
#ifdef WINPR_THREAD_POOL
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
#ifdef _WIN32
|
|
Packit |
1fb8d4 |
static INIT_ONCE init_once_module = INIT_ONCE_STATIC_INIT;
|
|
Packit |
1fb8d4 |
static PTP_WORK(WINAPI* pCreateThreadpoolWork)(PTP_WORK_CALLBACK pfnwk, PVOID pv,
|
|
Packit Service |
5a9772 |
PTP_CALLBACK_ENVIRON pcbe);
|
|
Packit Service |
5a9772 |
static VOID(WINAPI* pCloseThreadpoolWork)(PTP_WORK pwk);
|
|
Packit Service |
5a9772 |
static VOID(WINAPI* pSubmitThreadpoolWork)(PTP_WORK pwk);
|
|
Packit Service |
5a9772 |
static BOOL(WINAPI* pTrySubmitThreadpoolCallback)(PTP_SIMPLE_CALLBACK pfns, PVOID pv,
|
|
Packit Service |
5a9772 |
PTP_CALLBACK_ENVIRON pcbe);
|
|
Packit Service |
5a9772 |
static VOID(WINAPI* pWaitForThreadpoolWorkCallbacks)(PTP_WORK pwk, BOOL fCancelPendingCallbacks);
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
static BOOL CALLBACK init_module(PINIT_ONCE once, PVOID param, PVOID* context)
|
|
Packit |
1fb8d4 |
{
|
|
Packit |
1fb8d4 |
HMODULE kernel32 = LoadLibraryA("kernel32.dll");
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
if (kernel32)
|
|
Packit |
1fb8d4 |
{
|
|
Packit |
1fb8d4 |
pCreateThreadpoolWork = (void*)GetProcAddress(kernel32, "CreateThreadpoolWork");
|
|
Packit |
1fb8d4 |
pCloseThreadpoolWork = (void*)GetProcAddress(kernel32, "CloseThreadpoolWork");
|
|
Packit |
1fb8d4 |
pSubmitThreadpoolWork = (void*)GetProcAddress(kernel32, "SubmitThreadpoolWork");
|
|
Packit Service |
5a9772 |
pTrySubmitThreadpoolCallback =
|
|
Packit Service |
5a9772 |
(void*)GetProcAddress(kernel32, "TrySubmitThreadpoolCallback");
|
|
Packit Service |
5a9772 |
pWaitForThreadpoolWorkCallbacks =
|
|
Packit Service |
5a9772 |
(void*)GetProcAddress(kernel32, "WaitForThreadpoolWorkCallbacks");
|
|
Packit |
1fb8d4 |
}
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
return TRUE;
|
|
Packit |
1fb8d4 |
}
|
|
Packit |
1fb8d4 |
#endif
|
|
Packit |
1fb8d4 |
|
|
Packit Service |
5a9772 |
static TP_CALLBACK_ENVIRON DEFAULT_CALLBACK_ENVIRONMENT = {
|
|
Packit Service |
5a9772 |
1, /* Version */
|
|
Packit |
1fb8d4 |
NULL, /* Pool */
|
|
Packit |
1fb8d4 |
NULL, /* CleanupGroup */
|
|
Packit |
1fb8d4 |
NULL, /* CleanupGroupCancelCallback */
|
|
Packit |
1fb8d4 |
NULL, /* RaceDll */
|
|
Packit |
1fb8d4 |
NULL, /* ActivationContext */
|
|
Packit |
1fb8d4 |
NULL, /* FinalizationCallback */
|
|
Packit |
1fb8d4 |
{ 0 } /* Flags */
|
|
Packit |
1fb8d4 |
};
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
PTP_WORK winpr_CreateThreadpoolWork(PTP_WORK_CALLBACK pfnwk, PVOID pv, PTP_CALLBACK_ENVIRON pcbe)
|
|
Packit |
1fb8d4 |
{
|
|
Packit |
1fb8d4 |
PTP_WORK work = NULL;
|
|
Packit |
1fb8d4 |
#ifdef _WIN32
|
|
Packit |
1fb8d4 |
InitOnceExecuteOnce(&init_once_module, init_module, NULL, NULL);
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
if (pCreateThreadpoolWork)
|
|
Packit |
1fb8d4 |
return pCreateThreadpoolWork(pfnwk, pv, pcbe);
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
#endif
|
|
Packit Service |
5a9772 |
work = (PTP_WORK)calloc(1, sizeof(TP_WORK));
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
if (work)
|
|
Packit |
1fb8d4 |
{
|
|
Packit |
1fb8d4 |
if (!pcbe)
|
|
Packit |
1fb8d4 |
{
|
|
Packit |
1fb8d4 |
pcbe = &DEFAULT_CALLBACK_ENVIRONMENT;
|
|
Packit |
1fb8d4 |
pcbe->Pool = GetDefaultThreadpool();
|
|
Packit |
1fb8d4 |
}
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
work->CallbackEnvironment = pcbe;
|
|
Packit |
1fb8d4 |
work->WorkCallback = pfnwk;
|
|
Packit |
1fb8d4 |
work->CallbackParameter = pv;
|
|
Packit |
1fb8d4 |
#ifndef _WIN32
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
if (pcbe->CleanupGroup)
|
|
Packit |
1fb8d4 |
ArrayList_Add(pcbe->CleanupGroup->groups, work);
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
#endif
|
|
Packit |
1fb8d4 |
}
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
return work;
|
|
Packit |
1fb8d4 |
}
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
VOID winpr_CloseThreadpoolWork(PTP_WORK pwk)
|
|
Packit |
1fb8d4 |
{
|
|
Packit |
1fb8d4 |
#ifdef _WIN32
|
|
Packit |
1fb8d4 |
InitOnceExecuteOnce(&init_once_module, init_module, NULL, NULL);
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
if (pCloseThreadpoolWork)
|
|
Packit |
1fb8d4 |
{
|
|
Packit |
1fb8d4 |
pCloseThreadpoolWork(pwk);
|
|
Packit |
1fb8d4 |
return;
|
|
Packit |
1fb8d4 |
}
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
#else
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
if (pwk->CallbackEnvironment->CleanupGroup)
|
|
Packit |
1fb8d4 |
ArrayList_Remove(pwk->CallbackEnvironment->CleanupGroup->groups, pwk);
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
#endif
|
|
Packit |
1fb8d4 |
free(pwk);
|
|
Packit |
1fb8d4 |
}
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
VOID winpr_SubmitThreadpoolWork(PTP_WORK pwk)
|
|
Packit |
1fb8d4 |
{
|
|
Packit |
1fb8d4 |
PTP_POOL pool;
|
|
Packit |
1fb8d4 |
PTP_CALLBACK_INSTANCE callbackInstance;
|
|
Packit |
1fb8d4 |
#ifdef _WIN32
|
|
Packit |
1fb8d4 |
InitOnceExecuteOnce(&init_once_module, init_module, NULL, NULL);
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
if (pSubmitThreadpoolWork)
|
|
Packit |
1fb8d4 |
{
|
|
Packit |
1fb8d4 |
pSubmitThreadpoolWork(pwk);
|
|
Packit |
1fb8d4 |
return;
|
|
Packit |
1fb8d4 |
}
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
#endif
|
|
Packit |
1fb8d4 |
pool = pwk->CallbackEnvironment->Pool;
|
|
Packit Service |
5a9772 |
callbackInstance = (PTP_CALLBACK_INSTANCE)calloc(1, sizeof(TP_CALLBACK_INSTANCE));
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
if (callbackInstance)
|
|
Packit |
1fb8d4 |
{
|
|
Packit |
1fb8d4 |
callbackInstance->Work = pwk;
|
|
Packit |
1fb8d4 |
CountdownEvent_AddCount(pool->WorkComplete, 1);
|
|
Packit |
1fb8d4 |
Queue_Enqueue(pool->PendingQueue, callbackInstance);
|
|
Packit |
1fb8d4 |
}
|
|
Packit |
1fb8d4 |
}
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
BOOL winpr_TrySubmitThreadpoolCallback(PTP_SIMPLE_CALLBACK pfns, PVOID pv,
|
|
Packit |
1fb8d4 |
PTP_CALLBACK_ENVIRON pcbe)
|
|
Packit |
1fb8d4 |
{
|
|
Packit |
1fb8d4 |
#ifdef _WIN32
|
|
Packit |
1fb8d4 |
InitOnceExecuteOnce(&init_once_module, init_module, NULL, NULL);
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
if (pTrySubmitThreadpoolCallback)
|
|
Packit |
1fb8d4 |
return pTrySubmitThreadpoolCallback(pfns, pv, pcbe);
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
#endif
|
|
Packit |
1fb8d4 |
WLog_ERR(TAG, "TrySubmitThreadpoolCallback is not implemented");
|
|
Packit |
1fb8d4 |
return FALSE;
|
|
Packit |
1fb8d4 |
}
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
VOID winpr_WaitForThreadpoolWorkCallbacks(PTP_WORK pwk, BOOL fCancelPendingCallbacks)
|
|
Packit |
1fb8d4 |
{
|
|
Packit |
1fb8d4 |
HANDLE event;
|
|
Packit |
1fb8d4 |
PTP_POOL pool;
|
|
Packit |
1fb8d4 |
#ifdef _WIN32
|
|
Packit |
1fb8d4 |
InitOnceExecuteOnce(&init_once_module, init_module, NULL, NULL);
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
if (pWaitForThreadpoolWorkCallbacks)
|
|
Packit |
1fb8d4 |
{
|
|
Packit |
1fb8d4 |
pWaitForThreadpoolWorkCallbacks(pwk, fCancelPendingCallbacks);
|
|
Packit |
1fb8d4 |
return;
|
|
Packit |
1fb8d4 |
}
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
#endif
|
|
Packit |
1fb8d4 |
pool = pwk->CallbackEnvironment->Pool;
|
|
Packit |
1fb8d4 |
event = CountdownEvent_WaitHandle(pool->WorkComplete);
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
if (WaitForSingleObject(event, INFINITE) != WAIT_OBJECT_0)
|
|
Packit |
1fb8d4 |
WLog_ERR(TAG, "error waiting on work completion");
|
|
Packit |
1fb8d4 |
}
|
|
Packit |
1fb8d4 |
|
|
Packit |
1fb8d4 |
#endif /* WINPR_THREAD_POOL defined */
|