blob: e5b1fb94a6d12eb556b7488ae9b90e346e05971c [file] [log] [blame]
Arend van Spriel9f440b72013-02-08 15:53:36 +01001/*
2 * Copyright (c) 2012 Broadcom Corporation
3 *
4 * Permission to use, copy, modify, and/or distribute this software for any
5 * purpose with or without fee is hereby granted, provided that the above
6 * copyright notice and this permission notice appear in all copies.
7 *
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
11 * SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
13 * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
14 * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15 */
16#include <linux/slab.h>
17#include <linux/netdevice.h>
18#include <net/cfg80211.h>
19
20#include <brcmu_wifi.h>
21#include <brcmu_utils.h>
22#include <defs.h>
23#include <dhd.h>
24#include <dhd_dbg.h>
25#include "fwil.h"
Arend van Sprield3c0b632013-02-08 15:53:37 +010026#include "fwil_types.h"
Arend van Spriel9f440b72013-02-08 15:53:36 +010027#include "p2p.h"
28#include "wl_cfg80211.h"
29
30/* parameters used for p2p escan */
31#define P2PAPI_SCAN_NPROBES 1
32#define P2PAPI_SCAN_DWELL_TIME_MS 80
33#define P2PAPI_SCAN_SOCIAL_DWELL_TIME_MS 40
34#define P2PAPI_SCAN_HOME_TIME_MS 60
35#define P2PAPI_SCAN_NPROBS_TIME_MS 30
36#define P2PAPI_SCAN_AF_SEARCH_DWELL_TIME_MS 100
37#define WL_SCAN_CONNECT_DWELL_TIME_MS 200
38#define WL_SCAN_JOIN_PROBE_INTERVAL_MS 20
39
40#define BRCMF_P2P_WILDCARD_SSID "DIRECT-"
41#define BRCMF_P2P_WILDCARD_SSID_LEN (sizeof(BRCMF_P2P_WILDCARD_SSID) - 1)
42
43#define SOCIAL_CHAN_1 1
44#define SOCIAL_CHAN_2 6
45#define SOCIAL_CHAN_3 11
46#define SOCIAL_CHAN_CNT 3
47#define AF_PEER_SEARCH_CNT 2
48
Arend van Sprield3c0b632013-02-08 15:53:37 +010049#define BRCMF_SCB_TIMEOUT_VALUE 20
50
Hante Meulemane6da3402013-02-08 15:53:48 +010051#define P2P_VER 9 /* P2P version: 9=WiFi P2P v1.0 */
52#define P2P_PUB_AF_CATEGORY 0x04
53#define P2P_PUB_AF_ACTION 0x09
54#define P2P_AF_CATEGORY 0x7f
55#define P2P_OUI "\x50\x6F\x9A" /* P2P OUI */
56#define P2P_OUI_LEN 3 /* P2P OUI length */
57
58/* WiFi P2P Public Action Frame OUI Subtypes */
59#define P2P_PAF_GON_REQ 0 /* Group Owner Negotiation Req */
60#define P2P_PAF_GON_RSP 1 /* Group Owner Negotiation Rsp */
61#define P2P_PAF_GON_CONF 2 /* Group Owner Negotiation Confirm */
62#define P2P_PAF_INVITE_REQ 3 /* P2P Invitation Request */
63#define P2P_PAF_INVITE_RSP 4 /* P2P Invitation Response */
64#define P2P_PAF_DEVDIS_REQ 5 /* Device Discoverability Request */
65#define P2P_PAF_DEVDIS_RSP 6 /* Device Discoverability Response */
66#define P2P_PAF_PROVDIS_REQ 7 /* Provision Discovery Request */
67#define P2P_PAF_PROVDIS_RSP 8 /* Provision Discovery Response */
68#define P2P_PAF_SUBTYPE_INVALID 255 /* Invalid Subtype */
69
70/* WiFi P2P Action Frame OUI Subtypes */
71#define P2P_AF_NOTICE_OF_ABSENCE 0 /* Notice of Absence */
72#define P2P_AF_PRESENCE_REQ 1 /* P2P Presence Request */
73#define P2P_AF_PRESENCE_RSP 2 /* P2P Presence Response */
74#define P2P_AF_GO_DISC_REQ 3 /* GO Discoverability Request */
75
76/* P2P Service Discovery related */
77#define P2PSD_ACTION_CATEGORY 0x04 /* Public action frame */
78#define P2PSD_ACTION_ID_GAS_IREQ 0x0a /* GAS Initial Request AF */
79#define P2PSD_ACTION_ID_GAS_IRESP 0x0b /* GAS Initial Response AF */
80#define P2PSD_ACTION_ID_GAS_CREQ 0x0c /* GAS Comback Request AF */
81#define P2PSD_ACTION_ID_GAS_CRESP 0x0d /* GAS Comback Response AF */
82
Arend van Spriel9f440b72013-02-08 15:53:36 +010083/**
84 * struct brcmf_p2p_disc_st_le - set discovery state in firmware.
85 *
86 * @state: requested discovery state (see enum brcmf_p2p_disc_state).
87 * @chspec: channel parameter for %WL_P2P_DISC_ST_LISTEN state.
88 * @dwell: dwell time in ms for %WL_P2P_DISC_ST_LISTEN state.
89 */
90struct brcmf_p2p_disc_st_le {
91 u8 state;
92 __le16 chspec;
93 __le16 dwell;
94};
95
96/**
97 * enum brcmf_p2p_disc_state - P2P discovery state values
98 *
99 * @WL_P2P_DISC_ST_SCAN: P2P discovery with wildcard SSID and P2P IE.
100 * @WL_P2P_DISC_ST_LISTEN: P2P discovery off-channel for specified time.
101 * @WL_P2P_DISC_ST_SEARCH: P2P discovery with P2P wildcard SSID and P2P IE.
102 */
103enum brcmf_p2p_disc_state {
104 WL_P2P_DISC_ST_SCAN,
105 WL_P2P_DISC_ST_LISTEN,
106 WL_P2P_DISC_ST_SEARCH
107};
108
109/**
110 * struct brcmf_p2p_scan_le - P2P specific scan request.
111 *
112 * @type: type of scan method requested (values: 'E' or 'S').
113 * @reserved: reserved (ignored).
114 * @eparams: parameters used for type 'E'.
115 * @sparams: parameters used for type 'S'.
116 */
117struct brcmf_p2p_scan_le {
118 u8 type;
119 u8 reserved[3];
120 union {
121 struct brcmf_escan_params_le eparams;
122 struct brcmf_scan_params_le sparams;
123 };
124};
125
Arend van Spriel9f440b72013-02-08 15:53:36 +0100126/**
Hante Meulemane6da3402013-02-08 15:53:48 +0100127 * struct brcmf_p2p_pub_act_frame - WiFi P2P Public Action Frame
128 *
129 * @category: P2P_PUB_AF_CATEGORY
130 * @action: P2P_PUB_AF_ACTION
131 * @oui[3]: P2P_OUI
132 * @oui_type: OUI type - P2P_VER
133 * @subtype: OUI subtype - P2P_TYPE_*
134 * @dialog_token: nonzero, identifies req/rsp transaction
135 * @elts[1]: Variable length information elements.
136 */
137struct brcmf_p2p_pub_act_frame {
138 u8 category;
139 u8 action;
140 u8 oui[3];
141 u8 oui_type;
142 u8 subtype;
143 u8 dialog_token;
144 u8 elts[1];
145};
146
147/**
148 * struct brcmf_p2p_action_frame - WiFi P2P Action Frame
149 *
150 * @category: P2P_AF_CATEGORY
151 * @OUI[3]: OUI - P2P_OUI
152 * @type: OUI Type - P2P_VER
153 * @subtype: OUI Subtype - P2P_AF_*
154 * @dialog_token: nonzero, identifies req/resp tranaction
155 * @elts[1]: Variable length information elements.
156 */
157struct brcmf_p2p_action_frame {
158 u8 category;
159 u8 oui[3];
160 u8 type;
161 u8 subtype;
162 u8 dialog_token;
163 u8 elts[1];
164};
165
166/**
167 * struct brcmf_p2psd_gas_pub_act_frame - Wi-Fi GAS Public Action Frame
168 *
169 * @category: 0x04 Public Action Frame
170 * @action: 0x6c Advertisement Protocol
171 * @dialog_token: nonzero, identifies req/rsp transaction
172 * @query_data[1]: Query Data. SD gas ireq SD gas iresp
173 */
174struct brcmf_p2psd_gas_pub_act_frame {
175 u8 category;
176 u8 action;
177 u8 dialog_token;
178 u8 query_data[1];
179};
180
181
182/**
183 * brcmf_p2p_is_pub_action() - true if p2p public type frame.
184 *
185 * @frame: action frame data.
186 * @frame_len: length of action frame data.
187 *
188 * Determine if action frame is p2p public action type
189 */
190static bool brcmf_p2p_is_pub_action(void *frame, u32 frame_len)
191{
192 struct brcmf_p2p_pub_act_frame *pact_frm;
193
194 if (frame == NULL)
195 return false;
196
197 pact_frm = (struct brcmf_p2p_pub_act_frame *)frame;
198 if (frame_len < sizeof(struct brcmf_p2p_pub_act_frame) - 1)
199 return false;
200
201 if (pact_frm->category == P2P_PUB_AF_CATEGORY &&
202 pact_frm->action == P2P_PUB_AF_ACTION &&
203 pact_frm->oui_type == P2P_VER &&
204 memcmp(pact_frm->oui, P2P_OUI, P2P_OUI_LEN) == 0)
205 return true;
206
207 return false;
208}
209
210/**
211 * brcmf_p2p_is_p2p_action() - true if p2p action type frame.
212 *
213 * @frame: action frame data.
214 * @frame_len: length of action frame data.
215 *
216 * Determine if action frame is p2p action type
217 */
218static bool brcmf_p2p_is_p2p_action(void *frame, u32 frame_len)
219{
220 struct brcmf_p2p_action_frame *act_frm;
221
222 if (frame == NULL)
223 return false;
224
225 act_frm = (struct brcmf_p2p_action_frame *)frame;
226 if (frame_len < sizeof(struct brcmf_p2p_action_frame) - 1)
227 return false;
228
229 if (act_frm->category == P2P_AF_CATEGORY &&
230 act_frm->type == P2P_VER &&
231 memcmp(act_frm->oui, P2P_OUI, P2P_OUI_LEN) == 0)
232 return true;
233
234 return false;
235}
236
237/**
238 * brcmf_p2p_is_gas_action() - true if p2p gas action type frame.
239 *
240 * @frame: action frame data.
241 * @frame_len: length of action frame data.
242 *
243 * Determine if action frame is p2p gas action type
244 */
245static bool brcmf_p2p_is_gas_action(void *frame, u32 frame_len)
246{
247 struct brcmf_p2psd_gas_pub_act_frame *sd_act_frm;
248
249 if (frame == NULL)
250 return false;
251
252 sd_act_frm = (struct brcmf_p2psd_gas_pub_act_frame *)frame;
253 if (frame_len < sizeof(struct brcmf_p2psd_gas_pub_act_frame) - 1)
254 return false;
255
256 if (sd_act_frm->category != P2PSD_ACTION_CATEGORY)
257 return false;
258
259 if (sd_act_frm->action == P2PSD_ACTION_ID_GAS_IREQ ||
260 sd_act_frm->action == P2PSD_ACTION_ID_GAS_IRESP ||
261 sd_act_frm->action == P2PSD_ACTION_ID_GAS_CREQ ||
262 sd_act_frm->action == P2PSD_ACTION_ID_GAS_CRESP)
263 return true;
264
265 return false;
266}
267
268/**
269 * brcmf_p2p_print_actframe() - debug print routine.
270 *
271 * @tx: Received or to be transmitted
272 * @frame: action frame data.
273 * @frame_len: length of action frame data.
274 *
275 * Print information about the p2p action frame
276 */
277static void brcmf_p2p_print_actframe(bool tx, void *frame, u32 frame_len)
278{
279 struct brcmf_p2p_pub_act_frame *pact_frm;
280 struct brcmf_p2p_action_frame *act_frm;
281 struct brcmf_p2psd_gas_pub_act_frame *sd_act_frm;
282
283 if (!frame || frame_len <= 2)
284 return;
285
286 if (brcmf_p2p_is_pub_action(frame, frame_len)) {
287 pact_frm = (struct brcmf_p2p_pub_act_frame *)frame;
288 switch (pact_frm->subtype) {
289 case P2P_PAF_GON_REQ:
290 brcmf_dbg(TRACE, "%s P2P Group Owner Negotiation Req Frame\n",
291 (tx) ? "TX" : "RX");
292 break;
293 case P2P_PAF_GON_RSP:
294 brcmf_dbg(TRACE, "%s P2P Group Owner Negotiation Rsp Frame\n",
295 (tx) ? "TX" : "RX");
296 break;
297 case P2P_PAF_GON_CONF:
298 brcmf_dbg(TRACE, "%s P2P Group Owner Negotiation Confirm Frame\n",
299 (tx) ? "TX" : "RX");
300 break;
301 case P2P_PAF_INVITE_REQ:
302 brcmf_dbg(TRACE, "%s P2P Invitation Request Frame\n",
303 (tx) ? "TX" : "RX");
304 break;
305 case P2P_PAF_INVITE_RSP:
306 brcmf_dbg(TRACE, "%s P2P Invitation Response Frame\n",
307 (tx) ? "TX" : "RX");
308 break;
309 case P2P_PAF_DEVDIS_REQ:
310 brcmf_dbg(TRACE, "%s P2P Device Discoverability Request Frame\n",
311 (tx) ? "TX" : "RX");
312 break;
313 case P2P_PAF_DEVDIS_RSP:
314 brcmf_dbg(TRACE, "%s P2P Device Discoverability Response Frame\n",
315 (tx) ? "TX" : "RX");
316 break;
317 case P2P_PAF_PROVDIS_REQ:
318 brcmf_dbg(TRACE, "%s P2P Provision Discovery Request Frame\n",
319 (tx) ? "TX" : "RX");
320 break;
321 case P2P_PAF_PROVDIS_RSP:
322 brcmf_dbg(TRACE, "%s P2P Provision Discovery Response Frame\n",
323 (tx) ? "TX" : "RX");
324 break;
325 default:
326 brcmf_dbg(TRACE, "%s Unknown P2P Public Action Frame\n",
327 (tx) ? "TX" : "RX");
328 break;
329 }
330 } else if (brcmf_p2p_is_p2p_action(frame, frame_len)) {
331 act_frm = (struct brcmf_p2p_action_frame *)frame;
332 switch (act_frm->subtype) {
333 case P2P_AF_NOTICE_OF_ABSENCE:
334 brcmf_dbg(TRACE, "%s P2P Notice of Absence Frame\n",
335 (tx) ? "TX" : "RX");
336 break;
337 case P2P_AF_PRESENCE_REQ:
338 brcmf_dbg(TRACE, "%s P2P Presence Request Frame\n",
339 (tx) ? "TX" : "RX");
340 break;
341 case P2P_AF_PRESENCE_RSP:
342 brcmf_dbg(TRACE, "%s P2P Presence Response Frame\n",
343 (tx) ? "TX" : "RX");
344 break;
345 case P2P_AF_GO_DISC_REQ:
346 brcmf_dbg(TRACE, "%s P2P Discoverability Request Frame\n",
347 (tx) ? "TX" : "RX");
348 break;
349 default:
350 brcmf_dbg(TRACE, "%s Unknown P2P Action Frame\n",
351 (tx) ? "TX" : "RX");
352 }
353
354 } else if (brcmf_p2p_is_gas_action(frame, frame_len)) {
355 sd_act_frm = (struct brcmf_p2psd_gas_pub_act_frame *)frame;
356 switch (sd_act_frm->action) {
357 case P2PSD_ACTION_ID_GAS_IREQ:
358 brcmf_dbg(TRACE, "%s P2P GAS Initial Request\n",
359 (tx) ? "TX" : "RX");
360 break;
361 case P2PSD_ACTION_ID_GAS_IRESP:
362 brcmf_dbg(TRACE, "%s P2P GAS Initial Response\n",
363 (tx) ? "TX" : "RX");
364 break;
365 case P2PSD_ACTION_ID_GAS_CREQ:
366 brcmf_dbg(TRACE, "%s P2P GAS Comback Request\n",
367 (tx) ? "TX" : "RX");
368 break;
369 case P2PSD_ACTION_ID_GAS_CRESP:
370 brcmf_dbg(TRACE, "%s P2P GAS Comback Response\n",
371 (tx) ? "TX" : "RX");
372 break;
373 default:
374 brcmf_dbg(TRACE, "%s Unknown P2P GAS Frame\n",
375 (tx) ? "TX" : "RX");
376 break;
377 }
378 }
379}
380
381/**
Arend van Spriel9f440b72013-02-08 15:53:36 +0100382 * brcmf_p2p_set_firmware() - prepare firmware for peer-to-peer operation.
383 *
384 * @p2p: P2P specific data.
385 */
386static int brcmf_p2p_set_firmware(struct brcmf_p2p_info *p2p)
387{
388 struct net_device *ndev = cfg_to_ndev(p2p->cfg);
389 u8 null_eth_addr[] = { 0, 0, 0, 0, 0, 0 };
390 s32 ret = 0;
391
392 brcmf_fil_iovar_int_set(netdev_priv(ndev), "apsta", 1);
393
394 /* In case of COB type, firmware has default mac address
395 * After Initializing firmware, we have to set current mac address to
396 * firmware for P2P device address
397 */
398 ret = brcmf_fil_iovar_data_set(netdev_priv(ndev), "p2p_da_override",
399 null_eth_addr, sizeof(null_eth_addr));
400 if (ret)
401 brcmf_err("failed to update device address ret %d\n", ret);
402
403 return ret;
404}
405
406/**
407 * brcmf_p2p_generate_bss_mac() - derive mac addresses for P2P.
408 *
409 * @p2p: P2P specific data.
410 *
411 * P2P needs mac addresses for P2P device and interface. These are
412 * derived from the primary net device, ie. the permanent ethernet
413 * address of the device.
414 */
415static void brcmf_p2p_generate_bss_mac(struct brcmf_p2p_info *p2p)
416{
Arend van Spriel9f440b72013-02-08 15:53:36 +0100417 /* Generate the P2P Device Address. This consists of the device's
418 * primary MAC address with the locally administered bit set.
419 */
Arend van Sprielde66efc2013-02-08 15:53:40 +0100420 memcpy(p2p->dev_addr, p2p->cfg->pub->mac, ETH_ALEN);
Arend van Spriel9f440b72013-02-08 15:53:36 +0100421 p2p->dev_addr[0] |= 0x02;
422
423 /* Generate the P2P Interface Address. If the discovery and connection
424 * BSSCFGs need to simultaneously co-exist, then this address must be
425 * different from the P2P Device Address, but also locally administered.
426 */
427 memcpy(p2p->int_addr, p2p->dev_addr, ETH_ALEN);
428 p2p->int_addr[4] ^= 0x80;
429}
430
431/**
432 * brcmf_p2p_scan_is_p2p_request() - is cfg80211 scan request a P2P scan.
433 *
434 * @request: the scan request as received from cfg80211.
435 *
436 * returns true if one of the ssids in the request matches the
437 * P2P wildcard ssid; otherwise returns false.
438 */
439static bool brcmf_p2p_scan_is_p2p_request(struct cfg80211_scan_request *request)
440{
441 struct cfg80211_ssid *ssids = request->ssids;
442 int i;
443
444 for (i = 0; i < request->n_ssids; i++) {
445 if (ssids[i].ssid_len != BRCMF_P2P_WILDCARD_SSID_LEN)
446 continue;
447
448 brcmf_dbg(INFO, "comparing ssid \"%s\"", ssids[i].ssid);
449 if (!memcmp(BRCMF_P2P_WILDCARD_SSID, ssids[i].ssid,
450 BRCMF_P2P_WILDCARD_SSID_LEN))
451 return true;
452 }
453 return false;
454}
455
456/**
457 * brcmf_p2p_set_discover_state - set discover state in firmware.
458 *
459 * @ifp: low-level interface object.
460 * @state: discover state to set.
461 * @chanspec: channel parameters (for state @WL_P2P_DISC_ST_LISTEN only).
462 * @listen_ms: duration to listen (for state @WL_P2P_DISC_ST_LISTEN only).
463 */
464static s32 brcmf_p2p_set_discover_state(struct brcmf_if *ifp, u8 state,
465 u16 chanspec, u16 listen_ms)
466{
467 struct brcmf_p2p_disc_st_le discover_state;
468 s32 ret = 0;
469 brcmf_dbg(TRACE, "enter\n");
470
471 discover_state.state = state;
472 discover_state.chspec = cpu_to_le16(chanspec);
473 discover_state.dwell = cpu_to_le16(listen_ms);
474 ret = brcmf_fil_bsscfg_data_set(ifp, "p2p_state", &discover_state,
475 sizeof(discover_state));
476 return ret;
477}
478
479/**
Arend van Spriel9f440b72013-02-08 15:53:36 +0100480 * brcmf_p2p_init_discovery() - enable discovery in the firmware.
481 *
482 * @p2p: P2P specific data.
483 *
484 * Configures the firmware to allow P2P peer discovery. Creates the
485 * virtual interface and consequently the P2P device for it.
486 */
487static s32 brcmf_p2p_init_discovery(struct brcmf_p2p_info *p2p)
488{
489 struct net_device *ndev = cfg_to_ndev(p2p->cfg);
490 struct brcmf_cfg80211_vif *vif;
491 struct brcmf_if *ifp;
492 struct p2p_bss *bss_dev;
493 s32 index;
494 s32 ret;
495
496 brcmf_dbg(TRACE, "enter\n");
497
498 bss_dev = &p2p->bss_idx[P2PAPI_BSSCFG_DEVICE];
499 if (bss_dev->vif != NULL) {
500 brcmf_dbg(INFO, "do nothing, already initialized\n");
501 return 0;
502 }
503
504 /* Enable P2P Discovery in the firmware */
505 ret = brcmf_fil_iovar_int_set(netdev_priv(ndev), "p2p_disc", 1);
506 if (ret < 0) {
507 brcmf_err("set discover error\n");
508 return ret;
509 }
510
511 /* obtain bsscfg index for P2P discovery */
512 ret = brcmf_fil_iovar_int_get(netdev_priv(ndev), "p2p_dev", &index);
513 if (ret < 0) {
514 brcmf_err("retrieving discover bsscfg index failed\n");
515 return ret;
516 }
517
518 /*
519 * need brcmf_if for setting the discovery state.
520 */
521 ifp = kzalloc(sizeof(*vif->ifp), GFP_KERNEL);
522 if (!ifp) {
523 brcmf_err("could not create discovery if\n");
524 return -ENOMEM;
525 }
526
527 /* set required fields */
528 ifp->drvr = p2p->cfg->pub;
529 ifp->ifidx = 0;
530 ifp->bssidx = index;
531
532 /* Set the initial discovery state to SCAN */
533 ret = brcmf_p2p_set_discover_state(ifp, WL_P2P_DISC_ST_SCAN, 0, 0);
534
535 if (ret != 0) {
536 brcmf_err("unable to set WL_P2P_DISC_ST_SCAN\n");
537 (void)brcmf_fil_iovar_int_set(netdev_priv(ndev), "p2p_disc", 0);
538 kfree(ifp);
539 return ret;
540 }
541
542 /* create a vif for it */
Arend van Sprield3c0b632013-02-08 15:53:37 +0100543 vif = brcmf_alloc_vif(p2p->cfg, NL80211_IFTYPE_P2P_DEVICE, false);
Arend van Spriel9f440b72013-02-08 15:53:36 +0100544 if (IS_ERR(vif)) {
545 brcmf_err("could not create discovery vif\n");
546 kfree(ifp);
547 return PTR_ERR(vif);
548 }
549
550 vif->ifp = ifp;
551 ifp->vif = vif;
552 bss_dev->vif = vif;
553
554 return 0;
555}
556
557/**
558 * brcmf_p2p_deinit_discovery() - disable P2P device discovery.
559 *
560 * @p2p: P2P specific data.
561 *
562 * Resets the discovery state and disables it in firmware. The virtual
563 * interface and P2P device are freed.
564 */
565static s32 brcmf_p2p_deinit_discovery(struct brcmf_p2p_info *p2p)
566{
567 struct net_device *ndev = cfg_to_ndev(p2p->cfg);
568 struct brcmf_if *ifp;
569 struct p2p_bss *bss_dev;
570 brcmf_dbg(TRACE, "enter\n");
571
572 bss_dev = &p2p->bss_idx[P2PAPI_BSSCFG_DEVICE];
Arend van Spriel9f440b72013-02-08 15:53:36 +0100573 ifp = bss_dev->vif->ifp;
574
575 /* Set the discovery state to SCAN */
576 (void)brcmf_p2p_set_discover_state(ifp, WL_P2P_DISC_ST_SCAN, 0, 0);
577
578 /* Disable P2P discovery in the firmware */
579 (void)brcmf_fil_iovar_int_set(netdev_priv(ndev), "p2p_disc", 0);
580
581 /* remove discovery interface */
582 brcmf_free_vif(bss_dev->vif);
583 bss_dev->vif = NULL;
584 kfree(ifp);
585
586 return 0;
587}
588
589/**
590 * brcmf_p2p_enable_discovery() - initialize and configure discovery.
591 *
592 * @p2p: P2P specific data.
593 * @ie: buffer containing information elements.
594 * @ie_len: length of @ie buffer.
595 *
596 * Initializes the discovery device and configure the virtual interface.
597 */
Hante Meuleman0de8aac2013-02-08 15:53:38 +0100598static int brcmf_p2p_enable_discovery(struct brcmf_p2p_info *p2p)
Arend van Spriel9f440b72013-02-08 15:53:36 +0100599{
600 struct brcmf_cfg80211_vif *vif;
601 s32 ret = 0;
602
603 brcmf_dbg(TRACE, "enter\n");
604 vif = p2p->bss_idx[P2PAPI_BSSCFG_DEVICE].vif;
605 if (vif) {
Hante Meuleman0de8aac2013-02-08 15:53:38 +0100606 brcmf_dbg(INFO, "DISCOVERY init already done\n");
607 goto exit;
Arend van Spriel9f440b72013-02-08 15:53:36 +0100608 }
609
610 ret = brcmf_p2p_init_discovery(p2p);
611 if (ret < 0) {
612 brcmf_err("init discovery error %d\n", ret);
613 goto exit;
614 }
615
616 vif = p2p->bss_idx[P2PAPI_BSSCFG_DEVICE].vif;
617
618 /*
619 * Set wsec to any non-zero value in the discovery bsscfg
620 * to ensure our P2P probe responses have the privacy bit
621 * set in the 802.11 WPA IE. Some peer devices may not
622 * initiate WPS with us if this bit is not set.
623 */
624 ret = brcmf_fil_bsscfg_int_set(vif->ifp, "wsec", AES_ENABLED);
625 if (ret < 0)
626 brcmf_err("wsec error %d\n", ret);
627
Arend van Spriel9f440b72013-02-08 15:53:36 +0100628exit:
629 return ret;
630}
631
Hante Meuleman0de8aac2013-02-08 15:53:38 +0100632/**
Arend van Spriel9f440b72013-02-08 15:53:36 +0100633 * brcmf_p2p_escan() - initiate a P2P scan.
634 *
635 * @p2p: P2P specific data.
636 * @num_chans: number of channels to scan.
637 * @chanspecs: channel parameters for @num_chans channels.
638 * @search_state: P2P discover state to use.
639 * @action: scan action to pass to firmware.
640 * @bss_type: type of P2P bss.
641 */
642static s32 brcmf_p2p_escan(struct brcmf_p2p_info *p2p, u32 num_chans,
643 u16 chanspecs[], s32 search_state, u16 action,
644 enum p2p_bss_type bss_type)
645{
646 s32 ret = 0;
647 s32 memsize = offsetof(struct brcmf_p2p_scan_le,
648 eparams.params_le.channel_list);
649 s32 nprobes;
650 s32 active;
651 u32 i;
652 u8 *memblk;
653 struct brcmf_cfg80211_vif *vif;
654 struct brcmf_p2p_scan_le *p2p_params;
655 struct brcmf_scan_params_le *sparams;
656 struct brcmf_ssid ssid;
657
Arend van Spriel9f440b72013-02-08 15:53:36 +0100658 memsize += num_chans * sizeof(__le16);
659 memblk = kzalloc(memsize, GFP_KERNEL);
660 if (!memblk)
661 return -ENOMEM;
662
663 vif = p2p->bss_idx[bss_type].vif;
664 if (vif == NULL) {
665 brcmf_err("no vif for bss type %d\n", bss_type);
666 ret = -EINVAL;
667 goto exit;
668 }
669
670 switch (search_state) {
671 case WL_P2P_DISC_ST_SEARCH:
672 /*
673 * If we in SEARCH STATE, we don't need to set SSID explictly
674 * because dongle use P2P WILDCARD internally by default
675 */
676 /* use null ssid */
677 ssid.SSID_len = 0;
678 memset(ssid.SSID, 0, sizeof(ssid.SSID));
679 break;
680 case WL_P2P_DISC_ST_SCAN:
681 /*
682 * wpa_supplicant has p2p_find command with type social or
683 * progressive. For progressive, we need to set the ssid to
684 * P2P WILDCARD because we just do broadcast scan unless
685 * setting SSID.
686 */
687 ssid.SSID_len = BRCMF_P2P_WILDCARD_SSID_LEN;
688 memcpy(ssid.SSID, BRCMF_P2P_WILDCARD_SSID, ssid.SSID_len);
689 break;
690 default:
691 brcmf_err(" invalid search state %d\n", search_state);
692 ret = -EINVAL;
693 goto exit;
694 }
695
696 brcmf_p2p_set_discover_state(vif->ifp, search_state, 0, 0);
697
698 /*
699 * set p2p scan parameters.
700 */
701 p2p_params = (struct brcmf_p2p_scan_le *)memblk;
702 p2p_params->type = 'E';
703
704 /* determine the scan engine parameters */
705 sparams = &p2p_params->eparams.params_le;
706 sparams->bss_type = DOT11_BSSTYPE_ANY;
707 if (p2p->cfg->active_scan)
708 sparams->scan_type = 0;
709 else
710 sparams->scan_type = 1;
711
712 memset(&sparams->bssid, 0xFF, ETH_ALEN);
713 if (ssid.SSID_len)
714 memcpy(sparams->ssid_le.SSID, ssid.SSID, ssid.SSID_len);
715 sparams->ssid_le.SSID_len = cpu_to_le32(ssid.SSID_len);
716 sparams->home_time = cpu_to_le32(P2PAPI_SCAN_HOME_TIME_MS);
717
718 /*
719 * SOCIAL_CHAN_CNT + 1 takes care of the Progressive scan
720 * supported by the supplicant.
721 */
722 if (num_chans == SOCIAL_CHAN_CNT || num_chans == (SOCIAL_CHAN_CNT + 1))
723 active = P2PAPI_SCAN_SOCIAL_DWELL_TIME_MS;
724 else if (num_chans == AF_PEER_SEARCH_CNT)
725 active = P2PAPI_SCAN_AF_SEARCH_DWELL_TIME_MS;
726 else if (wl_get_vif_state_all(p2p->cfg, BRCMF_VIF_STATUS_CONNECTED))
727 active = -1;
728 else
729 active = P2PAPI_SCAN_DWELL_TIME_MS;
730
731 /* Override scan params to find a peer for a connection */
732 if (num_chans == 1) {
733 active = WL_SCAN_CONNECT_DWELL_TIME_MS;
734 /* XXX WAR to sync with presence period of VSDB GO.
735 * send probe request more frequently
736 */
737 nprobes = active / WL_SCAN_JOIN_PROBE_INTERVAL_MS;
738 } else {
739 nprobes = active / P2PAPI_SCAN_NPROBS_TIME_MS;
740 }
741
742 if (nprobes <= 0)
743 nprobes = 1;
744
745 brcmf_dbg(INFO, "nprobes # %d, active_time %d\n", nprobes, active);
746 sparams->active_time = cpu_to_le32(active);
747 sparams->nprobes = cpu_to_le32(nprobes);
748 sparams->passive_time = cpu_to_le32(-1);
749 sparams->channel_num = cpu_to_le32(num_chans &
750 BRCMF_SCAN_PARAMS_COUNT_MASK);
751 for (i = 0; i < num_chans; i++)
752 sparams->channel_list[i] = cpu_to_le16(chanspecs[i]);
753
754 /* set the escan specific parameters */
755 p2p_params->eparams.version = cpu_to_le32(BRCMF_ESCAN_REQ_VERSION);
756 p2p_params->eparams.action = cpu_to_le16(action);
757 p2p_params->eparams.sync_id = cpu_to_le16(0x1234);
758 /* perform p2p scan on primary device */
759 ret = brcmf_fil_bsscfg_data_set(vif->ifp, "p2p_scan", memblk, memsize);
760 if (!ret)
761 set_bit(BRCMF_SCAN_STATUS_BUSY, &p2p->cfg->scan_status);
762exit:
763 kfree(memblk);
764 return ret;
765}
766
767/**
768 * brcmf_p2p_run_escan() - escan callback for peer-to-peer.
769 *
770 * @cfg: driver private data for cfg80211 interface.
771 * @ndev: net device for which scan is requested.
772 * @request: scan request from cfg80211.
773 * @action: scan action.
774 *
775 * Determines the P2P discovery state based to scan request parameters and
776 * validates the channels in the request.
777 */
778static s32 brcmf_p2p_run_escan(struct brcmf_cfg80211_info *cfg,
779 struct net_device *ndev,
780 struct cfg80211_scan_request *request,
781 u16 action)
782{
783 struct brcmf_p2p_info *p2p = &cfg->p2p;
784 s32 err = 0;
785 s32 search_state = WL_P2P_DISC_ST_SCAN;
786 struct brcmf_cfg80211_vif *vif;
787 struct net_device *dev = NULL;
788 int i, num_nodfs = 0;
789 u16 *chanspecs;
790
791 brcmf_dbg(TRACE, "enter\n");
792
793 if (!request) {
794 err = -EINVAL;
795 goto exit;
796 }
797
798 if (request->n_channels) {
799 chanspecs = kcalloc(request->n_channels, sizeof(*chanspecs),
800 GFP_KERNEL);
801 if (!chanspecs) {
802 err = -ENOMEM;
803 goto exit;
804 }
805 vif = p2p->bss_idx[P2PAPI_BSSCFG_CONNECTION].vif;
806 if (vif)
807 dev = vif->wdev.netdev;
808 if (request->n_channels == 3 &&
809 request->channels[0]->hw_value == SOCIAL_CHAN_1 &&
810 request->channels[1]->hw_value == SOCIAL_CHAN_2 &&
811 request->channels[2]->hw_value == SOCIAL_CHAN_3) {
812 /* SOCIAL CHANNELS 1, 6, 11 */
813 search_state = WL_P2P_DISC_ST_SEARCH;
814 brcmf_dbg(INFO, "P2P SEARCH PHASE START\n");
815 } else if (dev != NULL && vif->mode == WL_MODE_AP) {
816 /* If you are already a GO, then do SEARCH only */
817 brcmf_dbg(INFO, "Already a GO. Do SEARCH Only\n");
818 search_state = WL_P2P_DISC_ST_SEARCH;
819 } else {
820 brcmf_dbg(INFO, "P2P SCAN STATE START\n");
821 }
822
823 /*
824 * no P2P scanning on passive or DFS channels.
825 */
826 for (i = 0; i < request->n_channels; i++) {
827 struct ieee80211_channel *chan = request->channels[i];
828
829 if (chan->flags & (IEEE80211_CHAN_RADAR |
830 IEEE80211_CHAN_PASSIVE_SCAN))
831 continue;
832
833 chanspecs[i] = channel_to_chanspec(chan);
834 brcmf_dbg(INFO, "%d: chan=%d, channel spec=%x\n",
835 num_nodfs, chan->hw_value, chanspecs[i]);
836 num_nodfs++;
837 }
838 err = brcmf_p2p_escan(p2p, num_nodfs, chanspecs, search_state,
839 action, P2PAPI_BSSCFG_DEVICE);
840 }
841exit:
842 if (err)
843 brcmf_err("error (%d)\n", err);
844 return err;
845}
846
847/**
848 * brcmf_p2p_scan_prep() - prepare scan based on request.
849 *
850 * @wiphy: wiphy device.
851 * @request: scan request from cfg80211.
Hante Meuleman0f8ffe12013-02-08 15:53:42 +0100852 * @vif: vif on which scan request is to be executed.
Arend van Spriel9f440b72013-02-08 15:53:36 +0100853 *
854 * Prepare the scan appropriately for type of scan requested. Overrides the
855 * escan .run() callback for peer-to-peer scanning.
856 */
857int brcmf_p2p_scan_prep(struct wiphy *wiphy,
Hante Meuleman0f8ffe12013-02-08 15:53:42 +0100858 struct cfg80211_scan_request *request,
859 struct brcmf_cfg80211_vif *vif)
Arend van Spriel9f440b72013-02-08 15:53:36 +0100860{
861 struct brcmf_cfg80211_info *cfg = wiphy_to_cfg(wiphy);
862 struct brcmf_p2p_info *p2p = &cfg->p2p;
863 int err = 0;
864
865 if (brcmf_p2p_scan_is_p2p_request(request)) {
866 /* find my listen channel */
867 err = cfg80211_get_p2p_attr(request->ie, request->ie_len,
868 IEEE80211_P2P_ATTR_LISTEN_CHANNEL,
869 &p2p->listen_channel, 1);
870 if (err < 0)
871 return err;
872
873 clear_bit(BRCMF_P2P_STATUS_GO_NEG_PHASE, &p2p->status);
874 brcmf_dbg(INFO, "P2P: GO_NEG_PHASE status cleared\n");
875
Hante Meuleman0de8aac2013-02-08 15:53:38 +0100876 err = brcmf_p2p_enable_discovery(p2p);
Hante Meuleman0f8ffe12013-02-08 15:53:42 +0100877 if (err)
878 return err;
Arend van Spriel9f440b72013-02-08 15:53:36 +0100879
Hante Meuleman0f8ffe12013-02-08 15:53:42 +0100880 vif = p2p->bss_idx[P2PAPI_BSSCFG_DEVICE].vif;
881
882 /* override .run_escan() callback. */
Arend van Spriel9f440b72013-02-08 15:53:36 +0100883 cfg->escan_info.run = brcmf_p2p_run_escan;
Arend van Spriel9f440b72013-02-08 15:53:36 +0100884 }
Hante Meuleman0f8ffe12013-02-08 15:53:42 +0100885 err = brcmf_vif_set_mgmt_ie(vif, BRCMF_VNDR_IE_PRBREQ_FLAG,
886 request->ie, request->ie_len);
Arend van Spriel9f440b72013-02-08 15:53:36 +0100887 return err;
888}
889
Hante Meuleman0de8aac2013-02-08 15:53:38 +0100890
891/**
892 * brcmf_p2p_remain_on_channel() - put device on channel and stay there.
893 *
894 * @wiphy: wiphy device.
895 * @channel: channel to stay on.
896 * @duration: time in ms to remain on channel.
897 *
898 */
899int brcmf_p2p_remain_on_channel(struct wiphy *wiphy, struct wireless_dev *wdev,
900 struct ieee80211_channel *channel,
901 unsigned int duration, u64 *cookie)
902{
903 struct brcmf_cfg80211_info *cfg = wiphy_to_cfg(wiphy);
904 struct brcmf_p2p_info *p2p = &cfg->p2p;
905 struct brcmf_cfg80211_vif *vif;
906 s32 err;
907 u16 chanspec;
908
909 brcmf_dbg(TRACE, "Enter, channel: %d, duration ms (%d)\n",
910 ieee80211_frequency_to_channel(channel->center_freq),
911 duration);
912
913 *cookie = 0;
914 err = brcmf_p2p_enable_discovery(p2p);
915 if (err)
916 goto exit;
917
918 chanspec = channel_to_chanspec(channel);
919 vif = p2p->bss_idx[P2PAPI_BSSCFG_DEVICE].vif;
920 err = brcmf_p2p_set_discover_state(vif->ifp, WL_P2P_DISC_ST_LISTEN,
921 chanspec, (u16)duration);
922 if (err)
923 goto exit;
924
925 memcpy(&p2p->remain_on_channel, channel,
926 sizeof(p2p->remain_on_channel));
927
928 set_bit(BRCMF_P2P_STATUS_REMAIN_ON_CHANNEL, &p2p->status);
929
930exit:
931 cfg80211_ready_on_channel(wdev, *cookie, channel, duration, GFP_KERNEL);
932 return err;
933}
934
935
936/**
937 * brcmf_p2p_notify_listen_complete() - p2p listen has completed.
938 *
939 * @ifp: interfac control.
940 * @e: event message. Not used, to make it usable for fweh event dispatcher.
941 * @data: payload of message. Not used.
942 *
943 */
944int brcmf_p2p_notify_listen_complete(struct brcmf_if *ifp,
945 const struct brcmf_event_msg *e,
946 void *data)
947{
948 struct brcmf_cfg80211_info *cfg = ifp->drvr->config;
949 struct brcmf_p2p_info *p2p = &cfg->p2p;
950
951 brcmf_dbg(TRACE, "Enter\n");
952 if (test_and_clear_bit(BRCMF_P2P_STATUS_REMAIN_ON_CHANNEL,
953 &p2p->status))
954 cfg80211_remain_on_channel_expired(&ifp->vif->wdev, 0,
955 &p2p->remain_on_channel,
956 GFP_KERNEL);
957 return 0;
958}
959
960
961/**
962 * brcmf_p2p_cancel_remain_on_channel() - cancel p2p listen state.
963 *
964 * @ifp: interfac control.
965 *
966 */
967void brcmf_p2p_cancel_remain_on_channel(struct brcmf_if *ifp)
968{
969 if (!ifp)
970 return;
971 brcmf_p2p_set_discover_state(ifp, WL_P2P_DISC_ST_SCAN, 0, 0);
972 brcmf_p2p_notify_listen_complete(ifp, NULL, NULL);
973}
974
975
Arend van Spriel9f440b72013-02-08 15:53:36 +0100976/**
Hante Meulemane6da3402013-02-08 15:53:48 +0100977 * brcmf_p2p_notify_action_frame_rx() - received action frame.
978 *
979 * @ifp: interfac control.
980 * @e: event message. Not used, to make it usable for fweh event dispatcher.
981 * @data: payload of message, containing action frame data.
982 *
983 */
984int brcmf_p2p_notify_action_frame_rx(struct brcmf_if *ifp,
985 const struct brcmf_event_msg *e,
986 void *data)
987{
988 struct wireless_dev *wdev;
989 u32 mgmt_frame_len = e->datalen - sizeof(struct brcmf_rx_mgmt_data);
990 struct brcmf_rx_mgmt_data *rxframe = (struct brcmf_rx_mgmt_data *)data;
991 u16 chanspec = be16_to_cpu(rxframe->chanspec);
992 struct ieee80211_mgmt *mgmt_frame;
993 s32 err;
994 s32 freq;
995 u16 mgmt_type;
996
997 /* Check if wpa_supplicant has registered for this frame */
998 brcmf_dbg(INFO, "ifp->vif->mgmt_rx_reg %04x\n", ifp->vif->mgmt_rx_reg);
999 mgmt_type = (IEEE80211_STYPE_ACTION & IEEE80211_FCTL_STYPE) >> 4;
1000 if ((ifp->vif->mgmt_rx_reg & BIT(mgmt_type)) == 0)
1001 return 0;
1002
1003 brcmf_p2p_print_actframe(false, (u8 *)(rxframe + 1), mgmt_frame_len);
1004
1005 mgmt_frame = kzalloc(offsetof(struct ieee80211_mgmt, u) +
1006 mgmt_frame_len, GFP_KERNEL);
1007 if (!mgmt_frame) {
1008 brcmf_err("No memory available for action frame\n");
1009 return -ENOMEM;
1010 }
1011 memcpy(mgmt_frame->da, ifp->mac_addr, ETH_ALEN);
1012 err = brcmf_fil_cmd_data_get(ifp, BRCMF_C_GET_BSSID, mgmt_frame->bssid,
1013 ETH_ALEN);
1014 if (err < 0)
1015 brcmf_err("BRCMF_C_GET_BSSID error %d\n", err);
1016 memcpy(mgmt_frame->sa, e->addr, ETH_ALEN);
1017 mgmt_frame->frame_control = cpu_to_le16(IEEE80211_STYPE_ACTION);
1018 memcpy(&mgmt_frame->u, (u8 *)(rxframe + 1), mgmt_frame_len);
1019 mgmt_frame_len += offsetof(struct ieee80211_mgmt, u);
1020
1021 freq = ieee80211_channel_to_frequency(CHSPEC_CHANNEL(chanspec),
1022 CHSPEC_IS2G(chanspec) ?
1023 IEEE80211_BAND_2GHZ :
1024 IEEE80211_BAND_5GHZ);
1025 wdev = ifp->ndev->ieee80211_ptr;
1026 cfg80211_rx_mgmt(wdev, freq, 0, (u8 *)mgmt_frame, mgmt_frame_len,
1027 GFP_ATOMIC);
1028
1029 kfree(mgmt_frame);
1030 return 0;
1031}
1032
1033
1034/**
Arend van Spriel9f440b72013-02-08 15:53:36 +01001035 * brcmf_p2p_attach() - attach for P2P.
1036 *
1037 * @cfg: driver private data for cfg80211 interface.
1038 */
Hante Meulemana0f07952013-02-08 15:53:47 +01001039void brcmf_p2p_attach(struct brcmf_cfg80211_info *cfg,
1040 struct brcmf_cfg80211_vif *vif)
Arend van Spriel9f440b72013-02-08 15:53:36 +01001041{
1042 struct brcmf_p2p_info *p2p;
1043
1044 p2p = &cfg->p2p;
1045
1046 p2p->cfg = cfg;
Hante Meulemana0f07952013-02-08 15:53:47 +01001047 p2p->bss_idx[P2PAPI_BSSCFG_PRIMARY].vif = vif;
Arend van Spriel9f440b72013-02-08 15:53:36 +01001048 brcmf_p2p_generate_bss_mac(p2p);
Hante Meulemana0f07952013-02-08 15:53:47 +01001049 brcmf_p2p_set_firmware(p2p);
Arend van Spriel9f440b72013-02-08 15:53:36 +01001050}
1051
1052/**
1053 * brcmf_p2p_detach() - detach P2P.
1054 *
1055 * @p2p: P2P specific data.
1056 */
1057void brcmf_p2p_detach(struct brcmf_p2p_info *p2p)
1058{
Hante Meuleman0de8aac2013-02-08 15:53:38 +01001059 if (p2p->bss_idx[P2PAPI_BSSCFG_DEVICE].vif != NULL) {
1060 brcmf_p2p_cancel_remain_on_channel(
1061 p2p->bss_idx[P2PAPI_BSSCFG_DEVICE].vif->ifp);
1062 brcmf_p2p_deinit_discovery(p2p);
1063 }
Arend van Spriel9f440b72013-02-08 15:53:36 +01001064 /* just set it all to zero */
1065 memset(p2p, 0, sizeof(*p2p));
1066}
1067
Hante Meuleman7a5c1f62013-02-08 15:53:44 +01001068/**
1069 * brcmf_p2p_get_current_chanspec() - Get current operation channel.
1070 *
1071 * @p2p: P2P specific data.
1072 * @chanspec: chanspec to be returned.
1073 */
1074static void brcmf_p2p_get_current_chanspec(struct brcmf_p2p_info *p2p,
1075 u16 *chanspec)
1076{
1077 struct brcmf_if *ifp;
1078 struct brcmf_fil_chan_info_le ci;
1079 s32 err;
1080
1081 ifp = p2p->bss_idx[P2PAPI_BSSCFG_PRIMARY].vif->ifp;
1082
1083 *chanspec = 11 & WL_CHANSPEC_CHAN_MASK;
1084
1085 err = brcmf_fil_cmd_data_get(ifp, BRCMF_C_GET_CHANNEL, &ci, sizeof(ci));
1086 if (!err) {
1087 *chanspec = le32_to_cpu(ci.hw_channel) & WL_CHANSPEC_CHAN_MASK;
1088 if (*chanspec < CH_MAX_2G_CHANNEL)
1089 *chanspec |= WL_CHANSPEC_BAND_2G;
1090 else
1091 *chanspec |= WL_CHANSPEC_BAND_5G;
1092 }
1093 *chanspec |= WL_CHANSPEC_BW_20 | WL_CHANSPEC_CTL_SB_NONE;
1094}
1095
1096/**
1097 * Change a P2P Role.
1098 * Parameters:
1099 * @mac: MAC address of the BSS to change a role
1100 * Returns 0 if success.
1101 */
1102int brcmf_p2p_ifchange(struct brcmf_cfg80211_info *cfg,
1103 enum brcmf_fil_p2p_if_types if_type)
1104{
1105 struct brcmf_p2p_info *p2p = &cfg->p2p;
1106 struct brcmf_cfg80211_vif *vif;
1107 struct brcmf_fil_p2p_if_le if_request;
1108 s32 err;
1109 u16 chanspec;
1110
1111 brcmf_dbg(TRACE, "Enter\n");
1112
1113 vif = p2p->bss_idx[P2PAPI_BSSCFG_PRIMARY].vif;
1114 if (!vif) {
1115 brcmf_err("vif for P2PAPI_BSSCFG_PRIMARY does not exist\n");
1116 return -EPERM;
1117 }
1118 brcmf_notify_escan_complete(cfg, vif->ifp->ndev, true, true);
1119 vif = p2p->bss_idx[P2PAPI_BSSCFG_CONNECTION].vif;
1120 if (!vif) {
1121 brcmf_err("vif for P2PAPI_BSSCFG_CONNECTION does not exist\n");
1122 return -EPERM;
1123 }
1124 brcmf_set_mpc(vif->ifp->ndev, 0);
1125
1126 /* In concurrency case, STA may be already associated in a particular */
1127 /* channel. so retrieve the current channel of primary interface and */
1128 /* then start the virtual interface on that. */
1129 brcmf_p2p_get_current_chanspec(p2p, &chanspec);
1130
1131 if_request.type = cpu_to_le16((u16)if_type);
1132 if_request.chspec = cpu_to_le16(chanspec);
1133 memcpy(if_request.addr, p2p->int_addr, sizeof(if_request.addr));
1134
1135 brcmf_cfg80211_arm_vif_event(cfg, vif);
1136 err = brcmf_fil_iovar_data_set(vif->ifp, "p2p_ifupd", &if_request,
1137 sizeof(if_request));
1138 if (err) {
1139 brcmf_err("p2p_ifupd FAILED, err=%d\n", err);
1140 brcmf_cfg80211_arm_vif_event(cfg, NULL);
1141 return err;
1142 }
1143 err = brcmf_cfg80211_wait_vif_event_timeout(cfg, BRCMF_E_IF_CHANGE,
1144 msecs_to_jiffies(1500));
1145 brcmf_cfg80211_arm_vif_event(cfg, NULL);
1146 if (!err) {
1147 brcmf_err("No BRCMF_E_IF_CHANGE event received\n");
1148 return -EIO;
1149 }
1150
1151 err = brcmf_fil_cmd_int_set(vif->ifp, BRCMF_C_SET_SCB_TIMEOUT,
1152 BRCMF_SCB_TIMEOUT_VALUE);
1153
1154 return err;
1155}
1156
1157static int brcmf_p2p_request_p2p_if(struct brcmf_p2p_info *p2p,
1158 struct brcmf_if *ifp, u8 ea[ETH_ALEN],
Arend van Sprield3c0b632013-02-08 15:53:37 +01001159 enum brcmf_fil_p2p_if_types iftype)
1160{
1161 struct brcmf_fil_p2p_if_le if_request;
Arend van Sprield3c0b632013-02-08 15:53:37 +01001162 int err;
Hante Meuleman7a5c1f62013-02-08 15:53:44 +01001163 u16 chanspec;
Arend van Sprield3c0b632013-02-08 15:53:37 +01001164
1165 /* we need a default channel */
Hante Meuleman7a5c1f62013-02-08 15:53:44 +01001166 brcmf_p2p_get_current_chanspec(p2p, &chanspec);
Arend van Sprield3c0b632013-02-08 15:53:37 +01001167
1168 /* fill the firmware request */
1169 memcpy(if_request.addr, ea, ETH_ALEN);
Hante Meuleman7ee2d922013-02-08 15:53:43 +01001170 if_request.type = cpu_to_le16((u16)iftype);
Arend van Sprield3c0b632013-02-08 15:53:37 +01001171 if_request.chspec = cpu_to_le16(chanspec);
1172
1173 err = brcmf_fil_iovar_data_set(ifp, "p2p_ifadd", &if_request,
1174 sizeof(if_request));
1175 if (err)
1176 return err;
1177
Arend van Sprield3c0b632013-02-08 15:53:37 +01001178 return err;
1179}
1180
Arend van Spriel5f4f9f12013-02-08 15:53:41 +01001181static int brcmf_p2p_disable_p2p_if(struct brcmf_cfg80211_vif *vif)
1182{
1183 struct brcmf_cfg80211_info *cfg = wdev_to_cfg(&vif->wdev);
1184 struct net_device *pri_ndev = cfg_to_ndev(cfg);
1185 struct brcmf_if *ifp = netdev_priv(pri_ndev);
1186 u8 *addr = vif->wdev.netdev->dev_addr;
1187
1188 return brcmf_fil_iovar_data_set(ifp, "p2p_ifdis", addr, ETH_ALEN);
1189}
1190
1191static int brcmf_p2p_release_p2p_if(struct brcmf_cfg80211_vif *vif)
1192{
1193 struct brcmf_cfg80211_info *cfg = wdev_to_cfg(&vif->wdev);
1194 struct net_device *pri_ndev = cfg_to_ndev(cfg);
1195 struct brcmf_if *ifp = netdev_priv(pri_ndev);
1196 u8 *addr = vif->wdev.netdev->dev_addr;
1197
1198 return brcmf_fil_iovar_data_set(ifp, "p2p_ifdel", addr, ETH_ALEN);
1199}
1200
Arend van Spriel9f440b72013-02-08 15:53:36 +01001201/**
1202 * brcmf_p2p_add_vif() - create a new P2P virtual interface.
1203 *
1204 * @wiphy: wiphy device of new interface.
1205 * @name: name of the new interface.
1206 * @type: nl80211 interface type.
1207 * @flags: TBD
1208 * @params: TBD
Arend van Spriel9f440b72013-02-08 15:53:36 +01001209 */
1210struct wireless_dev *brcmf_p2p_add_vif(struct wiphy *wiphy, const char *name,
1211 enum nl80211_iftype type, u32 *flags,
1212 struct vif_params *params)
1213{
Arend van Sprield3c0b632013-02-08 15:53:37 +01001214 struct brcmf_cfg80211_info *cfg = wiphy_to_cfg(wiphy);
1215 struct brcmf_if *ifp = netdev_priv(cfg_to_ndev(cfg));
1216 struct brcmf_cfg80211_vif *vif;
1217 enum brcmf_fil_p2p_if_types iftype;
1218 enum wl_mode mode;
1219 int err;
1220
1221 if (brcmf_cfg80211_vif_event_armed(cfg))
1222 return ERR_PTR(-EBUSY);
1223
Arend van Spriel9f440b72013-02-08 15:53:36 +01001224 brcmf_dbg(INFO, "adding vif \"%s\" (type=%d)\n", name, type);
Arend van Sprield3c0b632013-02-08 15:53:37 +01001225
1226 switch (type) {
1227 case NL80211_IFTYPE_P2P_CLIENT:
1228 iftype = BRCMF_FIL_P2P_IF_CLIENT;
1229 mode = WL_MODE_BSS;
1230 break;
1231 case NL80211_IFTYPE_P2P_GO:
1232 iftype = BRCMF_FIL_P2P_IF_GO;
1233 mode = WL_MODE_AP;
1234 break;
1235 default:
1236 return ERR_PTR(-EOPNOTSUPP);
1237 }
1238
1239 vif = brcmf_alloc_vif(cfg, type, false);
Hante Meuleman7ee2d922013-02-08 15:53:43 +01001240 if (IS_ERR(vif))
1241 return (struct wireless_dev *)vif;
Arend van Sprield3c0b632013-02-08 15:53:37 +01001242 brcmf_cfg80211_arm_vif_event(cfg, vif);
1243
Hante Meuleman7a5c1f62013-02-08 15:53:44 +01001244 err = brcmf_p2p_request_p2p_if(&cfg->p2p, ifp, cfg->p2p.int_addr,
1245 iftype);
Hante Meuleman7ee2d922013-02-08 15:53:43 +01001246 if (err) {
1247 brcmf_cfg80211_arm_vif_event(cfg, NULL);
Arend van Sprield3c0b632013-02-08 15:53:37 +01001248 goto fail;
Hante Meuleman7ee2d922013-02-08 15:53:43 +01001249 }
Arend van Sprield3c0b632013-02-08 15:53:37 +01001250
1251 /* wait for firmware event */
1252 err = brcmf_cfg80211_wait_vif_event_timeout(cfg, BRCMF_E_IF_ADD,
1253 msecs_to_jiffies(1500));
1254 brcmf_cfg80211_arm_vif_event(cfg, NULL);
1255 if (!err) {
1256 brcmf_err("timeout occurred\n");
1257 err = -EIO;
1258 goto fail;
1259 }
1260
1261 /* interface created in firmware */
1262 ifp = vif->ifp;
1263 if (!ifp) {
1264 brcmf_err("no if pointer provided\n");
1265 err = -ENOENT;
Hante Meuleman7ee2d922013-02-08 15:53:43 +01001266 goto fail;
Arend van Sprield3c0b632013-02-08 15:53:37 +01001267 }
1268
1269 strncpy(ifp->ndev->name, name, sizeof(ifp->ndev->name) - 1);
1270 brcmf_cfg80211_vif_complete(cfg);
Hante Meuleman7ee2d922013-02-08 15:53:43 +01001271 cfg->p2p.bss_idx[P2PAPI_BSSCFG_CONNECTION].vif = vif;
1272 /* Disable firmware roaming for P2P interface */
1273 brcmf_fil_iovar_int_set(ifp, "roam_off", 1);
1274 if (iftype == BRCMF_FIL_P2P_IF_GO) {
1275 /* set station timeout for p2p */
1276 brcmf_fil_cmd_int_set(ifp, BRCMF_C_SET_SCB_TIMEOUT,
1277 BRCMF_SCB_TIMEOUT_VALUE);
1278 }
Arend van Sprield3c0b632013-02-08 15:53:37 +01001279 return &ifp->vif->wdev;
1280
1281fail:
1282 brcmf_free_vif(vif);
1283 return ERR_PTR(err);
Arend van Spriel9f440b72013-02-08 15:53:36 +01001284}
1285
1286/**
1287 * brcmf_p2p_del_vif() - delete a P2P virtual interface.
1288 *
1289 * @wiphy: wiphy device of interface.
1290 * @wdev: wireless device of interface.
1291 *
1292 * TODO: not yet supported.
1293 */
1294int brcmf_p2p_del_vif(struct wiphy *wiphy, struct wireless_dev *wdev)
1295{
Arend van Sprield3c0b632013-02-08 15:53:37 +01001296 struct brcmf_cfg80211_info *cfg = wiphy_priv(wiphy);
Arend van Spriel9f440b72013-02-08 15:53:36 +01001297 struct brcmf_cfg80211_vif *vif;
Arend van Spriel5f4f9f12013-02-08 15:53:41 +01001298 unsigned long jiffie_timeout = msecs_to_jiffies(1500);
1299 bool wait_for_disable = false;
Arend van Sprield3c0b632013-02-08 15:53:37 +01001300 int err;
Arend van Spriel9f440b72013-02-08 15:53:36 +01001301
Arend van Spriel5f4f9f12013-02-08 15:53:41 +01001302 brcmf_dbg(TRACE, "delete P2P vif\n");
Arend van Spriel9f440b72013-02-08 15:53:36 +01001303 vif = container_of(wdev, struct brcmf_cfg80211_vif, wdev);
Arend van Sprield3c0b632013-02-08 15:53:37 +01001304
Arend van Spriel5f4f9f12013-02-08 15:53:41 +01001305 switch (vif->wdev.iftype) {
1306 case NL80211_IFTYPE_P2P_CLIENT:
1307 if (test_bit(BRCMF_VIF_STATUS_DISCONNECTING, &vif->sme_state))
1308 wait_for_disable = true;
1309 break;
1310
1311 case NL80211_IFTYPE_P2P_GO:
1312 if (!brcmf_p2p_disable_p2p_if(vif))
1313 wait_for_disable = true;
1314 break;
1315
1316 case NL80211_IFTYPE_P2P_DEVICE:
1317 default:
1318 return -ENOTSUPP;
1319 break;
1320 }
1321
1322 if (wait_for_disable)
Hante Meuleman7ee2d922013-02-08 15:53:43 +01001323 wait_for_completion_timeout(&cfg->vif_disabled,
1324 msecs_to_jiffies(500));
Arend van Spriel5f4f9f12013-02-08 15:53:41 +01001325
1326 brcmf_vif_clear_mgmt_ies(vif);
Arend van Sprield3c0b632013-02-08 15:53:37 +01001327
1328 brcmf_cfg80211_arm_vif_event(cfg, vif);
Arend van Spriel5f4f9f12013-02-08 15:53:41 +01001329 err = brcmf_p2p_release_p2p_if(vif);
Hante Meuleman7ee2d922013-02-08 15:53:43 +01001330 if (!err) {
Arend van Spriel5f4f9f12013-02-08 15:53:41 +01001331 /* wait for firmware event */
1332 err = brcmf_cfg80211_wait_vif_event_timeout(cfg, BRCMF_E_IF_DEL,
1333 jiffie_timeout);
Hante Meuleman7ee2d922013-02-08 15:53:43 +01001334 if (!err)
1335 err = -EIO;
1336 else
1337 err = 0;
1338 }
Arend van Sprield3c0b632013-02-08 15:53:37 +01001339 brcmf_cfg80211_arm_vif_event(cfg, NULL);
Arend van Spriel5f4f9f12013-02-08 15:53:41 +01001340 brcmf_free_vif(vif);
Hante Meuleman7ee2d922013-02-08 15:53:43 +01001341 cfg->p2p.bss_idx[P2PAPI_BSSCFG_CONNECTION].vif = NULL;
Arend van Spriel5f4f9f12013-02-08 15:53:41 +01001342
1343 return err;
Arend van Spriel9f440b72013-02-08 15:53:36 +01001344}