Blame winpr/libwinpr/pool/work.c

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 */