]> git.ipfire.org Git - thirdparty/hostap.git/blame - wpa_supplicant/ap.c
wpa_supplicant AP: More thorough AP mode deinit
[thirdparty/hostap.git] / wpa_supplicant / ap.c
CommitLineData
f1a48710
JM
1/*
2 * WPA Supplicant - Basic AP mode support routines
3 * Copyright (c) 2003-2009, Jouni Malinen <j@w1.fi>
4 * Copyright (c) 2009, Atheros Communications
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License version 2 as
8 * published by the Free Software Foundation.
9 *
10 * Alternatively, this software may be distributed under the terms of BSD
11 * license.
12 *
13 * See README and COPYING for more details.
14 */
15
6226e38d 16#include "utils/includes.h"
f1a48710 17
6226e38d 18#include "utils/common.h"
58c26600 19#include "common/ieee802_11_defs.h"
1057d78e 20#include "ap/hostapd.h"
6226e38d 21#include "ap/ap_config.h"
fe6bdb77 22#ifdef NEED_AP_MLME
1057d78e 23#include "ap/ieee802_11.h"
fe6bdb77 24#endif /* NEED_AP_MLME */
a8e0505b 25#include "ap/ieee802_1x.h"
363b9e60 26#include "ap/wps_hostapd.h"
0e2d35c6 27#include "ap/ctrl_iface_ap.h"
f1a48710
JM
28#include "eap_common/eap_defs.h"
29#include "eap_server/eap_methods.h"
30#include "eap_common/eap_wsc_common.h"
3ec97afe 31#include "wps/wps.h"
1f1b62a0 32#include "config_ssid.h"
094393b1 33#include "config.h"
1f1b62a0 34#include "wpa_supplicant_i.h"
2d5b792d
JM
35#include "driver_i.h"
36#include "ap.h"
f1a48710
JM
37
38
07f117ed
JM
39static int wpa_supplicant_conf_ap(struct wpa_supplicant *wpa_s,
40 struct wpa_ssid *ssid,
41 struct hostapd_config *conf)
42{
43 struct hostapd_bss_config *bss = &conf->bss[0];
a911a6e6 44 int pairwise;
c5121837 45
a911a6e6 46 conf->driver = wpa_s->driver;
07f117ed
JM
47
48 os_strlcpy(bss->iface, wpa_s->ifname, sizeof(bss->iface));
49
50 if (ssid->frequency == 0) {
51 /* default channel 11 */
52 conf->hw_mode = HOSTAPD_MODE_IEEE80211G;
53 conf->channel = 11;
54 } else if (ssid->frequency >= 2412 && ssid->frequency <= 2472) {
55 conf->hw_mode = HOSTAPD_MODE_IEEE80211G;
56 conf->channel = (ssid->frequency - 2407) / 5;
57 } else if ((ssid->frequency >= 5180 && ssid->frequency <= 5240) ||
58 (ssid->frequency >= 5745 && ssid->frequency <= 5825)) {
b615a25e 59 conf->hw_mode = HOSTAPD_MODE_IEEE80211A;
07f117ed
JM
60 conf->channel = (ssid->frequency - 5000) / 5;
61 } else {
62 wpa_printf(MSG_ERROR, "Unsupported AP mode frequency: %d MHz",
63 ssid->frequency);
64 return -1;
65 }
66
67 /* TODO: enable HT if driver supports it;
68 * drop to 11b if driver does not support 11g */
69
70 if (ssid->ssid_len == 0) {
71 wpa_printf(MSG_ERROR, "No SSID configured for AP mode");
72 return -1;
73 }
74 os_memcpy(bss->ssid.ssid, ssid->ssid, ssid->ssid_len);
75 bss->ssid.ssid[ssid->ssid_len] = '\0';
76 bss->ssid.ssid_len = ssid->ssid_len;
77 bss->ssid.ssid_set = 1;
78
79 if (wpa_key_mgmt_wpa_psk(ssid->key_mgmt))
80 bss->wpa = ssid->proto;
81 bss->wpa_key_mgmt = ssid->key_mgmt;
82 bss->wpa_pairwise = ssid->pairwise_cipher;
83 if (ssid->passphrase) {
84 bss->ssid.wpa_passphrase = os_strdup(ssid->passphrase);
07f117ed
JM
85 } else if (ssid->psk_set) {
86 os_free(bss->ssid.wpa_psk);
87 bss->ssid.wpa_psk = os_zalloc(sizeof(struct hostapd_wpa_psk));
88 if (bss->ssid.wpa_psk == NULL)
89 return -1;
90 os_memcpy(bss->ssid.wpa_psk->psk, ssid->psk, PMK_LEN);
91 bss->ssid.wpa_psk->group = 1;
92 }
93
07d9a552
JM
94 /* Select group cipher based on the enabled pairwise cipher suites */
95 pairwise = 0;
96 if (bss->wpa & 1)
97 pairwise |= bss->wpa_pairwise;
98 if (bss->wpa & 2) {
99 if (bss->rsn_pairwise == 0)
100 bss->rsn_pairwise = bss->wpa_pairwise;
101 pairwise |= bss->rsn_pairwise;
102 }
103 if (pairwise & WPA_CIPHER_TKIP)
104 bss->wpa_group = WPA_CIPHER_TKIP;
105 else
106 bss->wpa_group = WPA_CIPHER_CCMP;
107
108 if (bss->wpa && bss->ieee802_1x)
109 bss->ssid.security_policy = SECURITY_WPA;
110 else if (bss->wpa)
111 bss->ssid.security_policy = SECURITY_WPA_PSK;
112 else if (bss->ieee802_1x) {
113 bss->ssid.security_policy = SECURITY_IEEE_802_1X;
114 bss->ssid.wep.default_len = bss->default_wep_key_len;
115 } else if (bss->ssid.wep.keys_set)
116 bss->ssid.security_policy = SECURITY_STATIC_WEP;
117 else
118 bss->ssid.security_policy = SECURITY_PLAINTEXT;
119
3ec97afe
JM
120#ifdef CONFIG_WPS
121 /*
122 * Enable WPS by default, but require user interaction to actually use
123 * it. Only the internal Registrar is supported.
124 */
125 bss->eap_server = 1;
126 bss->wps_state = 2;
127 bss->ap_setup_locked = 1;
094393b1
JM
128 if (wpa_s->conf->config_methods)
129 bss->config_methods = os_strdup(wpa_s->conf->config_methods);
130 if (wpa_s->conf->device_type)
131 bss->device_type = os_strdup(wpa_s->conf->device_type);
3ec97afe
JM
132#endif /* CONFIG_WPS */
133
07f117ed
JM
134 return 0;
135}
136
137
2d5b792d
JM
138int wpa_supplicant_create_ap(struct wpa_supplicant *wpa_s,
139 struct wpa_ssid *ssid)
140{
141 struct wpa_driver_associate_params params;
142 struct hostapd_iface *hapd_iface;
143 struct hostapd_config *conf;
144 size_t i;
f1a48710 145
2d5b792d
JM
146 if (ssid->ssid == NULL || ssid->ssid_len == 0) {
147 wpa_printf(MSG_ERROR, "No SSID configured for AP mode");
148 return -1;
f1a48710
JM
149 }
150
2d5b792d 151 wpa_supplicant_ap_deinit(wpa_s);
d2440ba0
JM
152
153 wpa_printf(MSG_DEBUG, "Setting up AP (SSID='%s')",
154 wpa_ssid_txt(ssid->ssid, ssid->ssid_len));
155
156 os_memset(&params, 0, sizeof(params));
157 params.ssid = ssid->ssid;
158 params.ssid_len = ssid->ssid_len;
d7dcba70
JM
159 switch (ssid->mode) {
160 case WPAS_MODE_INFRA:
161 params.mode = IEEE80211_MODE_INFRA;
162 break;
163 case WPAS_MODE_IBSS:
164 params.mode = IEEE80211_MODE_IBSS;
165 break;
166 case WPAS_MODE_AP:
167 params.mode = IEEE80211_MODE_AP;
168 break;
169 }
d2440ba0
JM
170 params.freq = ssid->frequency;
171
508545f3
JM
172 if (ssid->key_mgmt & WPA_KEY_MGMT_PSK)
173 wpa_s->key_mgmt = WPA_KEY_MGMT_PSK;
174 else
175 wpa_s->key_mgmt = WPA_KEY_MGMT_NONE;
176 params.key_mgmt_suite = key_mgmt2driver(wpa_s->key_mgmt);
177
178 if (ssid->pairwise_cipher & WPA_CIPHER_CCMP)
179 wpa_s->pairwise_cipher = WPA_CIPHER_CCMP;
180 else if (ssid->pairwise_cipher & WPA_CIPHER_TKIP)
181 wpa_s->pairwise_cipher = WPA_CIPHER_TKIP;
182 else if (ssid->pairwise_cipher & WPA_CIPHER_NONE)
183 wpa_s->pairwise_cipher = WPA_CIPHER_NONE;
184 else {
185 wpa_printf(MSG_WARNING, "WPA: Failed to select pairwise "
186 "cipher.");
187 return -1;
188 }
189 params.pairwise_suite = cipher_suite2driver(wpa_s->pairwise_cipher);
190 params.group_suite = params.pairwise_suite;
191
d2440ba0
JM
192 if (wpa_drv_associate(wpa_s, &params) < 0) {
193 wpa_msg(wpa_s, MSG_INFO, "Failed to start AP functionality");
194 return -1;
195 }
196
2d5b792d
JM
197 wpa_s->ap_iface = hapd_iface = os_zalloc(sizeof(*wpa_s->ap_iface));
198 if (hapd_iface == NULL)
199 return -1;
0f2b2c19 200 hapd_iface->owner = wpa_s;
1f1b62a0 201
2d5b792d
JM
202 wpa_s->ap_iface->conf = conf = hostapd_config_defaults();
203 if (conf == NULL) {
204 wpa_supplicant_ap_deinit(wpa_s);
205 return -1;
206 }
1f1b62a0 207
07f117ed
JM
208 if (wpa_supplicant_conf_ap(wpa_s, ssid, conf)) {
209 wpa_printf(MSG_ERROR, "Failed to create AP configuration");
210 wpa_supplicant_ap_deinit(wpa_s);
211 return -1;
212 }
213
2d5b792d
JM
214 hapd_iface->num_bss = conf->num_bss;
215 hapd_iface->bss = os_zalloc(conf->num_bss *
216 sizeof(struct hostapd_data *));
217 if (hapd_iface->bss == NULL) {
218 wpa_supplicant_ap_deinit(wpa_s);
219 return -1;
220 }
1f1b62a0 221
2d5b792d
JM
222 for (i = 0; i < conf->num_bss; i++) {
223 hapd_iface->bss[i] =
224 hostapd_alloc_bss_data(hapd_iface, conf,
225 &conf->bss[i]);
226 if (hapd_iface->bss[i] == NULL) {
227 wpa_supplicant_ap_deinit(wpa_s);
228 return -1;
229 }
4f760fcc
JM
230
231 hapd_iface->bss[i]->msg_ctx = wpa_s;
2d5b792d
JM
232 }
233
a911a6e6
JM
234 os_memcpy(hapd_iface->bss[0]->own_addr, wpa_s->own_addr, ETH_ALEN);
235 hapd_iface->bss[0]->driver = wpa_s->driver;
236 hapd_iface->bss[0]->drv_priv = wpa_s->drv_priv;
237
238 if (hostapd_setup_interface(wpa_s->ap_iface)) {
2d5b792d
JM
239 wpa_printf(MSG_ERROR, "Failed to initialize AP interface");
240 wpa_supplicant_ap_deinit(wpa_s);
241 return -1;
1f1b62a0
JM
242 }
243
b2cc8056 244 wpa_s->current_ssid = ssid;
43fb5297 245 os_memcpy(wpa_s->bssid, wpa_s->own_addr, ETH_ALEN);
137fb724
WS
246 wpa_supplicant_set_state(wpa_s, WPA_COMPLETED);
247
4b768ed0
JM
248 if (wpa_s->ap_configured_cb)
249 wpa_s->ap_configured_cb(wpa_s->ap_configured_cb_ctx,
250 wpa_s->ap_configured_cb_data);
251
2d5b792d
JM
252 return 0;
253}
254
255
256void wpa_supplicant_ap_deinit(struct wpa_supplicant *wpa_s)
257{
258 if (wpa_s->ap_iface == NULL)
259 return;
260
7a649c7d 261 wpa_s->current_ssid = NULL;
2d5b792d 262 hostapd_interface_deinit(wpa_s->ap_iface);
f7c47833 263 hostapd_interface_free(wpa_s->ap_iface);
2d5b792d 264 wpa_s->ap_iface = NULL;
7a649c7d 265 wpa_drv_deinit_ap(wpa_s);
1f1b62a0 266}
0915d02c
JM
267
268
269void ap_tx_status(void *ctx, const u8 *addr,
270 const u8 *buf, size_t len, int ack)
271{
f8b1f695 272#ifdef NEED_AP_MLME
0915d02c
JM
273 struct wpa_supplicant *wpa_s = ctx;
274 hostapd_tx_status(wpa_s->ap_iface->bss[0], addr, buf, len, ack);
f8b1f695 275#endif /* NEED_AP_MLME */
0915d02c
JM
276}
277
278
fdbe50ed 279void ap_rx_from_unknown_sta(void *ctx, const u8 *frame, size_t len)
0915d02c 280{
f8b1f695 281#ifdef NEED_AP_MLME
0915d02c 282 struct wpa_supplicant *wpa_s = ctx;
fdbe50ed
JM
283 const struct ieee80211_hdr *hdr =
284 (const struct ieee80211_hdr *) frame;
fbbfcbac
FF
285 u16 fc = le_to_host16(hdr->frame_control);
286 ieee802_11_rx_from_unknown(wpa_s->ap_iface->bss[0], hdr->addr2,
287 (fc & (WLAN_FC_TODS | WLAN_FC_FROMDS)) ==
288 (WLAN_FC_TODS | WLAN_FC_FROMDS));
f8b1f695 289#endif /* NEED_AP_MLME */
0915d02c
JM
290}
291
292
2a8b7416 293void ap_mgmt_rx(void *ctx, struct rx_mgmt *rx_mgmt)
0915d02c 294{
f8b1f695 295#ifdef NEED_AP_MLME
0915d02c 296 struct wpa_supplicant *wpa_s = ctx;
2a8b7416
JM
297 struct hostapd_frame_info fi;
298 os_memset(&fi, 0, sizeof(fi));
299 fi.datarate = rx_mgmt->datarate;
300 fi.ssi_signal = rx_mgmt->ssi_signal;
301 ieee802_11_mgmt(wpa_s->ap_iface->bss[0], rx_mgmt->frame,
302 rx_mgmt->frame_len, &fi);
f8b1f695 303#endif /* NEED_AP_MLME */
0915d02c
JM
304}
305
306
f8b1f695 307void ap_mgmt_tx_cb(void *ctx, const u8 *buf, size_t len, u16 stype, int ok)
0915d02c 308{
f8b1f695 309#ifdef NEED_AP_MLME
0915d02c
JM
310 struct wpa_supplicant *wpa_s = ctx;
311 ieee802_11_mgmt_cb(wpa_s->ap_iface->bss[0], buf, len, stype, ok);
fe6bdb77 312#endif /* NEED_AP_MLME */
f8b1f695 313}
db149ac9
JM
314
315
316void wpa_supplicant_ap_rx_eapol(struct wpa_supplicant *wpa_s,
317 const u8 *src_addr, const u8 *buf, size_t len)
318{
a8e0505b 319 ieee802_1x_receive(wpa_s->ap_iface->bss[0], src_addr, buf, len);
db149ac9 320}
3ec97afe
JM
321
322
323#ifdef CONFIG_WPS
324
325int wpa_supplicant_ap_wps_pbc(struct wpa_supplicant *wpa_s, const u8 *bssid)
326{
327 return hostapd_wps_button_pushed(wpa_s->ap_iface->bss[0]);
328}
329
330
331int wpa_supplicant_ap_wps_pin(struct wpa_supplicant *wpa_s, const u8 *bssid,
332 const char *pin, char *buf, size_t buflen)
333{
334 int ret, ret_len = 0;
335
336 if (pin == NULL) {
337 unsigned int rpin = wps_generate_pin();
338 ret_len = os_snprintf(buf, buflen, "%d", rpin);
339 pin = buf;
340 }
341
077a781f 342 ret = hostapd_wps_add_pin(wpa_s->ap_iface->bss[0], "any", pin, 0);
3ec97afe
JM
343 if (ret)
344 return -1;
345 return ret_len;
346}
347
348#endif /* CONFIG_WPS */
e653b622
JM
349
350
35deb646
JM
351#ifdef CONFIG_CTRL_IFACE
352
e653b622
JM
353int ap_ctrl_iface_sta_first(struct wpa_supplicant *wpa_s,
354 char *buf, size_t buflen)
355{
356 if (wpa_s->ap_iface == NULL)
357 return -1;
358 return hostapd_ctrl_iface_sta_first(wpa_s->ap_iface->bss[0],
359 buf, buflen);
360}
361
362
363int ap_ctrl_iface_sta(struct wpa_supplicant *wpa_s, const char *txtaddr,
364 char *buf, size_t buflen)
365{
366 if (wpa_s->ap_iface == NULL)
367 return -1;
368 return hostapd_ctrl_iface_sta(wpa_s->ap_iface->bss[0], txtaddr,
369 buf, buflen);
370}
371
372
373int ap_ctrl_iface_sta_next(struct wpa_supplicant *wpa_s, const char *txtaddr,
374 char *buf, size_t buflen)
375{
376 if (wpa_s->ap_iface == NULL)
377 return -1;
378 return hostapd_ctrl_iface_sta_next(wpa_s->ap_iface->bss[0], txtaddr,
379 buf, buflen);
380}
35deb646 381
43fb5297
JM
382
383int ap_ctrl_iface_wpa_get_status(struct wpa_supplicant *wpa_s, char *buf,
384 size_t buflen, int verbose)
385{
386 char *pos = buf, *end = buf + buflen;
387 int ret;
388 struct hostapd_bss_config *conf;
389
390 if (wpa_s->ap_iface == NULL)
391 return -1;
392
393 conf = wpa_s->ap_iface->bss[0]->conf;
394 if (conf->wpa == 0)
395 return 0;
396
397 ret = os_snprintf(pos, end - pos,
398 "pairwise_cipher=%s\n"
399 "group_cipher=%s\n"
400 "key_mgmt=%s\n",
401 wpa_cipher_txt(conf->rsn_pairwise),
402 wpa_cipher_txt(conf->wpa_group),
403 wpa_key_mgmt_txt(conf->wpa_key_mgmt,
404 conf->wpa));
405 if (ret < 0 || ret >= end - pos)
406 return pos - buf;
407 pos += ret;
408 return pos - buf;
409}
410
35deb646 411#endif /* CONFIG_CTRL_IFACE */