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
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
33 #include "wpa_supplicant_i.h"
37 #define WPASVC_NAME TEXT("wpasvc")
39 #ifndef WPASVC_DISPLAY_NAME
40 #define WPASVC_DISPLAY_NAME TEXT("wpa_supplicant service")
42 #ifndef WPASVC_DESCRIPTION
43 #define WPASVC_DESCRIPTION \
44 TEXT("Provides IEEE 802.1X and WPA/WPA2 supplicant functionality")
47 static HANDLE kill_svc;
49 static SERVICE_STATUS_HANDLE svc_status_handle;
50 static SERVICE_STATUS svc_status;
54 #define WPA_KEY_ROOT HKEY_LOCAL_MACHINE
56 #ifndef WPA_KEY_PREFIX
57 #define WPA_KEY_PREFIX TEXT("SOFTWARE\\wpa_supplicant")
67 static int read_interface(struct wpa_global *global, HKEY _hk,
72 TCHAR adapter[TBUFLEN], config[TBUFLEN], ctrl_interface[TBUFLEN];
75 struct wpa_interface iface;
77 ret = RegOpenKeyEx(_hk, name, 0, KEY_QUERY_VALUE, &hk);
78 if (ret != ERROR_SUCCESS) {
79 printf("Could not open wpa_supplicant interface key\n");
83 os_memset(&iface, 0, sizeof(iface));
84 iface.driver = "ndis";
86 buflen = sizeof(ctrl_interface);
87 ret = RegQueryValueEx(hk, TEXT("ctrl_interface"), NULL, NULL,
88 (LPBYTE) ctrl_interface, &buflen);
89 if (ret == ERROR_SUCCESS) {
90 ctrl_interface[TBUFLEN - 1] = TEXT('\0');
91 wpa_unicode2ascii_inplace(ctrl_interface);
92 printf("ctrl_interface[len=%d] '%s'\n",
93 (int) buflen, (char *) ctrl_interface);
94 iface.ctrl_interface = (char *) ctrl_interface;
97 buflen = sizeof(adapter);
98 ret = RegQueryValueEx(hk, TEXT("adapter"), NULL, NULL,
99 (LPBYTE) adapter, &buflen);
100 if (ret == ERROR_SUCCESS) {
101 adapter[TBUFLEN - 1] = TEXT('\0');
102 wpa_unicode2ascii_inplace(adapter);
103 printf("adapter[len=%d] '%s'\n",
104 (int) buflen, (char *) adapter);
105 iface.ifname = (char *) adapter;
108 buflen = sizeof(config);
109 ret = RegQueryValueEx(hk, TEXT("config"), NULL, NULL,
110 (LPBYTE) config, &buflen);
111 if (ret == ERROR_SUCCESS) {
112 config[sizeof(config) - 1] = '\0';
113 wpa_unicode2ascii_inplace(config);
114 printf("config[len=%d] '%s'\n",
115 (int) buflen, (char *) config);
116 iface.confname = (char *) config;
121 if (wpa_supplicant_add_iface(global, &iface) == NULL)
128 static int wpa_supplicant_thread(void)
131 struct wpa_params params;
132 struct wpa_global *global;
134 DWORD val, buflen, i;
137 if (os_program_init())
140 os_memset(¶ms, 0, sizeof(params));
141 params.wpa_debug_level = MSG_INFO;
143 ret = RegOpenKeyEx(WPA_KEY_ROOT, WPA_KEY_PREFIX,
144 0, KEY_QUERY_VALUE, &hk);
145 if (ret != ERROR_SUCCESS) {
146 printf("Could not open wpa_supplicant registry key\n");
150 buflen = sizeof(val);
151 ret = RegQueryValueEx(hk, TEXT("debug_level"), NULL, NULL,
152 (LPBYTE) &val, &buflen);
153 if (ret == ERROR_SUCCESS && buflen == sizeof(val)) {
154 params.wpa_debug_level = val;
157 buflen = sizeof(val);
158 ret = RegQueryValueEx(hk, TEXT("debug_show_keys"), NULL, NULL,
159 (LPBYTE) &val, &buflen);
160 if (ret == ERROR_SUCCESS && buflen == sizeof(val)) {
161 params.wpa_debug_show_keys = val;
164 buflen = sizeof(val);
165 ret = RegQueryValueEx(hk, TEXT("debug_timestamp"), NULL, NULL,
166 (LPBYTE) &val, &buflen);
167 if (ret == ERROR_SUCCESS && buflen == sizeof(val)) {
168 params.wpa_debug_timestamp = val;
171 buflen = sizeof(val);
172 ret = RegQueryValueEx(hk, TEXT("debug_use_file"), NULL, NULL,
173 (LPBYTE) &val, &buflen);
174 if (ret == ERROR_SUCCESS && buflen == sizeof(val) && val) {
175 params.wpa_debug_file_path = "\\Temp\\wpa_supplicant-log.txt";
179 global = wpa_supplicant_init(¶ms);
180 if (global == NULL) {
181 printf("Failed to initialize wpa_supplicant\n");
185 ret = RegOpenKeyEx(hk, TEXT("interfaces"), 0, KEY_ENUMERATE_SUB_KEYS,
188 if (ret != ERROR_SUCCESS) {
189 printf("Could not open wpa_supplicant interfaces registry "
199 ret = RegEnumKeyEx(ihk, i, name, &namelen, NULL, NULL, NULL,
202 if (ret == ERROR_NO_MORE_ITEMS)
205 if (ret != ERROR_SUCCESS) {
206 printf("RegEnumKeyEx failed: 0x%x\n",
213 name[namelen] = '\0';
215 wpa_printf(MSG_DEBUG, "interface %d: %s\n", (int) i, name);
216 if (read_interface(global, ihk, name) < 0)
223 exitcode = wpa_supplicant_run(global);
225 wpa_supplicant_deinit(global);
233 static DWORD svc_thread(LPDWORD param)
235 int ret = wpa_supplicant_thread();
237 svc_status.dwCurrentState = SERVICE_STOPPED;
238 svc_status.dwWaitHint = 0;
239 if (!SetServiceStatus(svc_status_handle, &svc_status)) {
240 printf("SetServiceStatus() failed: %d\n",
241 (int) GetLastError());
248 static int register_service(const TCHAR *exe)
251 SERVICE_DESCRIPTION sd;
253 printf("Registering service: " TSTR "\n", WPASVC_NAME);
255 scm = OpenSCManager(0, 0, SC_MANAGER_CREATE_SERVICE);
257 printf("OpenSCManager failed: %d\n", (int) GetLastError());
261 svc = CreateService(scm, WPASVC_NAME, WPASVC_DISPLAY_NAME,
262 SERVICE_ALL_ACCESS, SERVICE_WIN32_OWN_PROCESS,
263 SERVICE_DEMAND_START, SERVICE_ERROR_NORMAL,
264 exe, NULL, NULL, NULL, NULL, NULL);
267 printf("CreateService failed: %d\n\n", (int) GetLastError());
268 CloseServiceHandle(scm);
272 os_memset(&sd, 0, sizeof(sd));
273 sd.lpDescription = WPASVC_DESCRIPTION;
274 if (!ChangeServiceConfig2(svc, SERVICE_CONFIG_DESCRIPTION, &sd)) {
275 printf("ChangeServiceConfig2 failed: %d\n",
276 (int) GetLastError());
277 /* This is not a fatal error, so continue anyway. */
280 CloseServiceHandle(svc);
281 CloseServiceHandle(scm);
283 printf("Service registered successfully.\n");
289 static int unregister_service(void)
292 SERVICE_STATUS status;
294 printf("Unregistering service: " TSTR "\n", WPASVC_NAME);
296 scm = OpenSCManager(0, 0, SC_MANAGER_CREATE_SERVICE);
298 printf("OpenSCManager failed: %d\n", (int) GetLastError());
302 svc = OpenService(scm, WPASVC_NAME, SERVICE_ALL_ACCESS | DELETE);
304 printf("OpenService failed: %d\n\n", (int) GetLastError());
305 CloseServiceHandle(scm);
309 if (QueryServiceStatus(svc, &status)) {
310 if (status.dwCurrentState != SERVICE_STOPPED) {
311 printf("Service currently active - stopping "
313 if (!ControlService(svc, SERVICE_CONTROL_STOP,
315 printf("ControlService failed: %d\n",
316 (int) GetLastError());
322 if (DeleteService(svc)) {
323 printf("Service unregistered successfully.\n");
325 printf("DeleteService failed: %d\n", (int) GetLastError());
328 CloseServiceHandle(svc);
329 CloseServiceHandle(scm);
335 static void WINAPI service_ctrl_handler(DWORD control_code)
337 switch (control_code) {
338 case SERVICE_CONTROL_INTERROGATE:
340 case SERVICE_CONTROL_SHUTDOWN:
341 case SERVICE_CONTROL_STOP:
342 svc_status.dwCurrentState = SERVICE_STOP_PENDING;
343 svc_status.dwWaitHint = 2000;
349 if (!SetServiceStatus(svc_status_handle, &svc_status)) {
350 printf("SetServiceStatus() failed: %d\n",
351 (int) GetLastError());
356 static void WINAPI service_start(DWORD argc, LPTSTR *argv)
360 svc_status_handle = RegisterServiceCtrlHandler(WPASVC_NAME,
361 service_ctrl_handler);
362 if (svc_status_handle == (SERVICE_STATUS_HANDLE) 0) {
363 printf("RegisterServiceCtrlHandler failed: %d\n",
364 (int) GetLastError());
368 os_memset(&svc_status, 0, sizeof(svc_status));
369 svc_status.dwServiceType = SERVICE_WIN32_OWN_PROCESS;
370 svc_status.dwCurrentState = SERVICE_START_PENDING;
371 svc_status.dwWaitHint = 1000;
373 if (!SetServiceStatus(svc_status_handle, &svc_status)) {
374 printf("SetServiceStatus() failed: %d\n",
375 (int) GetLastError());
379 kill_svc = CreateEvent(0, TRUE, FALSE, 0);
381 printf("CreateEvent failed: %d\n", (int) GetLastError());
385 if (CreateThread(0, 0, (LPTHREAD_START_ROUTINE) svc_thread, 0, 0, &id)
387 printf("CreateThread failed: %d\n", (int) GetLastError());
391 if (svc_status.dwCurrentState == SERVICE_START_PENDING) {
392 svc_status.dwCurrentState = SERVICE_RUNNING;
393 svc_status.dwWaitHint = 0;
394 svc_status.dwControlsAccepted = SERVICE_ACCEPT_STOP |
395 SERVICE_ACCEPT_SHUTDOWN;
398 if (!SetServiceStatus(svc_status_handle, &svc_status)) {
399 printf("SetServiceStatus() failed: %d\n",
400 (int) GetLastError());
404 /* wait until service gets killed */
405 WaitForSingleObject(kill_svc, INFINITE);
409 int main(int argc, char *argv[])
411 SERVICE_TABLE_ENTRY dt[] = {
412 { WPASVC_NAME, service_start },
417 if (os_strcmp(argv[1], "reg") == 0) {
422 path = os_malloc(MAX_PATH * sizeof(TCHAR));
425 if (!GetModuleFileName(NULL, path, MAX_PATH)) {
426 printf("GetModuleFileName failed: "
427 "%d\n", (int) GetLastError());
432 path = wpa_strdup_tchar(argv[2]);
436 ret = register_service(path);
439 } else if (os_strcmp(argv[1], "unreg") == 0) {
440 return unregister_service();
441 } else if (os_strcmp(argv[1], "app") == 0) {
442 return wpa_supplicant_thread();
446 if (!StartServiceCtrlDispatcher(dt)) {
447 printf("StartServiceCtrlDispatcher failed: %d\n",
448 (int) GetLastError());