driver_broadcom.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600
  1. /*
  2. * WPA Supplicant - driver interaction with Broadcom wl.o driver
  3. * Copyright (c) 2004, Nikki Chumkov <nikki@gattaca.ru>
  4. * Copyright (c) 2004, Jouni Malinen <j@w1.fi>
  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. #include "includes.h"
  16. #include <sys/ioctl.h>
  17. #include "common.h"
  18. #if 0
  19. #include <netpacket/packet.h>
  20. #include <net/ethernet.h> /* the L2 protocols */
  21. #else
  22. #include <linux/if_packet.h>
  23. #include <linux/if_ether.h> /* The L2 protocols */
  24. #endif
  25. #include <net/if.h>
  26. #include <typedefs.h>
  27. /* wlioctl.h is a Broadcom header file and it is available, e.g., from Linksys
  28. * WRT54G GPL tarball. */
  29. #include <wlioctl.h>
  30. #include "driver.h"
  31. #include "eloop.h"
  32. struct wpa_driver_broadcom_data {
  33. void *ctx;
  34. int ioctl_sock;
  35. int event_sock;
  36. char ifname[IFNAMSIZ + 1];
  37. };
  38. #ifndef WLC_DEAUTHENTICATE
  39. #define WLC_DEAUTHENTICATE 143
  40. #endif
  41. #ifndef WLC_DEAUTHENTICATE_WITH_REASON
  42. #define WLC_DEAUTHENTICATE_WITH_REASON 201
  43. #endif
  44. #ifndef WLC_SET_TKIP_COUNTERMEASURES
  45. #define WLC_SET_TKIP_COUNTERMEASURES 202
  46. #endif
  47. #if !defined(PSK_ENABLED) /* NEW driver interface */
  48. #define WL_VERSION 360130
  49. /* wireless authentication bit vector */
  50. #define WPA_ENABLED 1
  51. #define PSK_ENABLED 2
  52. #define WAUTH_WPA_ENABLED(wauth) ((wauth) & WPA_ENABLED)
  53. #define WAUTH_PSK_ENABLED(wauth) ((wauth) & PSK_ENABLED)
  54. #define WAUTH_ENABLED(wauth) ((wauth) & (WPA_ENABLED | PSK_ENABLED))
  55. #define WSEC_PRIMARY_KEY WL_PRIMARY_KEY
  56. typedef wl_wsec_key_t wsec_key_t;
  57. #endif
  58. typedef struct {
  59. uint32 val;
  60. struct ether_addr ea;
  61. uint16 res;
  62. } wlc_deauth_t;
  63. static void wpa_driver_broadcom_scan_timeout(void *eloop_ctx,
  64. void *timeout_ctx);
  65. static int broadcom_ioctl(struct wpa_driver_broadcom_data *drv, int cmd,
  66. void *buf, int len)
  67. {
  68. struct ifreq ifr;
  69. wl_ioctl_t ioc;
  70. int ret = 0;
  71. wpa_printf(MSG_MSGDUMP, "BROADCOM: wlioctl(%s,%d,len=%d,val=%p)",
  72. drv->ifname, cmd, len, buf);
  73. /* wpa_hexdump(MSG_MSGDUMP, "BROADCOM: wlioctl buf", buf, len); */
  74. ioc.cmd = cmd;
  75. ioc.buf = buf;
  76. ioc.len = len;
  77. os_strlcpy(ifr.ifr_name, drv->ifname, IFNAMSIZ);
  78. ifr.ifr_data = (caddr_t) &ioc;
  79. if ((ret = ioctl(drv->ioctl_sock, SIOCDEVPRIVATE, &ifr)) < 0) {
  80. if (cmd != WLC_GET_MAGIC)
  81. perror(ifr.ifr_name);
  82. wpa_printf(MSG_MSGDUMP, "BROADCOM: wlioctl cmd=%d res=%d",
  83. cmd, ret);
  84. }
  85. return ret;
  86. }
  87. static int wpa_driver_broadcom_get_bssid(void *priv, u8 *bssid)
  88. {
  89. struct wpa_driver_broadcom_data *drv = priv;
  90. if (broadcom_ioctl(drv, WLC_GET_BSSID, bssid, ETH_ALEN) == 0)
  91. return 0;
  92. os_memset(bssid, 0, ETH_ALEN);
  93. return -1;
  94. }
  95. static int wpa_driver_broadcom_get_ssid(void *priv, u8 *ssid)
  96. {
  97. struct wpa_driver_broadcom_data *drv = priv;
  98. wlc_ssid_t s;
  99. if (broadcom_ioctl(drv, WLC_GET_SSID, &s, sizeof(s)) == -1)
  100. return -1;
  101. os_memcpy(ssid, s.SSID, s.SSID_len);
  102. return s.SSID_len;
  103. }
  104. static int wpa_driver_broadcom_set_wpa(void *priv, int enable)
  105. {
  106. struct wpa_driver_broadcom_data *drv = priv;
  107. unsigned int wauth, wsec;
  108. struct ether_addr ea;
  109. os_memset(&ea, enable ? 0xff : 0, sizeof(ea));
  110. if (broadcom_ioctl(drv, WLC_GET_WPA_AUTH, &wauth, sizeof(wauth)) ==
  111. -1 ||
  112. broadcom_ioctl(drv, WLC_GET_WSEC, &wsec, sizeof(wsec)) == -1)
  113. return -1;
  114. if (enable) {
  115. wauth = PSK_ENABLED;
  116. wsec = TKIP_ENABLED;
  117. } else {
  118. wauth = 255;
  119. wsec &= ~(TKIP_ENABLED | AES_ENABLED);
  120. }
  121. if (broadcom_ioctl(drv, WLC_SET_WPA_AUTH, &wauth, sizeof(wauth)) ==
  122. -1 ||
  123. broadcom_ioctl(drv, WLC_SET_WSEC, &wsec, sizeof(wsec)) == -1)
  124. return -1;
  125. /* FIX: magic number / error handling? */
  126. broadcom_ioctl(drv, 122, &ea, sizeof(ea));
  127. return 0;
  128. }
  129. static int wpa_driver_broadcom_set_key(void *priv, wpa_alg alg,
  130. const u8 *addr, int key_idx, int set_tx,
  131. const u8 *seq, size_t seq_len,
  132. const u8 *key, size_t key_len)
  133. {
  134. struct wpa_driver_broadcom_data *drv = priv;
  135. int ret;
  136. wsec_key_t wkt;
  137. os_memset(&wkt, 0, sizeof wkt);
  138. wpa_printf(MSG_MSGDUMP, "BROADCOM: SET %sKEY[%d] alg=%d",
  139. set_tx ? "PRIMARY " : "", key_idx, alg);
  140. if (key && key_len > 0)
  141. wpa_hexdump_key(MSG_MSGDUMP, "BROADCOM: key", key, key_len);
  142. switch (alg) {
  143. case WPA_ALG_NONE:
  144. wkt.algo = CRYPTO_ALGO_OFF;
  145. break;
  146. case WPA_ALG_WEP:
  147. wkt.algo = CRYPTO_ALGO_WEP128; /* CRYPTO_ALGO_WEP1? */
  148. break;
  149. case WPA_ALG_TKIP:
  150. wkt.algo = 0; /* CRYPTO_ALGO_TKIP? */
  151. break;
  152. case WPA_ALG_CCMP:
  153. wkt.algo = 0; /* CRYPTO_ALGO_AES_CCM;
  154. * AES_OCB_MSDU, AES_OCB_MPDU? */
  155. break;
  156. default:
  157. wkt.algo = CRYPTO_ALGO_NALG;
  158. break;
  159. }
  160. if (seq && seq_len > 0)
  161. wpa_hexdump(MSG_MSGDUMP, "BROADCOM: SEQ", seq, seq_len);
  162. if (addr)
  163. wpa_hexdump(MSG_MSGDUMP, "BROADCOM: addr", addr, ETH_ALEN);
  164. wkt.index = key_idx;
  165. wkt.len = key_len;
  166. if (key && key_len > 0) {
  167. os_memcpy(wkt.data, key, key_len);
  168. if (key_len == 32) {
  169. /* hack hack hack XXX */
  170. os_memcpy(&wkt.data[16], &key[24], 8);
  171. os_memcpy(&wkt.data[24], &key[16], 8);
  172. }
  173. }
  174. /* wkt.algo = CRYPTO_ALGO_...; */
  175. wkt.flags = set_tx ? 0 : WSEC_PRIMARY_KEY;
  176. if (addr && set_tx)
  177. os_memcpy(&wkt.ea, addr, sizeof(wkt.ea));
  178. ret = broadcom_ioctl(drv, WLC_SET_KEY, &wkt, sizeof(wkt));
  179. if (addr && set_tx) {
  180. /* FIX: magic number / error handling? */
  181. broadcom_ioctl(drv, 121, &wkt.ea, sizeof(wkt.ea));
  182. }
  183. return ret;
  184. }
  185. static void wpa_driver_broadcom_event_receive(int sock, void *ctx,
  186. void *sock_ctx)
  187. {
  188. char buf[8192];
  189. int left;
  190. wl_wpa_header_t *wwh;
  191. union wpa_event_data data;
  192. if ((left = recv(sock, buf, sizeof buf, 0)) < 0)
  193. return;
  194. wpa_hexdump(MSG_DEBUG, "RECEIVE EVENT", (u8 *) buf, left);
  195. if ((size_t) left < sizeof(wl_wpa_header_t))
  196. return;
  197. wwh = (wl_wpa_header_t *) buf;
  198. if (wwh->snap.type != WL_WPA_ETHER_TYPE)
  199. return;
  200. if (os_memcmp(&wwh->snap, wl_wpa_snap_template, 6) != 0)
  201. return;
  202. os_memset(&data, 0, sizeof(data));
  203. switch (wwh->type) {
  204. case WLC_ASSOC_MSG:
  205. left -= WL_WPA_HEADER_LEN;
  206. wpa_printf(MSG_DEBUG, "BROADCOM: ASSOC MESSAGE (left: %d)",
  207. left);
  208. if (left > 0) {
  209. data.assoc_info.resp_ies = os_malloc(left);
  210. if (data.assoc_info.resp_ies == NULL)
  211. return;
  212. os_memcpy(data.assoc_info.resp_ies,
  213. buf + WL_WPA_HEADER_LEN, left);
  214. data.assoc_info.resp_ies_len = left;
  215. wpa_hexdump(MSG_MSGDUMP, "BROADCOM: copying %d bytes "
  216. "into resp_ies",
  217. data.assoc_info.resp_ies, left);
  218. }
  219. /* data.assoc_info.req_ies = NULL; */
  220. /* data.assoc_info.req_ies_len = 0; */
  221. wpa_supplicant_event(ctx, EVENT_ASSOCINFO, &data);
  222. wpa_supplicant_event(ctx, EVENT_ASSOC, NULL);
  223. break;
  224. case WLC_DISASSOC_MSG:
  225. wpa_printf(MSG_DEBUG, "BROADCOM: DISASSOC MESSAGE");
  226. wpa_supplicant_event(ctx, EVENT_DISASSOC, NULL);
  227. break;
  228. case WLC_PTK_MIC_MSG:
  229. wpa_printf(MSG_DEBUG, "BROADCOM: PTK MIC MSG MESSAGE");
  230. data.michael_mic_failure.unicast = 1;
  231. wpa_supplicant_event(ctx, EVENT_MICHAEL_MIC_FAILURE, &data);
  232. break;
  233. case WLC_GTK_MIC_MSG:
  234. wpa_printf(MSG_DEBUG, "BROADCOM: GTK MIC MSG MESSAGE");
  235. data.michael_mic_failure.unicast = 0;
  236. wpa_supplicant_event(ctx, EVENT_MICHAEL_MIC_FAILURE, &data);
  237. break;
  238. default:
  239. wpa_printf(MSG_DEBUG, "BROADCOM: UNKNOWN MESSAGE (%d)",
  240. wwh->type);
  241. break;
  242. }
  243. os_free(data.assoc_info.resp_ies);
  244. }
  245. static void * wpa_driver_broadcom_init(void *ctx, const char *ifname)
  246. {
  247. int s;
  248. struct sockaddr_ll ll;
  249. struct wpa_driver_broadcom_data *drv;
  250. struct ifreq ifr;
  251. /* open socket to kernel */
  252. if ((s = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
  253. perror("socket");
  254. return NULL;
  255. }
  256. /* do it */
  257. os_strlcpy(ifr.ifr_name, ifname, IFNAMSIZ);
  258. if (ioctl(s, SIOCGIFINDEX, &ifr) < 0) {
  259. perror(ifr.ifr_name);
  260. return NULL;
  261. }
  262. drv = os_zalloc(sizeof(*drv));
  263. if (drv == NULL)
  264. return NULL;
  265. drv->ctx = ctx;
  266. os_strlcpy(drv->ifname, ifname, sizeof(drv->ifname));
  267. drv->ioctl_sock = s;
  268. s = socket(PF_PACKET, SOCK_RAW, ntohs(ETH_P_802_2));
  269. if (s < 0) {
  270. perror("socket(PF_PACKET, SOCK_RAW, ntohs(ETH_P_802_2))");
  271. close(drv->ioctl_sock);
  272. os_free(drv);
  273. return NULL;
  274. }
  275. os_memset(&ll, 0, sizeof(ll));
  276. ll.sll_family = AF_PACKET;
  277. ll.sll_protocol = ntohs(ETH_P_802_2);
  278. ll.sll_ifindex = ifr.ifr_ifindex;
  279. ll.sll_hatype = 0;
  280. ll.sll_pkttype = PACKET_HOST;
  281. ll.sll_halen = 0;
  282. if (bind(s, (struct sockaddr *) &ll, sizeof(ll)) < 0) {
  283. perror("bind(netlink)");
  284. close(s);
  285. close(drv->ioctl_sock);
  286. os_free(drv);
  287. return NULL;
  288. }
  289. eloop_register_read_sock(s, wpa_driver_broadcom_event_receive, ctx,
  290. NULL);
  291. drv->event_sock = s;
  292. return drv;
  293. }
  294. static void wpa_driver_broadcom_deinit(void *priv)
  295. {
  296. struct wpa_driver_broadcom_data *drv = priv;
  297. eloop_cancel_timeout(wpa_driver_broadcom_scan_timeout, drv, drv->ctx);
  298. eloop_unregister_read_sock(drv->event_sock);
  299. close(drv->event_sock);
  300. close(drv->ioctl_sock);
  301. os_free(drv);
  302. }
  303. static int wpa_driver_broadcom_set_countermeasures(void *priv,
  304. int enabled)
  305. {
  306. #if 0
  307. struct wpa_driver_broadcom_data *drv = priv;
  308. /* FIX: ? */
  309. return broadcom_ioctl(drv, WLC_SET_TKIP_COUNTERMEASURES, &enabled,
  310. sizeof(enabled));
  311. #else
  312. return 0;
  313. #endif
  314. }
  315. static int wpa_driver_broadcom_set_drop_unencrypted(void *priv, int enabled)
  316. {
  317. struct wpa_driver_broadcom_data *drv = priv;
  318. /* SET_EAP_RESTRICT, SET_WEP_RESTRICT */
  319. int _restrict = (enabled ? 1 : 0);
  320. if (broadcom_ioctl(drv, WLC_SET_WEP_RESTRICT,
  321. &_restrict, sizeof(_restrict)) < 0 ||
  322. broadcom_ioctl(drv, WLC_SET_EAP_RESTRICT,
  323. &_restrict, sizeof(_restrict)) < 0)
  324. return -1;
  325. return 0;
  326. }
  327. static void wpa_driver_broadcom_scan_timeout(void *eloop_ctx,
  328. void *timeout_ctx)
  329. {
  330. wpa_printf(MSG_DEBUG, "Scan timeout - try to get results");
  331. wpa_supplicant_event(timeout_ctx, EVENT_SCAN_RESULTS, NULL);
  332. }
  333. static int wpa_driver_broadcom_scan(void *priv, const u8 *ssid,
  334. size_t ssid_len)
  335. {
  336. struct wpa_driver_broadcom_data *drv = priv;
  337. wlc_ssid_t wst = { 0, "" };
  338. if (ssid && ssid_len > 0 && ssid_len <= sizeof(wst.SSID)) {
  339. wst.SSID_len = ssid_len;
  340. os_memcpy(wst.SSID, ssid, ssid_len);
  341. }
  342. if (broadcom_ioctl(drv, WLC_SCAN, &wst, sizeof(wst)) < 0)
  343. return -1;
  344. eloop_cancel_timeout(wpa_driver_broadcom_scan_timeout, drv, drv->ctx);
  345. eloop_register_timeout(3, 0, wpa_driver_broadcom_scan_timeout, drv,
  346. drv->ctx);
  347. return 0;
  348. }
  349. static const int frequency_list[] = {
  350. 2412, 2417, 2422, 2427, 2432, 2437, 2442,
  351. 2447, 2452, 2457, 2462, 2467, 2472, 2484
  352. };
  353. struct bss_ie_hdr {
  354. u8 elem_id;
  355. u8 len;
  356. u8 oui[3];
  357. /* u8 oui_type; */
  358. /* u16 version; */
  359. } __attribute__ ((packed));
  360. static int
  361. wpa_driver_broadcom_get_scan_results(void *priv,
  362. struct wpa_scan_result *results,
  363. size_t max_size)
  364. {
  365. struct wpa_driver_broadcom_data *drv = priv;
  366. char *buf;
  367. wl_scan_results_t *wsr;
  368. wl_bss_info_t *wbi;
  369. size_t ap_num;
  370. buf = os_malloc(WLC_IOCTL_MAXLEN);
  371. if (buf == NULL)
  372. return -1;
  373. wsr = (wl_scan_results_t *) buf;
  374. wsr->buflen = WLC_IOCTL_MAXLEN - sizeof(wsr);
  375. wsr->version = 107;
  376. wsr->count = 0;
  377. if (broadcom_ioctl(drv, WLC_SCAN_RESULTS, buf, WLC_IOCTL_MAXLEN) < 0) {
  378. os_free(buf);
  379. return -1;
  380. }
  381. os_memset(results, 0, max_size * sizeof(struct wpa_scan_result));
  382. for (ap_num = 0, wbi = wsr->bss_info; ap_num < wsr->count; ++ap_num) {
  383. int left;
  384. struct bss_ie_hdr *ie;
  385. os_memcpy(results[ap_num].bssid, &wbi->BSSID, ETH_ALEN);
  386. os_memcpy(results[ap_num].ssid, wbi->SSID, wbi->SSID_len);
  387. results[ap_num].ssid_len = wbi->SSID_len;
  388. results[ap_num].freq = frequency_list[wbi->channel - 1];
  389. /* get ie's */
  390. wpa_hexdump(MSG_MSGDUMP, "BROADCOM: AP IEs",
  391. (u8 *) wbi + sizeof(*wbi), wbi->ie_length);
  392. ie = (struct bss_ie_hdr *) ((u8 *) wbi + sizeof(*wbi));
  393. for (left = wbi->ie_length; left > 0;
  394. left -= (ie->len + 2), ie = (struct bss_ie_hdr *)
  395. ((u8 *) ie + 2 + ie->len)) {
  396. wpa_printf(MSG_MSGDUMP, "BROADCOM: IE: id:%x, len:%d",
  397. ie->elem_id, ie->len);
  398. if (ie->len >= 3)
  399. wpa_printf(MSG_MSGDUMP,
  400. "BROADCOM: oui:%02x%02x%02x",
  401. ie->oui[0], ie->oui[1], ie->oui[2]);
  402. if (ie->elem_id != 0xdd ||
  403. ie->len < 6 ||
  404. os_memcmp(ie->oui, WPA_OUI, 3) != 0)
  405. continue;
  406. os_memcpy(results[ap_num].wpa_ie, ie, ie->len + 2);
  407. results[ap_num].wpa_ie_len = ie->len + 2;
  408. break;
  409. }
  410. wbi = (wl_bss_info_t *) ((u8 *) wbi + wbi->length);
  411. }
  412. wpa_printf(MSG_MSGDUMP, "Received %d bytes of scan results (%lu "
  413. "BSSes)",
  414. wsr->buflen, (unsigned long) ap_num);
  415. os_free(buf);
  416. return ap_num;
  417. }
  418. static int wpa_driver_broadcom_deauthenticate(void *priv, const u8 *addr,
  419. int reason_code)
  420. {
  421. struct wpa_driver_broadcom_data *drv = priv;
  422. wlc_deauth_t wdt;
  423. wdt.val = reason_code;
  424. os_memcpy(&wdt.ea, addr, sizeof wdt.ea);
  425. wdt.res = 0x7fff;
  426. return broadcom_ioctl(drv, WLC_DEAUTHENTICATE_WITH_REASON, &wdt,
  427. sizeof(wdt));
  428. }
  429. static int wpa_driver_broadcom_disassociate(void *priv, const u8 *addr,
  430. int reason_code)
  431. {
  432. struct wpa_driver_broadcom_data *drv = priv;
  433. return broadcom_ioctl(drv, WLC_DISASSOC, NULL, 0);
  434. }
  435. static int
  436. wpa_driver_broadcom_associate(void *priv,
  437. struct wpa_driver_associate_params *params)
  438. {
  439. struct wpa_driver_broadcom_data *drv = priv;
  440. wlc_ssid_t s;
  441. int infra = 1;
  442. int auth = 0;
  443. int wsec = 4;
  444. int dummy;
  445. int wpa_auth;
  446. s.SSID_len = params->ssid_len;
  447. os_memcpy(s.SSID, params->ssid, params->ssid_len);
  448. switch (params->pairwise_suite) {
  449. case CIPHER_WEP40:
  450. case CIPHER_WEP104:
  451. wsec = 1;
  452. break;
  453. case CIPHER_TKIP:
  454. wsec = 2;
  455. break;
  456. case CIPHER_CCMP:
  457. wsec = 4;
  458. break;
  459. default:
  460. wsec = 0;
  461. break;
  462. }
  463. switch (params->key_mgmt_suite) {
  464. case KEY_MGMT_802_1X:
  465. wpa_auth = 1;
  466. break;
  467. case KEY_MGMT_PSK:
  468. wpa_auth = 2;
  469. break;
  470. default:
  471. wpa_auth = 255;
  472. break;
  473. }
  474. /* printf("broadcom_associate: %u %u %u\n", pairwise_suite,
  475. * group_suite, key_mgmt_suite);
  476. * broadcom_ioctl(ifname, WLC_GET_WSEC, &wsec, sizeof(wsec));
  477. * wl join uses wlc_sec_wep here, not wlc_set_wsec */
  478. if (broadcom_ioctl(drv, WLC_SET_WSEC, &wsec, sizeof(wsec)) < 0 ||
  479. broadcom_ioctl(drv, WLC_SET_WPA_AUTH, &wpa_auth,
  480. sizeof(wpa_auth)) < 0 ||
  481. broadcom_ioctl(drv, WLC_GET_WEP, &dummy, sizeof(dummy)) < 0 ||
  482. broadcom_ioctl(drv, WLC_SET_INFRA, &infra, sizeof(infra)) < 0 ||
  483. broadcom_ioctl(drv, WLC_SET_AUTH, &auth, sizeof(auth)) < 0 ||
  484. broadcom_ioctl(drv, WLC_SET_WEP, &wsec, sizeof(wsec)) < 0 ||
  485. broadcom_ioctl(drv, WLC_SET_SSID, &s, sizeof(s)) < 0)
  486. return -1;
  487. return 0;
  488. }
  489. const struct wpa_driver_ops wpa_driver_broadcom_ops = {
  490. .name = "broadcom",
  491. .desc = "Broadcom wl.o driver",
  492. .get_bssid = wpa_driver_broadcom_get_bssid,
  493. .get_ssid = wpa_driver_broadcom_get_ssid,
  494. .set_wpa = wpa_driver_broadcom_set_wpa,
  495. .set_key = wpa_driver_broadcom_set_key,
  496. .init = wpa_driver_broadcom_init,
  497. .deinit = wpa_driver_broadcom_deinit,
  498. .set_countermeasures = wpa_driver_broadcom_set_countermeasures,
  499. .set_drop_unencrypted = wpa_driver_broadcom_set_drop_unencrypted,
  500. .scan = wpa_driver_broadcom_scan,
  501. .get_scan_results = wpa_driver_broadcom_get_scan_results,
  502. .deauthenticate = wpa_driver_broadcom_deauthenticate,
  503. .disassociate = wpa_driver_broadcom_disassociate,
  504. .associate = wpa_driver_broadcom_associate,
  505. };