1 /*
2 * PROJECT: ReactOS api tests
3 * LICENSE: LGPLv2.1+ - See COPYING.LIB in the top level directory
4 * PURPOSE: Test for service arguments
5 * PROGRAMMER: Jacek Caban for CodeWeavers
6 * Thomas Faber <thomas.faber@reactos.org>
7 */
8
9 #include "precomp.h"
10
11 static char **argv;
12 static int argc;
13
14 static HANDLE pipe_handle = INVALID_HANDLE_VALUE;
15 static char service_nameA[100];
16 static WCHAR service_nameW[100];
17 static WCHAR named_pipe_name[100];
18
19 /* Test process global variables */
20 static SC_HANDLE scm_handle;
21 static SERVICE_STATUS_HANDLE service_handle;
22
send_msg(const char * type,const char * msg)23 static void send_msg(const char *type, const char *msg)
24 {
25 DWORD written = 0;
26 char buf[512];
27
28 StringCbPrintfA(buf, sizeof(buf), "%s:%s", type, msg);
29 WriteFile(pipe_handle, buf, lstrlenA(buf)+1, &written, NULL);
30 }
31
32 #if 0
33 static inline void service_trace(const char *msg, ...)
34 {
35 va_list valist;
36 char buf[512];
37
38 va_start(valist, msg);
39 StringCbVPrintfA(buf, sizeof(buf), msg, valist);
40 va_end(valist);
41
42 send_msg("TRACE", buf);
43 }
44 #endif
45
service_ok(int cnd,const char * msg,...)46 static void service_ok(int cnd, const char *msg, ...)
47 {
48 va_list valist;
49 char buf[512];
50
51 va_start(valist, msg);
52 StringCbVPrintfA(buf, sizeof(buf), msg, valist);
53 va_end(valist);
54
55 send_msg(cnd ? "OK" : "FAIL", buf);
56 }
57
service_handler(DWORD ctrl)58 static VOID WINAPI service_handler(DWORD ctrl)
59 {
60 SERVICE_STATUS status;
61
62 status.dwServiceType = SERVICE_WIN32;
63 status.dwControlsAccepted = SERVICE_ACCEPT_STOP;
64 status.dwWin32ExitCode = 0;
65 status.dwServiceSpecificExitCode = 0;
66 status.dwCheckPoint = 0;
67 status.dwWaitHint = 0;
68
69 switch(ctrl)
70 {
71 case SERVICE_CONTROL_STOP:
72 status.dwCurrentState = SERVICE_STOP_PENDING;
73 status.dwControlsAccepted = 0;
74 SetServiceStatus(service_handle, &status);
75 default:
76 status.dwCurrentState = SERVICE_RUNNING;
77 SetServiceStatus(service_handle, &status);
78 }
79 }
80
service_main_common(void)81 static void service_main_common(void)
82 {
83 SERVICE_STATUS status;
84 BOOL res;
85
86 service_handle = RegisterServiceCtrlHandlerW(service_nameW, service_handler);
87 service_ok(service_handle != NULL, "RegisterServiceCtrlHandler failed: %lu\n", GetLastError());
88 if (!service_handle)
89 return;
90
91 status.dwServiceType = SERVICE_WIN32;
92 status.dwCurrentState = SERVICE_RUNNING;
93 status.dwControlsAccepted = SERVICE_ACCEPT_STOP | SERVICE_ACCEPT_SHUTDOWN;
94 status.dwWin32ExitCode = 0;
95 status.dwServiceSpecificExitCode = 0;
96 status.dwCheckPoint = 0;
97 status.dwWaitHint = 10000;
98 res = SetServiceStatus(service_handle, &status);
99 service_ok(res, "SetServiceStatus(SERVICE_RUNNING) failed: %lu", GetLastError());
100
101 Sleep(100);
102
103 status.dwCurrentState = SERVICE_STOPPED;
104 status.dwControlsAccepted = 0;
105 res = SetServiceStatus(service_handle, &status);
106 service_ok(res, "SetServiceStatus(SERVICE_STOPPED) failed: %lu", GetLastError());
107 }
108
109 /*
110 * A/W argument layout XP/2003:
111 * [argv array of pointers][parameter 1][parameter 2]...[service name]
112 *
113 * A/W argument layout Vista:
114 * [argv array of pointers][align to 8 bytes][parameter 1][parameter 2]...[service name]
115 *
116 * A/W argument layout Win7/8:
117 * [argv array of pointers][service name]
118 * [parameter 1][align to 4 bytes][parameter 2][align to 4 bytes]...
119 *
120 * Space for parameters and service name is always enough to store
121 * the WCHAR version including null terminator.
122 */
123
service_mainA(DWORD service_argc,char ** service_argv)124 static void WINAPI service_mainA(DWORD service_argc, char **service_argv)
125 {
126 int i;
127 char *next_arg;
128 char *next_arg_aligned;
129
130 service_ok(service_argc == argc - 3, "service_argc = %d, expected %d", service_argc, argc - 3);
131 if (service_argc == argc - 3)
132 {
133 service_ok(!strcmp(service_argv[0], service_nameA), "service_argv[0] = %s, expected %s", service_argv[0], service_nameA);
134 service_ok(service_argv[0] == (char *)&service_argv[service_argc] ||
135 service_argv[1] == (char *)&service_argv[service_argc] ||
136 service_argv[1] == (char *)(((ULONG_PTR)&service_argv[service_argc] + 7) & ~7), "service_argv[0] = %p, [1] = %p, expected one of them to be %p", service_argv[0], service_argv[1], &service_argv[service_argc]);
137 //service_trace("service_argv[0] = %p", service_argv[0]);
138 next_arg_aligned = next_arg = NULL;
139 for (i = 1; i < service_argc; i++)
140 {
141 //service_trace("service_argv[%d] = %p", i, service_argv[i]);
142 service_ok(!strcmp(service_argv[i], argv[i + 3]), "service_argv[%d] = %s, expected %s", i, service_argv[i], argv[i + 3]);
143 service_ok(next_arg == NULL ||
144 service_argv[i] == next_arg ||
145 service_argv[i] == next_arg_aligned, "service_argv[%d] = %p, expected %p or %p", i, service_argv[i], next_arg, next_arg_aligned);
146 next_arg = service_argv[i];
147 next_arg += 2 * (strlen(next_arg) + 1);
148 next_arg_aligned = (char *)(((ULONG_PTR)next_arg + 3) & ~3);
149 }
150 }
151
152 service_main_common();
153 }
154
service_mainW(DWORD service_argc,WCHAR ** service_argv)155 static void WINAPI service_mainW(DWORD service_argc, WCHAR **service_argv)
156 {
157 int i;
158 WCHAR argW[32];
159 WCHAR *next_arg;
160 WCHAR *next_arg_aligned;
161
162 service_ok(service_argc == argc - 3, "service_argc = %d, expected %d", service_argc, argc - 3);
163 if (service_argc == argc - 3)
164 {
165 service_ok(!wcscmp(service_argv[0], service_nameW), "service_argv[0] = %ls, expected %ls", service_argv[0], service_nameW);
166 service_ok(service_argv[0] == (WCHAR *)&service_argv[service_argc] ||
167 service_argv[1] == (WCHAR *)&service_argv[service_argc] ||
168 service_argv[1] == (WCHAR *)(((ULONG_PTR)&service_argv[service_argc] + 7) & ~7), "service_argv[0] = %p, [1] = %p, expected one of them to be %p", service_argv[0], service_argv[1], &service_argv[service_argc]);
169 //service_trace("service_argv[0] = %p", service_argv[0]);
170 next_arg_aligned = next_arg = NULL;
171 for (i = 1; i < service_argc; i++)
172 {
173 //service_trace("service_argv[%d] = %p", i, service_argv[i]);
174 MultiByteToWideChar(CP_ACP, 0, argv[i + 3], -1, argW, _countof(argW));
175 service_ok(!wcscmp(service_argv[i], argW), "service_argv[%d] = %ls, expected %ls", i, service_argv[i], argW);
176 service_ok(next_arg == NULL ||
177 service_argv[i] == next_arg ||
178 service_argv[i] == next_arg_aligned, "service_argv[%d] = %p, expected %p or %p", i, service_argv[i], next_arg, next_arg_aligned);
179 next_arg = service_argv[i];
180 next_arg += wcslen(next_arg) + 1;
181 next_arg_aligned = (WCHAR *)(((ULONG_PTR)next_arg + 3) & ~3);
182 }
183 }
184
185 service_main_common();
186 }
187
service_process(BOOLEAN unicode)188 static void service_process(BOOLEAN unicode)
189 {
190 BOOL res;
191
192 SERVICE_TABLE_ENTRYA servtblA[] =
193 {
194 { service_nameA, service_mainA },
195 { NULL, NULL }
196 };
197 SERVICE_TABLE_ENTRYW servtblW[] =
198 {
199 { service_nameW, service_mainW },
200 { NULL, NULL }
201 };
202
203 res = WaitNamedPipeW(named_pipe_name, NMPWAIT_USE_DEFAULT_WAIT);
204 if (!res)
205 return;
206
207 pipe_handle = CreateFileW(named_pipe_name, GENERIC_WRITE, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
208 if (pipe_handle == INVALID_HANDLE_VALUE)
209 return;
210
211 //service_trace("Starting...");
212 if (unicode)
213 {
214 res = StartServiceCtrlDispatcherW(servtblW);
215 service_ok(res, "StartServiceCtrlDispatcherW failed: %lu\n", GetLastError());
216 }
217 else
218 {
219 res = StartServiceCtrlDispatcherA(servtblA);
220 service_ok(res, "StartServiceCtrlDispatcherA failed: %lu\n", GetLastError());
221 }
222
223 CloseHandle(pipe_handle);
224 }
225
register_service(PCWSTR extra_args)226 static SC_HANDLE register_service(PCWSTR extra_args)
227 {
228 WCHAR service_cmd[MAX_PATH+150];
229 SC_HANDLE service;
230
231 GetModuleFileNameW(NULL, service_cmd, MAX_PATH);
232
233 StringCbCatW(service_cmd, sizeof(service_cmd), L" ServiceArgs ");
234 StringCbCatW(service_cmd, sizeof(service_cmd), service_nameW);
235 StringCbCatW(service_cmd, sizeof(service_cmd), extra_args);
236
237 trace("service_cmd \"%ls\"\n", service_cmd);
238
239 service = CreateServiceW(scm_handle, service_nameW, service_nameW, SERVICE_ALL_ACCESS,
240 SERVICE_WIN32_OWN_PROCESS, SERVICE_DEMAND_START, SERVICE_ERROR_IGNORE,
241 service_cmd, NULL, NULL, NULL, NULL, NULL);
242 if (!service && GetLastError() == ERROR_ACCESS_DENIED)
243 {
244 skip("Not enough access right to create service\n");
245 return NULL;
246 }
247
248 ok(service != NULL, "CreateService failed: %lu\n", GetLastError());
249 return service;
250 }
251
pipe_thread(void * arg)252 static DWORD WINAPI pipe_thread(void *arg)
253 {
254 char buf[512];
255 DWORD read;
256 BOOL res;
257
258 res = ConnectNamedPipe(pipe_handle, NULL);
259 ok(res || GetLastError() == ERROR_PIPE_CONNECTED, "ConnectNamedPipe failed: %lu\n", GetLastError());
260
261 while (1)
262 {
263 res = ReadFile(pipe_handle, buf, sizeof(buf), &read, NULL);
264 if (!res)
265 {
266 ok(GetLastError() == ERROR_BROKEN_PIPE || GetLastError() == ERROR_INVALID_HANDLE,
267 "ReadFile failed: %lu\n", GetLastError());
268 break;
269 }
270
271 if (!strncmp(buf, "TRACE:", 6))
272 {
273 trace("service trace: %s\n", buf+6);
274 }
275 else if (!strncmp(buf, "OK:", 3))
276 {
277 ok(1, "service: %s\n", buf+3);
278 }
279 else if (!strncmp(buf, "FAIL:", 5))
280 {
281 ok(0, "service: %s\n", buf+5);
282 }
283 else
284 {
285 ok(0, "malformed service message: %s\n", buf);
286 }
287 }
288
289 DisconnectNamedPipe(pipe_handle);
290 //trace("pipe disconnected\n");
291 return 0;
292 }
293
test_startA(SC_HANDLE service_handle,int service_argc,const char ** service_argv)294 static void test_startA(SC_HANDLE service_handle, int service_argc, const char **service_argv)
295 {
296 SERVICE_STATUS status;
297 BOOL res;
298
299 res = StartServiceA(service_handle, service_argc, service_argv);
300 ok(res, "StartService failed: %lu\n", GetLastError());
301 if (!res)
302 return;
303
304 do
305 {
306 Sleep(100);
307 ZeroMemory(&status, sizeof(status));
308 res = QueryServiceStatus(service_handle, &status);
309 } while (res && status.dwCurrentState != SERVICE_STOPPED);
310 ok(res, "QueryServiceStatus failed: %lu\n", GetLastError());
311 ok(status.dwCurrentState == SERVICE_STOPPED, "status.dwCurrentState = %lx\n", status.dwCurrentState);
312 }
313
test_startW(SC_HANDLE service_handle,int service_argc,const WCHAR ** service_argv)314 static void test_startW(SC_HANDLE service_handle, int service_argc, const WCHAR **service_argv)
315 {
316 SERVICE_STATUS status;
317 BOOL res;
318
319 res = StartServiceW(service_handle, service_argc, service_argv);
320 ok(res, "StartService failed: %lu\n", GetLastError());
321 if (!res)
322 return;
323
324 do
325 {
326 Sleep(100);
327 ZeroMemory(&status, sizeof(status));
328 res = QueryServiceStatus(service_handle, &status);
329 } while (res && status.dwCurrentState != SERVICE_STOPPED);
330 ok(res, "QueryServiceStatus failed: %lu\n", GetLastError());
331 ok(status.dwCurrentState == SERVICE_STOPPED, "status.dwCurrentState = %lx\n", status.dwCurrentState);
332 }
333
test_runner(BOOLEAN unicode,PCWSTR extra_args,int service_argc,void * service_argv)334 static void test_runner(BOOLEAN unicode, PCWSTR extra_args, int service_argc, void *service_argv)
335 {
336 HANDLE thread;
337 SC_HANDLE service_handle;
338 BOOL res;
339
340 StringCbPrintfW(service_nameW, sizeof(service_nameW), L"WineTestService%lu", GetTickCount());
341 WideCharToMultiByte(CP_ACP, 0, service_nameW, -1, service_nameA, _countof(service_nameA), NULL, NULL);
342 //trace("service_name: %ls\n", service_nameW);
343 StringCbPrintfW(named_pipe_name, sizeof(named_pipe_name), L"\\\\.\\pipe\\%ls_pipe", service_nameW);
344
345 pipe_handle = CreateNamedPipeW(named_pipe_name, PIPE_ACCESS_INBOUND,
346 PIPE_TYPE_MESSAGE|PIPE_READMODE_MESSAGE|PIPE_WAIT, 10, 2048, 2048, 10000, NULL);
347 ok(pipe_handle != INVALID_HANDLE_VALUE, "CreateNamedPipe failed: %lu\n", GetLastError());
348 if (pipe_handle == INVALID_HANDLE_VALUE)
349 return;
350
351 thread = CreateThread(NULL, 0, pipe_thread, NULL, 0, NULL);
352 ok(thread != NULL, "CreateThread failed: %lu\n", GetLastError());
353 if (!thread)
354 goto Quit;
355
356 service_handle = register_service(extra_args);
357 if (!service_handle)
358 goto Quit;
359
360 //trace("starting...\n");
361
362 if (unicode)
363 test_startW(service_handle, service_argc, service_argv);
364 else
365 test_startA(service_handle, service_argc, service_argv);
366
367 res = DeleteService(service_handle);
368 ok(res, "DeleteService failed: %lu\n", GetLastError());
369
370 CloseServiceHandle(service_handle);
371
372 ok(WaitForSingleObject(thread, 10000) == WAIT_OBJECT_0, "Timeout waiting for thread\n");
373
374 Quit:
375 if (thread)
376 CloseHandle(thread);
377
378 if (pipe_handle != INVALID_HANDLE_VALUE)
379 CloseHandle(pipe_handle);
380 }
381
START_TEST(ServiceArgs)382 START_TEST(ServiceArgs)
383 {
384 argc = winetest_get_mainargs(&argv);
385
386 scm_handle = OpenSCManagerW(NULL, NULL, SC_MANAGER_ALL_ACCESS);
387 ok(scm_handle != NULL, "OpenSCManager failed: %lu\n", GetLastError());
388 if (!scm_handle)
389 {
390 skip("Failed to open service control manager. Skipping test\n");
391 return;
392 }
393
394 if (argc < 3)
395 {
396 char *service_argvA[10];
397 WCHAR *service_argvW[10];
398
399 test_runner(FALSE, L" A", 0, NULL);
400 test_runner(FALSE, L" W", 0, NULL);
401 test_runner(TRUE, L" A", 0, NULL);
402 test_runner(TRUE, L" W", 0, NULL);
403
404 service_argvA[0] = "x";
405 service_argvW[0] = L"x";
406 test_runner(FALSE, L" A x", 1, service_argvA);
407 test_runner(FALSE, L" W x", 1, service_argvA);
408 test_runner(TRUE, L" A x", 1, service_argvW);
409 test_runner(TRUE, L" W x", 1, service_argvW);
410
411 service_argvA[1] = "y";
412 service_argvW[1] = L"y";
413 test_runner(FALSE, L" A x y", 2, service_argvA);
414 test_runner(FALSE, L" W x y", 2, service_argvA);
415 test_runner(TRUE, L" A x y", 2, service_argvW);
416 test_runner(TRUE, L" W x y", 2, service_argvW);
417
418 service_argvA[0] = "ab";
419 service_argvW[0] = L"ab";
420 test_runner(FALSE, L" A ab y", 2, service_argvA);
421 test_runner(FALSE, L" W ab y", 2, service_argvA);
422 test_runner(TRUE, L" A ab y", 2, service_argvW);
423 test_runner(TRUE, L" W ab y", 2, service_argvW);
424
425 service_argvA[0] = "abc";
426 service_argvW[0] = L"abc";
427 test_runner(FALSE, L" A abc y", 2, service_argvA);
428 test_runner(FALSE, L" W abc y", 2, service_argvA);
429 test_runner(TRUE, L" A abc y", 2, service_argvW);
430 test_runner(TRUE, L" W abc y", 2, service_argvW);
431
432 service_argvA[0] = "abcd";
433 service_argvW[0] = L"abcd";
434 test_runner(FALSE, L" A abcd y", 2, service_argvA);
435 test_runner(FALSE, L" W abcd y", 2, service_argvA);
436 test_runner(TRUE, L" A abcd y", 2, service_argvW);
437 test_runner(TRUE, L" W abcd y", 2, service_argvW);
438
439 service_argvA[0] = "abcde";
440 service_argvW[0] = L"abcde";
441 test_runner(FALSE, L" A abcde y", 2, service_argvA);
442 test_runner(FALSE, L" W abcde y", 2, service_argvA);
443 test_runner(TRUE, L" A abcde y", 2, service_argvW);
444 test_runner(TRUE, L" W abcde y", 2, service_argvW);
445
446 service_argvA[0] = "abcdef";
447 service_argvW[0] = L"abcdef";
448 test_runner(FALSE, L" A abcdef y", 2, service_argvA);
449 test_runner(FALSE, L" W abcdef y", 2, service_argvA);
450 test_runner(TRUE, L" A abcdef y", 2, service_argvW);
451 test_runner(TRUE, L" W abcdef y", 2, service_argvW);
452
453 service_argvA[0] = "abcdefg";
454 service_argvW[0] = L"abcdefg";
455 test_runner(FALSE, L" A abcdefg y", 2, service_argvA);
456 test_runner(FALSE, L" W abcdefg y", 2, service_argvA);
457 test_runner(TRUE, L" A abcdefg y", 2, service_argvW);
458 test_runner(TRUE, L" W abcdefg y", 2, service_argvW);
459
460 service_argvA[0] = "";
461 service_argvW[0] = L"";
462 test_runner(FALSE, L" A \"\" y", 2, service_argvA);
463 test_runner(FALSE, L" W \"\" y", 2, service_argvA);
464 test_runner(TRUE, L" A \"\" y", 2, service_argvW);
465 test_runner(TRUE, L" W \"\" y", 2, service_argvW);
466 }
467 else
468 {
469 strcpy(service_nameA, argv[2]);
470 MultiByteToWideChar(CP_ACP, 0, service_nameA, -1, service_nameW, _countof(service_nameW));
471 StringCbPrintfW(named_pipe_name, sizeof(named_pipe_name), L"\\\\.\\pipe\\%ls_pipe", service_nameW);
472 if (!strcmp(argv[3], "A"))
473 service_process(FALSE);
474 else
475 service_process(TRUE);
476 }
477
478 CloseServiceHandle(scm_handle);
479 }
480