1 /*
2  * PROJECT:     ReactOS api tests
3  * LICENSE:     LGPL-2.0-or-later (https://spdx.org/licenses/LGPL-2.0-or-later)
4  * PURPOSE:     Tests for QueueUserAPC, SleepEx, WaitForSingleObjectEx etc.
5  * COPYRIGHT:   Copyright 2020 Katayama Hirofumi MZ (katayama.hirofumi.mz@gmail.com)
6  */
7 #include "precomp.h"
8 
9 #define MAX_RECORD 30
10 
11 static LONG s_record_count = 0;
12 static DWORD s_record[MAX_RECORD + 1] = { 0 };
13 static BOOL s_terminate_all = FALSE;
14 
15 static const DWORD s_expected[] =
16 {
17     0, 1, 7, 8, 4,
18     2, 1, 9, 10, 5,
19     2, 1, 11, 12, 13,
20     6, 2, 3, 14, 15,
21     16
22 };
23 static const SIZE_T s_expected_count = _countof(s_expected);
24 
25 static void AddValueToRecord(DWORD dwValue)
26 {
27     LONG next = InterlockedIncrement(&s_record_count) - 1;
28     if (next < MAX_RECORD)
29         s_record[next] = dwValue;
30 }
31 
32 static VOID CheckRecord(void)
33 {
34     SIZE_T i, count = min(s_record_count, s_expected_count);
35 
36     for (i = 0; i < count; ++i)
37     {
38         ok(s_record[i] == s_expected[i], "s_record[%u]: got %lu vs expected %lu\n",
39            (INT)i, s_record[i], s_expected[i]);
40     }
41 
42     count = abs((int)s_record_count - (int)s_expected_count);
43     for (i = 0; i < count; ++i)
44     {
45         ok(s_record_count == s_expected_count,
46            "s_record_count: got %u vs expected %u\n",
47            (int)s_record_count, (int)s_expected_count);
48     }
49 }
50 
51 static DWORD WINAPI ThreadFunc1(LPVOID arg)
52 {
53     AddValueToRecord(0);
54     while (!s_terminate_all)
55     {
56         AddValueToRecord(1);
57         ok_long(SleepEx(INFINITE, TRUE), WAIT_IO_COMPLETION);
58         AddValueToRecord(2);
59     }
60     AddValueToRecord(3);
61     return 0;
62 }
63 
64 static DWORD WINAPI ThreadFunc2(LPVOID arg)
65 {
66     AddValueToRecord(0);
67     while (!s_terminate_all)
68     {
69         AddValueToRecord(1);
70         ok_long(WaitForSingleObjectEx(GetCurrentThread(), INFINITE, TRUE), WAIT_IO_COMPLETION);
71         AddValueToRecord(2);
72     }
73     AddValueToRecord(3);
74     return 0;
75 }
76 
77 static VOID NTAPI DoUserAPC1(ULONG_PTR Parameter)
78 {
79     ok_int((int)Parameter, 1);
80     AddValueToRecord(4);
81 }
82 
83 static VOID NTAPI DoUserAPC2(ULONG_PTR Parameter)
84 {
85     ok_int((int)Parameter, 2);
86     AddValueToRecord(5);
87 }
88 
89 static VOID NTAPI DoUserAPC3(ULONG_PTR Parameter)
90 {
91     ok_int((int)Parameter, 3);
92     AddValueToRecord(6);
93     s_terminate_all = TRUE;
94 }
95 
96 static void JustDoIt(LPTHREAD_START_ROUTINE fn)
97 {
98     HANDLE hThread;
99     DWORD dwThreadId;
100 
101     s_terminate_all = FALSE;
102     s_record_count = 0;
103     ZeroMemory(s_record, sizeof(s_record));
104 
105     hThread = CreateThread(NULL, 0, fn, NULL, 0, &dwThreadId);
106     ok(hThread != NULL, "hThread was NULL\n");
107 
108     Sleep(100);
109 
110     AddValueToRecord(7);
111     ok_long(QueueUserAPC(DoUserAPC1, hThread, 1), 1);
112     AddValueToRecord(8);
113 
114     Sleep(100);
115 
116     AddValueToRecord(9);
117     ok_long(QueueUserAPC(DoUserAPC2, hThread, 2), 1);
118     AddValueToRecord(10);
119 
120     Sleep(100);
121 
122     AddValueToRecord(11);
123     ok_long(QueueUserAPC(DoUserAPC3, hThread, 3), 1);
124     AddValueToRecord(12);
125 
126     AddValueToRecord(13);
127     ok_long(WaitForSingleObject(hThread, 5 * 1000), WAIT_OBJECT_0);
128     AddValueToRecord(14);
129 
130     AddValueToRecord(15);
131     ok_int(CloseHandle(hThread), TRUE);
132     hThread = NULL;
133     AddValueToRecord(16);
134 
135     CheckRecord();
136 }
137 
138 static void TestForSleepEx(void)
139 {
140     JustDoIt(ThreadFunc1);
141 }
142 
143 static void TestForWaitForSingleObjectEx(void)
144 {
145     JustDoIt(ThreadFunc2);
146 }
147 
148 static DWORD WINAPI ThreadFunc3(LPVOID arg)
149 {
150     return 0;
151 }
152 
153 static void TestMultipleUserAPCs(void)
154 {
155     HANDLE hThread;
156     DWORD dwThreadId;
157 
158     s_record_count = 0;
159 
160     hThread = CreateThread(NULL, 0, ThreadFunc3, NULL, CREATE_SUSPENDED, &dwThreadId);
161     ok(hThread != NULL, "hThread was NULL\n");
162 
163     ok_long(QueueUserAPC(DoUserAPC1, hThread, 1), 1);
164     ok_long(QueueUserAPC(DoUserAPC2, hThread, 2), 1);
165     ok_long(QueueUserAPC(DoUserAPC3, hThread, 3), 1);
166 
167     ok_long(s_record_count, 0);
168 
169     ResumeThread(hThread);
170 
171     ok_long(WaitForSingleObject(hThread, 5 * 1000), WAIT_OBJECT_0);
172     ok_int(CloseHandle(hThread), TRUE);
173 
174     ok_long(s_record_count, 3);
175     ok_long(s_record[0], 4);
176     ok_long(s_record[1], 5);
177     ok_long(s_record[2], 6);
178 }
179 
180 START_TEST(QueueUserAPC)
181 {
182     TestForSleepEx();
183     TestForWaitForSingleObjectEx();
184     TestMultipleUserAPCs();
185 }
186