18d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt/*
28d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt * WPA Supplicant - Windows/NDIS driver interface - event processing
38d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt * Copyright (c) 2004-2005, Jouni Malinen <j@w1.fi>
48d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt *
58d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt * This program is free software; you can redistribute it and/or modify
68d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt * it under the terms of the GNU General Public License version 2 as
78d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt * published by the Free Software Foundation.
88d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt *
98d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt * Alternatively, this software may be distributed under the terms of BSD
108d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt * license.
118d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt *
128d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt * See README and COPYING for more details.
138d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt */
148d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt
158d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt#include "includes.h"
168d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt
178d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt#include "common.h"
188d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt#include "driver.h"
198d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt#include "eloop.h"
208d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt
218d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt/* Keep this event processing in a separate file and without WinPcap headers to
228d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt * avoid conflicts with some of the header files. */
238d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidtstruct _ADAPTER;
248d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidttypedef struct _ADAPTER * LPADAPTER;
258d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt#include "driver_ndis.h"
268d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt
278d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt
288d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidtvoid wpa_driver_ndis_event_connect(struct wpa_driver_ndis_data *drv);
298d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidtvoid wpa_driver_ndis_event_disconnect(struct wpa_driver_ndis_data *drv);
308d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidtvoid wpa_driver_ndis_event_media_specific(struct wpa_driver_ndis_data *drv,
318d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt					  const u8 *data, size_t data_len);
328d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidtvoid wpa_driver_ndis_event_adapter_arrival(struct wpa_driver_ndis_data *drv);
338d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidtvoid wpa_driver_ndis_event_adapter_removal(struct wpa_driver_ndis_data *drv);
348d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt
358d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt
368d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidtenum event_types { EVENT_CONNECT, EVENT_DISCONNECT,
378d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		   EVENT_MEDIA_SPECIFIC, EVENT_ADAPTER_ARRIVAL,
388d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		   EVENT_ADAPTER_REMOVAL };
398d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt
408d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt/* Event data:
418d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt * enum event_types (as int, i.e., 4 octets)
428d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt * data length (2 octets (big endian), optional)
438d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt * data (variable len, optional)
448d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt */
458d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt
468d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt
478d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidtstatic void wpa_driver_ndis_event_process(struct wpa_driver_ndis_data *drv,
488d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt					  u8 *buf, size_t len)
498d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt{
508d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	u8 *pos, *data = NULL;
518d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	enum event_types type;
528d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	size_t data_len = 0;
538d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt
548d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	wpa_hexdump(MSG_MSGDUMP, "NDIS: received event data", buf, len);
558d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	if (len < sizeof(int))
568d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		return;
578d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	type = *((int *) buf);
588d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	pos = buf + sizeof(int);
598d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	wpa_printf(MSG_DEBUG, "NDIS: event - type %d", type);
608d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt
618d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	if (buf + len - pos > 2) {
628d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		data_len = (int) *pos++ << 8;
638d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		data_len += *pos++;
648d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		if (data_len > (size_t) (buf + len - pos)) {
658d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt			wpa_printf(MSG_DEBUG, "NDIS: event data overflow");
668d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt			return;
678d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		}
688d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		data = pos;
698d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		wpa_hexdump(MSG_MSGDUMP, "NDIS: event data", data, data_len);
708d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	}
718d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt
728d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	switch (type) {
738d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	case EVENT_CONNECT:
748d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		wpa_driver_ndis_event_connect(drv);
758d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		break;
768d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	case EVENT_DISCONNECT:
778d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		wpa_driver_ndis_event_disconnect(drv);
788d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		break;
798d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	case EVENT_MEDIA_SPECIFIC:
808d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		wpa_driver_ndis_event_media_specific(drv, data, data_len);
818d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		break;
828d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	case EVENT_ADAPTER_ARRIVAL:
838d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		wpa_driver_ndis_event_adapter_arrival(drv);
848d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		break;
858d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	case EVENT_ADAPTER_REMOVAL:
868d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		wpa_driver_ndis_event_adapter_removal(drv);
878d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		break;
888d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	}
898d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt}
908d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt
918d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt
928d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidtvoid wpa_driver_ndis_event_pipe_cb(void *eloop_data, void *user_data)
938d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt{
948d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	struct wpa_driver_ndis_data *drv = eloop_data;
958d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	u8 buf[512];
968d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	DWORD len;
978d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt
988d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	ResetEvent(drv->event_avail);
998d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	if (ReadFile(drv->events_pipe, buf, sizeof(buf), &len, NULL))
1008d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		wpa_driver_ndis_event_process(drv, buf, len);
1018d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	else {
1028d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt		wpa_printf(MSG_DEBUG, "%s: ReadFile() failed: %d", __func__,
1038d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt			   (int) GetLastError());
1048d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt	}
1058d520ff1dc2da35cdca849e982051b86468016d8Dmitry Shmidt}
106