xref: /dflybsd-src/contrib/wpa_supplicant/src/ap/ap_list.c (revision 3a84a4273475ed07d0ab1c2dfeffdfedef35d9cd)
13ff40c12SJohn Marino /*
23ff40c12SJohn Marino  * hostapd / AP table
33ff40c12SJohn Marino  * Copyright (c) 2002-2009, Jouni Malinen <j@w1.fi>
43ff40c12SJohn Marino  * Copyright (c) 2003-2004, Instant802 Networks, Inc.
53ff40c12SJohn Marino  * Copyright (c) 2006, Devicescape Software, Inc.
63ff40c12SJohn Marino  *
73ff40c12SJohn Marino  * This software may be distributed under the terms of the BSD license.
83ff40c12SJohn Marino  * See README for more details.
93ff40c12SJohn Marino  */
103ff40c12SJohn Marino 
113ff40c12SJohn Marino #include "utils/includes.h"
123ff40c12SJohn Marino 
133ff40c12SJohn Marino #include "utils/common.h"
143ff40c12SJohn Marino #include "utils/eloop.h"
153ff40c12SJohn Marino #include "common/ieee802_11_defs.h"
163ff40c12SJohn Marino #include "common/ieee802_11_common.h"
173ff40c12SJohn Marino #include "hostapd.h"
183ff40c12SJohn Marino #include "ap_config.h"
193ff40c12SJohn Marino #include "ieee802_11.h"
203ff40c12SJohn Marino #include "sta_info.h"
213ff40c12SJohn Marino #include "beacon.h"
223ff40c12SJohn Marino #include "ap_list.h"
233ff40c12SJohn Marino 
243ff40c12SJohn Marino 
253ff40c12SJohn Marino /* AP list is a double linked list with head->prev pointing to the end of the
263ff40c12SJohn Marino  * list and tail->next = NULL. Entries are moved to the head of the list
273ff40c12SJohn Marino  * whenever a beacon has been received from the AP in question. The tail entry
283ff40c12SJohn Marino  * in this link will thus be the least recently used entry. */
293ff40c12SJohn Marino 
303ff40c12SJohn Marino 
ap_list_beacon_olbc(struct hostapd_iface * iface,struct ap_info * ap)313ff40c12SJohn Marino static int ap_list_beacon_olbc(struct hostapd_iface *iface, struct ap_info *ap)
323ff40c12SJohn Marino {
333ff40c12SJohn Marino 	int i;
343ff40c12SJohn Marino 
35*a1157835SDaniel Fojt 	if (iface->current_mode == NULL ||
36*a1157835SDaniel Fojt 	    iface->current_mode->mode != HOSTAPD_MODE_IEEE80211G ||
373ff40c12SJohn Marino 	    iface->conf->channel != ap->channel)
383ff40c12SJohn Marino 		return 0;
393ff40c12SJohn Marino 
403ff40c12SJohn Marino 	if (ap->erp != -1 && (ap->erp & ERP_INFO_NON_ERP_PRESENT))
413ff40c12SJohn Marino 		return 1;
423ff40c12SJohn Marino 
433ff40c12SJohn Marino 	for (i = 0; i < WLAN_SUPP_RATES_MAX; i++) {
443ff40c12SJohn Marino 		int rate = (ap->supported_rates[i] & 0x7f) * 5;
453ff40c12SJohn Marino 		if (rate == 60 || rate == 90 || rate > 110)
463ff40c12SJohn Marino 			return 0;
473ff40c12SJohn Marino 	}
483ff40c12SJohn Marino 
493ff40c12SJohn Marino 	return 1;
503ff40c12SJohn Marino }
513ff40c12SJohn Marino 
523ff40c12SJohn Marino 
ap_get_ap(struct hostapd_iface * iface,const u8 * ap)533ff40c12SJohn Marino static struct ap_info * ap_get_ap(struct hostapd_iface *iface, const u8 *ap)
543ff40c12SJohn Marino {
553ff40c12SJohn Marino 	struct ap_info *s;
563ff40c12SJohn Marino 
573ff40c12SJohn Marino 	s = iface->ap_hash[STA_HASH(ap)];
583ff40c12SJohn Marino 	while (s != NULL && os_memcmp(s->addr, ap, ETH_ALEN) != 0)
593ff40c12SJohn Marino 		s = s->hnext;
603ff40c12SJohn Marino 	return s;
613ff40c12SJohn Marino }
623ff40c12SJohn Marino 
633ff40c12SJohn Marino 
ap_ap_list_add(struct hostapd_iface * iface,struct ap_info * ap)643ff40c12SJohn Marino static void ap_ap_list_add(struct hostapd_iface *iface, struct ap_info *ap)
653ff40c12SJohn Marino {
663ff40c12SJohn Marino 	if (iface->ap_list) {
673ff40c12SJohn Marino 		ap->prev = iface->ap_list->prev;
683ff40c12SJohn Marino 		iface->ap_list->prev = ap;
693ff40c12SJohn Marino 	} else
703ff40c12SJohn Marino 		ap->prev = ap;
713ff40c12SJohn Marino 	ap->next = iface->ap_list;
723ff40c12SJohn Marino 	iface->ap_list = ap;
733ff40c12SJohn Marino }
743ff40c12SJohn Marino 
753ff40c12SJohn Marino 
ap_ap_list_del(struct hostapd_iface * iface,struct ap_info * ap)763ff40c12SJohn Marino static void ap_ap_list_del(struct hostapd_iface *iface, struct ap_info *ap)
773ff40c12SJohn Marino {
783ff40c12SJohn Marino 	if (iface->ap_list == ap)
793ff40c12SJohn Marino 		iface->ap_list = ap->next;
803ff40c12SJohn Marino 	else
813ff40c12SJohn Marino 		ap->prev->next = ap->next;
823ff40c12SJohn Marino 
833ff40c12SJohn Marino 	if (ap->next)
843ff40c12SJohn Marino 		ap->next->prev = ap->prev;
853ff40c12SJohn Marino 	else if (iface->ap_list)
863ff40c12SJohn Marino 		iface->ap_list->prev = ap->prev;
873ff40c12SJohn Marino }
883ff40c12SJohn Marino 
893ff40c12SJohn Marino 
ap_ap_hash_add(struct hostapd_iface * iface,struct ap_info * ap)903ff40c12SJohn Marino static void ap_ap_hash_add(struct hostapd_iface *iface, struct ap_info *ap)
913ff40c12SJohn Marino {
923ff40c12SJohn Marino 	ap->hnext = iface->ap_hash[STA_HASH(ap->addr)];
933ff40c12SJohn Marino 	iface->ap_hash[STA_HASH(ap->addr)] = ap;
943ff40c12SJohn Marino }
953ff40c12SJohn Marino 
963ff40c12SJohn Marino 
ap_ap_hash_del(struct hostapd_iface * iface,struct ap_info * ap)973ff40c12SJohn Marino static void ap_ap_hash_del(struct hostapd_iface *iface, struct ap_info *ap)
983ff40c12SJohn Marino {
993ff40c12SJohn Marino 	struct ap_info *s;
1003ff40c12SJohn Marino 
1013ff40c12SJohn Marino 	s = iface->ap_hash[STA_HASH(ap->addr)];
1023ff40c12SJohn Marino 	if (s == NULL) return;
1033ff40c12SJohn Marino 	if (os_memcmp(s->addr, ap->addr, ETH_ALEN) == 0) {
1043ff40c12SJohn Marino 		iface->ap_hash[STA_HASH(ap->addr)] = s->hnext;
1053ff40c12SJohn Marino 		return;
1063ff40c12SJohn Marino 	}
1073ff40c12SJohn Marino 
1083ff40c12SJohn Marino 	while (s->hnext != NULL &&
1093ff40c12SJohn Marino 	       os_memcmp(s->hnext->addr, ap->addr, ETH_ALEN) != 0)
1103ff40c12SJohn Marino 		s = s->hnext;
1113ff40c12SJohn Marino 	if (s->hnext != NULL)
1123ff40c12SJohn Marino 		s->hnext = s->hnext->hnext;
1133ff40c12SJohn Marino 	else
114*a1157835SDaniel Fojt 		wpa_printf(MSG_INFO, "AP: could not remove AP " MACSTR
115*a1157835SDaniel Fojt 			   " from hash table",  MAC2STR(ap->addr));
1163ff40c12SJohn Marino }
1173ff40c12SJohn Marino 
1183ff40c12SJohn Marino 
ap_free_ap(struct hostapd_iface * iface,struct ap_info * ap)1193ff40c12SJohn Marino static void ap_free_ap(struct hostapd_iface *iface, struct ap_info *ap)
1203ff40c12SJohn Marino {
1213ff40c12SJohn Marino 	ap_ap_hash_del(iface, ap);
1223ff40c12SJohn Marino 	ap_ap_list_del(iface, ap);
1233ff40c12SJohn Marino 
1243ff40c12SJohn Marino 	iface->num_ap--;
1253ff40c12SJohn Marino 	os_free(ap);
1263ff40c12SJohn Marino }
1273ff40c12SJohn Marino 
1283ff40c12SJohn Marino 
hostapd_free_aps(struct hostapd_iface * iface)1293ff40c12SJohn Marino static void hostapd_free_aps(struct hostapd_iface *iface)
1303ff40c12SJohn Marino {
1313ff40c12SJohn Marino 	struct ap_info *ap, *prev;
1323ff40c12SJohn Marino 
1333ff40c12SJohn Marino 	ap = iface->ap_list;
1343ff40c12SJohn Marino 
1353ff40c12SJohn Marino 	while (ap) {
1363ff40c12SJohn Marino 		prev = ap;
1373ff40c12SJohn Marino 		ap = ap->next;
1383ff40c12SJohn Marino 		ap_free_ap(iface, prev);
1393ff40c12SJohn Marino 	}
1403ff40c12SJohn Marino 
1413ff40c12SJohn Marino 	iface->ap_list = NULL;
1423ff40c12SJohn Marino }
1433ff40c12SJohn Marino 
1443ff40c12SJohn Marino 
ap_ap_add(struct hostapd_iface * iface,const u8 * addr)1453ff40c12SJohn Marino static struct ap_info * ap_ap_add(struct hostapd_iface *iface, const u8 *addr)
1463ff40c12SJohn Marino {
1473ff40c12SJohn Marino 	struct ap_info *ap;
1483ff40c12SJohn Marino 
1493ff40c12SJohn Marino 	ap = os_zalloc(sizeof(struct ap_info));
1503ff40c12SJohn Marino 	if (ap == NULL)
1513ff40c12SJohn Marino 		return NULL;
1523ff40c12SJohn Marino 
1533ff40c12SJohn Marino 	/* initialize AP info data */
1543ff40c12SJohn Marino 	os_memcpy(ap->addr, addr, ETH_ALEN);
1553ff40c12SJohn Marino 	ap_ap_list_add(iface, ap);
1563ff40c12SJohn Marino 	iface->num_ap++;
1573ff40c12SJohn Marino 	ap_ap_hash_add(iface, ap);
1583ff40c12SJohn Marino 
1593ff40c12SJohn Marino 	if (iface->num_ap > iface->conf->ap_table_max_size && ap != ap->prev) {
1603ff40c12SJohn Marino 		wpa_printf(MSG_DEBUG, "Removing the least recently used AP "
1613ff40c12SJohn Marino 			   MACSTR " from AP table", MAC2STR(ap->prev->addr));
1623ff40c12SJohn Marino 		ap_free_ap(iface, ap->prev);
1633ff40c12SJohn Marino 	}
1643ff40c12SJohn Marino 
1653ff40c12SJohn Marino 	return ap;
1663ff40c12SJohn Marino }
1673ff40c12SJohn Marino 
1683ff40c12SJohn Marino 
ap_list_process_beacon(struct hostapd_iface * iface,const struct ieee80211_mgmt * mgmt,struct ieee802_11_elems * elems,struct hostapd_frame_info * fi)1693ff40c12SJohn Marino void ap_list_process_beacon(struct hostapd_iface *iface,
1703ff40c12SJohn Marino 			    const struct ieee80211_mgmt *mgmt,
1713ff40c12SJohn Marino 			    struct ieee802_11_elems *elems,
1723ff40c12SJohn Marino 			    struct hostapd_frame_info *fi)
1733ff40c12SJohn Marino {
1743ff40c12SJohn Marino 	struct ap_info *ap;
1753ff40c12SJohn Marino 	int new_ap = 0;
1763ff40c12SJohn Marino 	int set_beacon = 0;
1773ff40c12SJohn Marino 
1783ff40c12SJohn Marino 	if (iface->conf->ap_table_max_size < 1)
1793ff40c12SJohn Marino 		return;
1803ff40c12SJohn Marino 
1813ff40c12SJohn Marino 	ap = ap_get_ap(iface, mgmt->bssid);
1823ff40c12SJohn Marino 	if (!ap) {
1833ff40c12SJohn Marino 		ap = ap_ap_add(iface, mgmt->bssid);
1843ff40c12SJohn Marino 		if (!ap) {
185*a1157835SDaniel Fojt 			wpa_printf(MSG_INFO,
186*a1157835SDaniel Fojt 				   "Failed to allocate AP information entry");
1873ff40c12SJohn Marino 			return;
1883ff40c12SJohn Marino 		}
1893ff40c12SJohn Marino 		new_ap = 1;
1903ff40c12SJohn Marino 	}
1913ff40c12SJohn Marino 
1923ff40c12SJohn Marino 	merge_byte_arrays(ap->supported_rates, WLAN_SUPP_RATES_MAX,
1933ff40c12SJohn Marino 			  elems->supp_rates, elems->supp_rates_len,
1943ff40c12SJohn Marino 			  elems->ext_supp_rates, elems->ext_supp_rates_len);
1953ff40c12SJohn Marino 
196*a1157835SDaniel Fojt 	if (elems->erp_info)
1973ff40c12SJohn Marino 		ap->erp = elems->erp_info[0];
1983ff40c12SJohn Marino 	else
1993ff40c12SJohn Marino 		ap->erp = -1;
2003ff40c12SJohn Marino 
201*a1157835SDaniel Fojt 	if (elems->ds_params)
2023ff40c12SJohn Marino 		ap->channel = elems->ds_params[0];
203*a1157835SDaniel Fojt 	else if (elems->ht_operation)
2043ff40c12SJohn Marino 		ap->channel = elems->ht_operation[0];
2053ff40c12SJohn Marino 	else if (fi)
2063ff40c12SJohn Marino 		ap->channel = fi->channel;
2073ff40c12SJohn Marino 
2083ff40c12SJohn Marino 	if (elems->ht_capabilities)
2093ff40c12SJohn Marino 		ap->ht_support = 1;
2103ff40c12SJohn Marino 	else
2113ff40c12SJohn Marino 		ap->ht_support = 0;
2123ff40c12SJohn Marino 
2133ff40c12SJohn Marino 	os_get_reltime(&ap->last_beacon);
2143ff40c12SJohn Marino 
2153ff40c12SJohn Marino 	if (!new_ap && ap != iface->ap_list) {
2163ff40c12SJohn Marino 		/* move AP entry into the beginning of the list so that the
2173ff40c12SJohn Marino 		 * oldest entry is always in the end of the list */
2183ff40c12SJohn Marino 		ap_ap_list_del(iface, ap);
2193ff40c12SJohn Marino 		ap_ap_list_add(iface, ap);
2203ff40c12SJohn Marino 	}
2213ff40c12SJohn Marino 
2223ff40c12SJohn Marino 	if (!iface->olbc &&
2233ff40c12SJohn Marino 	    ap_list_beacon_olbc(iface, ap)) {
2243ff40c12SJohn Marino 		iface->olbc = 1;
2253ff40c12SJohn Marino 		wpa_printf(MSG_DEBUG, "OLBC AP detected: " MACSTR
2263ff40c12SJohn Marino 			   " (channel %d) - enable protection",
2273ff40c12SJohn Marino 			   MAC2STR(ap->addr), ap->channel);
2283ff40c12SJohn Marino 		set_beacon++;
2293ff40c12SJohn Marino 	}
2303ff40c12SJohn Marino 
2313ff40c12SJohn Marino #ifdef CONFIG_IEEE80211N
2323ff40c12SJohn Marino 	if (!iface->olbc_ht && !ap->ht_support &&
2333ff40c12SJohn Marino 	    (ap->channel == 0 ||
2343ff40c12SJohn Marino 	     ap->channel == iface->conf->channel ||
2353ff40c12SJohn Marino 	     ap->channel == iface->conf->channel +
2363ff40c12SJohn Marino 	     iface->conf->secondary_channel * 4)) {
2373ff40c12SJohn Marino 		iface->olbc_ht = 1;
2383ff40c12SJohn Marino 		hostapd_ht_operation_update(iface);
2393ff40c12SJohn Marino 		wpa_printf(MSG_DEBUG, "OLBC HT AP detected: " MACSTR
2403ff40c12SJohn Marino 			   " (channel %d) - enable protection",
2413ff40c12SJohn Marino 			   MAC2STR(ap->addr), ap->channel);
2423ff40c12SJohn Marino 		set_beacon++;
2433ff40c12SJohn Marino 	}
2443ff40c12SJohn Marino #endif /* CONFIG_IEEE80211N */
2453ff40c12SJohn Marino 
2463ff40c12SJohn Marino 	if (set_beacon)
2473ff40c12SJohn Marino 		ieee802_11_update_beacons(iface);
2483ff40c12SJohn Marino }
2493ff40c12SJohn Marino 
2503ff40c12SJohn Marino 
ap_list_timer(struct hostapd_iface * iface)251*a1157835SDaniel Fojt void ap_list_timer(struct hostapd_iface *iface)
2523ff40c12SJohn Marino {
2533ff40c12SJohn Marino 	struct os_reltime now;
2543ff40c12SJohn Marino 	struct ap_info *ap;
2553ff40c12SJohn Marino 	int set_beacon = 0;
2563ff40c12SJohn Marino 
2573ff40c12SJohn Marino 	if (!iface->ap_list)
2583ff40c12SJohn Marino 		return;
2593ff40c12SJohn Marino 
2603ff40c12SJohn Marino 	os_get_reltime(&now);
2613ff40c12SJohn Marino 
2623ff40c12SJohn Marino 	while (iface->ap_list) {
2633ff40c12SJohn Marino 		ap = iface->ap_list->prev;
2643ff40c12SJohn Marino 		if (!os_reltime_expired(&now, &ap->last_beacon,
2653ff40c12SJohn Marino 					iface->conf->ap_table_expiration_time))
2663ff40c12SJohn Marino 			break;
2673ff40c12SJohn Marino 
2683ff40c12SJohn Marino 		ap_free_ap(iface, ap);
2693ff40c12SJohn Marino 	}
2703ff40c12SJohn Marino 
2713ff40c12SJohn Marino 	if (iface->olbc || iface->olbc_ht) {
2723ff40c12SJohn Marino 		int olbc = 0;
2733ff40c12SJohn Marino 		int olbc_ht = 0;
2743ff40c12SJohn Marino 
2753ff40c12SJohn Marino 		ap = iface->ap_list;
2763ff40c12SJohn Marino 		while (ap && (olbc == 0 || olbc_ht == 0)) {
2773ff40c12SJohn Marino 			if (ap_list_beacon_olbc(iface, ap))
2783ff40c12SJohn Marino 				olbc = 1;
2793ff40c12SJohn Marino 			if (!ap->ht_support)
2803ff40c12SJohn Marino 				olbc_ht = 1;
2813ff40c12SJohn Marino 			ap = ap->next;
2823ff40c12SJohn Marino 		}
2833ff40c12SJohn Marino 		if (!olbc && iface->olbc) {
2843ff40c12SJohn Marino 			wpa_printf(MSG_DEBUG, "OLBC not detected anymore");
2853ff40c12SJohn Marino 			iface->olbc = 0;
2863ff40c12SJohn Marino 			set_beacon++;
2873ff40c12SJohn Marino 		}
2883ff40c12SJohn Marino #ifdef CONFIG_IEEE80211N
2893ff40c12SJohn Marino 		if (!olbc_ht && iface->olbc_ht) {
2903ff40c12SJohn Marino 			wpa_printf(MSG_DEBUG, "OLBC HT not detected anymore");
2913ff40c12SJohn Marino 			iface->olbc_ht = 0;
2923ff40c12SJohn Marino 			hostapd_ht_operation_update(iface);
2933ff40c12SJohn Marino 			set_beacon++;
2943ff40c12SJohn Marino 		}
2953ff40c12SJohn Marino #endif /* CONFIG_IEEE80211N */
2963ff40c12SJohn Marino 	}
2973ff40c12SJohn Marino 
2983ff40c12SJohn Marino 	if (set_beacon)
2993ff40c12SJohn Marino 		ieee802_11_update_beacons(iface);
3003ff40c12SJohn Marino }
3013ff40c12SJohn Marino 
3023ff40c12SJohn Marino 
ap_list_init(struct hostapd_iface * iface)3033ff40c12SJohn Marino int ap_list_init(struct hostapd_iface *iface)
3043ff40c12SJohn Marino {
3053ff40c12SJohn Marino 	return 0;
3063ff40c12SJohn Marino }
3073ff40c12SJohn Marino 
3083ff40c12SJohn Marino 
ap_list_deinit(struct hostapd_iface * iface)3093ff40c12SJohn Marino void ap_list_deinit(struct hostapd_iface *iface)
3103ff40c12SJohn Marino {
3113ff40c12SJohn Marino 	hostapd_free_aps(iface);
3123ff40c12SJohn Marino }
313