xref: /freebsd/contrib/wpa/src/drivers/ndis_events.c (revision 7d99ab9fd0cc2c1ce2ecef0ed6d0672c2a50b0cb)
1 /*
2  * ndis_events - Receive NdisMIndicateStatus() events using WMI
3  * Copyright (c) 2004-2006, Jouni Malinen <j@w1.fi>
4  *
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.
8  *
9  * Alternatively, this software may be distributed under the terms of BSD
10  * license.
11  *
12  * See README and COPYING for more details.
13  */
14 
15 #define _WIN32_WINNT    0x0400
16 
17 #include "includes.h"
18 
19 #ifndef COBJMACROS
20 #define COBJMACROS
21 #endif /* COBJMACROS */
22 #include <wbemidl.h>
23 
24 #include "common.h"
25 
26 
27 static int wmi_refcnt = 0;
28 static int wmi_first = 1;
29 
30 struct ndis_events_data {
31 	IWbemObjectSink sink;
32 	IWbemObjectSinkVtbl sink_vtbl;
33 
34 	IWbemServices *pSvc;
35 	IWbemLocator *pLoc;
36 
37 	HANDLE read_pipe, write_pipe, event_avail;
38 	UINT ref;
39 	int terminating;
40 	char *ifname; /* {GUID..} */
41 	WCHAR *adapter_desc;
42 };
43 
44 #define BstrAlloc(x) (x) ? SysAllocString(x) : NULL
45 #define BstrFree(x) if (x) SysFreeString(x)
46 
47 /* WBEM / WMI wrapper functions, to perform in-place conversion of WCHARs to
48  * BSTRs */
49 HRESULT STDMETHODCALLTYPE call_IWbemServices_ExecQuery(
50 	IWbemServices *pSvc, LPCWSTR strQueryLanguage, LPCWSTR strQuery,
51 	long lFlags, IWbemContext *pCtx, IEnumWbemClassObject **ppEnum)
52 {
53 	BSTR bsQueryLanguage, bsQuery;
54 	HRESULT hr;
55 
56 	bsQueryLanguage = BstrAlloc(strQueryLanguage);
57 	bsQuery = BstrAlloc(strQuery);
58 
59 	hr = IWbemServices_ExecQuery(pSvc, bsQueryLanguage, bsQuery, lFlags,
60 				     pCtx, ppEnum);
61 
62 	BstrFree(bsQueryLanguage);
63 	BstrFree(bsQuery);
64 
65 	return hr;
66 }
67 
68 
69 HRESULT STDMETHODCALLTYPE call_IWbemServices_ExecNotificationQueryAsync(
70 	IWbemServices *pSvc, LPCWSTR strQueryLanguage, LPCWSTR strQuery,
71 	long lFlags, IWbemContext *pCtx, IWbemObjectSink *pResponseHandler)
72 {
73 	BSTR bsQueryLanguage, bsQuery;
74 	HRESULT hr;
75 
76 	bsQueryLanguage = BstrAlloc(strQueryLanguage);
77 	bsQuery = BstrAlloc(strQuery);
78 
79 	hr = IWbemServices_ExecNotificationQueryAsync(pSvc, bsQueryLanguage,
80 						      bsQuery, lFlags, pCtx,
81 						      pResponseHandler);
82 
83 	BstrFree(bsQueryLanguage);
84 	BstrFree(bsQuery);
85 
86 	return hr;
87 }
88 
89 
90 HRESULT STDMETHODCALLTYPE call_IWbemLocator_ConnectServer(
91 	IWbemLocator *pLoc, LPCWSTR strNetworkResource, LPCWSTR strUser,
92 	LPCWSTR strPassword, LPCWSTR strLocale, long lSecurityFlags,
93 	LPCWSTR strAuthority, IWbemContext *pCtx, IWbemServices **ppNamespace)
94 {
95 	BSTR bsNetworkResource, bsUser, bsPassword, bsLocale, bsAuthority;
96 	HRESULT hr;
97 
98 	bsNetworkResource = BstrAlloc(strNetworkResource);
99 	bsUser = BstrAlloc(strUser);
100 	bsPassword = BstrAlloc(strPassword);
101 	bsLocale = BstrAlloc(strLocale);
102 	bsAuthority = BstrAlloc(strAuthority);
103 
104 	hr = IWbemLocator_ConnectServer(pLoc, bsNetworkResource, bsUser,
105 					bsPassword, bsLocale, lSecurityFlags,
106 					bsAuthority, pCtx, ppNamespace);
107 
108 	BstrFree(bsNetworkResource);
109 	BstrFree(bsUser);
110 	BstrFree(bsPassword);
111 	BstrFree(bsLocale);
112 	BstrFree(bsAuthority);
113 
114 	return hr;
115 }
116 
117 
118 enum event_types { EVENT_CONNECT, EVENT_DISCONNECT, EVENT_MEDIA_SPECIFIC,
119 		   EVENT_ADAPTER_ARRIVAL, EVENT_ADAPTER_REMOVAL };
120 
121 static int ndis_events_get_adapter(struct ndis_events_data *events,
122 				   const char *ifname, const char *desc);
123 
124 
125 static int ndis_events_constructor(struct ndis_events_data *events)
126 {
127 	events->ref = 1;
128 
129 	if (!CreatePipe(&events->read_pipe, &events->write_pipe, NULL, 512)) {
130 		wpa_printf(MSG_ERROR, "CreatePipe() failed: %d",
131 			   (int) GetLastError());
132 		return -1;
133 	}
134 	events->event_avail = CreateEvent(NULL, TRUE, FALSE, NULL);
135 	if (events->event_avail == NULL) {
136 		wpa_printf(MSG_ERROR, "CreateEvent() failed: %d",
137 			   (int) GetLastError());
138 		CloseHandle(events->read_pipe);
139 		CloseHandle(events->write_pipe);
140 		return -1;
141 	}
142 
143 	return 0;
144 }
145 
146 
147 static void ndis_events_destructor(struct ndis_events_data *events)
148 {
149 	CloseHandle(events->read_pipe);
150 	CloseHandle(events->write_pipe);
151 	CloseHandle(events->event_avail);
152 	IWbemServices_Release(events->pSvc);
153 	IWbemLocator_Release(events->pLoc);
154 	if (--wmi_refcnt == 0)
155 		CoUninitialize();
156 }
157 
158 
159 static HRESULT STDMETHODCALLTYPE
160 ndis_events_query_interface(IWbemObjectSink *this, REFIID riid, void **obj)
161 {
162 	*obj = NULL;
163 
164 	if (IsEqualIID(riid, &IID_IUnknown) ||
165 	    IsEqualIID(riid, &IID_IWbemObjectSink)) {
166 		*obj = this;
167 		IWbemObjectSink_AddRef(this);
168 		return NOERROR;
169 	}
170 
171 	return E_NOINTERFACE;
172 }
173 
174 
175 static ULONG STDMETHODCALLTYPE ndis_events_add_ref(IWbemObjectSink *this)
176 {
177 	struct ndis_events_data *events = (struct ndis_events_data *) this;
178 	return ++events->ref;
179 }
180 
181 
182 static ULONG STDMETHODCALLTYPE ndis_events_release(IWbemObjectSink *this)
183 {
184 	struct ndis_events_data *events = (struct ndis_events_data *) this;
185 
186 	if (--events->ref != 0)
187 		return events->ref;
188 
189 	ndis_events_destructor(events);
190 	wpa_printf(MSG_DEBUG, "ndis_events: terminated");
191 	os_free(events->adapter_desc);
192 	os_free(events->ifname);
193 	os_free(events);
194 	return 0;
195 }
196 
197 
198 static int ndis_events_send_event(struct ndis_events_data *events,
199 				  enum event_types type,
200 				  char *data, size_t data_len)
201 {
202 	char buf[512], *pos, *end;
203 	int _type;
204 	DWORD written;
205 
206 	end = buf + sizeof(buf);
207 	_type = (int) type;
208 	os_memcpy(buf, &_type, sizeof(_type));
209 	pos = buf + sizeof(_type);
210 
211 	if (data) {
212 		if (2 + data_len > (size_t) (end - pos)) {
213 			wpa_printf(MSG_DEBUG, "Not enough room for send_event "
214 				   "data (%d)", data_len);
215 			return -1;
216 		}
217 		*pos++ = data_len >> 8;
218 		*pos++ = data_len & 0xff;
219 		os_memcpy(pos, data, data_len);
220 		pos += data_len;
221 	}
222 
223 	if (WriteFile(events->write_pipe, buf, pos - buf, &written, NULL)) {
224 		SetEvent(events->event_avail);
225 		return 0;
226 	}
227 	wpa_printf(MSG_INFO, "WriteFile() failed: %d", (int) GetLastError());
228 	return -1;
229 }
230 
231 
232 static void ndis_events_media_connect(struct ndis_events_data *events)
233 {
234 	wpa_printf(MSG_DEBUG, "MSNdis_StatusMediaConnect");
235 	ndis_events_send_event(events, EVENT_CONNECT, NULL, 0);
236 }
237 
238 
239 static void ndis_events_media_disconnect(struct ndis_events_data *events)
240 {
241 	wpa_printf(MSG_DEBUG, "MSNdis_StatusMediaDisconnect");
242 	ndis_events_send_event(events, EVENT_DISCONNECT, NULL, 0);
243 }
244 
245 
246 static void ndis_events_media_specific(struct ndis_events_data *events,
247 				       IWbemClassObject *pObj)
248 {
249 	VARIANT vt;
250 	HRESULT hr;
251 	LONG lower, upper, k;
252 	UCHAR ch;
253 	char *data, *pos;
254 	size_t data_len;
255 
256 	wpa_printf(MSG_DEBUG, "MSNdis_StatusMediaSpecificIndication");
257 
258 	/* This is the StatusBuffer from NdisMIndicateStatus() call */
259 	hr = IWbemClassObject_Get(pObj, L"NdisStatusMediaSpecificIndication",
260 				  0, &vt, NULL, NULL);
261 	if (FAILED(hr)) {
262 		wpa_printf(MSG_DEBUG, "Could not get "
263 			   "NdisStatusMediaSpecificIndication from "
264 			   "the object?!");
265 		return;
266 	}
267 
268 	SafeArrayGetLBound(V_ARRAY(&vt), 1, &lower);
269 	SafeArrayGetUBound(V_ARRAY(&vt), 1, &upper);
270 	data_len = upper - lower + 1;
271 	data = os_malloc(data_len);
272 	if (data == NULL) {
273 		wpa_printf(MSG_DEBUG, "Failed to allocate buffer for event "
274 			   "data");
275 		VariantClear(&vt);
276 		return;
277 	}
278 
279 	pos = data;
280 	for (k = lower; k <= upper; k++) {
281 		SafeArrayGetElement(V_ARRAY(&vt), &k, &ch);
282 		*pos++ = ch;
283 	}
284 	wpa_hexdump(MSG_DEBUG, "MediaSpecificEvent", (u8 *) data, data_len);
285 
286 	VariantClear(&vt);
287 
288 	ndis_events_send_event(events, EVENT_MEDIA_SPECIFIC, data, data_len);
289 
290 	os_free(data);
291 }
292 
293 
294 static void ndis_events_adapter_arrival(struct ndis_events_data *events)
295 {
296 	wpa_printf(MSG_DEBUG, "MSNdis_NotifyAdapterArrival");
297 	ndis_events_send_event(events, EVENT_ADAPTER_ARRIVAL, NULL, 0);
298 }
299 
300 
301 static void ndis_events_adapter_removal(struct ndis_events_data *events)
302 {
303 	wpa_printf(MSG_DEBUG, "MSNdis_NotifyAdapterRemoval");
304 	ndis_events_send_event(events, EVENT_ADAPTER_REMOVAL, NULL, 0);
305 }
306 
307 
308 static HRESULT STDMETHODCALLTYPE
309 ndis_events_indicate(IWbemObjectSink *this, long lObjectCount,
310 		     IWbemClassObject __RPC_FAR *__RPC_FAR *ppObjArray)
311 {
312 	struct ndis_events_data *events = (struct ndis_events_data *) this;
313 	long i;
314 
315 	if (events->terminating) {
316 		wpa_printf(MSG_DEBUG, "ndis_events_indicate: Ignore "
317 			   "indication - terminating");
318 		return WBEM_NO_ERROR;
319 	}
320 	/* wpa_printf(MSG_DEBUG, "Notification received - %d object(s)",
321 	   lObjectCount); */
322 
323 	for (i = 0; i < lObjectCount; i++) {
324 		IWbemClassObject *pObj = ppObjArray[i];
325 		HRESULT hr;
326 		VARIANT vtClass, vt;
327 
328 		hr = IWbemClassObject_Get(pObj, L"__CLASS", 0, &vtClass, NULL,
329 					  NULL);
330 		if (FAILED(hr)) {
331 			wpa_printf(MSG_DEBUG, "Failed to get __CLASS from "
332 				   "event.");
333 			break;
334 		}
335 		/* wpa_printf(MSG_DEBUG, "CLASS: '%S'", vtClass.bstrVal); */
336 
337 		hr = IWbemClassObject_Get(pObj, L"InstanceName", 0, &vt, NULL,
338 					  NULL);
339 		if (FAILED(hr)) {
340 			wpa_printf(MSG_DEBUG, "Failed to get InstanceName "
341 				   "from event.");
342 			VariantClear(&vtClass);
343 			break;
344 		}
345 
346 		if (wcscmp(vtClass.bstrVal,
347 			   L"MSNdis_NotifyAdapterArrival") == 0) {
348 			wpa_printf(MSG_DEBUG, "ndis_events_indicate: Try to "
349 				   "update adapter description since it may "
350 				   "have changed with new adapter instance");
351 			ndis_events_get_adapter(events, events->ifname, NULL);
352 		}
353 
354 		if (wcscmp(events->adapter_desc, vt.bstrVal) != 0) {
355 			wpa_printf(MSG_DEBUG, "ndis_events_indicate: Ignore "
356 				   "indication for foreign adapter: "
357 				   "InstanceName: '%S' __CLASS: '%S'",
358 				   vt.bstrVal, vtClass.bstrVal);
359 			VariantClear(&vtClass);
360 			VariantClear(&vt);
361 			continue;
362 		}
363 		VariantClear(&vt);
364 
365 		if (wcscmp(vtClass.bstrVal,
366 			   L"MSNdis_StatusMediaSpecificIndication") == 0) {
367 			ndis_events_media_specific(events, pObj);
368 		} else if (wcscmp(vtClass.bstrVal,
369 				  L"MSNdis_StatusMediaConnect") == 0) {
370 			ndis_events_media_connect(events);
371 		} else if (wcscmp(vtClass.bstrVal,
372 				  L"MSNdis_StatusMediaDisconnect") == 0) {
373 			ndis_events_media_disconnect(events);
374 		} else if (wcscmp(vtClass.bstrVal,
375 				  L"MSNdis_NotifyAdapterArrival") == 0) {
376 			ndis_events_adapter_arrival(events);
377 		} else if (wcscmp(vtClass.bstrVal,
378 				  L"MSNdis_NotifyAdapterRemoval") == 0) {
379 			ndis_events_adapter_removal(events);
380 		} else {
381 			wpa_printf(MSG_DEBUG, "Unepected event - __CLASS: "
382 				   "'%S'", vtClass.bstrVal);
383 		}
384 
385 		VariantClear(&vtClass);
386 	}
387 
388 	return WBEM_NO_ERROR;
389 }
390 
391 
392 static HRESULT STDMETHODCALLTYPE
393 ndis_events_set_status(IWbemObjectSink *this, long lFlags, HRESULT hResult,
394 		       BSTR strParam, IWbemClassObject __RPC_FAR *pObjParam)
395 {
396 	return WBEM_NO_ERROR;
397 }
398 
399 
400 static int notification_query(IWbemObjectSink *pDestSink,
401 			      IWbemServices *pSvc, const char *class_name)
402 {
403 	HRESULT hr;
404 	WCHAR query[256];
405 
406 	_snwprintf(query, 256,
407 		  L"SELECT * FROM %S", class_name);
408 	wpa_printf(MSG_DEBUG, "ndis_events: WMI: %S", query);
409 	hr = call_IWbemServices_ExecNotificationQueryAsync(
410 		pSvc, L"WQL", query, 0, 0, pDestSink);
411 	if (FAILED(hr)) {
412 		wpa_printf(MSG_DEBUG, "ExecNotificationQueryAsync for %s "
413 			   "failed with hresult of 0x%x",
414 			   class_name, (int) hr);
415 		return -1;
416 	}
417 
418 	return 0;
419 }
420 
421 
422 static int register_async_notification(IWbemObjectSink *pDestSink,
423 				       IWbemServices *pSvc)
424 {
425 	int i;
426 	const char *class_list[] = {
427 		"MSNdis_StatusMediaConnect",
428 		"MSNdis_StatusMediaDisconnect",
429 		"MSNdis_StatusMediaSpecificIndication",
430 		"MSNdis_NotifyAdapterArrival",
431 		"MSNdis_NotifyAdapterRemoval",
432 		NULL
433 	};
434 
435 	for (i = 0; class_list[i]; i++) {
436 		if (notification_query(pDestSink, pSvc, class_list[i]) < 0)
437 			return -1;
438 	}
439 
440 	return 0;
441 }
442 
443 
444 void ndis_events_deinit(struct ndis_events_data *events)
445 {
446 	events->terminating = 1;
447 	IWbemServices_CancelAsyncCall(events->pSvc, &events->sink);
448 	IWbemObjectSink_Release(&events->sink);
449 	/*
450 	 * Rest of deinitialization is done in ndis_events_destructor() once
451 	 * all reference count drops to zero.
452 	 */
453 }
454 
455 
456 static int ndis_events_use_desc(struct ndis_events_data *events,
457 				const char *desc)
458 {
459 	char *tmp, *pos;
460 	size_t len;
461 
462 	if (desc == NULL) {
463 		if (events->adapter_desc == NULL)
464 			return -1;
465 		/* Continue using old description */
466 		return 0;
467 	}
468 
469 	tmp = os_strdup(desc);
470 	if (tmp == NULL)
471 		return -1;
472 
473 	pos = os_strstr(tmp, " (Microsoft's Packet Scheduler)");
474 	if (pos)
475 		*pos = '\0';
476 
477 	len = os_strlen(tmp);
478 	events->adapter_desc = os_malloc((len + 1) * sizeof(WCHAR));
479 	if (events->adapter_desc == NULL) {
480 		os_free(tmp);
481 		return -1;
482 	}
483 	_snwprintf(events->adapter_desc, len + 1, L"%S", tmp);
484 	os_free(tmp);
485 	return 0;
486 }
487 
488 
489 static int ndis_events_get_adapter(struct ndis_events_data *events,
490 				   const char *ifname, const char *desc)
491 {
492 	HRESULT hr;
493 	IWbemServices *pSvc;
494 #define MAX_QUERY_LEN 256
495 	WCHAR query[MAX_QUERY_LEN];
496 	IEnumWbemClassObject *pEnumerator;
497 	IWbemClassObject *pObj;
498 	ULONG uReturned;
499 	VARIANT vt;
500 	int len, pos;
501 
502 	/*
503 	 * Try to get adapter descriptor through WMI CIMv2 Win32_NetworkAdapter
504 	 * to have better probability of matching with InstanceName from
505 	 * MSNdis events. If this fails, use the provided description.
506 	 */
507 
508 	os_free(events->adapter_desc);
509 	events->adapter_desc = NULL;
510 
511 	hr = call_IWbemLocator_ConnectServer(
512 		events->pLoc, L"ROOT\\CIMV2", NULL, NULL, 0, 0, 0, 0, &pSvc);
513 	if (FAILED(hr)) {
514 		wpa_printf(MSG_ERROR, "ndis_events: Could not connect to WMI "
515 			   "server (ROOT\\CIMV2) - error 0x%x", (int) hr);
516 		return ndis_events_use_desc(events, desc);
517 	}
518 	wpa_printf(MSG_DEBUG, "ndis_events: Connected to ROOT\\CIMV2.");
519 
520 	_snwprintf(query, MAX_QUERY_LEN,
521 		  L"SELECT Index FROM Win32_NetworkAdapterConfiguration "
522 		  L"WHERE SettingID='%S'", ifname);
523 	wpa_printf(MSG_DEBUG, "ndis_events: WMI: %S", query);
524 
525 	hr = call_IWbemServices_ExecQuery(
526 		pSvc, L"WQL", query,
527 		WBEM_FLAG_FORWARD_ONLY | WBEM_FLAG_RETURN_IMMEDIATELY,
528 		NULL, &pEnumerator);
529 	if (!SUCCEEDED(hr)) {
530 		wpa_printf(MSG_DEBUG, "ndis_events: Failed to query interface "
531 			   "GUID from Win32_NetworkAdapterConfiguration: "
532 			   "0x%x", (int) hr);
533 		IWbemServices_Release(pSvc);
534 		return ndis_events_use_desc(events, desc);
535 	}
536 
537 	uReturned = 0;
538 	hr = IEnumWbemClassObject_Next(pEnumerator, WBEM_INFINITE, 1,
539 				       &pObj, &uReturned);
540 	if (!SUCCEEDED(hr) || uReturned == 0) {
541 		wpa_printf(MSG_DEBUG, "ndis_events: Failed to find interface "
542 			   "GUID from Win32_NetworkAdapterConfiguration: "
543 			   "0x%x", (int) hr);
544 		IEnumWbemClassObject_Release(pEnumerator);
545 		IWbemServices_Release(pSvc);
546 		return ndis_events_use_desc(events, desc);
547 	}
548 	IEnumWbemClassObject_Release(pEnumerator);
549 
550 	VariantInit(&vt);
551 	hr = IWbemClassObject_Get(pObj, L"Index", 0, &vt, NULL, NULL);
552 	if (!SUCCEEDED(hr)) {
553 		wpa_printf(MSG_DEBUG, "ndis_events: Failed to get Index from "
554 			   "Win32_NetworkAdapterConfiguration: 0x%x",
555 			   (int) hr);
556 		IWbemServices_Release(pSvc);
557 		return ndis_events_use_desc(events, desc);
558 	}
559 
560 	_snwprintf(query, MAX_QUERY_LEN,
561 		  L"SELECT Name,PNPDeviceID FROM Win32_NetworkAdapter WHERE "
562 		  L"Index=%d",
563 		  vt.uintVal);
564 	wpa_printf(MSG_DEBUG, "ndis_events: WMI: %S", query);
565 	VariantClear(&vt);
566 	IWbemClassObject_Release(pObj);
567 
568 	hr = call_IWbemServices_ExecQuery(
569 		pSvc, L"WQL", query,
570 		WBEM_FLAG_FORWARD_ONLY | WBEM_FLAG_RETURN_IMMEDIATELY,
571 		NULL, &pEnumerator);
572 	if (!SUCCEEDED(hr)) {
573 		wpa_printf(MSG_DEBUG, "ndis_events: Failed to query interface "
574 			   "from Win32_NetworkAdapter: 0x%x", (int) hr);
575 		IWbemServices_Release(pSvc);
576 		return ndis_events_use_desc(events, desc);
577 	}
578 
579 	uReturned = 0;
580 	hr = IEnumWbemClassObject_Next(pEnumerator, WBEM_INFINITE, 1,
581 				       &pObj, &uReturned);
582 	if (!SUCCEEDED(hr) || uReturned == 0) {
583 		wpa_printf(MSG_DEBUG, "ndis_events: Failed to find interface "
584 			   "from Win32_NetworkAdapter: 0x%x", (int) hr);
585 		IEnumWbemClassObject_Release(pEnumerator);
586 		IWbemServices_Release(pSvc);
587 		return ndis_events_use_desc(events, desc);
588 	}
589 	IEnumWbemClassObject_Release(pEnumerator);
590 
591 	hr = IWbemClassObject_Get(pObj, L"Name", 0, &vt, NULL, NULL);
592 	if (!SUCCEEDED(hr)) {
593 		wpa_printf(MSG_DEBUG, "ndis_events: Failed to get Name from "
594 			   "Win32_NetworkAdapter: 0x%x", (int) hr);
595 		IWbemClassObject_Release(pObj);
596 		IWbemServices_Release(pSvc);
597 		return ndis_events_use_desc(events, desc);
598 	}
599 
600 	wpa_printf(MSG_DEBUG, "ndis_events: Win32_NetworkAdapter::Name='%S'",
601 		   vt.bstrVal);
602 	events->adapter_desc = _wcsdup(vt.bstrVal);
603 	VariantClear(&vt);
604 
605 	/*
606 	 * Try to get even better candidate for matching with InstanceName
607 	 * from Win32_PnPEntity. This is needed at least for some USB cards
608 	 * that can change the InstanceName whenever being unplugged and
609 	 * plugged again.
610 	 */
611 
612 	hr = IWbemClassObject_Get(pObj, L"PNPDeviceID", 0, &vt, NULL, NULL);
613 	if (!SUCCEEDED(hr)) {
614 		wpa_printf(MSG_DEBUG, "ndis_events: Failed to get PNPDeviceID "
615 			   "from Win32_NetworkAdapter: 0x%x", (int) hr);
616 		IWbemClassObject_Release(pObj);
617 		IWbemServices_Release(pSvc);
618 		if (events->adapter_desc == NULL)
619 			return ndis_events_use_desc(events, desc);
620 		return 0; /* use Win32_NetworkAdapter::Name */
621 	}
622 
623 	wpa_printf(MSG_DEBUG, "ndis_events: Win32_NetworkAdapter::PNPDeviceID="
624 		   "'%S'", vt.bstrVal);
625 
626 	len = _snwprintf(query, MAX_QUERY_LEN,
627 			L"SELECT Name FROM Win32_PnPEntity WHERE DeviceID='");
628 	if (len < 0 || len >= MAX_QUERY_LEN - 1) {
629 		VariantClear(&vt);
630 		IWbemClassObject_Release(pObj);
631 		IWbemServices_Release(pSvc);
632 		if (events->adapter_desc == NULL)
633 			return ndis_events_use_desc(events, desc);
634 		return 0; /* use Win32_NetworkAdapter::Name */
635 	}
636 
637 	/* Escape \ as \\ */
638 	for (pos = 0; vt.bstrVal[pos] && len < MAX_QUERY_LEN - 2; pos++) {
639 		if (vt.bstrVal[pos] == '\\') {
640 			if (len >= MAX_QUERY_LEN - 3)
641 				break;
642 			query[len++] = '\\';
643 		}
644 		query[len++] = vt.bstrVal[pos];
645 	}
646 	query[len++] = L'\'';
647 	query[len] = L'\0';
648 	VariantClear(&vt);
649 	IWbemClassObject_Release(pObj);
650 	wpa_printf(MSG_DEBUG, "ndis_events: WMI: %S", query);
651 
652 	hr = call_IWbemServices_ExecQuery(
653 		pSvc, L"WQL", query,
654 		WBEM_FLAG_FORWARD_ONLY | WBEM_FLAG_RETURN_IMMEDIATELY,
655 		NULL, &pEnumerator);
656 	if (!SUCCEEDED(hr)) {
657 		wpa_printf(MSG_DEBUG, "ndis_events: Failed to query interface "
658 			   "Name from Win32_PnPEntity: 0x%x", (int) hr);
659 		IWbemServices_Release(pSvc);
660 		if (events->adapter_desc == NULL)
661 			return ndis_events_use_desc(events, desc);
662 		return 0; /* use Win32_NetworkAdapter::Name */
663 	}
664 
665 	uReturned = 0;
666 	hr = IEnumWbemClassObject_Next(pEnumerator, WBEM_INFINITE, 1,
667 				       &pObj, &uReturned);
668 	if (!SUCCEEDED(hr) || uReturned == 0) {
669 		wpa_printf(MSG_DEBUG, "ndis_events: Failed to find interface "
670 			   "from Win32_PnPEntity: 0x%x", (int) hr);
671 		IEnumWbemClassObject_Release(pEnumerator);
672 		IWbemServices_Release(pSvc);
673 		if (events->adapter_desc == NULL)
674 			return ndis_events_use_desc(events, desc);
675 		return 0; /* use Win32_NetworkAdapter::Name */
676 	}
677 	IEnumWbemClassObject_Release(pEnumerator);
678 
679 	hr = IWbemClassObject_Get(pObj, L"Name", 0, &vt, NULL, NULL);
680 	if (!SUCCEEDED(hr)) {
681 		wpa_printf(MSG_DEBUG, "ndis_events: Failed to get Name from "
682 			   "Win32_PnPEntity: 0x%x", (int) hr);
683 		IWbemClassObject_Release(pObj);
684 		IWbemServices_Release(pSvc);
685 		if (events->adapter_desc == NULL)
686 			return ndis_events_use_desc(events, desc);
687 		return 0; /* use Win32_NetworkAdapter::Name */
688 	}
689 
690 	wpa_printf(MSG_DEBUG, "ndis_events: Win32_PnPEntity::Name='%S'",
691 		   vt.bstrVal);
692 	os_free(events->adapter_desc);
693 	events->adapter_desc = _wcsdup(vt.bstrVal);
694 	VariantClear(&vt);
695 
696 	IWbemClassObject_Release(pObj);
697 
698 	IWbemServices_Release(pSvc);
699 
700 	if (events->adapter_desc == NULL)
701 		return ndis_events_use_desc(events, desc);
702 
703 	return 0;
704 }
705 
706 
707 struct ndis_events_data *
708 ndis_events_init(HANDLE *read_pipe, HANDLE *event_avail,
709 		 const char *ifname, const char *desc)
710 {
711 	HRESULT hr;
712 	IWbemObjectSink *pSink;
713 	struct ndis_events_data *events;
714 
715 	events = os_zalloc(sizeof(*events));
716 	if (events == NULL) {
717 		wpa_printf(MSG_ERROR, "Could not allocate sink for events.");
718 		return NULL;
719 	}
720 	events->ifname = os_strdup(ifname);
721 	if (events->ifname == NULL) {
722 		os_free(events);
723 		return NULL;
724 	}
725 
726 	if (wmi_refcnt++ == 0) {
727 		hr = CoInitializeEx(0, COINIT_MULTITHREADED);
728 		if (FAILED(hr)) {
729 			wpa_printf(MSG_ERROR, "CoInitializeEx() failed - "
730 				   "returned 0x%x", (int) hr);
731 			os_free(events);
732 			return NULL;
733 		}
734 	}
735 
736 	if (wmi_first) {
737 		/* CoInitializeSecurity() must be called once and only once
738 		 * per process, so let's use wmi_first flag to protect against
739 		 * multiple calls. */
740 		wmi_first = 0;
741 
742 		hr = CoInitializeSecurity(NULL, -1, NULL, NULL,
743 					  RPC_C_AUTHN_LEVEL_PKT_PRIVACY,
744 					  RPC_C_IMP_LEVEL_IMPERSONATE,
745 					  NULL, EOAC_SECURE_REFS, NULL);
746 		if (FAILED(hr)) {
747 			wpa_printf(MSG_ERROR, "CoInitializeSecurity() failed "
748 				   "- returned 0x%x", (int) hr);
749 			os_free(events);
750 			return NULL;
751 		}
752 	}
753 
754 	hr = CoCreateInstance(&CLSID_WbemLocator, 0, CLSCTX_INPROC_SERVER,
755 			      &IID_IWbemLocator,
756 			      (LPVOID *) (void *) &events->pLoc);
757 	if (FAILED(hr)) {
758 		wpa_printf(MSG_ERROR, "CoCreateInstance() failed - returned "
759 			   "0x%x", (int) hr);
760 		CoUninitialize();
761 		os_free(events);
762 		return NULL;
763 	}
764 
765 	if (ndis_events_get_adapter(events, ifname, desc) < 0) {
766 		CoUninitialize();
767 		os_free(events);
768 		return NULL;
769 	}
770 	wpa_printf(MSG_DEBUG, "ndis_events: use adapter descriptor '%S'",
771 		   events->adapter_desc);
772 
773 	hr = call_IWbemLocator_ConnectServer(
774 		events->pLoc, L"ROOT\\WMI", NULL, NULL,
775 		0, 0, 0, 0, &events->pSvc);
776 	if (FAILED(hr)) {
777 		wpa_printf(MSG_ERROR, "Could not connect to server - error "
778 			   "0x%x", (int) hr);
779 		CoUninitialize();
780 		os_free(events->adapter_desc);
781 		os_free(events);
782 		return NULL;
783 	}
784 	wpa_printf(MSG_DEBUG, "Connected to ROOT\\WMI.");
785 
786 	ndis_events_constructor(events);
787 	pSink = &events->sink;
788 	pSink->lpVtbl = &events->sink_vtbl;
789 	events->sink_vtbl.QueryInterface = ndis_events_query_interface;
790 	events->sink_vtbl.AddRef = ndis_events_add_ref;
791 	events->sink_vtbl.Release = ndis_events_release;
792 	events->sink_vtbl.Indicate = ndis_events_indicate;
793 	events->sink_vtbl.SetStatus = ndis_events_set_status;
794 
795 	if (register_async_notification(pSink, events->pSvc) < 0) {
796 		wpa_printf(MSG_DEBUG, "Failed to register async "
797 			   "notifications");
798 		ndis_events_destructor(events);
799 		os_free(events->adapter_desc);
800 		os_free(events);
801 		return NULL;
802 	}
803 
804 	*read_pipe = events->read_pipe;
805 	*event_avail = events->event_avail;
806 
807 	return events;
808 }
809