driver_iphone.m 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466
  1. /*
  2. * WPA Supplicant - iPhone/iPod touch Apple80211 driver interface
  3. * Copyright (c) 2007, Jouni Malinen <j@w1.fi>
  4. *
  5. * This program is free software; you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License version 2 as
  7. * published by the Free Software Foundation.
  8. *
  9. * Alternatively, this software may be distributed under the terms of BSD
  10. * license.
  11. *
  12. * See README and COPYING for more details.
  13. */
  14. #include "includes.h"
  15. #define Boolean __DummyBoolean
  16. #include <CoreFoundation/CoreFoundation.h>
  17. #undef Boolean
  18. #include "common.h"
  19. #include "driver.h"
  20. #include "eloop.h"
  21. #include "common/ieee802_11_defs.h"
  22. #include "MobileApple80211.h"
  23. struct wpa_driver_iphone_data {
  24. void *ctx;
  25. Apple80211Ref wireless_ctx;
  26. CFArrayRef scan_results;
  27. int ctrl_power;
  28. };
  29. static const void * cfdict_get_key_str(CFDictionaryRef dict, const char *key)
  30. {
  31. const void *res;
  32. CFStringRef str = CFStringCreateWithCString(kCFAllocatorDefault, key,
  33. kCFStringEncodingMacRoman);
  34. if (str == NULL)
  35. return NULL;
  36. res = CFDictionaryGetValue(dict, str);
  37. CFRelease(str);
  38. return res;
  39. }
  40. static int wpa_driver_iphone_get_ssid(void *priv, u8 *ssid)
  41. {
  42. struct wpa_driver_iphone_data *drv = priv;
  43. CFDataRef data;
  44. int err, len;
  45. err = Apple80211CopyValue(drv->wireless_ctx, APPLE80211_VALUE_SSID, 0,
  46. &data);
  47. if (err != 0) {
  48. wpa_printf(MSG_DEBUG, "iPhone: Apple80211CopyValue(SSID) "
  49. "failed: %d", err);
  50. return -1;
  51. }
  52. len = CFDataGetLength(data);
  53. if (len > 32) {
  54. CFRelease(data);
  55. return -1;
  56. }
  57. os_memcpy(ssid, CFDataGetBytePtr(data), len);
  58. CFRelease(data);
  59. return len;
  60. }
  61. static int wpa_driver_iphone_get_bssid(void *priv, u8 *bssid)
  62. {
  63. struct wpa_driver_iphone_data *drv = priv;
  64. CFStringRef data;
  65. int err;
  66. int a1, a2, a3, a4, a5, a6;
  67. err = Apple80211CopyValue(drv->wireless_ctx, APPLE80211_VALUE_BSSID, 0,
  68. &data);
  69. if (err != 0) {
  70. wpa_printf(MSG_DEBUG, "iPhone: Apple80211CopyValue(BSSID) "
  71. "failed: %d", err);
  72. return -1;
  73. }
  74. sscanf(CFStringGetCStringPtr(data, kCFStringEncodingMacRoman),
  75. "%x:%x:%x:%x:%x:%x", &a1, &a2, &a3, &a4, &a5, &a6);
  76. bssid[0] = a1;
  77. bssid[1] = a2;
  78. bssid[2] = a3;
  79. bssid[3] = a4;
  80. bssid[4] = a5;
  81. bssid[5] = a6;
  82. CFRelease(data);
  83. return 0;
  84. }
  85. static void wpa_driver_iphone_scan_timeout(void *eloop_ctx, void *timeout_ctx)
  86. {
  87. wpa_supplicant_event(timeout_ctx, EVENT_SCAN_RESULTS, NULL);
  88. }
  89. static int wpa_driver_iphone_scan(void *priv, const u8 *ssid, size_t ssid_len)
  90. {
  91. struct wpa_driver_iphone_data *drv = priv;
  92. int err;
  93. if (drv->scan_results) {
  94. CFRelease(drv->scan_results);
  95. drv->scan_results = NULL;
  96. }
  97. err = Apple80211Scan(drv->wireless_ctx, &drv->scan_results, NULL);
  98. if (err) {
  99. wpa_printf(MSG_DEBUG, "iPhone: Apple80211Scan failed: %d",
  100. err);
  101. return -1;
  102. }
  103. eloop_register_timeout(0, 0, wpa_driver_iphone_scan_timeout, drv,
  104. drv->ctx);
  105. return 0;
  106. }
  107. static int wpa_driver_iphone_get_scan_results(void *priv,
  108. struct wpa_scan_result *results,
  109. size_t max_size)
  110. {
  111. struct wpa_driver_iphone_data *drv = priv;
  112. size_t i, num;
  113. if (drv->scan_results == NULL)
  114. return 0;
  115. num = CFArrayGetCount(drv->scan_results);
  116. if (num > max_size)
  117. num = max_size;
  118. os_memset(results, 0, num * sizeof(struct wpa_scan_result));
  119. for (i = 0; i < num; i++) {
  120. struct wpa_scan_result *res = &results[i];
  121. CFDictionaryRef dict =
  122. CFArrayGetValueAtIndex(drv->scan_results, i);
  123. CFDataRef data;
  124. CFStringRef str;
  125. CFNumberRef num;
  126. int val;
  127. data = cfdict_get_key_str(dict, "SSID");
  128. if (data) {
  129. res->ssid_len = CFDataGetLength(data);
  130. if (res->ssid_len > 32)
  131. res->ssid_len = 32;
  132. os_memcpy(res->ssid, CFDataGetBytePtr(data),
  133. res->ssid_len);
  134. }
  135. str = cfdict_get_key_str(dict, "BSSID");
  136. if (str) {
  137. int a1, a2, a3, a4, a5, a6;
  138. sscanf(CFStringGetCStringPtr(
  139. str, kCFStringEncodingMacRoman),
  140. "%x:%x:%x:%x:%x:%x",
  141. &a1, &a2, &a3, &a4, &a5, &a6);
  142. res->bssid[0] = a1;
  143. res->bssid[1] = a2;
  144. res->bssid[2] = a3;
  145. res->bssid[3] = a4;
  146. res->bssid[4] = a5;
  147. res->bssid[5] = a6;
  148. }
  149. num = cfdict_get_key_str(dict, "CAPABILITIES");
  150. if (num) {
  151. if (CFNumberGetValue(num, kCFNumberSInt32Type, &val))
  152. res->caps = val;
  153. }
  154. num = cfdict_get_key_str(dict, "CHANNEL");
  155. if (num) {
  156. if (CFNumberGetValue(num, kCFNumberSInt32Type, &val))
  157. res->freq = 2407 + val * 5;
  158. }
  159. num = cfdict_get_key_str(dict, "RSSI");
  160. if (num) {
  161. if (CFNumberGetValue(num, kCFNumberSInt32Type, &val))
  162. res->level = val;
  163. }
  164. num = cfdict_get_key_str(dict, "NOISE");
  165. if (num) {
  166. if (CFNumberGetValue(num, kCFNumberSInt32Type, &val))
  167. res->noise = val;
  168. }
  169. data = cfdict_get_key_str(dict, "IE");
  170. if (data) {
  171. u8 *ptr = (u8 *) CFDataGetBytePtr(data);
  172. int len = CFDataGetLength(data);
  173. u8 *pos = ptr, *end = ptr + len;
  174. while (pos + 2 < end) {
  175. if (pos + 2 + pos[1] > end)
  176. break;
  177. if (pos[0] == WLAN_EID_RSN &&
  178. pos[1] <= SSID_MAX_WPA_IE_LEN) {
  179. os_memcpy(res->rsn_ie, pos,
  180. 2 + pos[1]);
  181. res->rsn_ie_len = 2 + pos[1];
  182. }
  183. if (pos[0] == WLAN_EID_VENDOR_SPECIFIC &&
  184. pos[1] > 4 && pos[2] == 0x00 &&
  185. pos[3] == 0x50 && pos[4] == 0xf2 &&
  186. pos[5] == 0x01) {
  187. os_memcpy(res->wpa_ie, pos,
  188. 2 + pos[1]);
  189. res->wpa_ie_len = 2 + pos[1];
  190. }
  191. pos = pos + 2 + pos[1];
  192. }
  193. }
  194. }
  195. return num;
  196. }
  197. static void wpa_driver_iphone_assoc_timeout(void *eloop_ctx, void *timeout_ctx)
  198. {
  199. struct wpa_driver_iphone_data *drv = eloop_ctx;
  200. u8 bssid[ETH_ALEN];
  201. if (wpa_driver_iphone_get_bssid(drv, bssid) != 0) {
  202. eloop_register_timeout(1, 0, wpa_driver_iphone_assoc_timeout,
  203. drv, drv->ctx);
  204. return;
  205. }
  206. wpa_supplicant_event(timeout_ctx, EVENT_ASSOC, NULL);
  207. }
  208. static int wpa_driver_iphone_associate(
  209. void *priv, struct wpa_driver_associate_params *params)
  210. {
  211. struct wpa_driver_iphone_data *drv = priv;
  212. int i, num, err;
  213. size_t ssid_len;
  214. CFDictionaryRef bss = NULL;
  215. /*
  216. * TODO: Consider generating parameters instead of just using an entry
  217. * from scan results in order to support ap_scan=2.
  218. */
  219. if (drv->scan_results == NULL) {
  220. wpa_printf(MSG_DEBUG, "iPhone: No scan results - cannot "
  221. "associate");
  222. return -1;
  223. }
  224. num = CFArrayGetCount(drv->scan_results);
  225. for (i = 0; i < num; i++) {
  226. CFDictionaryRef dict =
  227. CFArrayGetValueAtIndex(drv->scan_results, i);
  228. CFDataRef data;
  229. data = cfdict_get_key_str(dict, "SSID");
  230. if (data == NULL)
  231. continue;
  232. ssid_len = CFDataGetLength(data);
  233. if (ssid_len != params->ssid_len ||
  234. os_memcmp(CFDataGetBytePtr(data), params->ssid, ssid_len)
  235. != 0)
  236. continue;
  237. bss = dict;
  238. break;
  239. }
  240. if (bss == NULL) {
  241. wpa_printf(MSG_DEBUG, "iPhone: Could not find SSID from scan "
  242. "results - cannot associate");
  243. return -1;
  244. }
  245. wpa_printf(MSG_DEBUG, "iPhone: Trying to associate with a BSS found "
  246. "from scan results");
  247. err = Apple80211Associate(drv->wireless_ctx, bss, NULL);
  248. if (err) {
  249. wpa_printf(MSG_DEBUG, "iPhone: Apple80211Associate() failed: "
  250. "%d", err);
  251. return -1;
  252. }
  253. /*
  254. * Driver is actually already associated; report association from an
  255. * eloop callback.
  256. */
  257. eloop_cancel_timeout(wpa_driver_iphone_assoc_timeout, drv, drv->ctx);
  258. eloop_register_timeout(0, 0, wpa_driver_iphone_assoc_timeout, drv,
  259. drv->ctx);
  260. return 0;
  261. }
  262. static int wpa_driver_iphone_set_key(void *priv, wpa_alg alg, const u8 *addr,
  263. int key_idx, int set_tx, const u8 *seq,
  264. size_t seq_len, const u8 *key,
  265. size_t key_len)
  266. {
  267. /*
  268. * TODO: Need to either support configuring PMK for 4-way handshake or
  269. * PTK for TKIP/CCMP.
  270. */
  271. return -1;
  272. }
  273. static int wpa_driver_iphone_get_capa(void *priv, struct wpa_driver_capa *capa)
  274. {
  275. os_memset(capa, 0, sizeof(*capa));
  276. capa->key_mgmt = WPA_DRIVER_CAPA_KEY_MGMT_WPA |
  277. WPA_DRIVER_CAPA_KEY_MGMT_WPA2 |
  278. WPA_DRIVER_CAPA_KEY_MGMT_WPA_PSK |
  279. WPA_DRIVER_CAPA_KEY_MGMT_WPA2_PSK;
  280. capa->enc = WPA_DRIVER_CAPA_ENC_WEP40 | WPA_DRIVER_CAPA_ENC_WEP104 |
  281. WPA_DRIVER_CAPA_ENC_TKIP | WPA_DRIVER_CAPA_ENC_CCMP;
  282. capa->auth = WPA_DRIVER_AUTH_OPEN | WPA_DRIVER_AUTH_SHARED |
  283. WPA_DRIVER_AUTH_LEAP;
  284. capa->flags = WPA_DRIVER_FLAGS_4WAY_HANDSHAKE;
  285. return 0;
  286. }
  287. static void * wpa_driver_iphone_init(void *ctx, const char *ifname)
  288. {
  289. struct wpa_driver_iphone_data *drv;
  290. int err;
  291. char power;
  292. CFStringRef name;
  293. CFDictionaryRef dict;
  294. drv = os_zalloc(sizeof(*drv));
  295. if (drv == NULL)
  296. return NULL;
  297. drv->ctx = ctx;
  298. err = Apple80211Open(&drv->wireless_ctx);
  299. if (err) {
  300. wpa_printf(MSG_ERROR, "iPhone: Apple80211Open failed: %d",
  301. err);
  302. os_free(drv);
  303. return NULL;
  304. }
  305. name = CFStringCreateWithCString(kCFAllocatorDefault, ifname,
  306. kCFStringEncodingISOLatin1);
  307. if (name == NULL) {
  308. wpa_printf(MSG_ERROR, "iPhone: ifname -> CFString failed");
  309. Apple80211Close(drv->wireless_ctx);
  310. os_free(drv);
  311. return NULL;
  312. }
  313. err = Apple80211BindToInterface(drv->wireless_ctx, name);
  314. CFRelease(name);
  315. if (err) {
  316. wpa_printf(MSG_ERROR, "iPhone: Apple80211BindToInterface "
  317. "failed: %d", err);
  318. Apple80211Close(drv->wireless_ctx);
  319. os_free(drv);
  320. return NULL;
  321. }
  322. err = Apple80211GetPower(drv->wireless_ctx, &power);
  323. if (err)
  324. wpa_printf(MSG_DEBUG, "iPhone: Apple80211GetPower failed: %d",
  325. err);
  326. wpa_printf(MSG_DEBUG, "iPhone: Power=%d", power);
  327. if (!power) {
  328. drv->ctrl_power = 1;
  329. err = Apple80211SetPower(drv->wireless_ctx, 1);
  330. if (err) {
  331. wpa_printf(MSG_DEBUG, "iPhone: Apple80211SetPower "
  332. "failed: %d", err);
  333. Apple80211Close(drv->wireless_ctx);
  334. os_free(drv);
  335. return NULL;
  336. }
  337. }
  338. err = Apple80211GetInfoCopy(drv->wireless_ctx, &dict);
  339. if (err == 0) {
  340. CFShow(dict);
  341. CFRelease(dict);
  342. } else {
  343. printf("Apple80211GetInfoCopy: %d\n", err);
  344. }
  345. return drv;
  346. }
  347. static void wpa_driver_iphone_deinit(void *priv)
  348. {
  349. struct wpa_driver_iphone_data *drv = priv;
  350. int err;
  351. eloop_cancel_timeout(wpa_driver_iphone_scan_timeout, drv, drv->ctx);
  352. eloop_cancel_timeout(wpa_driver_iphone_assoc_timeout, drv, drv->ctx);
  353. if (drv->ctrl_power) {
  354. wpa_printf(MSG_DEBUG, "iPhone: Power down the interface");
  355. err = Apple80211SetPower(drv->wireless_ctx, 0);
  356. if (err) {
  357. wpa_printf(MSG_DEBUG, "iPhone: Apple80211SetPower(0) "
  358. "failed: %d", err);
  359. }
  360. }
  361. err = Apple80211Close(drv->wireless_ctx);
  362. if (err) {
  363. wpa_printf(MSG_DEBUG, "iPhone: Apple80211Close failed: %d",
  364. err);
  365. }
  366. if (drv->scan_results)
  367. CFRelease(drv->scan_results);
  368. os_free(drv);
  369. }
  370. const struct wpa_driver_ops wpa_driver_iphone_ops = {
  371. .name = "iphone",
  372. .desc = "iPhone/iPod touch Apple80211 driver",
  373. .get_ssid = wpa_driver_iphone_get_ssid,
  374. .get_bssid = wpa_driver_iphone_get_bssid,
  375. .init = wpa_driver_iphone_init,
  376. .deinit = wpa_driver_iphone_deinit,
  377. .scan = wpa_driver_iphone_scan,
  378. .get_scan_results = wpa_driver_iphone_get_scan_results,
  379. .associate = wpa_driver_iphone_associate,
  380. .set_key = wpa_driver_iphone_set_key,
  381. .get_capa = wpa_driver_iphone_get_capa,
  382. };