Fix cross compilation (e.g. on Darwin). Following changes to make.tmpl,
[AROS.git] / workbench / network / WirelessManager / wpa_supplicant / main_winsvc.c
blob4a46ed5f104915ca70cc24e1cfe9fe1c8ead6588
1 /*
2 * WPA Supplicant / main() function for Win32 service
3 * Copyright (c) 2003-2006, Jouni Malinen <j@w1.fi>
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License version 2 as
7 * published by the Free Software Foundation.
9 * Alternatively, this software may be distributed under the terms of BSD
10 * license.
12 * See README and COPYING for more details.
14 * The root of wpa_supplicant configuration in registry is
15 * HKEY_LOCAL_MACHINE\\SOFTWARE\\%wpa_supplicant. This level includes global
16 * parameters and a 'interfaces' subkey with all the interface configuration
17 * (adapter to confname mapping). Each such mapping is a subkey that has
18 * 'adapter' and 'config' values.
20 * This program can be run either as a normal command line application, e.g.,
21 * for debugging, with 'wpasvc.exe app' or as a Windows service. Service need
22 * to be registered with 'wpasvc.exe reg <full path to wpasvc.exe>'. After
23 * this, it can be started like any other Windows service (e.g., 'net start
24 * wpasvc') or it can be configured to start automatically through the Services
25 * tool in administrative tasks. The service can be unregistered with
26 * 'wpasvc.exe unreg'.
29 #include "includes.h"
30 #include <windows.h>
32 #include "common.h"
33 #include "wpa_supplicant_i.h"
34 #include "eloop.h"
36 #ifndef WPASVC_NAME
37 #define WPASVC_NAME TEXT("wpasvc")
38 #endif
39 #ifndef WPASVC_DISPLAY_NAME
40 #define WPASVC_DISPLAY_NAME TEXT("wpa_supplicant service")
41 #endif
42 #ifndef WPASVC_DESCRIPTION
43 #define WPASVC_DESCRIPTION \
44 TEXT("Provides IEEE 802.1X and WPA/WPA2 supplicant functionality")
45 #endif
47 static HANDLE kill_svc;
49 static SERVICE_STATUS_HANDLE svc_status_handle;
50 static SERVICE_STATUS svc_status;
53 #ifndef WPA_KEY_ROOT
54 #define WPA_KEY_ROOT HKEY_LOCAL_MACHINE
55 #endif
56 #ifndef WPA_KEY_PREFIX
57 #define WPA_KEY_PREFIX TEXT("SOFTWARE\\wpa_supplicant")
58 #endif
60 #ifdef UNICODE
61 #define TSTR "%S"
62 #else /* UNICODE */
63 #define TSTR "%s"
64 #endif /* UNICODE */
67 static int read_interface(struct wpa_global *global, HKEY _hk,
68 const TCHAR *name)
70 HKEY hk;
71 #define TBUFLEN 255
72 TCHAR adapter[TBUFLEN], config[TBUFLEN], ctrl_interface[TBUFLEN];
73 DWORD buflen, val;
74 LONG ret;
75 struct wpa_interface iface;
76 int skip_on_error = 0;
78 ret = RegOpenKeyEx(_hk, name, 0, KEY_QUERY_VALUE, &hk);
79 if (ret != ERROR_SUCCESS) {
80 printf("Could not open wpa_supplicant interface key\n");
81 return -1;
84 os_memset(&iface, 0, sizeof(iface));
85 iface.driver = "ndis";
87 buflen = sizeof(ctrl_interface);
88 ret = RegQueryValueEx(hk, TEXT("ctrl_interface"), NULL, NULL,
89 (LPBYTE) ctrl_interface, &buflen);
90 if (ret == ERROR_SUCCESS) {
91 ctrl_interface[TBUFLEN - 1] = TEXT('\0');
92 wpa_unicode2ascii_inplace(ctrl_interface);
93 printf("ctrl_interface[len=%d] '%s'\n",
94 (int) buflen, (char *) ctrl_interface);
95 iface.ctrl_interface = (char *) ctrl_interface;
98 buflen = sizeof(adapter);
99 ret = RegQueryValueEx(hk, TEXT("adapter"), NULL, NULL,
100 (LPBYTE) adapter, &buflen);
101 if (ret == ERROR_SUCCESS) {
102 adapter[TBUFLEN - 1] = TEXT('\0');
103 wpa_unicode2ascii_inplace(adapter);
104 printf("adapter[len=%d] '%s'\n",
105 (int) buflen, (char *) adapter);
106 iface.ifname = (char *) adapter;
109 buflen = sizeof(config);
110 ret = RegQueryValueEx(hk, TEXT("config"), NULL, NULL,
111 (LPBYTE) config, &buflen);
112 if (ret == ERROR_SUCCESS) {
113 config[sizeof(config) - 1] = '\0';
114 wpa_unicode2ascii_inplace(config);
115 printf("config[len=%d] '%s'\n",
116 (int) buflen, (char *) config);
117 iface.confname = (char *) config;
120 buflen = sizeof(val);
121 ret = RegQueryValueEx(hk, TEXT("skip_on_error"), NULL, NULL,
122 (LPBYTE) &val, &buflen);
123 if (ret == ERROR_SUCCESS && buflen == sizeof(val))
124 skip_on_error = val;
126 RegCloseKey(hk);
128 if (wpa_supplicant_add_iface(global, &iface) == NULL) {
129 if (skip_on_error)
130 wpa_printf(MSG_DEBUG, "Skipped interface '%s' due to "
131 "initialization failure", iface.ifname);
132 else
133 return -1;
136 return 0;
140 static int wpa_supplicant_thread(void)
142 int exitcode;
143 struct wpa_params params;
144 struct wpa_global *global;
145 HKEY hk, ihk;
146 DWORD val, buflen, i;
147 LONG ret;
149 if (os_program_init())
150 return -1;
152 os_memset(&params, 0, sizeof(params));
153 params.wpa_debug_level = MSG_INFO;
155 ret = RegOpenKeyEx(WPA_KEY_ROOT, WPA_KEY_PREFIX,
156 0, KEY_QUERY_VALUE, &hk);
157 if (ret != ERROR_SUCCESS) {
158 printf("Could not open wpa_supplicant registry key\n");
159 return -1;
162 buflen = sizeof(val);
163 ret = RegQueryValueEx(hk, TEXT("debug_level"), NULL, NULL,
164 (LPBYTE) &val, &buflen);
165 if (ret == ERROR_SUCCESS && buflen == sizeof(val)) {
166 params.wpa_debug_level = val;
169 buflen = sizeof(val);
170 ret = RegQueryValueEx(hk, TEXT("debug_show_keys"), NULL, NULL,
171 (LPBYTE) &val, &buflen);
172 if (ret == ERROR_SUCCESS && buflen == sizeof(val)) {
173 params.wpa_debug_show_keys = val;
176 buflen = sizeof(val);
177 ret = RegQueryValueEx(hk, TEXT("debug_timestamp"), NULL, NULL,
178 (LPBYTE) &val, &buflen);
179 if (ret == ERROR_SUCCESS && buflen == sizeof(val)) {
180 params.wpa_debug_timestamp = val;
183 buflen = sizeof(val);
184 ret = RegQueryValueEx(hk, TEXT("debug_use_file"), NULL, NULL,
185 (LPBYTE) &val, &buflen);
186 if (ret == ERROR_SUCCESS && buflen == sizeof(val) && val) {
187 params.wpa_debug_file_path = "\\Temp\\wpa_supplicant-log.txt";
190 exitcode = 0;
191 global = wpa_supplicant_init(&params);
192 if (global == NULL) {
193 printf("Failed to initialize wpa_supplicant\n");
194 exitcode = -1;
197 ret = RegOpenKeyEx(hk, TEXT("interfaces"), 0, KEY_ENUMERATE_SUB_KEYS,
198 &ihk);
199 RegCloseKey(hk);
200 if (ret != ERROR_SUCCESS) {
201 printf("Could not open wpa_supplicant interfaces registry "
202 "key\n");
203 return -1;
206 for (i = 0; ; i++) {
207 TCHAR name[255];
208 DWORD namelen;
210 namelen = 255;
211 ret = RegEnumKeyEx(ihk, i, name, &namelen, NULL, NULL, NULL,
212 NULL);
214 if (ret == ERROR_NO_MORE_ITEMS)
215 break;
217 if (ret != ERROR_SUCCESS) {
218 printf("RegEnumKeyEx failed: 0x%x\n",
219 (unsigned int) ret);
220 break;
223 if (namelen >= 255)
224 namelen = 255 - 1;
225 name[namelen] = '\0';
227 wpa_printf(MSG_DEBUG, "interface %d: %s\n", (int) i, name);
228 if (read_interface(global, ihk, name) < 0)
229 exitcode = -1;
232 RegCloseKey(ihk);
234 if (exitcode == 0)
235 exitcode = wpa_supplicant_run(global);
237 wpa_supplicant_deinit(global);
239 os_program_deinit();
241 return exitcode;
245 static DWORD svc_thread(LPDWORD param)
247 int ret = wpa_supplicant_thread();
249 svc_status.dwCurrentState = SERVICE_STOPPED;
250 svc_status.dwWaitHint = 0;
251 if (!SetServiceStatus(svc_status_handle, &svc_status)) {
252 printf("SetServiceStatus() failed: %d\n",
253 (int) GetLastError());
256 return ret;
260 static int register_service(const TCHAR *exe)
262 SC_HANDLE svc, scm;
263 SERVICE_DESCRIPTION sd;
265 printf("Registering service: " TSTR "\n", WPASVC_NAME);
267 scm = OpenSCManager(0, 0, SC_MANAGER_CREATE_SERVICE);
268 if (!scm) {
269 printf("OpenSCManager failed: %d\n", (int) GetLastError());
270 return -1;
273 svc = CreateService(scm, WPASVC_NAME, WPASVC_DISPLAY_NAME,
274 SERVICE_ALL_ACCESS, SERVICE_WIN32_OWN_PROCESS,
275 SERVICE_DEMAND_START, SERVICE_ERROR_NORMAL,
276 exe, NULL, NULL, NULL, NULL, NULL);
278 if (!svc) {
279 printf("CreateService failed: %d\n\n", (int) GetLastError());
280 CloseServiceHandle(scm);
281 return -1;
284 os_memset(&sd, 0, sizeof(sd));
285 sd.lpDescription = WPASVC_DESCRIPTION;
286 if (!ChangeServiceConfig2(svc, SERVICE_CONFIG_DESCRIPTION, &sd)) {
287 printf("ChangeServiceConfig2 failed: %d\n",
288 (int) GetLastError());
289 /* This is not a fatal error, so continue anyway. */
292 CloseServiceHandle(svc);
293 CloseServiceHandle(scm);
295 printf("Service registered successfully.\n");
297 return 0;
301 static int unregister_service(void)
303 SC_HANDLE svc, scm;
304 SERVICE_STATUS status;
306 printf("Unregistering service: " TSTR "\n", WPASVC_NAME);
308 scm = OpenSCManager(0, 0, SC_MANAGER_CREATE_SERVICE);
309 if (!scm) {
310 printf("OpenSCManager failed: %d\n", (int) GetLastError());
311 return -1;
314 svc = OpenService(scm, WPASVC_NAME, SERVICE_ALL_ACCESS | DELETE);
315 if (!svc) {
316 printf("OpenService failed: %d\n\n", (int) GetLastError());
317 CloseServiceHandle(scm);
318 return -1;
321 if (QueryServiceStatus(svc, &status)) {
322 if (status.dwCurrentState != SERVICE_STOPPED) {
323 printf("Service currently active - stopping "
324 "service...\n");
325 if (!ControlService(svc, SERVICE_CONTROL_STOP,
326 &status)) {
327 printf("ControlService failed: %d\n",
328 (int) GetLastError());
330 Sleep(500);
334 if (DeleteService(svc)) {
335 printf("Service unregistered successfully.\n");
336 } else {
337 printf("DeleteService failed: %d\n", (int) GetLastError());
340 CloseServiceHandle(svc);
341 CloseServiceHandle(scm);
343 return 0;
347 static void WINAPI service_ctrl_handler(DWORD control_code)
349 switch (control_code) {
350 case SERVICE_CONTROL_INTERROGATE:
351 break;
352 case SERVICE_CONTROL_SHUTDOWN:
353 case SERVICE_CONTROL_STOP:
354 svc_status.dwCurrentState = SERVICE_STOP_PENDING;
355 svc_status.dwWaitHint = 2000;
356 eloop_terminate();
357 SetEvent(kill_svc);
358 break;
361 if (!SetServiceStatus(svc_status_handle, &svc_status)) {
362 printf("SetServiceStatus() failed: %d\n",
363 (int) GetLastError());
368 static void WINAPI service_start(DWORD argc, LPTSTR *argv)
370 DWORD id;
372 svc_status_handle = RegisterServiceCtrlHandler(WPASVC_NAME,
373 service_ctrl_handler);
374 if (svc_status_handle == (SERVICE_STATUS_HANDLE) 0) {
375 printf("RegisterServiceCtrlHandler failed: %d\n",
376 (int) GetLastError());
377 return;
380 os_memset(&svc_status, 0, sizeof(svc_status));
381 svc_status.dwServiceType = SERVICE_WIN32_OWN_PROCESS;
382 svc_status.dwCurrentState = SERVICE_START_PENDING;
383 svc_status.dwWaitHint = 1000;
385 if (!SetServiceStatus(svc_status_handle, &svc_status)) {
386 printf("SetServiceStatus() failed: %d\n",
387 (int) GetLastError());
388 return;
391 kill_svc = CreateEvent(0, TRUE, FALSE, 0);
392 if (!kill_svc) {
393 printf("CreateEvent failed: %d\n", (int) GetLastError());
394 return;
397 if (CreateThread(0, 0, (LPTHREAD_START_ROUTINE) svc_thread, 0, 0, &id)
398 == 0) {
399 printf("CreateThread failed: %d\n", (int) GetLastError());
400 return;
403 if (svc_status.dwCurrentState == SERVICE_START_PENDING) {
404 svc_status.dwCurrentState = SERVICE_RUNNING;
405 svc_status.dwWaitHint = 0;
406 svc_status.dwControlsAccepted = SERVICE_ACCEPT_STOP |
407 SERVICE_ACCEPT_SHUTDOWN;
410 if (!SetServiceStatus(svc_status_handle, &svc_status)) {
411 printf("SetServiceStatus() failed: %d\n",
412 (int) GetLastError());
413 return;
416 /* wait until service gets killed */
417 WaitForSingleObject(kill_svc, INFINITE);
421 int main(int argc, char *argv[])
423 SERVICE_TABLE_ENTRY dt[] = {
424 { WPASVC_NAME, service_start },
425 { NULL, NULL }
428 if (argc > 1) {
429 if (os_strcmp(argv[1], "reg") == 0) {
430 TCHAR *path;
431 int ret;
433 if (argc < 3) {
434 path = os_malloc(MAX_PATH * sizeof(TCHAR));
435 if (path == NULL)
436 return -1;
437 if (!GetModuleFileName(NULL, path, MAX_PATH)) {
438 printf("GetModuleFileName failed: "
439 "%d\n", (int) GetLastError());
440 os_free(path);
441 return -1;
443 } else {
444 path = wpa_strdup_tchar(argv[2]);
445 if (path == NULL)
446 return -1;
448 ret = register_service(path);
449 os_free(path);
450 return ret;
451 } else if (os_strcmp(argv[1], "unreg") == 0) {
452 return unregister_service();
453 } else if (os_strcmp(argv[1], "app") == 0) {
454 return wpa_supplicant_thread();
458 if (!StartServiceCtrlDispatcher(dt)) {
459 printf("StartServiceCtrlDispatcher failed: %d\n",
460 (int) GetLastError());
463 return 0;