ieee802_11_common.c 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326
  1. /*
  2. * IEEE 802.11 Common routines
  3. * Copyright (c) 2002-2009, 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. #include "common.h"
  16. #include "ieee802_11_defs.h"
  17. #include "ieee802_11_common.h"
  18. static int ieee802_11_parse_vendor_specific(const u8 *pos, size_t elen,
  19. struct ieee802_11_elems *elems,
  20. int show_errors)
  21. {
  22. unsigned int oui;
  23. /* first 3 bytes in vendor specific information element are the IEEE
  24. * OUI of the vendor. The following byte is used a vendor specific
  25. * sub-type. */
  26. if (elen < 4) {
  27. if (show_errors) {
  28. wpa_printf(MSG_MSGDUMP, "short vendor specific "
  29. "information element ignored (len=%lu)",
  30. (unsigned long) elen);
  31. }
  32. return -1;
  33. }
  34. oui = WPA_GET_BE24(pos);
  35. switch (oui) {
  36. case OUI_MICROSOFT:
  37. /* Microsoft/Wi-Fi information elements are further typed and
  38. * subtyped */
  39. switch (pos[3]) {
  40. case 1:
  41. /* Microsoft OUI (00:50:F2) with OUI Type 1:
  42. * real WPA information element */
  43. elems->wpa_ie = pos;
  44. elems->wpa_ie_len = elen;
  45. break;
  46. case WMM_OUI_TYPE:
  47. /* WMM information element */
  48. if (elen < 5) {
  49. wpa_printf(MSG_MSGDUMP, "short WMM "
  50. "information element ignored "
  51. "(len=%lu)",
  52. (unsigned long) elen);
  53. return -1;
  54. }
  55. switch (pos[4]) {
  56. case WMM_OUI_SUBTYPE_INFORMATION_ELEMENT:
  57. case WMM_OUI_SUBTYPE_PARAMETER_ELEMENT:
  58. /*
  59. * Share same pointer since only one of these
  60. * is used and they start with same data.
  61. * Length field can be used to distinguish the
  62. * IEs.
  63. */
  64. elems->wmm = pos;
  65. elems->wmm_len = elen;
  66. break;
  67. case WMM_OUI_SUBTYPE_TSPEC_ELEMENT:
  68. elems->wmm_tspec = pos;
  69. elems->wmm_tspec_len = elen;
  70. break;
  71. default:
  72. wpa_printf(MSG_EXCESSIVE, "unknown WMM "
  73. "information element ignored "
  74. "(subtype=%d len=%lu)",
  75. pos[4], (unsigned long) elen);
  76. return -1;
  77. }
  78. break;
  79. case 4:
  80. /* Wi-Fi Protected Setup (WPS) IE */
  81. elems->wps_ie = pos;
  82. elems->wps_ie_len = elen;
  83. break;
  84. default:
  85. wpa_printf(MSG_EXCESSIVE, "Unknown Microsoft "
  86. "information element ignored "
  87. "(type=%d len=%lu)",
  88. pos[3], (unsigned long) elen);
  89. return -1;
  90. }
  91. break;
  92. case OUI_BROADCOM:
  93. switch (pos[3]) {
  94. case VENDOR_HT_CAPAB_OUI_TYPE:
  95. elems->vendor_ht_cap = pos;
  96. elems->vendor_ht_cap_len = elen;
  97. break;
  98. default:
  99. wpa_printf(MSG_EXCESSIVE, "Unknown Broadcom "
  100. "information element ignored "
  101. "(type=%d len=%lu)",
  102. pos[3], (unsigned long) elen);
  103. return -1;
  104. }
  105. break;
  106. default:
  107. wpa_printf(MSG_EXCESSIVE, "unknown vendor specific "
  108. "information element ignored (vendor OUI "
  109. "%02x:%02x:%02x len=%lu)",
  110. pos[0], pos[1], pos[2], (unsigned long) elen);
  111. return -1;
  112. }
  113. return 0;
  114. }
  115. /**
  116. * ieee802_11_parse_elems - Parse information elements in management frames
  117. * @start: Pointer to the start of IEs
  118. * @len: Length of IE buffer in octets
  119. * @elems: Data structure for parsed elements
  120. * @show_errors: Whether to show parsing errors in debug log
  121. * Returns: Parsing result
  122. */
  123. ParseRes ieee802_11_parse_elems(const u8 *start, size_t len,
  124. struct ieee802_11_elems *elems,
  125. int show_errors)
  126. {
  127. size_t left = len;
  128. const u8 *pos = start;
  129. int unknown = 0;
  130. os_memset(elems, 0, sizeof(*elems));
  131. while (left >= 2) {
  132. u8 id, elen;
  133. id = *pos++;
  134. elen = *pos++;
  135. left -= 2;
  136. if (elen > left) {
  137. if (show_errors) {
  138. wpa_printf(MSG_DEBUG, "IEEE 802.11 element "
  139. "parse failed (id=%d elen=%d "
  140. "left=%lu)",
  141. id, elen, (unsigned long) left);
  142. wpa_hexdump(MSG_MSGDUMP, "IEs", start, len);
  143. }
  144. return ParseFailed;
  145. }
  146. switch (id) {
  147. case WLAN_EID_SSID:
  148. elems->ssid = pos;
  149. elems->ssid_len = elen;
  150. break;
  151. case WLAN_EID_SUPP_RATES:
  152. elems->supp_rates = pos;
  153. elems->supp_rates_len = elen;
  154. break;
  155. case WLAN_EID_FH_PARAMS:
  156. elems->fh_params = pos;
  157. elems->fh_params_len = elen;
  158. break;
  159. case WLAN_EID_DS_PARAMS:
  160. elems->ds_params = pos;
  161. elems->ds_params_len = elen;
  162. break;
  163. case WLAN_EID_CF_PARAMS:
  164. elems->cf_params = pos;
  165. elems->cf_params_len = elen;
  166. break;
  167. case WLAN_EID_TIM:
  168. elems->tim = pos;
  169. elems->tim_len = elen;
  170. break;
  171. case WLAN_EID_IBSS_PARAMS:
  172. elems->ibss_params = pos;
  173. elems->ibss_params_len = elen;
  174. break;
  175. case WLAN_EID_CHALLENGE:
  176. elems->challenge = pos;
  177. elems->challenge_len = elen;
  178. break;
  179. case WLAN_EID_ERP_INFO:
  180. elems->erp_info = pos;
  181. elems->erp_info_len = elen;
  182. break;
  183. case WLAN_EID_EXT_SUPP_RATES:
  184. elems->ext_supp_rates = pos;
  185. elems->ext_supp_rates_len = elen;
  186. break;
  187. case WLAN_EID_VENDOR_SPECIFIC:
  188. if (ieee802_11_parse_vendor_specific(pos, elen,
  189. elems,
  190. show_errors))
  191. unknown++;
  192. break;
  193. case WLAN_EID_RSN:
  194. elems->rsn_ie = pos;
  195. elems->rsn_ie_len = elen;
  196. break;
  197. case WLAN_EID_PWR_CAPABILITY:
  198. elems->power_cap = pos;
  199. elems->power_cap_len = elen;
  200. break;
  201. case WLAN_EID_SUPPORTED_CHANNELS:
  202. elems->supp_channels = pos;
  203. elems->supp_channels_len = elen;
  204. break;
  205. case WLAN_EID_MOBILITY_DOMAIN:
  206. elems->mdie = pos;
  207. elems->mdie_len = elen;
  208. break;
  209. case WLAN_EID_FAST_BSS_TRANSITION:
  210. elems->ftie = pos;
  211. elems->ftie_len = elen;
  212. break;
  213. case WLAN_EID_TIMEOUT_INTERVAL:
  214. elems->timeout_int = pos;
  215. elems->timeout_int_len = elen;
  216. break;
  217. case WLAN_EID_HT_CAP:
  218. elems->ht_capabilities = pos;
  219. elems->ht_capabilities_len = elen;
  220. break;
  221. case WLAN_EID_HT_OPERATION:
  222. elems->ht_operation = pos;
  223. elems->ht_operation_len = elen;
  224. break;
  225. default:
  226. unknown++;
  227. if (!show_errors)
  228. break;
  229. wpa_printf(MSG_MSGDUMP, "IEEE 802.11 element parse "
  230. "ignored unknown element (id=%d elen=%d)",
  231. id, elen);
  232. break;
  233. }
  234. left -= elen;
  235. pos += elen;
  236. }
  237. if (left)
  238. return ParseFailed;
  239. return unknown ? ParseUnknown : ParseOK;
  240. }
  241. int ieee802_11_ie_count(const u8 *ies, size_t ies_len)
  242. {
  243. int count = 0;
  244. const u8 *pos, *end;
  245. if (ies == NULL)
  246. return 0;
  247. pos = ies;
  248. end = ies + ies_len;
  249. while (pos + 2 <= end) {
  250. if (pos + 2 + pos[1] > end)
  251. break;
  252. count++;
  253. pos += 2 + pos[1];
  254. }
  255. return count;
  256. }
  257. struct wpabuf * ieee802_11_vendor_ie_concat(const u8 *ies, size_t ies_len,
  258. u32 oui_type)
  259. {
  260. struct wpabuf *buf;
  261. const u8 *end, *pos, *ie;
  262. pos = ies;
  263. end = ies + ies_len;
  264. ie = NULL;
  265. while (pos + 1 < end) {
  266. if (pos + 2 + pos[1] > end)
  267. return NULL;
  268. if (pos[0] == WLAN_EID_VENDOR_SPECIFIC && pos[1] >= 4 &&
  269. WPA_GET_BE32(&pos[2]) == oui_type) {
  270. ie = pos;
  271. break;
  272. }
  273. pos += 2 + pos[1];
  274. }
  275. if (ie == NULL)
  276. return NULL; /* No specified vendor IE found */
  277. buf = wpabuf_alloc(ies_len);
  278. if (buf == NULL)
  279. return NULL;
  280. /*
  281. * There may be multiple vendor IEs in the message, so need to
  282. * concatenate their data fields.
  283. */
  284. while (pos + 1 < end) {
  285. if (pos + 2 + pos[1] > end)
  286. break;
  287. if (pos[0] == WLAN_EID_VENDOR_SPECIFIC && pos[1] >= 4 &&
  288. WPA_GET_BE32(&pos[2]) == oui_type)
  289. wpabuf_put_data(buf, pos + 6, pos[1] - 4);
  290. pos += 2 + pos[1];
  291. }
  292. return buf;
  293. }