2012-12-22 03:30:33 +04:00
|
|
|
/**
|
|
|
|
* WinPR: Windows Portable Runtime
|
|
|
|
* Thread Pool API (Callback Clean-up)
|
|
|
|
*
|
|
|
|
* Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2022-02-16 12:08:00 +03:00
|
|
|
#include <winpr/config.h>
|
2012-12-22 03:30:33 +04:00
|
|
|
|
|
|
|
#include <winpr/crt.h>
|
|
|
|
#include <winpr/pool.h>
|
2016-02-06 00:28:45 +03:00
|
|
|
#include <winpr/library.h>
|
2012-12-22 03:30:33 +04:00
|
|
|
|
2013-01-23 01:19:32 +04:00
|
|
|
#include "pool.h"
|
|
|
|
|
2016-06-04 18:04:12 +03:00
|
|
|
#ifdef WINPR_THREAD_POOL
|
2013-01-22 08:34:46 +04:00
|
|
|
|
2016-06-04 18:04:12 +03:00
|
|
|
#ifdef _WIN32
|
|
|
|
static INIT_ONCE init_once_module = INIT_ONCE_STATIC_INIT;
|
2019-11-06 17:24:51 +03:00
|
|
|
static VOID(WINAPI* pSetEventWhenCallbackReturns)(PTP_CALLBACK_INSTANCE pci, HANDLE evt);
|
|
|
|
static VOID(WINAPI* pReleaseSemaphoreWhenCallbackReturns)(PTP_CALLBACK_INSTANCE pci, HANDLE sem,
|
|
|
|
DWORD crel);
|
|
|
|
static VOID(WINAPI* pReleaseMutexWhenCallbackReturns)(PTP_CALLBACK_INSTANCE pci, HANDLE mut);
|
|
|
|
static VOID(WINAPI* pLeaveCriticalSectionWhenCallbackReturns)(PTP_CALLBACK_INSTANCE pci,
|
|
|
|
PCRITICAL_SECTION pcs);
|
|
|
|
static VOID(WINAPI* pFreeLibraryWhenCallbackReturns)(PTP_CALLBACK_INSTANCE pci, HMODULE mod);
|
|
|
|
static VOID(WINAPI* pDisassociateCurrentThreadFromCallback)(PTP_CALLBACK_INSTANCE pci);
|
2013-01-22 08:34:46 +04:00
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
static BOOL CALLBACK init_module(PINIT_ONCE once, PVOID param, PVOID* context)
|
2013-01-22 08:34:46 +04:00
|
|
|
{
|
2016-06-04 18:04:12 +03:00
|
|
|
HMODULE kernel32 = LoadLibraryA("kernel32.dll");
|
|
|
|
if (kernel32)
|
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
pSetEventWhenCallbackReturns =
|
|
|
|
(void*)GetProcAddress(kernel32, "SetEventWhenCallbackReturns");
|
|
|
|
pReleaseSemaphoreWhenCallbackReturns =
|
|
|
|
(void*)GetProcAddress(kernel32, "ReleaseSemaphoreWhenCallbackReturns");
|
|
|
|
pReleaseMutexWhenCallbackReturns =
|
|
|
|
(void*)GetProcAddress(kernel32, "ReleaseMutexWhenCallbackReturns");
|
|
|
|
pLeaveCriticalSectionWhenCallbackReturns =
|
|
|
|
(void*)GetProcAddress(kernel32, "LeaveCriticalSectionWhenCallbackReturns");
|
|
|
|
pFreeLibraryWhenCallbackReturns =
|
|
|
|
(void*)GetProcAddress(kernel32, "FreeLibraryWhenCallbackReturns");
|
|
|
|
pDisassociateCurrentThreadFromCallback =
|
|
|
|
(void*)GetProcAddress(kernel32, "DisassociateCurrentThreadFromCallback");
|
2016-06-04 18:04:12 +03:00
|
|
|
}
|
|
|
|
return TRUE;
|
2013-01-22 08:34:46 +04:00
|
|
|
}
|
|
|
|
#endif
|
2012-12-22 03:30:33 +04:00
|
|
|
|
|
|
|
VOID SetEventWhenCallbackReturns(PTP_CALLBACK_INSTANCE pci, HANDLE evt)
|
|
|
|
{
|
2013-01-22 08:34:46 +04:00
|
|
|
#ifdef _WIN32
|
2016-06-04 18:04:12 +03:00
|
|
|
InitOnceExecuteOnce(&init_once_module, init_module, NULL, NULL);
|
2013-01-22 08:34:46 +04:00
|
|
|
if (pSetEventWhenCallbackReturns)
|
2016-06-04 18:04:12 +03:00
|
|
|
{
|
2013-01-22 08:34:46 +04:00
|
|
|
pSetEventWhenCallbackReturns(pci, evt);
|
2016-06-04 18:04:12 +03:00
|
|
|
return;
|
|
|
|
}
|
2013-01-22 08:34:46 +04:00
|
|
|
#endif
|
2016-06-04 18:04:12 +03:00
|
|
|
/* No default implementation */
|
2012-12-22 03:30:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
VOID ReleaseSemaphoreWhenCallbackReturns(PTP_CALLBACK_INSTANCE pci, HANDLE sem, DWORD crel)
|
|
|
|
{
|
2013-01-22 08:34:46 +04:00
|
|
|
#ifdef _WIN32
|
2016-06-04 18:04:12 +03:00
|
|
|
InitOnceExecuteOnce(&init_once_module, init_module, NULL, NULL);
|
2013-01-22 08:34:46 +04:00
|
|
|
if (pReleaseSemaphoreWhenCallbackReturns)
|
2016-06-04 18:04:12 +03:00
|
|
|
{
|
2013-01-22 08:34:46 +04:00
|
|
|
pReleaseSemaphoreWhenCallbackReturns(pci, sem, crel);
|
2016-06-04 18:04:12 +03:00
|
|
|
return;
|
|
|
|
}
|
2013-01-22 08:34:46 +04:00
|
|
|
#endif
|
2016-06-04 18:04:12 +03:00
|
|
|
/* No default implementation */
|
2012-12-22 03:30:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
VOID ReleaseMutexWhenCallbackReturns(PTP_CALLBACK_INSTANCE pci, HANDLE mut)
|
|
|
|
{
|
2013-01-22 08:34:46 +04:00
|
|
|
#ifdef _WIN32
|
2016-06-04 18:04:12 +03:00
|
|
|
InitOnceExecuteOnce(&init_once_module, init_module, NULL, NULL);
|
2013-01-22 08:34:46 +04:00
|
|
|
if (pReleaseMutexWhenCallbackReturns)
|
2016-06-04 18:04:12 +03:00
|
|
|
{
|
2013-01-22 08:34:46 +04:00
|
|
|
pReleaseMutexWhenCallbackReturns(pci, mut);
|
2016-06-04 18:04:12 +03:00
|
|
|
return;
|
|
|
|
}
|
2013-01-22 08:34:46 +04:00
|
|
|
#endif
|
2016-06-04 18:04:12 +03:00
|
|
|
/* No default implementation */
|
2012-12-22 03:30:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
VOID LeaveCriticalSectionWhenCallbackReturns(PTP_CALLBACK_INSTANCE pci, PCRITICAL_SECTION pcs)
|
|
|
|
{
|
2013-01-22 08:34:46 +04:00
|
|
|
#ifdef _WIN32
|
2016-06-04 18:04:12 +03:00
|
|
|
InitOnceExecuteOnce(&init_once_module, init_module, NULL, NULL);
|
2013-01-22 08:34:46 +04:00
|
|
|
if (pLeaveCriticalSectionWhenCallbackReturns)
|
2016-06-04 18:04:12 +03:00
|
|
|
{
|
2013-01-22 08:34:46 +04:00
|
|
|
pLeaveCriticalSectionWhenCallbackReturns(pci, pcs);
|
2016-06-04 18:04:12 +03:00
|
|
|
}
|
2013-01-22 08:34:46 +04:00
|
|
|
#endif
|
2016-06-04 18:04:12 +03:00
|
|
|
/* No default implementation */
|
2012-12-22 03:30:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
VOID FreeLibraryWhenCallbackReturns(PTP_CALLBACK_INSTANCE pci, HMODULE mod)
|
|
|
|
{
|
2013-01-22 08:34:46 +04:00
|
|
|
#ifdef _WIN32
|
2016-06-04 18:04:12 +03:00
|
|
|
InitOnceExecuteOnce(&init_once_module, init_module, NULL, NULL);
|
2013-01-22 08:34:46 +04:00
|
|
|
if (pFreeLibraryWhenCallbackReturns)
|
2016-06-04 18:04:12 +03:00
|
|
|
{
|
2013-01-22 08:34:46 +04:00
|
|
|
pFreeLibraryWhenCallbackReturns(pci, mod);
|
2016-06-04 18:04:12 +03:00
|
|
|
return;
|
|
|
|
}
|
2013-01-22 08:34:46 +04:00
|
|
|
#endif
|
2016-06-04 18:04:12 +03:00
|
|
|
/* No default implementation */
|
2012-12-22 03:30:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
VOID DisassociateCurrentThreadFromCallback(PTP_CALLBACK_INSTANCE pci)
|
|
|
|
{
|
2013-01-22 08:34:46 +04:00
|
|
|
#ifdef _WIN32
|
2016-06-04 18:04:12 +03:00
|
|
|
InitOnceExecuteOnce(&init_once_module, init_module, NULL, NULL);
|
2013-01-22 08:34:46 +04:00
|
|
|
if (pDisassociateCurrentThreadFromCallback)
|
2016-06-04 18:04:12 +03:00
|
|
|
{
|
2013-01-22 08:34:46 +04:00
|
|
|
pDisassociateCurrentThreadFromCallback(pci);
|
2016-06-04 18:04:12 +03:00
|
|
|
return;
|
|
|
|
}
|
2013-01-22 08:34:46 +04:00
|
|
|
#endif
|
2016-06-04 18:04:12 +03:00
|
|
|
/* No default implementation */
|
2012-12-22 03:30:33 +04:00
|
|
|
}
|
|
|
|
|
2016-06-04 18:04:12 +03:00
|
|
|
#endif /* WINPR_THREAD_POOL defined */
|