xref: /openbsd-src/sys/net80211/ieee80211_ioctl.c (revision ada90a5b2c4a609a9b698d98b0173b8ee03c34c4)
1*ada90a5bSstsp /*	$OpenBSD: ieee80211_ioctl.c,v 1.81 2022/03/07 08:13:13 stsp Exp $	*/
291b2158bSmillert /*	$NetBSD: ieee80211_ioctl.c,v 1.15 2004/05/06 02:58:16 dyoung Exp $	*/
391b2158bSmillert 
491b2158bSmillert /*-
591b2158bSmillert  * Copyright (c) 2001 Atsushi Onoe
691b2158bSmillert  * Copyright (c) 2002, 2003 Sam Leffler, Errno Consulting
791b2158bSmillert  * All rights reserved.
891b2158bSmillert  *
991b2158bSmillert  * Redistribution and use in source and binary forms, with or without
1091b2158bSmillert  * modification, are permitted provided that the following conditions
1191b2158bSmillert  * are met:
1291b2158bSmillert  * 1. Redistributions of source code must retain the above copyright
1391b2158bSmillert  *    notice, this list of conditions and the following disclaimer.
1491b2158bSmillert  * 2. Redistributions in binary form must reproduce the above copyright
1591b2158bSmillert  *    notice, this list of conditions and the following disclaimer in the
1691b2158bSmillert  *    documentation and/or other materials provided with the distribution.
1791b2158bSmillert  * 3. The name of the author may not be used to endorse or promote products
1891b2158bSmillert  *    derived from this software without specific prior written permission.
1991b2158bSmillert  *
2091b2158bSmillert  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
2191b2158bSmillert  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
2291b2158bSmillert  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
2391b2158bSmillert  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
2491b2158bSmillert  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
2591b2158bSmillert  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
2691b2158bSmillert  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
2791b2158bSmillert  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
2891b2158bSmillert  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
2991b2158bSmillert  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
3091b2158bSmillert  */
3191b2158bSmillert 
3291b2158bSmillert /*
337eb95549Sreyk  * IEEE 802.11 ioctl support
3491b2158bSmillert  */
3591b2158bSmillert 
3691b2158bSmillert #include <sys/param.h>
3791b2158bSmillert #include <sys/kernel.h>
3891b2158bSmillert #include <sys/socket.h>
3991b2158bSmillert #include <sys/sockio.h>
4091b2158bSmillert #include <sys/systm.h>
4191b2158bSmillert #include <sys/endian.h>
423ce67372Sreyk #include <sys/tree.h>
4391b2158bSmillert 
4491b2158bSmillert #include <net/if.h>
4591b2158bSmillert #include <net/if_media.h>
4691b2158bSmillert 
4791b2158bSmillert #include <netinet/in.h>
4891b2158bSmillert #include <netinet/if_ether.h>
4991b2158bSmillert 
5091b2158bSmillert #include <net80211/ieee80211_var.h>
51f91ff320Sdamien #include <net80211/ieee80211_crypto.h>
5291b2158bSmillert #include <net80211/ieee80211_ioctl.h>
5391b2158bSmillert 
54f22d9adcSdamien void	 ieee80211_node2req(struct ieee80211com *,
55f22d9adcSdamien 	    const struct ieee80211_node *, struct ieee80211_nodereq *);
56f22d9adcSdamien void	 ieee80211_req2node(struct ieee80211com *,
57f22d9adcSdamien 	    const struct ieee80211_nodereq *, struct ieee80211_node *);
5891b2158bSmillert 
5914207dadSreyk void
ieee80211_node2req(struct ieee80211com * ic,const struct ieee80211_node * ni,struct ieee80211_nodereq * nr)60f22d9adcSdamien ieee80211_node2req(struct ieee80211com *ic, const struct ieee80211_node *ni,
6114207dadSreyk     struct ieee80211_nodereq *nr)
6291b2158bSmillert {
639827b6d2Sstsp 	uint8_t rssi;
649827b6d2Sstsp 
652c29abd8Sstsp 	memset(nr, 0, sizeof(*nr));
662c29abd8Sstsp 
672c29abd8Sstsp 	strlcpy(nr->nr_ifname, ic->ic_if.if_xname, sizeof(nr->nr_ifname));
682c29abd8Sstsp 
6914207dadSreyk 	/* Node address and name information */
7014207dadSreyk 	IEEE80211_ADDR_COPY(nr->nr_macaddr, ni->ni_macaddr);
7114207dadSreyk 	IEEE80211_ADDR_COPY(nr->nr_bssid, ni->ni_bssid);
7214207dadSreyk 	nr->nr_nwid_len = ni->ni_esslen;
7314207dadSreyk 	bcopy(ni->ni_essid, nr->nr_nwid, IEEE80211_NWID_LEN);
7491b2158bSmillert 
7514207dadSreyk 	/* Channel and rates */
7614207dadSreyk 	nr->nr_channel = ieee80211_chan2ieee(ic, ni->ni_chan);
776112cb09Sstsp 	if (ni->ni_chan != IEEE80211_CHAN_ANYC)
7814207dadSreyk 		nr->nr_chan_flags = ni->ni_chan->ic_flags;
79eb1db771Sstsp 	if (ic->ic_curmode != IEEE80211_MODE_11N)
80eb1db771Sstsp 		nr->nr_chan_flags &= ~IEEE80211_CHAN_HT;
8114207dadSreyk 	nr->nr_nrates = ni->ni_rates.rs_nrates;
8214207dadSreyk 	bcopy(ni->ni_rates.rs_rates, nr->nr_rates, IEEE80211_RATE_MAXSIZE);
8314207dadSreyk 
8414207dadSreyk 	/* Node status information */
859827b6d2Sstsp 	rssi = (*ic->ic_node_getrssi)(ic, ni);
869827b6d2Sstsp 	if (ic->ic_max_rssi) {
879827b6d2Sstsp 		/* Driver reports RSSI relative to ic_max_rssi. */
889827b6d2Sstsp 		nr->nr_rssi = rssi;
899827b6d2Sstsp 	} else {
909827b6d2Sstsp 		/*
919827b6d2Sstsp 		 * Driver reports RSSI value in dBm.
929827b6d2Sstsp 		 * Convert from unsigned to signed.
939827b6d2Sstsp 		 * Some drivers report a negative value, some don't.
949827b6d2Sstsp 		 * Reasonable range is -20dBm to -80dBm.
959827b6d2Sstsp 		 */
969827b6d2Sstsp 		nr->nr_rssi = (rssi < 128) ? -rssi : rssi;
979827b6d2Sstsp 	}
9877e58062Sreyk 	nr->nr_max_rssi = ic->ic_max_rssi;
9914207dadSreyk 	bcopy(ni->ni_tstamp, nr->nr_tstamp, sizeof(nr->nr_tstamp));
10014207dadSreyk 	nr->nr_intval = ni->ni_intval;
10114207dadSreyk 	nr->nr_capinfo = ni->ni_capinfo;
10214207dadSreyk 	nr->nr_erp = ni->ni_erp;
10314207dadSreyk 	nr->nr_pwrsave = ni->ni_pwrsave;
10414207dadSreyk 	nr->nr_associd = ni->ni_associd;
10514207dadSreyk 	nr->nr_txseq = ni->ni_txseq;
10614207dadSreyk 	nr->nr_rxseq = ni->ni_rxseq;
10714207dadSreyk 	nr->nr_fails = ni->ni_fails;
108799b58a5Sstsp 	nr->nr_assoc_fail = ni->ni_assoc_fail; /* flag values are the same */
10914207dadSreyk 	nr->nr_inact = ni->ni_inact;
11014207dadSreyk 	nr->nr_txrate = ni->ni_txrate;
11114207dadSreyk 	nr->nr_state = ni->ni_state;
112a3353d28Sstsp 
113a3353d28Sstsp 	/* RSN */
114a3353d28Sstsp 	nr->nr_rsnciphers = ni->ni_rsnciphers;
115a3353d28Sstsp 	nr->nr_rsnakms = 0;
116849fcdcaSstsp 	nr->nr_rsnprotos = 0;
117b2709672Sstsp 	if (ni->ni_supported_rsnprotos & IEEE80211_PROTO_RSN)
118849fcdcaSstsp 		nr->nr_rsnprotos |= IEEE80211_WPA_PROTO_WPA2;
119b2709672Sstsp 	if (ni->ni_supported_rsnprotos & IEEE80211_PROTO_WPA)
120b2709672Sstsp 		nr->nr_rsnprotos |= IEEE80211_WPA_PROTO_WPA1;
121b2709672Sstsp 	if (ni->ni_supported_rsnakms & IEEE80211_AKM_8021X)
122a3353d28Sstsp 		nr->nr_rsnakms |= IEEE80211_WPA_AKM_8021X;
123b2709672Sstsp 	if (ni->ni_supported_rsnakms & IEEE80211_AKM_PSK)
124a3353d28Sstsp 		nr->nr_rsnakms |= IEEE80211_WPA_AKM_PSK;
125b2709672Sstsp 	if (ni->ni_supported_rsnakms & IEEE80211_AKM_SHA256_8021X)
126a3353d28Sstsp 		nr->nr_rsnakms |= IEEE80211_WPA_AKM_SHA256_8021X;
127b2709672Sstsp 	if (ni->ni_supported_rsnakms & IEEE80211_AKM_SHA256_PSK)
128a3353d28Sstsp 		nr->nr_rsnakms |= IEEE80211_WPA_AKM_SHA256_PSK;
12914207dadSreyk 
13014207dadSreyk 	/* Node flags */
13114207dadSreyk 	nr->nr_flags = 0;
13214207dadSreyk 	if (bcmp(nr->nr_macaddr, nr->nr_bssid, IEEE80211_ADDR_LEN) == 0)
13314207dadSreyk 		nr->nr_flags |= IEEE80211_NODEREQ_AP;
13414207dadSreyk 	if (ni == ic->ic_bss)
13514207dadSreyk 		nr->nr_flags |= IEEE80211_NODEREQ_AP_BSS;
1365a8c2b9bSstsp 
1375a8c2b9bSstsp 	/* HT */
1385a8c2b9bSstsp 	nr->nr_htcaps = ni->ni_htcaps;
1395a8c2b9bSstsp 	memcpy(nr->nr_rxmcs, ni->ni_rxmcs, sizeof(nr->nr_rxmcs));
1405a8c2b9bSstsp 	nr->nr_max_rxrate = ni->ni_max_rxrate;
1415a8c2b9bSstsp 	nr->nr_tx_mcs_set = ni->ni_tx_mcs_set;
142ec43501fSstsp 	if (ni->ni_flags & IEEE80211_NODE_HT)
143ec43501fSstsp 		nr->nr_flags |= IEEE80211_NODEREQ_HT;
14450d1a4aeSstsp 
14550d1a4aeSstsp 	/* HT / VHT */
14650d1a4aeSstsp 	nr->nr_txmcs = ni->ni_txmcs;
14750d1a4aeSstsp 
14850d1a4aeSstsp 	/* VHT */
14950d1a4aeSstsp 	nr->nr_vht_ss = ni->ni_vht_ss;
15050d1a4aeSstsp 	if (ni->ni_flags & IEEE80211_NODE_VHT)
15150d1a4aeSstsp 		nr->nr_flags |= IEEE80211_NODEREQ_VHT;
15291b2158bSmillert }
15391b2158bSmillert 
15414207dadSreyk void
ieee80211_req2node(struct ieee80211com * ic,const struct ieee80211_nodereq * nr,struct ieee80211_node * ni)155f22d9adcSdamien ieee80211_req2node(struct ieee80211com *ic, const struct ieee80211_nodereq *nr,
15614207dadSreyk     struct ieee80211_node *ni)
15791b2158bSmillert {
15814207dadSreyk 	/* Node address and name information */
15914207dadSreyk 	IEEE80211_ADDR_COPY(ni->ni_macaddr, nr->nr_macaddr);
16014207dadSreyk 	IEEE80211_ADDR_COPY(ni->ni_bssid, nr->nr_bssid);
16114207dadSreyk 	ni->ni_esslen = nr->nr_nwid_len;
16214207dadSreyk 	bcopy(nr->nr_nwid, ni->ni_essid, IEEE80211_NWID_LEN);
16391b2158bSmillert 
16414207dadSreyk 	/* Rates */
16514207dadSreyk 	ni->ni_rates.rs_nrates = nr->nr_nrates;
16614207dadSreyk 	bcopy(nr->nr_rates, ni->ni_rates.rs_rates, IEEE80211_RATE_MAXSIZE);
16791b2158bSmillert 
16814207dadSreyk 	/* Node information */
16914207dadSreyk 	ni->ni_intval = nr->nr_intval;
17014207dadSreyk 	ni->ni_capinfo = nr->nr_capinfo;
17114207dadSreyk 	ni->ni_erp = nr->nr_erp;
17214207dadSreyk 	ni->ni_pwrsave = nr->nr_pwrsave;
17314207dadSreyk 	ni->ni_associd = nr->nr_associd;
17414207dadSreyk 	ni->ni_txseq = nr->nr_txseq;
17514207dadSreyk 	ni->ni_rxseq = nr->nr_rxseq;
17614207dadSreyk 	ni->ni_fails = nr->nr_fails;
17714207dadSreyk 	ni->ni_inact = nr->nr_inact;
17814207dadSreyk 	ni->ni_txrate = nr->nr_txrate;
17914207dadSreyk 	ni->ni_state = nr->nr_state;
18091b2158bSmillert }
18191b2158bSmillert 
182d5793db9Sstsp void
ieee80211_disable_wep(struct ieee80211com * ic)183d5793db9Sstsp ieee80211_disable_wep(struct ieee80211com *ic)
184d5793db9Sstsp {
185d5793db9Sstsp 	struct ieee80211_key *k;
186d5793db9Sstsp 	int i;
187d5793db9Sstsp 
188d5793db9Sstsp 	for (i = 0; i < IEEE80211_WEP_NKID; i++) {
189d5793db9Sstsp 		k = &ic->ic_nw_keys[i];
190d5793db9Sstsp 		if (k->k_cipher != IEEE80211_CIPHER_NONE)
191d5793db9Sstsp 			(*ic->ic_delete_key)(ic, NULL, k);
192ad281cb6Stb 		explicit_bzero(k, sizeof(*k));
193d5793db9Sstsp 	}
194d5793db9Sstsp 	ic->ic_flags &= ~IEEE80211_F_WEPON;
195d5793db9Sstsp }
196d5793db9Sstsp 
197d5793db9Sstsp void
ieee80211_disable_rsn(struct ieee80211com * ic)198d5793db9Sstsp ieee80211_disable_rsn(struct ieee80211com *ic)
199d5793db9Sstsp {
200d5793db9Sstsp 	ic->ic_flags &= ~(IEEE80211_F_PSK | IEEE80211_F_RSNON);
201ad281cb6Stb 	explicit_bzero(ic->ic_psk, sizeof(ic->ic_psk));
202d5793db9Sstsp 	ic->ic_rsnprotos = 0;
203d5793db9Sstsp 	ic->ic_rsnakms = 0;
204d5793db9Sstsp 	ic->ic_rsngroupcipher = 0;
205d5793db9Sstsp 	ic->ic_rsnciphers = 0;
206d5793db9Sstsp }
207d5793db9Sstsp 
208d6390b08Sphessler /* Keep in sync with ieee80211_node.c:ieee80211_ess_setnwkeys() */
209e03e709cSdamien static int
ieee80211_ioctl_setnwkeys(struct ieee80211com * ic,const struct ieee80211_nwkey * nwkey)210e03e709cSdamien ieee80211_ioctl_setnwkeys(struct ieee80211com *ic,
211e03e709cSdamien     const struct ieee80211_nwkey *nwkey)
212e03e709cSdamien {
213e03e709cSdamien 	struct ieee80211_key *k;
214e03e709cSdamien 	int error, i;
215e03e709cSdamien 
216e03e709cSdamien 	if (!(ic->ic_caps & IEEE80211_C_WEP))
217e03e709cSdamien 		return ENODEV;
218e03e709cSdamien 
219e03e709cSdamien 	if (nwkey->i_wepon == IEEE80211_NWKEY_OPEN) {
220e03e709cSdamien 		if (!(ic->ic_flags & IEEE80211_F_WEPON))
221e03e709cSdamien 			return 0;
222e03e709cSdamien 		ic->ic_flags &= ~IEEE80211_F_WEPON;
223e03e709cSdamien 		return ENETRESET;
224e03e709cSdamien 	}
225e03e709cSdamien 	if (nwkey->i_defkid < 1 || nwkey->i_defkid > IEEE80211_WEP_NKID)
226e03e709cSdamien 		return EINVAL;
227e03e709cSdamien 
228e03e709cSdamien 	for (i = 0; i < IEEE80211_WEP_NKID; i++) {
229e03e709cSdamien 		if (nwkey->i_key[i].i_keylen == 0 ||
230e03e709cSdamien 		    nwkey->i_key[i].i_keydat == NULL)
231e03e709cSdamien 			continue;	/* entry not set */
232e03e709cSdamien 		if (nwkey->i_key[i].i_keylen > IEEE80211_KEYBUF_SIZE)
233e03e709cSdamien 			return EINVAL;
234e03e709cSdamien 
235e03e709cSdamien 		/* map wep key to ieee80211_key */
236e03e709cSdamien 		k = &ic->ic_nw_keys[i];
237e03e709cSdamien 		if (k->k_cipher != IEEE80211_CIPHER_NONE)
238e03e709cSdamien 			(*ic->ic_delete_key)(ic, NULL, k);
239e03e709cSdamien 		memset(k, 0, sizeof(*k));
240e03e709cSdamien 		if (nwkey->i_key[i].i_keylen <= 5)
241e03e709cSdamien 			k->k_cipher = IEEE80211_CIPHER_WEP40;
242e03e709cSdamien 		else
243e03e709cSdamien 			k->k_cipher = IEEE80211_CIPHER_WEP104;
24445eec175Sdamien 		k->k_len = ieee80211_cipher_keylen(k->k_cipher);
245e03e709cSdamien 		k->k_flags = IEEE80211_KEY_GROUP | IEEE80211_KEY_TX;
24645eec175Sdamien 		error = copyin(nwkey->i_key[i].i_keydat, k->k_key, k->k_len);
247e03e709cSdamien 		if (error != 0)
248e03e709cSdamien 			return error;
2495ba53380Skrw 		error = (*ic->ic_set_key)(ic, NULL, k);
2505ba53380Skrw 		switch (error) {
2515ba53380Skrw 		case 0:
2525ba53380Skrw 		case EBUSY:
2535ba53380Skrw 			break;
2545ba53380Skrw 		default:
255e03e709cSdamien 			return error;
256e03e709cSdamien 		}
2575ba53380Skrw 	}
258e03e709cSdamien 
259e03e709cSdamien 	ic->ic_def_txkey = nwkey->i_defkid - 1;
260e03e709cSdamien 	ic->ic_flags |= IEEE80211_F_WEPON;
261d5793db9Sstsp 	if (ic->ic_flags & IEEE80211_F_RSNON)
262d5793db9Sstsp 		ieee80211_disable_rsn(ic);
263e03e709cSdamien 
264e03e709cSdamien 	return ENETRESET;
265e03e709cSdamien }
266e03e709cSdamien 
267e03e709cSdamien static int
ieee80211_ioctl_getnwkeys(struct ieee80211com * ic,struct ieee80211_nwkey * nwkey)268e03e709cSdamien ieee80211_ioctl_getnwkeys(struct ieee80211com *ic,
269e03e709cSdamien     struct ieee80211_nwkey *nwkey)
270e03e709cSdamien {
2717c4ecb61Sstsp 	int i;
272e03e709cSdamien 
273e03e709cSdamien 	if (ic->ic_flags & IEEE80211_F_WEPON)
274e03e709cSdamien 		nwkey->i_wepon = IEEE80211_NWKEY_WEP;
275e03e709cSdamien 	else
276e03e709cSdamien 		nwkey->i_wepon = IEEE80211_NWKEY_OPEN;
277e03e709cSdamien 
278e03e709cSdamien 	nwkey->i_defkid = ic->ic_wep_txkey + 1;
279e03e709cSdamien 
280e03e709cSdamien 	for (i = 0; i < IEEE80211_WEP_NKID; i++) {
281e03e709cSdamien 		if (nwkey->i_key[i].i_keydat == NULL)
282e03e709cSdamien 			continue;
2837c4ecb61Sstsp 		/* do not show any keys to userland */
2847c4ecb61Sstsp 		return EPERM;
285e03e709cSdamien 	}
286e03e709cSdamien 	return 0;
287e03e709cSdamien }
288e03e709cSdamien 
289d6390b08Sphessler /* Keep in sync with ieee80211_node.c:ieee80211_ess_setwpaparms() */
290e03e709cSdamien static int
ieee80211_ioctl_setwpaparms(struct ieee80211com * ic,const struct ieee80211_wpaparams * wpa)291e03e709cSdamien ieee80211_ioctl_setwpaparms(struct ieee80211com *ic,
292e03e709cSdamien     const struct ieee80211_wpaparams *wpa)
293e03e709cSdamien {
294e03e709cSdamien 	if (!(ic->ic_caps & IEEE80211_C_RSN))
295e03e709cSdamien 		return ENODEV;
296e03e709cSdamien 
297e03e709cSdamien 	if (!wpa->i_enabled) {
298e03e709cSdamien 		if (!(ic->ic_flags & IEEE80211_F_RSNON))
299e03e709cSdamien 			return 0;
300e03e709cSdamien 		ic->ic_flags &= ~IEEE80211_F_RSNON;
3018adea7f1Sphessler 		ic->ic_rsnprotos = 0;
3028adea7f1Sphessler 		ic->ic_rsnakms = 0;
3038adea7f1Sphessler 		ic->ic_rsngroupcipher = 0;
3048adea7f1Sphessler 		ic->ic_rsnciphers = 0;
305e03e709cSdamien 		return ENETRESET;
306e03e709cSdamien 	}
307e03e709cSdamien 
308e03e709cSdamien 	ic->ic_rsnprotos = 0;
309e03e709cSdamien 	if (wpa->i_protos & IEEE80211_WPA_PROTO_WPA1)
310e03e709cSdamien 		ic->ic_rsnprotos |= IEEE80211_PROTO_WPA;
311e03e709cSdamien 	if (wpa->i_protos & IEEE80211_WPA_PROTO_WPA2)
312e03e709cSdamien 		ic->ic_rsnprotos |= IEEE80211_PROTO_RSN;
31338f9dca0Sstsp 	if (ic->ic_rsnprotos == 0)	/* set to default (RSN) */
31438f9dca0Sstsp 		ic->ic_rsnprotos = IEEE80211_PROTO_RSN;
315e03e709cSdamien 
316e03e709cSdamien 	ic->ic_rsnakms = 0;
317e03e709cSdamien 	if (wpa->i_akms & IEEE80211_WPA_AKM_PSK)
318cdccf32bSdamien 		ic->ic_rsnakms |= IEEE80211_AKM_PSK;
319cdccf32bSdamien 	if (wpa->i_akms & IEEE80211_WPA_AKM_SHA256_PSK)
320cdccf32bSdamien 		ic->ic_rsnakms |= IEEE80211_AKM_SHA256_PSK;
321cdccf32bSdamien 	if (wpa->i_akms & IEEE80211_WPA_AKM_8021X)
322cdccf32bSdamien 		ic->ic_rsnakms |= IEEE80211_AKM_8021X;
323cdccf32bSdamien 	if (wpa->i_akms & IEEE80211_WPA_AKM_SHA256_8021X)
324cdccf32bSdamien 		ic->ic_rsnakms |= IEEE80211_AKM_SHA256_8021X;
325cdccf32bSdamien 	if (ic->ic_rsnakms == 0)	/* set to default (PSK) */
326cdccf32bSdamien 		ic->ic_rsnakms = IEEE80211_AKM_PSK;
327e03e709cSdamien 
328e03e709cSdamien 	if (wpa->i_groupcipher == IEEE80211_WPA_CIPHER_WEP40)
329e03e709cSdamien 		ic->ic_rsngroupcipher = IEEE80211_CIPHER_WEP40;
330e03e709cSdamien 	else if (wpa->i_groupcipher == IEEE80211_WPA_CIPHER_TKIP)
331e03e709cSdamien 		ic->ic_rsngroupcipher = IEEE80211_CIPHER_TKIP;
332e03e709cSdamien 	else if (wpa->i_groupcipher == IEEE80211_WPA_CIPHER_CCMP)
333e03e709cSdamien 		ic->ic_rsngroupcipher = IEEE80211_CIPHER_CCMP;
334e03e709cSdamien 	else if (wpa->i_groupcipher == IEEE80211_WPA_CIPHER_WEP104)
335e03e709cSdamien 		ic->ic_rsngroupcipher = IEEE80211_CIPHER_WEP104;
336e03e709cSdamien 	else  {	/* set to default */
337e03e709cSdamien 		if (ic->ic_rsnprotos & IEEE80211_PROTO_WPA)
338e03e709cSdamien 			ic->ic_rsngroupcipher = IEEE80211_CIPHER_TKIP;
339e03e709cSdamien 		else
340e03e709cSdamien 			ic->ic_rsngroupcipher = IEEE80211_CIPHER_CCMP;
341e03e709cSdamien 	}
342e03e709cSdamien 
343e03e709cSdamien 	ic->ic_rsnciphers = 0;
344e03e709cSdamien 	if (wpa->i_ciphers & IEEE80211_WPA_CIPHER_TKIP)
345e03e709cSdamien 		ic->ic_rsnciphers |= IEEE80211_CIPHER_TKIP;
346e03e709cSdamien 	if (wpa->i_ciphers & IEEE80211_WPA_CIPHER_CCMP)
347e03e709cSdamien 		ic->ic_rsnciphers |= IEEE80211_CIPHER_CCMP;
348e03e709cSdamien 	if (wpa->i_ciphers & IEEE80211_WPA_CIPHER_USEGROUP)
349e03e709cSdamien 		ic->ic_rsnciphers = IEEE80211_CIPHER_USEGROUP;
3505be96dfeSstsp 	if (ic->ic_rsnciphers == 0) { /* set to default (CCMP, TKIP if WPA1) */
35138f9dca0Sstsp 		ic->ic_rsnciphers = IEEE80211_CIPHER_CCMP;
3525be96dfeSstsp 		if (ic->ic_rsnprotos & IEEE80211_PROTO_WPA)
3535be96dfeSstsp 			ic->ic_rsnciphers |= IEEE80211_CIPHER_TKIP;
3545be96dfeSstsp 	}
355e03e709cSdamien 
356e03e709cSdamien 	ic->ic_flags |= IEEE80211_F_RSNON;
357e03e709cSdamien 
358e03e709cSdamien 	return ENETRESET;
359e03e709cSdamien }
360e03e709cSdamien 
361e03e709cSdamien static int
ieee80211_ioctl_getwpaparms(struct ieee80211com * ic,struct ieee80211_wpaparams * wpa)362e03e709cSdamien ieee80211_ioctl_getwpaparms(struct ieee80211com *ic,
363e03e709cSdamien     struct ieee80211_wpaparams *wpa)
364e03e709cSdamien {
365e03e709cSdamien 	wpa->i_enabled = (ic->ic_flags & IEEE80211_F_RSNON) ? 1 : 0;
366e03e709cSdamien 
367e03e709cSdamien 	wpa->i_protos = 0;
368e03e709cSdamien 	if (ic->ic_rsnprotos & IEEE80211_PROTO_WPA)
369e03e709cSdamien 		wpa->i_protos |= IEEE80211_WPA_PROTO_WPA1;
370e03e709cSdamien 	if (ic->ic_rsnprotos & IEEE80211_PROTO_RSN)
371e03e709cSdamien 		wpa->i_protos |= IEEE80211_WPA_PROTO_WPA2;
372e03e709cSdamien 
373e03e709cSdamien 	wpa->i_akms = 0;
374cdccf32bSdamien 	if (ic->ic_rsnakms & IEEE80211_AKM_PSK)
375e03e709cSdamien 		wpa->i_akms |= IEEE80211_WPA_AKM_PSK;
376cdccf32bSdamien 	if (ic->ic_rsnakms & IEEE80211_AKM_SHA256_PSK)
377cdccf32bSdamien 		wpa->i_akms |= IEEE80211_WPA_AKM_SHA256_PSK;
378cdccf32bSdamien 	if (ic->ic_rsnakms & IEEE80211_AKM_8021X)
379cdccf32bSdamien 		wpa->i_akms |= IEEE80211_WPA_AKM_8021X;
380cdccf32bSdamien 	if (ic->ic_rsnakms & IEEE80211_AKM_SHA256_8021X)
381cdccf32bSdamien 		wpa->i_akms |= IEEE80211_WPA_AKM_SHA256_8021X;
382e03e709cSdamien 
383e03e709cSdamien 	if (ic->ic_rsngroupcipher == IEEE80211_CIPHER_WEP40)
384e03e709cSdamien 		wpa->i_groupcipher = IEEE80211_WPA_CIPHER_WEP40;
385e03e709cSdamien 	else if (ic->ic_rsngroupcipher == IEEE80211_CIPHER_TKIP)
386e03e709cSdamien 		wpa->i_groupcipher = IEEE80211_WPA_CIPHER_TKIP;
387e03e709cSdamien 	else if (ic->ic_rsngroupcipher == IEEE80211_CIPHER_CCMP)
388e03e709cSdamien 		wpa->i_groupcipher = IEEE80211_WPA_CIPHER_CCMP;
389e03e709cSdamien 	else if (ic->ic_rsngroupcipher == IEEE80211_CIPHER_WEP104)
390e03e709cSdamien 		wpa->i_groupcipher = IEEE80211_WPA_CIPHER_WEP104;
391e03e709cSdamien 	else
392e03e709cSdamien 		wpa->i_groupcipher = IEEE80211_WPA_CIPHER_NONE;
393e03e709cSdamien 
394e03e709cSdamien 	wpa->i_ciphers = 0;
395e03e709cSdamien 	if (ic->ic_rsnciphers & IEEE80211_CIPHER_TKIP)
396e03e709cSdamien 		wpa->i_ciphers |= IEEE80211_WPA_CIPHER_TKIP;
397e03e709cSdamien 	if (ic->ic_rsnciphers & IEEE80211_CIPHER_CCMP)
398e03e709cSdamien 		wpa->i_ciphers |= IEEE80211_WPA_CIPHER_CCMP;
399e03e709cSdamien 	if (ic->ic_rsnciphers & IEEE80211_CIPHER_USEGROUP)
400e03e709cSdamien 		wpa->i_ciphers = IEEE80211_WPA_CIPHER_USEGROUP;
401e03e709cSdamien 
402e03e709cSdamien 	return 0;
403e03e709cSdamien }
404e03e709cSdamien 
405d7eb32e3Sphessler static void
ieee80211_ess_getwpaparms(struct ieee80211_ess * ess,struct ieee80211_wpaparams * wpa)406d7eb32e3Sphessler ieee80211_ess_getwpaparms(struct ieee80211_ess *ess,
407d7eb32e3Sphessler     struct ieee80211_wpaparams *wpa)
408d7eb32e3Sphessler {
409d7eb32e3Sphessler 	wpa->i_enabled = (ess->flags & IEEE80211_F_RSNON) ? 1 : 0;
410d7eb32e3Sphessler 
411d7eb32e3Sphessler 	wpa->i_protos = 0;
412d7eb32e3Sphessler 	if (ess->rsnprotos & IEEE80211_PROTO_WPA)
413d7eb32e3Sphessler 		wpa->i_protos |= IEEE80211_WPA_PROTO_WPA1;
414d7eb32e3Sphessler 	if (ess->rsnprotos & IEEE80211_PROTO_RSN)
415d7eb32e3Sphessler 		wpa->i_protos |= IEEE80211_WPA_PROTO_WPA2;
416d7eb32e3Sphessler 
417d7eb32e3Sphessler 	wpa->i_akms = 0;
418d7eb32e3Sphessler 	if (ess->rsnakms & IEEE80211_AKM_PSK)
419d7eb32e3Sphessler 		wpa->i_akms |= IEEE80211_WPA_AKM_PSK;
420d7eb32e3Sphessler 	if (ess->rsnakms & IEEE80211_AKM_SHA256_PSK)
421d7eb32e3Sphessler 		wpa->i_akms |= IEEE80211_WPA_AKM_SHA256_PSK;
422d7eb32e3Sphessler 	if (ess->rsnakms & IEEE80211_AKM_8021X)
423d7eb32e3Sphessler 		wpa->i_akms |= IEEE80211_WPA_AKM_8021X;
424d7eb32e3Sphessler 	if (ess->rsnakms & IEEE80211_AKM_SHA256_8021X)
425d7eb32e3Sphessler 		wpa->i_akms |= IEEE80211_WPA_AKM_SHA256_8021X;
426d7eb32e3Sphessler 
427d7eb32e3Sphessler 	if (ess->rsngroupcipher == IEEE80211_CIPHER_WEP40)
428d7eb32e3Sphessler 		wpa->i_groupcipher = IEEE80211_WPA_CIPHER_WEP40;
429d7eb32e3Sphessler 	else if (ess->rsngroupcipher == IEEE80211_CIPHER_TKIP)
430d7eb32e3Sphessler 		wpa->i_groupcipher = IEEE80211_WPA_CIPHER_TKIP;
431d7eb32e3Sphessler 	else if (ess->rsngroupcipher == IEEE80211_CIPHER_CCMP)
432d7eb32e3Sphessler 		wpa->i_groupcipher = IEEE80211_WPA_CIPHER_CCMP;
433d7eb32e3Sphessler 	else if (ess->rsngroupcipher == IEEE80211_CIPHER_WEP104)
434d7eb32e3Sphessler 		wpa->i_groupcipher = IEEE80211_WPA_CIPHER_WEP104;
435d7eb32e3Sphessler 	else
436d7eb32e3Sphessler 		wpa->i_groupcipher = IEEE80211_WPA_CIPHER_NONE;
437d7eb32e3Sphessler 
438d7eb32e3Sphessler 	wpa->i_ciphers = 0;
439d7eb32e3Sphessler 	if (ess->rsnciphers & IEEE80211_CIPHER_TKIP)
440d7eb32e3Sphessler 		wpa->i_ciphers |= IEEE80211_WPA_CIPHER_TKIP;
441d7eb32e3Sphessler 	if (ess->rsnciphers & IEEE80211_CIPHER_CCMP)
442d7eb32e3Sphessler 		wpa->i_ciphers |= IEEE80211_WPA_CIPHER_CCMP;
443d7eb32e3Sphessler 	if (ess->rsnciphers & IEEE80211_CIPHER_USEGROUP)
444d7eb32e3Sphessler 		wpa->i_ciphers = IEEE80211_WPA_CIPHER_USEGROUP;
445d7eb32e3Sphessler }
446d7eb32e3Sphessler 
44791b2158bSmillert int
ieee80211_ioctl(struct ifnet * ifp,u_long cmd,caddr_t data)44891b2158bSmillert ieee80211_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data)
44991b2158bSmillert {
45091b2158bSmillert 	struct ieee80211com *ic = (void *)ifp;
45191b2158bSmillert 	struct ifreq *ifr = (struct ifreq *)data;
45291b2158bSmillert 	int i, error = 0;
4530b1a3dc9Sbenno 	size_t len;
45491b2158bSmillert 	struct ieee80211_nwid nwid;
455020402a2Sphessler 	struct ieee80211_join join;
4560b1a3dc9Sbenno 	struct ieee80211_joinreq_all *ja;
457020402a2Sphessler 	struct ieee80211_ess *ess;
458e03e709cSdamien 	struct ieee80211_wpapsk *psk;
459f91ff320Sdamien 	struct ieee80211_keyavail *ka;
460f91ff320Sdamien 	struct ieee80211_keyrun *kr;
46191b2158bSmillert 	struct ieee80211_power *power;
46291b2158bSmillert 	struct ieee80211_bssid *bssid;
46391b2158bSmillert 	struct ieee80211chanreq *chanreq;
46491b2158bSmillert 	struct ieee80211_channel *chan;
465eb215967Sreyk 	struct ieee80211_txpower *txpower;
46691b2158bSmillert 	static const u_int8_t empty_macaddr[IEEE80211_ADDR_LEN] = {
46791b2158bSmillert 		0x00, 0x00, 0x00, 0x00, 0x00, 0x00
46891b2158bSmillert 	};
46914207dadSreyk 	struct ieee80211_nodereq *nr, nrbuf;
47014207dadSreyk 	struct ieee80211_nodereq_all *na;
471d6725d96Sreyk 	struct ieee80211_node *ni;
472*ada90a5bSstsp 	struct ieee80211_chaninfo chaninfo;
473*ada90a5bSstsp 	struct ieee80211_chanreq_all *allchans;
47475175d43Sreyk 	u_int32_t flags;
47591b2158bSmillert 
47691b2158bSmillert 	switch (cmd) {
47791b2158bSmillert 	case SIOCSIFMEDIA:
47891b2158bSmillert 	case SIOCGIFMEDIA:
47991b2158bSmillert 		error = ifmedia_ioctl(ifp, ifr, &ic->ic_media, cmd);
48091b2158bSmillert 		break;
48191b2158bSmillert 	case SIOCS80211NWID:
4823e676399Smpi 		if ((error = suser(curproc)) != 0)
4831386a18cSuwe 			break;
48491b2158bSmillert 		if ((error = copyin(ifr->ifr_data, &nwid, sizeof(nwid))) != 0)
48591b2158bSmillert 			break;
48691b2158bSmillert 		if (nwid.i_len > IEEE80211_NWID_LEN) {
48791b2158bSmillert 			error = EINVAL;
48891b2158bSmillert 			break;
48991b2158bSmillert 		}
49091b2158bSmillert 		memset(ic->ic_des_essid, 0, IEEE80211_NWID_LEN);
49191b2158bSmillert 		ic->ic_des_esslen = nwid.i_len;
49291b2158bSmillert 		memcpy(ic->ic_des_essid, nwid.i_nwid, nwid.i_len);
493b2cf04c7Sstsp 		if (ic->ic_des_esslen > 0) {
494b2cf04c7Sstsp 			/* 'nwid' disables auto-join magic */
495b2cf04c7Sstsp 			ic->ic_flags &= ~IEEE80211_F_AUTO_JOIN;
4966aa27acdSphessler 		} else if (!TAILQ_EMPTY(&ic->ic_ess)) {
497b2cf04c7Sstsp 			/* '-nwid' re-enables auto-join */
498b2cf04c7Sstsp 			ic->ic_flags |= IEEE80211_F_AUTO_JOIN;
499b2cf04c7Sstsp 		}
500b97e28d2Sphessler 		/* disable WPA/WEP */
501b97e28d2Sphessler 		ieee80211_disable_rsn(ic);
502b97e28d2Sphessler 		ieee80211_disable_wep(ic);
50391b2158bSmillert 		error = ENETRESET;
50491b2158bSmillert 		break;
50591b2158bSmillert 	case SIOCG80211NWID:
50691b2158bSmillert 		memset(&nwid, 0, sizeof(nwid));
50791b2158bSmillert 		switch (ic->ic_state) {
50891b2158bSmillert 		case IEEE80211_S_INIT:
50991b2158bSmillert 		case IEEE80211_S_SCAN:
51091b2158bSmillert 			nwid.i_len = ic->ic_des_esslen;
51191b2158bSmillert 			memcpy(nwid.i_nwid, ic->ic_des_essid, nwid.i_len);
51291b2158bSmillert 			break;
51391b2158bSmillert 		default:
51491b2158bSmillert 			nwid.i_len = ic->ic_bss->ni_esslen;
51591b2158bSmillert 			memcpy(nwid.i_nwid, ic->ic_bss->ni_essid, nwid.i_len);
51691b2158bSmillert 			break;
51791b2158bSmillert 		}
51891b2158bSmillert 		error = copyout(&nwid, ifr->ifr_data, sizeof(nwid));
51991b2158bSmillert 		break;
520020402a2Sphessler 	case SIOCS80211JOIN:
521020402a2Sphessler 		if ((error = suser(curproc)) != 0)
522020402a2Sphessler 			break;
52326bb14d9Sphessler 		if (ic->ic_opmode != IEEE80211_M_STA)
52426bb14d9Sphessler 			break;
525020402a2Sphessler 		if ((error = copyin(ifr->ifr_data, &join, sizeof(join))) != 0)
526020402a2Sphessler 			break;
527020402a2Sphessler 		if (join.i_len > IEEE80211_NWID_LEN) {
528020402a2Sphessler 			error = EINVAL;
529020402a2Sphessler 			break;
530020402a2Sphessler 		}
53152300c1eSphessler 		if (join.i_flags & IEEE80211_JOIN_DEL) {
53252300c1eSphessler 			int update_ic = 0;
53352300c1eSphessler 			if (ic->ic_des_esslen == join.i_len &&
53452300c1eSphessler 			    memcmp(join.i_nwid, ic->ic_des_essid,
53552300c1eSphessler 			    join.i_len) == 0)
53652300c1eSphessler 				update_ic = 1;
53752300c1eSphessler 			if (join.i_flags & IEEE80211_JOIN_DEL_ALL &&
53852300c1eSphessler 			    ieee80211_get_ess(ic, ic->ic_des_essid,
53952300c1eSphessler 			    ic->ic_des_esslen) != NULL)
54052300c1eSphessler 				update_ic = 1;
541157eb8fdSphessler 			ieee80211_del_ess(ic, join.i_nwid, join.i_len,
5428a6c2968Sphessler 			    join.i_flags & IEEE80211_JOIN_DEL_ALL ? 1 : 0);
54352300c1eSphessler 			if (update_ic == 1) {
54452300c1eSphessler 				/* Unconfigure this essid */
54552300c1eSphessler 				memset(ic->ic_des_essid, 0, IEEE80211_NWID_LEN);
54652300c1eSphessler 				ic->ic_des_esslen = 0;
54752300c1eSphessler 				/* disable WPA/WEP */
54852300c1eSphessler 				ieee80211_disable_rsn(ic);
54952300c1eSphessler 				ieee80211_disable_wep(ic);
55052300c1eSphessler 				error = ENETRESET;
55152300c1eSphessler 			}
55252300c1eSphessler 		} else {
55303adf27bSphessler 			if (ic->ic_des_esslen == join.i_len &&
55403adf27bSphessler 			    memcmp(join.i_nwid, ic->ic_des_essid,
55503adf27bSphessler 			    join.i_len) == 0) {
55626bb14d9Sphessler 				struct ieee80211_node *ni;
55726bb14d9Sphessler 
55803adf27bSphessler 				ieee80211_deselect_ess(ic);
55926bb14d9Sphessler 				ni = ieee80211_find_node(ic,
56026bb14d9Sphessler 				    ic->ic_bss->ni_bssid);
56126bb14d9Sphessler 				if (ni != NULL)
56226bb14d9Sphessler 					ieee80211_free_node(ic, ni);
56303adf27bSphessler 				error = ENETRESET;
56403adf27bSphessler 			}
565020402a2Sphessler 			/* save nwid for auto-join */
566d6390b08Sphessler 			if (ieee80211_add_ess(ic, &join) == 0)
567b2cf04c7Sstsp 				ic->ic_flags |= IEEE80211_F_AUTO_JOIN;
568b2cf04c7Sstsp 		}
569020402a2Sphessler 		break;
570020402a2Sphessler 	case SIOCG80211JOIN:
571020402a2Sphessler 		memset(&join, 0, sizeof(join));
572020402a2Sphessler 		error = ENOENT;
573020402a2Sphessler 		if (ic->ic_bss == NULL)
574020402a2Sphessler 			break;
575020402a2Sphessler 		TAILQ_FOREACH(ess, &ic->ic_ess, ess_next) {
576020402a2Sphessler 			if (memcmp(ess->essid, ic->ic_bss->ni_essid,
577020402a2Sphessler 			    IEEE80211_NWID_LEN) == 0) {
578020402a2Sphessler 				join.i_len = ic->ic_bss->ni_esslen;
579020402a2Sphessler 				memcpy(join.i_nwid, ic->ic_bss->ni_essid,
580020402a2Sphessler 				    join.i_len);
581b2cf04c7Sstsp 				if (ic->ic_flags & IEEE80211_F_AUTO_JOIN)
582020402a2Sphessler 					join.i_flags = IEEE80211_JOIN_FOUND;
583020402a2Sphessler 				error = copyout(&join, ifr->ifr_data,
584020402a2Sphessler 				    sizeof(join));
585020402a2Sphessler 				break;
586020402a2Sphessler 			}
587020402a2Sphessler 		}
588020402a2Sphessler 		break;
5890b1a3dc9Sbenno 	case SIOCG80211JOINALL:
5900b1a3dc9Sbenno 		ja = (struct ieee80211_joinreq_all *)data;
5910b1a3dc9Sbenno 		ja->ja_nodes = len = 0;
5920b1a3dc9Sbenno 		TAILQ_FOREACH(ess, &ic->ic_ess, ess_next) {
5930b1a3dc9Sbenno 			if (len + sizeof(ja->ja_node[0]) >= ja->ja_size) {
5940b1a3dc9Sbenno 				error = E2BIG;
5950b1a3dc9Sbenno 				break;
5960b1a3dc9Sbenno 			}
5970b1a3dc9Sbenno 			memset(&join, 0, sizeof(join));
5980b1a3dc9Sbenno 			join.i_len = ess->esslen;
5990b1a3dc9Sbenno 			memcpy(&join.i_nwid, ess->essid, join.i_len);
600d7eb32e3Sphessler 			if (ess->flags & IEEE80211_F_RSNON)
601d7eb32e3Sphessler 				join.i_flags |= IEEE80211_JOIN_WPA;
602d7eb32e3Sphessler 			if (ess->flags & IEEE80211_F_PSK)
603d7eb32e3Sphessler 				join.i_flags |= IEEE80211_JOIN_WPAPSK;
604d7eb32e3Sphessler 			if (ess->flags & IEEE80211_JOIN_8021X)
605d7eb32e3Sphessler 				join.i_flags |= IEEE80211_JOIN_8021X;
606d7eb32e3Sphessler 			if (ess->flags & IEEE80211_F_WEPON)
607d7eb32e3Sphessler 				join.i_flags |= IEEE80211_JOIN_NWKEY;
6088a6c2968Sphessler 			if (ess->flags & IEEE80211_JOIN_ANY)
6098a6c2968Sphessler 				join.i_flags |= IEEE80211_JOIN_ANY;
610d7eb32e3Sphessler 			ieee80211_ess_getwpaparms(ess, &join.i_wpaparams);
6110b1a3dc9Sbenno 			error = copyout(&join, &ja->ja_node[ja->ja_nodes],
6120b1a3dc9Sbenno 			    sizeof(ja->ja_node[0]));
6130b1a3dc9Sbenno 			if (error)
6140b1a3dc9Sbenno 				break;
6150b1a3dc9Sbenno 			len += sizeof(join);
6160b1a3dc9Sbenno 			ja->ja_nodes++;
6170b1a3dc9Sbenno 		}
6180b1a3dc9Sbenno 		break;
61991b2158bSmillert 	case SIOCS80211NWKEY:
6203e676399Smpi 		if ((error = suser(curproc)) != 0)
6211386a18cSuwe 			break;
622e03e709cSdamien 		error = ieee80211_ioctl_setnwkeys(ic, (void *)data);
623e03e709cSdamien 		break;
624e03e709cSdamien 	case SIOCG80211NWKEY:
625e03e709cSdamien 		error = ieee80211_ioctl_getnwkeys(ic, (void *)data);
626e03e709cSdamien 		break;
627e03e709cSdamien 	case SIOCS80211WPAPARMS:
6283e676399Smpi 		if ((error = suser(curproc)) != 0)
629e03e709cSdamien 			break;
630e03e709cSdamien 		error = ieee80211_ioctl_setwpaparms(ic, (void *)data);
631e03e709cSdamien 		break;
632e03e709cSdamien 	case SIOCG80211WPAPARMS:
633e03e709cSdamien 		error = ieee80211_ioctl_getwpaparms(ic, (void *)data);
634e03e709cSdamien 		break;
635e03e709cSdamien 	case SIOCS80211WPAPSK:
6363e676399Smpi 		if ((error = suser(curproc)) != 0)
637e03e709cSdamien 			break;
638e03e709cSdamien 		psk = (struct ieee80211_wpapsk *)data;
639e03e709cSdamien 		if (psk->i_enabled) {
640e03e709cSdamien 			ic->ic_flags |= IEEE80211_F_PSK;
641e03e709cSdamien 			memcpy(ic->ic_psk, psk->i_psk, sizeof(ic->ic_psk));
642d5793db9Sstsp 			if (ic->ic_flags & IEEE80211_F_WEPON)
643d5793db9Sstsp 				ieee80211_disable_wep(ic);
644e03e709cSdamien 		} else {
645e03e709cSdamien 			ic->ic_flags &= ~IEEE80211_F_PSK;
646e03e709cSdamien 			memset(ic->ic_psk, 0, sizeof(ic->ic_psk));
64791b2158bSmillert 		}
64891b2158bSmillert 		error = ENETRESET;
64991b2158bSmillert 		break;
650e03e709cSdamien 	case SIOCG80211WPAPSK:
651e03e709cSdamien 		psk = (struct ieee80211_wpapsk *)data;
652e03e709cSdamien 		if (ic->ic_flags & IEEE80211_F_PSK) {
6537c4ecb61Sstsp 			/* do not show any keys to userland */
654e03e709cSdamien 			psk->i_enabled = 2;
655e03e709cSdamien 			memset(psk->i_psk, 0, sizeof(psk->i_psk));
656e03e709cSdamien 			break;	/* return ok but w/o key */
657e03e709cSdamien 		} else
658e03e709cSdamien 			psk->i_enabled = 0;
65991b2158bSmillert 		break;
660f91ff320Sdamien 	case SIOCS80211KEYAVAIL:
6613e676399Smpi 		if ((error = suser(curproc)) != 0)
662f91ff320Sdamien 			break;
663f91ff320Sdamien 		ka = (struct ieee80211_keyavail *)data;
664f91ff320Sdamien 		(void)ieee80211_pmksa_add(ic, IEEE80211_AKM_8021X,
665f91ff320Sdamien 		    ka->i_macaddr, ka->i_key, ka->i_lifetime);
666f91ff320Sdamien 		break;
667f91ff320Sdamien 	case SIOCS80211KEYRUN:
6683e676399Smpi 		if ((error = suser(curproc)) != 0)
669f91ff320Sdamien 			break;
670f91ff320Sdamien 		kr = (struct ieee80211_keyrun *)data;
671f91ff320Sdamien 		error = ieee80211_keyrun(ic, kr->i_macaddr);
672d5793db9Sstsp 		if (error == 0 && (ic->ic_flags & IEEE80211_F_WEPON))
673d5793db9Sstsp 			ieee80211_disable_wep(ic);
674f91ff320Sdamien 		break;
67591b2158bSmillert 	case SIOCS80211POWER:
6763e676399Smpi 		if ((error = suser(curproc)) != 0)
6771386a18cSuwe 			break;
67891b2158bSmillert 		power = (struct ieee80211_power *)data;
67991b2158bSmillert 		ic->ic_lintval = power->i_maxsleep;
68091b2158bSmillert 		if (power->i_enabled != 0) {
68191b2158bSmillert 			if ((ic->ic_caps & IEEE80211_C_PMGT) == 0)
68291b2158bSmillert 				error = EINVAL;
68391b2158bSmillert 			else if ((ic->ic_flags & IEEE80211_F_PMGTON) == 0) {
68491b2158bSmillert 				ic->ic_flags |= IEEE80211_F_PMGTON;
68591b2158bSmillert 				error = ENETRESET;
68691b2158bSmillert 			}
68791b2158bSmillert 		} else {
68891b2158bSmillert 			if (ic->ic_flags & IEEE80211_F_PMGTON) {
68991b2158bSmillert 				ic->ic_flags &= ~IEEE80211_F_PMGTON;
69091b2158bSmillert 				error = ENETRESET;
69191b2158bSmillert 			}
69291b2158bSmillert 		}
69391b2158bSmillert 		break;
69491b2158bSmillert 	case SIOCG80211POWER:
69591b2158bSmillert 		power = (struct ieee80211_power *)data;
69691b2158bSmillert 		power->i_enabled = (ic->ic_flags & IEEE80211_F_PMGTON) ? 1 : 0;
69791b2158bSmillert 		power->i_maxsleep = ic->ic_lintval;
69891b2158bSmillert 		break;
69991b2158bSmillert 	case SIOCS80211BSSID:
7003e676399Smpi 		if ((error = suser(curproc)) != 0)
7011386a18cSuwe 			break;
70291b2158bSmillert 		bssid = (struct ieee80211_bssid *)data;
70391b2158bSmillert 		if (IEEE80211_ADDR_EQ(bssid->i_bssid, empty_macaddr))
70491b2158bSmillert 			ic->ic_flags &= ~IEEE80211_F_DESBSSID;
70591b2158bSmillert 		else {
70691b2158bSmillert 			ic->ic_flags |= IEEE80211_F_DESBSSID;
70791b2158bSmillert 			IEEE80211_ADDR_COPY(ic->ic_des_bssid, bssid->i_bssid);
70891b2158bSmillert 		}
709171ac09aSdamien #ifndef IEEE80211_STA_ONLY
71091b2158bSmillert 		if (ic->ic_opmode == IEEE80211_M_HOSTAP)
71191b2158bSmillert 			break;
712171ac09aSdamien #endif
71391b2158bSmillert 		switch (ic->ic_state) {
71491b2158bSmillert 		case IEEE80211_S_INIT:
71591b2158bSmillert 		case IEEE80211_S_SCAN:
71691b2158bSmillert 			error = ENETRESET;
71791b2158bSmillert 			break;
71891b2158bSmillert 		default:
71991b2158bSmillert 			if ((ic->ic_flags & IEEE80211_F_DESBSSID) &&
72091b2158bSmillert 			    !IEEE80211_ADDR_EQ(ic->ic_des_bssid,
72191b2158bSmillert 			    ic->ic_bss->ni_bssid))
72291b2158bSmillert 				error = ENETRESET;
72391b2158bSmillert 			break;
72491b2158bSmillert 		}
72591b2158bSmillert 		break;
72691b2158bSmillert 	case SIOCG80211BSSID:
72791b2158bSmillert 		bssid = (struct ieee80211_bssid *)data;
72891b2158bSmillert 		switch (ic->ic_state) {
72991b2158bSmillert 		case IEEE80211_S_INIT:
73091b2158bSmillert 		case IEEE80211_S_SCAN:
731171ac09aSdamien #ifndef IEEE80211_STA_ONLY
73291b2158bSmillert 			if (ic->ic_opmode == IEEE80211_M_HOSTAP)
73391b2158bSmillert 				IEEE80211_ADDR_COPY(bssid->i_bssid,
73491b2158bSmillert 				    ic->ic_myaddr);
735171ac09aSdamien 			else
736171ac09aSdamien #endif
737171ac09aSdamien 			if (ic->ic_flags & IEEE80211_F_DESBSSID)
73891b2158bSmillert 				IEEE80211_ADDR_COPY(bssid->i_bssid,
73991b2158bSmillert 				    ic->ic_des_bssid);
74091b2158bSmillert 			else
74191b2158bSmillert 				memset(bssid->i_bssid, 0, IEEE80211_ADDR_LEN);
74291b2158bSmillert 			break;
74391b2158bSmillert 		default:
74491b2158bSmillert 			IEEE80211_ADDR_COPY(bssid->i_bssid,
74591b2158bSmillert 			    ic->ic_bss->ni_bssid);
74691b2158bSmillert 			break;
74791b2158bSmillert 		}
74891b2158bSmillert 		break;
74991b2158bSmillert 	case SIOCS80211CHANNEL:
7503e676399Smpi 		if ((error = suser(curproc)) != 0)
7511386a18cSuwe 			break;
75291b2158bSmillert 		chanreq = (struct ieee80211chanreq *)data;
75391b2158bSmillert 		if (chanreq->i_channel == IEEE80211_CHAN_ANY)
75491b2158bSmillert 			ic->ic_des_chan = IEEE80211_CHAN_ANYC;
75591b2158bSmillert 		else if (chanreq->i_channel > IEEE80211_CHAN_MAX ||
75691b2158bSmillert 		    isclr(ic->ic_chan_active, chanreq->i_channel)) {
75791b2158bSmillert 			error = EINVAL;
75891b2158bSmillert 			break;
75991b2158bSmillert 		} else
76091b2158bSmillert 			ic->ic_ibss_chan = ic->ic_des_chan =
76191b2158bSmillert 			    &ic->ic_channels[chanreq->i_channel];
76291b2158bSmillert 		switch (ic->ic_state) {
76391b2158bSmillert 		case IEEE80211_S_INIT:
76491b2158bSmillert 		case IEEE80211_S_SCAN:
76591b2158bSmillert 			error = ENETRESET;
76691b2158bSmillert 			break;
76791b2158bSmillert 		default:
76891b2158bSmillert 			if (ic->ic_opmode == IEEE80211_M_STA) {
76991b2158bSmillert 				if (ic->ic_des_chan != IEEE80211_CHAN_ANYC &&
77091b2158bSmillert 				    ic->ic_bss->ni_chan != ic->ic_des_chan)
77191b2158bSmillert 					error = ENETRESET;
77291b2158bSmillert 			} else {
77391b2158bSmillert 				if (ic->ic_bss->ni_chan != ic->ic_ibss_chan)
77491b2158bSmillert 					error = ENETRESET;
77591b2158bSmillert 			}
77691b2158bSmillert 			break;
77791b2158bSmillert 		}
77891b2158bSmillert 		break;
77991b2158bSmillert 	case SIOCG80211CHANNEL:
78091b2158bSmillert 		chanreq = (struct ieee80211chanreq *)data;
78191b2158bSmillert 		switch (ic->ic_state) {
78291b2158bSmillert 		case IEEE80211_S_INIT:
78391b2158bSmillert 		case IEEE80211_S_SCAN:
78491b2158bSmillert 			if (ic->ic_opmode == IEEE80211_M_STA)
78591b2158bSmillert 				chan = ic->ic_des_chan;
78691b2158bSmillert 			else
78791b2158bSmillert 				chan = ic->ic_ibss_chan;
78891b2158bSmillert 			break;
78991b2158bSmillert 		default:
79091b2158bSmillert 			chan = ic->ic_bss->ni_chan;
79191b2158bSmillert 			break;
79291b2158bSmillert 		}
79391b2158bSmillert 		chanreq->i_channel = ieee80211_chan2ieee(ic, chan);
79491b2158bSmillert 		break;
79523a5964dSdamien 	case SIOCG80211ALLCHANS:
796*ada90a5bSstsp 		allchans = (struct ieee80211_chanreq_all *)data;
797*ada90a5bSstsp 		for (i = 0; i <= IEEE80211_CHAN_MAX; i++) {
798*ada90a5bSstsp 			chan = &ic->ic_channels[i];
799*ada90a5bSstsp 			chaninfo.ic_freq = chan->ic_freq;
800*ada90a5bSstsp 			chaninfo.ic_flags = 0;
801*ada90a5bSstsp 			if (chan->ic_flags & IEEE80211_CHAN_2GHZ)
802*ada90a5bSstsp 				chaninfo.ic_flags |= IEEE80211_CHANINFO_2GHZ;
803*ada90a5bSstsp 			if (chan->ic_flags & IEEE80211_CHAN_5GHZ)
804*ada90a5bSstsp 				chaninfo.ic_flags |= IEEE80211_CHANINFO_5GHZ;
805*ada90a5bSstsp 			if (chan->ic_flags & IEEE80211_CHAN_PASSIVE)
806*ada90a5bSstsp 				chaninfo.ic_flags |= IEEE80211_CHANINFO_PASSIVE;
807*ada90a5bSstsp 			error = copyout(&chaninfo, &allchans->i_chans[i],
808*ada90a5bSstsp 			    sizeof(chaninfo));
809*ada90a5bSstsp 			if (error)
810*ada90a5bSstsp 				break;
811*ada90a5bSstsp 		}
81223a5964dSdamien 		break;
8130fd4e251Sreyk #if 0
8140fd4e251Sreyk 	case SIOCG80211ZSTATS:
8150fd4e251Sreyk #endif
81691b2158bSmillert 	case SIOCG80211STATS:
81791b2158bSmillert 		ifr = (struct ifreq *)data;
818821cee17Smiod 		error = copyout(&ic->ic_stats, ifr->ifr_data,
819821cee17Smiod 		    sizeof(ic->ic_stats));
8200fd4e251Sreyk #if 0
8210fd4e251Sreyk 		if (cmd == SIOCG80211ZSTATS)
8220fd4e251Sreyk 			memset(&ic->ic_stats, 0, sizeof(ic->ic_stats));
8230fd4e251Sreyk #endif
82491b2158bSmillert 		break;
825eb215967Sreyk 	case SIOCS80211TXPOWER:
8263e676399Smpi 		if ((error = suser(curproc)) != 0)
8271386a18cSuwe 			break;
828eb215967Sreyk 		txpower = (struct ieee80211_txpower *)data;
829eb215967Sreyk 		if ((ic->ic_caps & IEEE80211_C_TXPMGT) == 0) {
830eb215967Sreyk 			error = EINVAL;
831eb215967Sreyk 			break;
832eb215967Sreyk 		}
8332deb40ddSjsg 		if (!(IEEE80211_TXPOWER_MIN <= txpower->i_val &&
8342deb40ddSjsg 			txpower->i_val <= IEEE80211_TXPOWER_MAX)) {
835eb215967Sreyk 			error = EINVAL;
836eb215967Sreyk 			break;
837eb215967Sreyk 		}
838eb215967Sreyk 		ic->ic_txpower = txpower->i_val;
839eb215967Sreyk 		error = ENETRESET;
840eb215967Sreyk 		break;
841eb215967Sreyk 	case SIOCG80211TXPOWER:
842eb215967Sreyk 		txpower = (struct ieee80211_txpower *)data;
843eb215967Sreyk 		if ((ic->ic_caps & IEEE80211_C_TXPMGT) == 0)
844eb215967Sreyk 			error = EINVAL;
845eb215967Sreyk 		else
846eb215967Sreyk 			txpower->i_val = ic->ic_txpower;
847eb215967Sreyk 		break;
84891b2158bSmillert 	case SIOCSIFMTU:
84991b2158bSmillert 		ifr = (struct ifreq *)data;
85091b2158bSmillert 		if (!(IEEE80211_MTU_MIN <= ifr->ifr_mtu &&
85191b2158bSmillert 		    ifr->ifr_mtu <= IEEE80211_MTU_MAX))
85291b2158bSmillert 			error = EINVAL;
85391b2158bSmillert 		else
85491b2158bSmillert 			ifp->if_mtu = ifr->ifr_mtu;
85591b2158bSmillert 		break;
85614207dadSreyk 	case SIOCS80211SCAN:
8572fa6698eSpirofti 		/* Disabled. SIOCG80211ALLNODES is enough. */
85814207dadSreyk 		break;
85914207dadSreyk 	case SIOCG80211NODE:
86014207dadSreyk 		nr = (struct ieee80211_nodereq *)data;
861799b58a5Sstsp 		if (ic->ic_bss &&
862799b58a5Sstsp 		    IEEE80211_ADDR_EQ(nr->nr_macaddr, ic->ic_bss->ni_macaddr))
863799b58a5Sstsp 			ni = ic->ic_bss;
864799b58a5Sstsp 		else
86514207dadSreyk 			ni = ieee80211_find_node(ic, nr->nr_macaddr);
86614207dadSreyk 		if (ni == NULL) {
86714207dadSreyk 			error = ENOENT;
86814207dadSreyk 			break;
86914207dadSreyk 		}
87014207dadSreyk 		ieee80211_node2req(ic, ni, nr);
87114207dadSreyk 		break;
87214207dadSreyk 	case SIOCS80211NODE:
8733e676399Smpi 		if ((error = suser(curproc)) != 0)
87414207dadSreyk 			break;
875171ac09aSdamien #ifndef IEEE80211_STA_ONLY
87614207dadSreyk 		if (ic->ic_opmode == IEEE80211_M_HOSTAP) {
87714207dadSreyk 			error = EINVAL;
87814207dadSreyk 			break;
87914207dadSreyk 		}
880171ac09aSdamien #endif
88114207dadSreyk 		nr = (struct ieee80211_nodereq *)data;
88214207dadSreyk 
88314207dadSreyk 		ni = ieee80211_find_node(ic, nr->nr_macaddr);
88414207dadSreyk 		if (ni == NULL)
88514207dadSreyk 			ni = ieee80211_alloc_node(ic, nr->nr_macaddr);
88614207dadSreyk 		if (ni == NULL) {
88714207dadSreyk 			error = ENOENT;
88814207dadSreyk 			break;
88914207dadSreyk 		}
89014207dadSreyk 
89114207dadSreyk 		if (nr->nr_flags & IEEE80211_NODEREQ_COPY)
89214207dadSreyk 			ieee80211_req2node(ic, nr, ni);
89314207dadSreyk 		break;
8942ee29fc2Skettenis #ifndef IEEE80211_STA_ONLY
89514207dadSreyk 	case SIOCS80211DELNODE:
8963e676399Smpi 		if ((error = suser(curproc)) != 0)
89714207dadSreyk 			break;
89814207dadSreyk 		nr = (struct ieee80211_nodereq *)data;
89914207dadSreyk 		ni = ieee80211_find_node(ic, nr->nr_macaddr);
900d6725d96Sreyk 		if (ni == NULL)
901d6725d96Sreyk 			error = ENOENT;
90214207dadSreyk 		else if (ni == ic->ic_bss)
90314207dadSreyk 			error = EPERM;
904d6725d96Sreyk 		else {
905d6725d96Sreyk 			if (ni->ni_state == IEEE80211_STA_COLLECT)
906d6725d96Sreyk 				break;
907d6725d96Sreyk 
908d6725d96Sreyk 			/* Disassociate station. */
909d6725d96Sreyk 			if (ni->ni_state == IEEE80211_STA_ASSOC)
910d6725d96Sreyk 				IEEE80211_SEND_MGMT(ic, ni,
911d6725d96Sreyk 				    IEEE80211_FC0_SUBTYPE_DISASSOC,
912d6725d96Sreyk 				    IEEE80211_REASON_ASSOC_LEAVE);
913d6725d96Sreyk 
914d6725d96Sreyk 			/* Deauth station. */
915d6725d96Sreyk 			if (ni->ni_state >= IEEE80211_STA_AUTH)
916d6725d96Sreyk 				IEEE80211_SEND_MGMT(ic, ni,
917d6725d96Sreyk 				    IEEE80211_FC0_SUBTYPE_DEAUTH,
918d6725d96Sreyk 				    IEEE80211_REASON_AUTH_LEAVE);
919d6725d96Sreyk 
9202ee29fc2Skettenis 			ieee80211_node_leave(ic, ni);
921d6725d96Sreyk 		}
922d6725d96Sreyk 		break;
9232ee29fc2Skettenis #endif
92414207dadSreyk 	case SIOCG80211ALLNODES:
9252fa6698eSpirofti 		if ((ifp->if_flags & (IFF_UP | IFF_RUNNING)) !=
9262fa6698eSpirofti 		    (IFF_UP | IFF_RUNNING)) {
9272fa6698eSpirofti 			error = ENETDOWN;
9282fa6698eSpirofti 			break;
9292fa6698eSpirofti 		}
9302fa6698eSpirofti 
93114207dadSreyk 		na = (struct ieee80211_nodereq_all *)data;
93214207dadSreyk 		na->na_nodes = i = 0;
9338529d8f0Sdlg 		ni = RBT_MIN(ieee80211_tree, &ic->ic_tree);
93414207dadSreyk 		while (ni && na->na_size >=
93514207dadSreyk 		    i + sizeof(struct ieee80211_nodereq)) {
93614207dadSreyk 			ieee80211_node2req(ic, ni, &nrbuf);
93714207dadSreyk 			error = copyout(&nrbuf, (caddr_t)na->na_node + i,
93814207dadSreyk 			    sizeof(struct ieee80211_nodereq));
939d6725d96Sreyk 			if (error)
940d6725d96Sreyk 				break;
94114207dadSreyk 			i += sizeof(struct ieee80211_nodereq);
94214207dadSreyk 			na->na_nodes++;
9438529d8f0Sdlg 			ni = RBT_NEXT(ieee80211_tree, ni);
944d6725d96Sreyk 		}
9450af58becSstsp 		if (suser(curproc) == 0)
9460af58becSstsp 			ieee80211_begin_bgscan(ifp);
947d6725d96Sreyk 		break;
94875175d43Sreyk 	case SIOCG80211FLAGS:
949534bf8f4Sstsp 		flags = ic->ic_userflags;
950171ac09aSdamien #ifndef IEEE80211_STA_ONLY
95175175d43Sreyk 		if (ic->ic_opmode != IEEE80211_M_HOSTAP)
952171ac09aSdamien #endif
95375175d43Sreyk 			flags &= ~IEEE80211_F_HOSTAPMASK;
954534bf8f4Sstsp 		ifr->ifr_flags = flags;
95575175d43Sreyk 		break;
95675175d43Sreyk 	case SIOCS80211FLAGS:
9573e676399Smpi 		if ((error = suser(curproc)) != 0)
95875175d43Sreyk 			break;
959534bf8f4Sstsp 		flags = ifr->ifr_flags;
960c77fac02Sdamien 		if (
961171ac09aSdamien #ifndef IEEE80211_STA_ONLY
962c77fac02Sdamien 		    ic->ic_opmode != IEEE80211_M_HOSTAP &&
963c77fac02Sdamien #endif
96475175d43Sreyk 		    (flags & IEEE80211_F_HOSTAPMASK)) {
96575175d43Sreyk 			error = EINVAL;
96675175d43Sreyk 			break;
96775175d43Sreyk 		}
968534bf8f4Sstsp 		ic->ic_userflags = flags;
96975175d43Sreyk 		error = ENETRESET;
97075175d43Sreyk 		break;
971c37a468cSmpi 	case SIOCADDMULTI:
972c37a468cSmpi 	case SIOCDELMULTI:
973c37a468cSmpi 		error = (cmd == SIOCADDMULTI) ?
974c37a468cSmpi 		    ether_addmulti(ifr, &ic->ic_ac) :
975c37a468cSmpi 		    ether_delmulti(ifr, &ic->ic_ac);
976c37a468cSmpi 		if (error == ENETRESET)
977c37a468cSmpi 			error = 0;
978c37a468cSmpi 		break;
97991b2158bSmillert 	default:
980775775feSbrad 		error = ether_ioctl(ifp, &ic->ic_ac, cmd, data);
98191b2158bSmillert 	}
982775775feSbrad 
98391b2158bSmillert 	return error;
98491b2158bSmillert }
985