Commit 87f3b50b authored by Piotr Caban's avatar Piotr Caban Committed by Alexandre Julliard

msvcp140_atomic_wait: Add __std_atomic_wait_direct implementation.

parent 4c26471a
...@@ -74,3 +74,10 @@ void __stdcall __std_atomic_notify_all_direct(void *addr) ...@@ -74,3 +74,10 @@ void __stdcall __std_atomic_notify_all_direct(void *addr)
TRACE("(%p)\n", addr); TRACE("(%p)\n", addr);
WakeByAddressAll(addr); WakeByAddressAll(addr);
} }
BOOL __stdcall __std_atomic_wait_direct(volatile void *addr, void *cmp,
SIZE_T size, DWORD timeout)
{
TRACE("(%p %p %Id %ld)\n", addr, cmp, size, timeout);
return WaitOnAddress(addr, cmp, size, timeout);
}
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
@ stdcall __std_atomic_notify_one_direct(ptr) @ stdcall __std_atomic_notify_one_direct(ptr)
@ stub __std_atomic_notify_one_indirect @ stub __std_atomic_notify_one_indirect
@ stub __std_atomic_set_api_level @ stub __std_atomic_set_api_level
@ stub __std_atomic_wait_direct @ stdcall __std_atomic_wait_direct(ptr ptr long long)
@ stub __std_atomic_wait_get_deadline @ stub __std_atomic_wait_get_deadline
@ stub __std_atomic_wait_get_remaining_timeout @ stub __std_atomic_wait_get_remaining_timeout
@ stub __std_atomic_wait_indirect @ stub __std_atomic_wait_indirect
......
...@@ -17,6 +17,7 @@ ...@@ -17,6 +17,7 @@
*/ */
#include <stdio.h> #include <stdio.h>
#include <process.h>
#include "windef.h" #include "windef.h"
#include "winbase.h" #include "winbase.h"
...@@ -29,6 +30,8 @@ static void (__stdcall *p___std_close_threadpool_work)(PTP_WORK); ...@@ -29,6 +30,8 @@ static void (__stdcall *p___std_close_threadpool_work)(PTP_WORK);
static PTP_WORK (__stdcall *p___std_create_threadpool_work)(PTP_WORK_CALLBACK, void*, PTP_CALLBACK_ENVIRON); static PTP_WORK (__stdcall *p___std_create_threadpool_work)(PTP_WORK_CALLBACK, void*, PTP_CALLBACK_ENVIRON);
static void (__stdcall *p___std_submit_threadpool_work)(PTP_WORK); static void (__stdcall *p___std_submit_threadpool_work)(PTP_WORK);
static void (__stdcall *p___std_wait_for_threadpool_work_callbacks)(PTP_WORK, BOOL); static void (__stdcall *p___std_wait_for_threadpool_work_callbacks)(PTP_WORK, BOOL);
static BOOL (__stdcall *p___std_atomic_wait_direct)(volatile void*, void*, size_t, DWORD);
static void (__stdcall *p___std_atomic_notify_one_direct)(void*);
#define SETNOFAIL(x,y) x = (void*)GetProcAddress(msvcp,y) #define SETNOFAIL(x,y) x = (void*)GetProcAddress(msvcp,y)
#define SET(x,y) do { SETNOFAIL(x,y); ok(x != NULL, "Export '%s' not found\n", y); } while(0) #define SET(x,y) do { SETNOFAIL(x,y); ok(x != NULL, "Export '%s' not found\n", y); } while(0)
...@@ -46,6 +49,8 @@ static HMODULE init(void) ...@@ -46,6 +49,8 @@ static HMODULE init(void)
SET(p___std_create_threadpool_work, "__std_create_threadpool_work"); SET(p___std_create_threadpool_work, "__std_create_threadpool_work");
SET(p___std_submit_threadpool_work, "__std_submit_threadpool_work"); SET(p___std_submit_threadpool_work, "__std_submit_threadpool_work");
SET(p___std_wait_for_threadpool_work_callbacks, "__std_wait_for_threadpool_work_callbacks"); SET(p___std_wait_for_threadpool_work_callbacks, "__std_wait_for_threadpool_work_callbacks");
SET(p___std_atomic_wait_direct, "__std_atomic_wait_direct");
SET(p___std_atomic_notify_one_direct, "__std_atomic_notify_one_direct");
return msvcp; return msvcp;
} }
...@@ -162,6 +167,76 @@ static void test_threadpool_work(void) ...@@ -162,6 +167,76 @@ static void test_threadpool_work(void)
ok(!work, "expected failure\n"); ok(!work, "expected failure\n");
} }
LONG64 address;
static void __cdecl atomic_wait_thread(void *arg)
{
LONG64 compare = 0;
int r;
r = p___std_atomic_wait_direct(&address, &compare, sizeof(address), 2000);
ok(r == 1, "r = %d\n", r);
}
static void test___std_atomic_wait_direct(void)
{
LONG64 compare;
HANDLE thread;
DWORD gle;
int r;
if (!GetProcAddress(GetModuleHandleA("kernelbase"), "WaitOnAddress"))
{
win_skip("WaitOnAddress not available\n");
return;
}
address = compare = 0;
SetLastError(0);
r = p___std_atomic_wait_direct(&address, &compare, 5, 0);
ok(!r, "r = %d\n", r);
gle = GetLastError();
ok(gle == ERROR_INVALID_PARAMETER, "expected %d, got %ld\n", ERROR_INVALID_PARAMETER, gle);
SetLastError(0);
r = p___std_atomic_wait_direct(&address, &compare, 1, 0);
ok(!r, "r = %d\n", r);
gle = GetLastError();
ok(gle == ERROR_TIMEOUT, "expected %d, got %ld\n", ERROR_TIMEOUT, gle);
r = p___std_atomic_wait_direct(&address, &compare, 2, 0);
ok(!r, "r = %d\n", r);
gle = GetLastError();
ok(gle == ERROR_TIMEOUT, "expected %d, got %ld\n", ERROR_TIMEOUT, gle);
r = p___std_atomic_wait_direct(&address, &compare, 4, 0);
ok(!r, "r = %d\n", r);
gle = GetLastError();
ok(gle == ERROR_TIMEOUT, "expected %d, got %ld\n", ERROR_TIMEOUT, gle);
r = p___std_atomic_wait_direct(&address, &compare, 8, 0);
ok(!r, "r = %d\n", r);
gle = GetLastError();
ok(gle == ERROR_TIMEOUT, "expected %d, got %ld\n", ERROR_TIMEOUT, gle);
SetLastError(0);
r = p___std_atomic_wait_direct(&address, &compare, 8, 1);
ok(!r, "r = %d\n", r);
gle = GetLastError();
ok(gle == ERROR_TIMEOUT, "expected %d, got %ld\n", ERROR_TIMEOUT, gle);
compare = 1;
SetLastError(0);
r = p___std_atomic_wait_direct(&address, &compare, 8, 0);
ok(r == 1, "r = %d\n", r);
gle = GetLastError();
ok(!gle, "expected 0, got %ld\n", gle);
thread = (HANDLE)_beginthread(atomic_wait_thread, 0, NULL);
ok(thread != INVALID_HANDLE_VALUE, "_beginthread failed\n");
Sleep(100);
address = 1;
p___std_atomic_notify_one_direct(&address);
WaitForSingleObject(thread, INFINITE);
CloseHandle(thread);
}
START_TEST(msvcp140_atomic_wait) START_TEST(msvcp140_atomic_wait)
{ {
HMODULE msvcp; HMODULE msvcp;
...@@ -172,5 +247,6 @@ START_TEST(msvcp140_atomic_wait) ...@@ -172,5 +247,6 @@ START_TEST(msvcp140_atomic_wait)
} }
test___std_parallel_algorithms_hw_threads(); test___std_parallel_algorithms_hw_threads();
test_threadpool_work(); test_threadpool_work();
test___std_atomic_wait_direct();
FreeLibrary(msvcp); FreeLibrary(msvcp);
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment