1 /* 2 * PROJECT: ReactOS API tests 3 * LICENSE: LGPLv2.1+ - See COPYING.LIB in the top level directory 4 * PURPOSE: Test for SendMessageTimeout 5 * PROGRAMMERS: Thomas Faber <thomas.faber@reactos.org> 6 */ 7 8 #include "precomp.h" 9 10 static DWORD dwThread1; 11 static DWORD dwThread2; 12 static HANDLE hThread1; 13 static HANDLE hThread2; 14 static HWND hWndThread1; 15 static HWND hWndThread2; 16 17 static 18 void 19 TestSendMessageTimeout( 20 _In_ HWND hWnd, 21 _In_ UINT Msg) 22 { 23 LRESULT ret; 24 DWORD_PTR result; 25 26 ret = SendMessageTimeoutW(hWnd, Msg, 0, 0, SMTO_NORMAL, 0, NULL); 27 ok(ret == 0, "ret = %Id\n", ret); 28 29 result = 0x55555555; 30 ret = SendMessageTimeoutW(hWnd, Msg, 0, 0, SMTO_NORMAL, 0, &result); 31 ok(ret == 0, "ret = %Id\n", ret); 32 ok(result == 0, "result = %Iu\n", result); 33 34 ret = SendMessageTimeoutA(hWnd, Msg, 0, 0, SMTO_NORMAL, 0, NULL); 35 ok(ret == 0, "ret = %Id\n", ret); 36 37 result = 0x55555555; 38 ret = SendMessageTimeoutA(hWnd, Msg, 0, 0, SMTO_NORMAL, 0, &result); 39 ok(ret == 0, "ret = %Id\n", ret); 40 ok(result == 0, "result = %Iu\n", result); 41 } 42 43 #define WM_SENDTOOTHERTHREAD (WM_USER + 14) 44 45 #define KILL_THREAD1_FLAG 0x40000000 46 #define KILL_THREAD2_FLAG 0x20000000 47 #define KILL_THREAD_FLAGS (KILL_THREAD1_FLAG | KILL_THREAD2_FLAG) 48 49 static 50 LRESULT 51 CALLBACK 52 WndProc( 53 _In_ HWND hWnd, 54 _In_ UINT message, 55 _In_ WPARAM wParam, 56 _In_ LPARAM lParam) 57 { 58 if (IsDWmMsg(message) || IseKeyMsg(message)) 59 return DefWindowProcW(hWnd, message, wParam, lParam); 60 61 if (hWnd == hWndThread1) 62 { 63 RECORD_MESSAGE(1, message, SENT, wParam, lParam); 64 } 65 else if (hWnd == hWndThread2) 66 { 67 RECORD_MESSAGE(2, message, SENT, wParam, lParam); 68 } 69 else 70 { 71 RECORD_MESSAGE(3, message, SENT, wParam, lParam); 72 } 73 74 switch (message) 75 { 76 case WM_SENDTOOTHERTHREAD: 77 if (GetCurrentThreadId() == dwThread1) 78 { 79 if ((wParam & KILL_THREAD2_FLAG) && 80 (wParam & ~KILL_THREAD_FLAGS) > 10) 81 { 82 TerminateThread(hThread2, 123); 83 } 84 if ((wParam & KILL_THREAD1_FLAG) && 85 (wParam & ~KILL_THREAD_FLAGS) > 10) 86 { 87 TerminateThread(hThread1, 456); 88 } 89 ok(lParam == dwThread2, "lParam = %Iu, expected %lu\n", lParam, dwThread2); 90 return SendMessage(hWndThread2, WM_SENDTOOTHERTHREAD, wParam + 1, GetCurrentThreadId()); 91 } 92 else 93 { 94 ok(lParam == dwThread1, "lParam = %Iu, expected %lu\n", lParam, dwThread1); 95 return SendMessage(hWndThread1, WM_SENDTOOTHERTHREAD, wParam + 1, GetCurrentThreadId()); 96 } 97 } 98 99 return DefWindowProcW(hWnd, message, wParam, lParam); 100 } 101 102 static 103 DWORD 104 WINAPI 105 Thread1( 106 _Inout_opt_ PVOID Parameter) 107 { 108 MSG msg; 109 110 hWndThread1 = CreateWindowExW(0, L"SendTest", NULL, 0, 10, 10, 20, 20, NULL, NULL, 0, NULL); 111 ok(hWndThread1 != NULL, "CreateWindow failed\n"); 112 113 while (PeekMessage(&msg, 0, 0, 0, PM_REMOVE)) 114 { 115 if (!(msg.message > WM_USER || IsDWmMsg(msg.message) || IseKeyMsg(msg.message))) 116 RECORD_MESSAGE(1, msg.message, POST, 0, 0); 117 DispatchMessageA(&msg); 118 } 119 120 ResumeThread(hThread2); 121 122 while (MsgWaitForMultipleObjectsEx(1, &hThread2, FALSE, QS_ALLEVENTS, MWMO_ALERTABLE) != WAIT_OBJECT_0) 123 { 124 while (PeekMessage(&msg, 0, 0, 0, PM_REMOVE)) 125 { 126 if (!(msg.message > WM_USER || IsDWmMsg(msg.message) || IseKeyMsg(msg.message))) 127 RECORD_MESSAGE(1, msg.message, POST, 0, 0); 128 DispatchMessageA(&msg); 129 } 130 } 131 132 DestroyWindow(hWndThread1); 133 while (PeekMessage(&msg, 0, 0, 0, PM_REMOVE)) 134 { 135 if (!(msg.message > WM_USER || IsDWmMsg(msg.message) || IseKeyMsg(msg.message))) 136 RECORD_MESSAGE(1, msg.message, POST, 0, 0); 137 DispatchMessageA(&msg); 138 } 139 140 return 6; 141 } 142 143 static 144 DWORD 145 WINAPI 146 Thread2( 147 _Inout_opt_ PVOID Parameter) 148 { 149 MSG msg; 150 LRESULT ret; 151 WPARAM wParam; 152 153 hWndThread2 = CreateWindowExW(0, L"SendTest", NULL, 0, 10, 10, 20, 20, NULL, NULL, 0, NULL); 154 ok(hWndThread2 != NULL, "CreateWindow failed\n"); 155 156 while (PeekMessage(&msg, 0, 0, 0, PM_REMOVE)) 157 { 158 if (!(msg.message > WM_USER || IsDWmMsg(msg.message) || IseKeyMsg(msg.message))) 159 RECORD_MESSAGE(2, msg.message, POST, 0, 0); 160 DispatchMessageA(&msg); 161 } 162 163 wParam = (WPARAM)Parameter; 164 ret = SendMessage(hWndThread1, WM_SENDTOOTHERTHREAD, wParam, dwThread2); 165 ok(ret == 0, "ret = %lu\n", ret); 166 167 while (PeekMessage(&msg, 0, 0, 0, PM_REMOVE)) 168 { 169 if (!(msg.message > WM_USER || IsDWmMsg(msg.message) || IseKeyMsg(msg.message))) 170 RECORD_MESSAGE(2, msg.message, POST, 0, 0); 171 DispatchMessageA(&msg); 172 } 173 174 DestroyWindow(hWndThread2); 175 while (PeekMessage(&msg, 0, 0, 0, PM_REMOVE)) 176 { 177 if (!(msg.message > WM_USER || IsDWmMsg(msg.message) || IseKeyMsg(msg.message))) 178 RECORD_MESSAGE(2, msg.message, POST, 0, 0); 179 DispatchMessageA(&msg); 180 } 181 182 return 7; 183 } 184 185 static 186 void 187 TestRecursiveInterThreadMessages( 188 _In_ BOOL KillThread1, 189 _In_ BOOL KillThread2) 190 { 191 PVOID Parameter; 192 HANDLE Handles[2]; 193 BOOL Ret; 194 DWORD ExitCode; 195 196 Parameter = (PVOID)((KillThread1 ? KILL_THREAD1_FLAG : 0) | 197 (KillThread2 ? KILL_THREAD2_FLAG : 0)); 198 hThread1 = CreateThread(NULL, 0, Thread1, Parameter, CREATE_SUSPENDED, &dwThread1); 199 hThread2 = CreateThread(NULL, 0, Thread2, Parameter, CREATE_SUSPENDED, &dwThread2); 200 201 ResumeThread(hThread1); 202 203 Handles[0] = hThread1; 204 Handles[1] = hThread2; 205 WaitForMultipleObjects(2, Handles, TRUE, INFINITE); 206 207 Ret = GetExitCodeThread(hThread1, &ExitCode); 208 ok(Ret == TRUE, "GetExitCodeThread failed with %lu\n", GetLastError()); 209 if (KillThread1) 210 ok(ExitCode == 456, "Thread1 exit code is %lu\n", ExitCode); 211 else 212 ok(ExitCode == 6, "Thread1 exit code is %lu\n", ExitCode); 213 214 Ret = GetExitCodeThread(hThread2, &ExitCode); 215 ok(Ret == TRUE, "GetExitCodeThread failed with %lu\n", GetLastError()); 216 if (KillThread2) 217 ok(ExitCode == 123, "Thread2 exit code is %lu\n", ExitCode); 218 else 219 ok(ExitCode == 7, "Thread2 exit code is %lu\n", ExitCode); 220 221 CloseHandle(hThread2); 222 CloseHandle(hThread1); 223 224 //TRACE_CACHE(); 225 } 226 227 START_TEST(SendMessageTimeout) 228 { 229 TestSendMessageTimeout(NULL, WM_USER); 230 TestSendMessageTimeout(NULL, WM_PAINT); 231 TestSendMessageTimeout(NULL, WM_GETICON); 232 233 RegisterSimpleClass(WndProc, L"SendTest"); 234 235 TestRecursiveInterThreadMessages(FALSE, FALSE); 236 TestRecursiveInterThreadMessages(FALSE, TRUE); 237 TestRecursiveInterThreadMessages(TRUE, FALSE); 238 TestRecursiveInterThreadMessages(TRUE, TRUE); 239 } 240