1 /*
2  * PROJECT:         ReactOS api tests
3  * LICENSE:         GPLv2+ - See COPYING in the top level directory
4  * PURPOSE:         Test for service networking
5  * PROGRAMMER:      Pierre Schweitzer
6  */
7 
8 #include "precomp.h"
9 
10 #include "svchlp.h"
11 
12 #define WIN32_NO_STATUS
13 #include <iphlpapi.h>
14 #include <winsock2.h>
15 
16 /*** Service part of the test ***/
17 
18 static SERVICE_STATUS_HANDLE status_handle;
19 
20 static void
21 report_service_status(DWORD dwCurrentState,
22                       DWORD dwWin32ExitCode,
23                       DWORD dwWaitHint)
24 {
25     BOOL res;
26     SERVICE_STATUS status;
27 
28     status.dwServiceType   = SERVICE_WIN32_OWN_PROCESS;
29     status.dwCurrentState  = dwCurrentState;
30     status.dwWin32ExitCode = dwWin32ExitCode;
31     status.dwWaitHint      = dwWaitHint;
32 
33     status.dwServiceSpecificExitCode = 0;
34     status.dwCheckPoint              = 0;
35 
36     if ( (dwCurrentState == SERVICE_START_PENDING) ||
37          (dwCurrentState == SERVICE_STOP_PENDING)  ||
38          (dwCurrentState == SERVICE_STOPPED) )
39     {
40         status.dwControlsAccepted = 0;
41     }
42     else
43     {
44         status.dwControlsAccepted = SERVICE_ACCEPT_STOP | SERVICE_ACCEPT_SHUTDOWN;
45     }
46 
47 #if 0
48     if ( (dwCurrentState == SERVICE_RUNNING) || (dwCurrentState == SERVICE_STOPPED) )
49         status.dwCheckPoint = 0;
50     else
51         status.dwCheckPoint = dwCheckPoint++;
52 #endif
53 
54     res = SetServiceStatus(status_handle, &status);
55     service_ok(res, "SetServiceStatus(%d) failed: %lu\n", dwCurrentState, GetLastError());
56 }
57 
58 static VOID WINAPI service_handler(DWORD ctrl)
59 {
60     switch(ctrl)
61     {
62     case SERVICE_CONTROL_STOP:
63     case SERVICE_CONTROL_SHUTDOWN:
64         report_service_status(SERVICE_STOP_PENDING, NO_ERROR, 0);
65     default:
66         report_service_status(SERVICE_RUNNING, NO_ERROR, 0);
67     }
68 }
69 
70 static DWORD GetExtendedTcpTableWithAlloc(PVOID *TcpTable, BOOL Order, DWORD Family, TCP_TABLE_CLASS Class)
71 {
72     DWORD ret;
73     DWORD Size = 0;
74 
75     *TcpTable = NULL;
76 
77     ret = GetExtendedTcpTable(*TcpTable, &Size, Order, Family, Class, 0);
78     if (ret == ERROR_INSUFFICIENT_BUFFER)
79     {
80         *TcpTable = HeapAlloc(GetProcessHeap(), 0, Size);
81         if (*TcpTable == NULL)
82         {
83             return ERROR_OUTOFMEMORY;
84         }
85 
86         ret = GetExtendedTcpTable(*TcpTable, &Size, Order, Family, Class, 0);
87         if (ret != NO_ERROR)
88         {
89             HeapFree(GetProcessHeap(), 0, *TcpTable);
90             *TcpTable = NULL;
91         }
92     }
93 
94     return ret;
95 }
96 
97 static DWORD GetExtendedUdpTableWithAlloc(PVOID *UdpTable, BOOL Order, DWORD Family, UDP_TABLE_CLASS Class)
98 {
99     DWORD ret;
100     DWORD Size = 0;
101 
102     *UdpTable = NULL;
103 
104     ret = GetExtendedUdpTable(*UdpTable, &Size, Order, Family, Class, 0);
105     if (ret == ERROR_INSUFFICIENT_BUFFER)
106     {
107         *UdpTable = HeapAlloc(GetProcessHeap(), 0, Size);
108         if (*UdpTable == NULL)
109         {
110             return ERROR_OUTOFMEMORY;
111         }
112 
113         ret = GetExtendedUdpTable(*UdpTable, &Size, Order, Family, Class, 0);
114         if (ret != NO_ERROR)
115         {
116             HeapFree(GetProcessHeap(), 0, *UdpTable);
117             *UdpTable = NULL;
118         }
119     }
120 
121     return ret;
122 }
123 
124 static void
125 test_tcp(LPWSTR svc_name, DWORD service_tag)
126 {
127     SOCKET sock;
128     SOCKADDR_IN server;
129     PMIB_TCPTABLE_OWNER_MODULE TcpTableOwnerMod;
130     DWORD i, ret;
131     BOOLEAN Found;
132     DWORD Pid = GetCurrentProcessId();
133 
134     sock = socket(AF_INET, SOCK_STREAM, 0);
135     service_ok(sock != INVALID_SOCKET, "Socket creation failed!\n");
136 
137     ZeroMemory(&server, sizeof(SOCKADDR_IN));
138     server.sin_family = AF_INET;
139     server.sin_addr.s_addr = htonl(INADDR_ANY);
140     server.sin_port = htons(9876);
141 
142     ret = bind(sock, (SOCKADDR*)&server, sizeof(SOCKADDR_IN));
143     service_ok(ret != SOCKET_ERROR, "binding failed\n");
144 
145     ret = listen(sock, SOMAXCONN);
146     service_ok(ret != SOCKET_ERROR, "listening failed\n");
147 
148     ret = GetExtendedTcpTableWithAlloc((PVOID *)&TcpTableOwnerMod, TRUE, AF_INET, TCP_TABLE_OWNER_MODULE_LISTENER);
149     service_ok(ret == ERROR_SUCCESS, "GetExtendedTcpTableWithAlloc failed: %x\n", ret);
150     if (ret == ERROR_SUCCESS)
151     {
152         service_ok(TcpTableOwnerMod->dwNumEntries > 0, "No TCP connections?!\n");
153 
154         Found = FALSE;
155         for (i = 0; i < TcpTableOwnerMod->dwNumEntries; ++i)
156         {
157             if (TcpTableOwnerMod->table[i].dwState == MIB_TCP_STATE_LISTEN &&
158                 TcpTableOwnerMod->table[i].dwLocalAddr == 0 &&
159                 TcpTableOwnerMod->table[i].dwLocalPort == htons(9876) &&
160                 TcpTableOwnerMod->table[i].dwRemoteAddr == 0)
161             {
162                 Found = TRUE;
163                 break;
164             }
165         }
166 
167         service_ok(Found, "Our socket wasn't found!\n");
168         if (Found)
169         {
170             DWORD Size = 0;
171             PTCPIP_OWNER_MODULE_BASIC_INFO BasicInfo = NULL;
172 
173             service_ok(TcpTableOwnerMod->table[i].dwOwningPid == Pid, "Invalid owner\n");
174             service_ok((DWORD)(TcpTableOwnerMod->table[i].OwningModuleInfo[0]) == service_tag, "Invalid tag: %x - %x\n", (DWORD)TcpTableOwnerMod->table[i].OwningModuleInfo[0], service_tag);
175 
176             ret = GetOwnerModuleFromTcpEntry(&TcpTableOwnerMod->table[i], TCPIP_OWNER_MODULE_INFO_BASIC, BasicInfo, &Size);
177             service_ok(ret == ERROR_INSUFFICIENT_BUFFER, "GetOwnerModuleFromTcpEntry failed with: %x\n", ret);
178             if (ERROR_INSUFFICIENT_BUFFER == ERROR_INSUFFICIENT_BUFFER)
179             {
180                 BasicInfo = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, Size);
181                 service_ok(BasicInfo != NULL, "HeapAlloc failed\n");
182 
183                 ret = GetOwnerModuleFromTcpEntry(&TcpTableOwnerMod->table[i], TCPIP_OWNER_MODULE_INFO_BASIC, BasicInfo, &Size);
184                 service_ok(ret == ERROR_SUCCESS, "GetOwnerModuleFromTcpEntry failed with: %x\n", ret);
185                 service_ok(_wcsicmp(svc_name, BasicInfo->pModulePath) == 0, "Mismatching names (%S, %S)\n", svc_name, BasicInfo->pModulePath);
186                 service_ok(_wcsicmp(svc_name, BasicInfo->pModuleName) == 0, "Mismatching names (%S, %S)\n", svc_name, BasicInfo->pModuleName);
187             }
188         }
189 
190         HeapFree(GetProcessHeap(), 0, TcpTableOwnerMod);
191     }
192 
193     closesocket(sock);
194 }
195 
196 static void
197 test_udp(LPWSTR svc_name, DWORD service_tag)
198 {
199     SOCKET sock;
200     SOCKADDR_IN server;
201     PMIB_UDPTABLE_OWNER_MODULE UdpTableOwnerMod;
202     DWORD i, ret;
203     BOOLEAN Found;
204     DWORD Pid = GetCurrentProcessId();
205 
206     sock = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
207     service_ok(sock != INVALID_SOCKET, "Socket creation failed!\n");
208 
209     ZeroMemory(&server, sizeof(SOCKADDR_IN));
210     server.sin_family = AF_INET;
211     server.sin_addr.s_addr = htonl(INADDR_ANY);
212     server.sin_port = htons(9876);
213 
214     ret = bind(sock, (SOCKADDR*)&server, sizeof(SOCKADDR_IN));
215     service_ok(ret != SOCKET_ERROR, "binding failed\n");
216 
217     ret = GetExtendedUdpTableWithAlloc((PVOID *)&UdpTableOwnerMod, TRUE, AF_INET, UDP_TABLE_OWNER_MODULE);
218     service_ok(ret == ERROR_SUCCESS, "GetExtendedUdpTableWithAlloc failed: %x\n", ret);
219     if (ret == ERROR_SUCCESS)
220     {
221         service_ok(UdpTableOwnerMod->dwNumEntries > 0, "No TCP connections?!\n");
222 
223         Found = FALSE;
224         for (i = 0; i < UdpTableOwnerMod->dwNumEntries; ++i)
225         {
226             if (UdpTableOwnerMod->table[i].dwLocalAddr == 0 &&
227                 UdpTableOwnerMod->table[i].dwLocalPort == htons(9876))
228             {
229                 Found = TRUE;
230                 break;
231             }
232         }
233 
234         service_ok(Found, "Our socket wasn't found!\n");
235         if (Found)
236         {
237             DWORD Size = 0;
238             PTCPIP_OWNER_MODULE_BASIC_INFO BasicInfo = NULL;
239 
240             service_ok(UdpTableOwnerMod->table[i].dwOwningPid == Pid, "Invalid owner\n");
241             service_ok((DWORD)(UdpTableOwnerMod->table[i].OwningModuleInfo[0]) == service_tag, "Invalid tag: %x - %x\n", (DWORD)UdpTableOwnerMod->table[i].OwningModuleInfo[0], service_tag);
242 
243             ret = GetOwnerModuleFromUdpEntry(&UdpTableOwnerMod->table[i], TCPIP_OWNER_MODULE_INFO_BASIC, BasicInfo, &Size);
244             service_ok(ret == ERROR_INSUFFICIENT_BUFFER, "GetOwnerModuleFromUdpEntry failed with: %x\n", ret);
245             if (ret == ERROR_INSUFFICIENT_BUFFER)
246             {
247                 BasicInfo = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, Size);
248                 service_ok(BasicInfo != NULL, "HeapAlloc failed\n");
249 
250                 ret = GetOwnerModuleFromUdpEntry(&UdpTableOwnerMod->table[i], TCPIP_OWNER_MODULE_INFO_BASIC, BasicInfo, &Size);
251                 service_ok(ret == ERROR_SUCCESS, "GetOwnerModuleFromUdpEntry failed with: %x\n", ret);
252                 service_ok(_wcsicmp(svc_name, BasicInfo->pModulePath) == 0, "Mismatching names (%S, %S)\n", svc_name, BasicInfo->pModulePath);
253                 service_ok(_wcsicmp(svc_name, BasicInfo->pModuleName) == 0, "Mismatching names (%S, %S)\n", svc_name, BasicInfo->pModuleName);
254             }
255         }
256 
257         HeapFree(GetProcessHeap(), 0, UdpTableOwnerMod);
258     }
259 
260     closesocket(sock);
261 }
262 
263 static void WINAPI
264 service_main(DWORD dwArgc, LPWSTR* lpszArgv)
265 {
266     // SERVICE_STATUS_HANDLE status_handle;
267     PTEB Teb;
268     WSADATA wsaData;
269 
270     UNREFERENCED_PARAMETER(dwArgc);
271 
272     /* Register our service for control (lpszArgv[0] holds the service name) */
273     status_handle = RegisterServiceCtrlHandlerW(lpszArgv[0], service_handler);
274     service_ok(status_handle != NULL, "RegisterServiceCtrlHandler failed: %lu\n", GetLastError());
275     if (!status_handle)
276         return;
277 
278     /* Report SERVICE_RUNNING status */
279     report_service_status(SERVICE_RUNNING, NO_ERROR, 4000);
280 
281     /* Check our tag is not 0 */
282     Teb = NtCurrentTeb();
283     service_ok(Teb->SubProcessTag != 0, "SubProcessTag is not defined!\n");
284 
285     if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0)
286     {
287         skip("Failed to init WS2\n");
288         goto quit;
289     }
290 
291     test_tcp(lpszArgv[0], (DWORD)Teb->SubProcessTag);
292     test_udp(lpszArgv[0], (DWORD)Teb->SubProcessTag);
293 
294     WSACleanup();
295 quit:
296     /* Work is done */
297     report_service_status(SERVICE_STOPPED, NO_ERROR, 0);
298 }
299 
300 static BOOL start_service(PCSTR service_nameA, PCWSTR service_nameW)
301 {
302     BOOL res;
303 
304     SERVICE_TABLE_ENTRYW servtbl[] =
305     {
306         { (PWSTR)service_nameW, service_main },
307         { NULL, NULL }
308     };
309 
310     res = StartServiceCtrlDispatcherW(servtbl);
311     service_ok(res, "StartServiceCtrlDispatcherW failed: %lu\n", GetLastError());
312     return res;
313 }
314 
315 
316 /*** Tester part of the test ***/
317 
318 static void
319 my_test_server(PCSTR service_nameA,
320                PCWSTR service_nameW,
321                void *param)
322 {
323     BOOL res;
324     SC_HANDLE hSC = NULL;
325     SC_HANDLE hService = NULL;
326     SERVICE_STATUS ServiceStatus;
327 
328     /* Open the SCM */
329     hSC = OpenSCManagerW(NULL, NULL, SC_MANAGER_ALL_ACCESS);
330     if (!hSC)
331     {
332         skip("OpenSCManagerW failed with error %lu!\n", GetLastError());
333         return;
334     }
335 
336     /* First create ourselves as a service running in the default LocalSystem account */
337     hService = register_service_exW(hSC, L"ServiceNetwork", service_nameW, NULL,
338                                     SERVICE_ALL_ACCESS,
339                                     SERVICE_WIN32_OWN_PROCESS | SERVICE_INTERACTIVE_PROCESS,
340                                     SERVICE_DEMAND_START,
341                                     SERVICE_ERROR_IGNORE,
342                                     NULL, NULL, NULL,
343                                     NULL, NULL);
344     if (!hService)
345     {
346         skip("CreateServiceW failed with error %lu!\n", GetLastError());
347         goto Cleanup;
348     }
349 
350     /* Start it */
351     if (!StartServiceW(hService, 0, NULL))
352     {
353         skip("StartServiceW failed with error %lu!\n", GetLastError());
354         goto Cleanup;
355     }
356 
357     /* Wait for the service to stop by itself */
358     do
359     {
360         Sleep(100);
361         ZeroMemory(&ServiceStatus, sizeof(ServiceStatus));
362         res = QueryServiceStatus(hService, &ServiceStatus);
363     } while (res && ServiceStatus.dwCurrentState != SERVICE_STOPPED);
364     ok(res, "QueryServiceStatus failed: %lu\n", GetLastError());
365     ok(ServiceStatus.dwCurrentState == SERVICE_STOPPED, "ServiceStatus.dwCurrentState = %lx\n", ServiceStatus.dwCurrentState);
366 
367     /* Be sure the service is really stopped */
368     res = ControlService(hService, SERVICE_CONTROL_STOP, &ServiceStatus);
369     if (!res && ServiceStatus.dwCurrentState != SERVICE_STOPPED &&
370         ServiceStatus.dwCurrentState != SERVICE_STOP_PENDING &&
371         GetLastError() != ERROR_SERVICE_NOT_ACTIVE)
372     {
373         skip("ControlService failed with error %lu!\n", GetLastError());
374         goto Cleanup;
375     }
376 
377 #if 0
378     trace("Service stopped. Going to restart it...\n");
379 
380     /* Now change the service configuration to make it start under the NetworkService account */
381     if (!ChangeServiceConfigW(hService,
382                               SERVICE_NO_CHANGE,
383                               SERVICE_NO_CHANGE,
384                               SERVICE_NO_CHANGE,
385                               NULL, NULL, NULL, NULL,
386                               L"NT AUTHORITY\\NetworkService", L"",
387                               NULL))
388     {
389         skip("ChangeServiceConfigW failed with error %lu!\n", GetLastError());
390         goto Cleanup;
391     }
392 
393     /* Start it */
394     if (!StartServiceW(hService, 0, NULL))
395     {
396         skip("StartServiceW failed with error %lu!\n", GetLastError());
397         goto Cleanup;
398     }
399 
400     /* Wait for the service to stop by itself */
401     do
402     {
403         Sleep(100);
404         ZeroMemory(&ServiceStatus, sizeof(ServiceStatus));
405         res = QueryServiceStatus(hService, &ServiceStatus);
406     } while (res && ServiceStatus.dwCurrentState != SERVICE_STOPPED);
407     ok(res, "QueryServiceStatus failed: %lu\n", GetLastError());
408     ok(ServiceStatus.dwCurrentState == SERVICE_STOPPED, "ServiceStatus.dwCurrentState = %lx\n", ServiceStatus.dwCurrentState);
409 
410     /* Be sure the service is really stopped */
411     res = ControlService(hService, SERVICE_CONTROL_STOP, &ServiceStatus);
412     if (!res && ServiceStatus.dwCurrentState != SERVICE_STOPPED &&
413         ServiceStatus.dwCurrentState != SERVICE_STOP_PENDING &&
414         GetLastError() != ERROR_SERVICE_NOT_ACTIVE)
415     {
416         skip("ControlService failed with error %lu!\n", GetLastError());
417         goto Cleanup;
418     }
419 #endif
420 
421 Cleanup:
422     if (hService)
423     {
424         res = DeleteService(hService);
425         ok(res, "DeleteService failed: %lu\n", GetLastError());
426         CloseServiceHandle(hService);
427     }
428 
429     if (hSC)
430         CloseServiceHandle(hSC);
431 }
432 
433 START_TEST(ServiceNetwork)
434 {
435     int argc;
436     char** argv;
437 
438     /* Check whether this test is started as a separated service process */
439     argc = winetest_get_mainargs(&argv);
440     if (argc >= 3)
441     {
442         service_process(start_service, argc, argv);
443         return;
444     }
445 
446     /* We are started as the real test */
447     test_runner(my_test_server, NULL);
448     // trace("Returned from test_runner\n");
449 }
450