rrm.c 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394
  1. /*
  2. * hostapd / Radio Measurement (RRM)
  3. * Copyright(c) 2013 - 2016 Intel Mobile Communications GmbH.
  4. * Copyright(c) 2011 - 2016 Intel Corporation. All rights reserved.
  5. *
  6. * This software may be distributed under the terms of the BSD license.
  7. * See README for more details.
  8. */
  9. #include "utils/includes.h"
  10. #include "utils/common.h"
  11. #include "hostapd.h"
  12. #include "ap_drv_ops.h"
  13. #include "sta_info.h"
  14. #include "eloop.h"
  15. #include "neighbor_db.h"
  16. #include "rrm.h"
  17. #define HOSTAPD_RRM_REQUEST_TIMEOUT 5
  18. static void hostapd_lci_rep_timeout_handler(void *eloop_data, void *user_ctx)
  19. {
  20. struct hostapd_data *hapd = eloop_data;
  21. wpa_printf(MSG_DEBUG, "RRM: LCI request (token %u) timed out",
  22. hapd->lci_req_token);
  23. hapd->lci_req_active = 0;
  24. }
  25. static void hostapd_handle_lci_report(struct hostapd_data *hapd, u8 token,
  26. const u8 *pos, size_t len)
  27. {
  28. if (!hapd->lci_req_active || hapd->lci_req_token != token) {
  29. wpa_printf(MSG_DEBUG, "Unexpected LCI report, token %u", token);
  30. return;
  31. }
  32. hapd->lci_req_active = 0;
  33. eloop_cancel_timeout(hostapd_lci_rep_timeout_handler, hapd, NULL);
  34. wpa_printf(MSG_DEBUG, "LCI report token %u len %zu", token, len);
  35. }
  36. static void hostapd_handle_radio_msmt_report(struct hostapd_data *hapd,
  37. const u8 *buf, size_t len)
  38. {
  39. const struct ieee80211_mgmt *mgmt = (const struct ieee80211_mgmt *) buf;
  40. const u8 *pos, *ie, *end;
  41. u8 token;
  42. end = buf + len;
  43. token = mgmt->u.action.u.rrm.dialog_token;
  44. pos = mgmt->u.action.u.rrm.variable;
  45. while ((ie = get_ie(pos, end - pos, WLAN_EID_MEASURE_REPORT))) {
  46. if (ie[1] < 5) {
  47. wpa_printf(MSG_DEBUG, "Bad Measurement Report element");
  48. break;
  49. }
  50. wpa_printf(MSG_DEBUG, "Measurement report type %u", ie[4]);
  51. switch (ie[4]) {
  52. case MEASURE_TYPE_LCI:
  53. hostapd_handle_lci_report(hapd, token, ie + 2, ie[1]);
  54. break;
  55. default:
  56. wpa_printf(MSG_DEBUG,
  57. "Measurement report type %u is not supported",
  58. ie[4]);
  59. break;
  60. }
  61. pos = ie + ie[1] + 2;
  62. }
  63. }
  64. static u16 hostapd_parse_location_lci_req_age(const u8 *buf, size_t len)
  65. {
  66. const u8 *subelem;
  67. /* Range Request element + Location Subject + Maximum Age subelement */
  68. if (len < 3 + 1 + 4)
  69. return 0;
  70. /* Subelements are arranged as IEs */
  71. subelem = get_ie(buf + 4, len - 4, LCI_REQ_SUBELEM_MAX_AGE);
  72. if (subelem && subelem[1] == 2)
  73. return *(u16 *) (subelem + 2);
  74. return 0;
  75. }
  76. static int hostapd_check_lci_age(struct hostapd_neighbor_entry *nr, u16 max_age)
  77. {
  78. struct os_time curr, diff;
  79. unsigned long diff_l;
  80. if (!max_age)
  81. return 0;
  82. if (max_age == 0xffff)
  83. return 1;
  84. if (os_get_time(&curr))
  85. return 0;
  86. os_time_sub(&curr, &nr->lci_date, &diff);
  87. /* avoid overflow */
  88. if (diff.sec > 0xffff)
  89. return 0;
  90. /* LCI age is calculated in 10th of a second units. */
  91. diff_l = diff.sec * 10 + diff.usec / 100000;
  92. return max_age > diff_l;
  93. }
  94. static size_t hostapd_neighbor_report_len(struct wpabuf *buf,
  95. struct hostapd_neighbor_entry *nr,
  96. int send_lci, int send_civic)
  97. {
  98. size_t len = 2 + wpabuf_len(nr->nr);
  99. if (send_lci && nr->lci)
  100. len += 2 + wpabuf_len(nr->lci);
  101. if (send_civic && nr->civic)
  102. len += 2 + wpabuf_len(nr->civic);
  103. return len;
  104. }
  105. static void hostapd_send_nei_report_resp(struct hostapd_data *hapd,
  106. const u8 *addr, u8 dialog_token,
  107. struct wpa_ssid_value *ssid, u8 lci,
  108. u8 civic, u16 lci_max_age)
  109. {
  110. struct hostapd_neighbor_entry *nr;
  111. struct wpabuf *buf;
  112. u8 *msmt_token;
  113. /*
  114. * The number and length of the Neighbor Report elements in a Neighbor
  115. * Report frame is limited by the maximum allowed MMPDU size; + 3 bytes
  116. * of RRM header.
  117. */
  118. buf = wpabuf_alloc(3 + IEEE80211_MAX_MMPDU_SIZE);
  119. if (!buf)
  120. return;
  121. wpabuf_put_u8(buf, WLAN_ACTION_RADIO_MEASUREMENT);
  122. wpabuf_put_u8(buf, WLAN_RRM_NEIGHBOR_REPORT_RESPONSE);
  123. wpabuf_put_u8(buf, dialog_token);
  124. dl_list_for_each(nr, &hapd->nr_db, struct hostapd_neighbor_entry,
  125. list) {
  126. int send_lci;
  127. size_t len;
  128. if (ssid->ssid_len != nr->ssid.ssid_len ||
  129. os_memcmp(ssid->ssid, nr->ssid.ssid, ssid->ssid_len) != 0)
  130. continue;
  131. send_lci = (lci != 0) && hostapd_check_lci_age(nr, lci_max_age);
  132. len = hostapd_neighbor_report_len(buf, nr, send_lci, civic);
  133. if (len - 2 > 0xff) {
  134. wpa_printf(MSG_DEBUG,
  135. "NR entry for " MACSTR " exceeds 0xFF bytes",
  136. MAC2STR(nr->bssid));
  137. continue;
  138. }
  139. if (len > wpabuf_tailroom(buf))
  140. break;
  141. wpabuf_put_u8(buf, WLAN_EID_NEIGHBOR_REPORT);
  142. wpabuf_put_u8(buf, len - 2);
  143. wpabuf_put_buf(buf, nr->nr);
  144. if (send_lci && nr->lci) {
  145. wpabuf_put_u8(buf, WLAN_EID_MEASURE_REPORT);
  146. wpabuf_put_u8(buf, wpabuf_len(nr->lci));
  147. /*
  148. * Override measurement token - the first byte of the
  149. * Measurement Report element.
  150. */
  151. msmt_token = wpabuf_put(buf, 0);
  152. wpabuf_put_buf(buf, nr->lci);
  153. *msmt_token = lci;
  154. }
  155. if (civic && nr->civic) {
  156. wpabuf_put_u8(buf, WLAN_EID_MEASURE_REPORT);
  157. wpabuf_put_u8(buf, wpabuf_len(nr->civic));
  158. /*
  159. * Override measurement token - the first byte of the
  160. * Measurement Report element.
  161. */
  162. msmt_token = wpabuf_put(buf, 0);
  163. wpabuf_put_buf(buf, nr->civic);
  164. *msmt_token = civic;
  165. }
  166. }
  167. hostapd_drv_send_action(hapd, hapd->iface->freq, 0, addr,
  168. wpabuf_head(buf), wpabuf_len(buf));
  169. wpabuf_free(buf);
  170. }
  171. static void hostapd_handle_nei_report_req(struct hostapd_data *hapd,
  172. const u8 *buf, size_t len)
  173. {
  174. const struct ieee80211_mgmt *mgmt = (const struct ieee80211_mgmt *) buf;
  175. const u8 *pos, *ie, *end;
  176. struct wpa_ssid_value ssid = {
  177. .ssid_len = 0
  178. };
  179. u8 token;
  180. u8 lci = 0, civic = 0; /* Measurement tokens */
  181. u16 lci_max_age = 0;
  182. if (!(hapd->conf->radio_measurements[0] &
  183. WLAN_RRM_CAPS_NEIGHBOR_REPORT))
  184. return;
  185. end = buf + len;
  186. token = mgmt->u.action.u.rrm.dialog_token;
  187. pos = mgmt->u.action.u.rrm.variable;
  188. len = end - pos;
  189. ie = get_ie(pos, len, WLAN_EID_SSID);
  190. if (ie && ie[1] && ie[1] <= SSID_MAX_LEN) {
  191. ssid.ssid_len = ie[1];
  192. os_memcpy(ssid.ssid, ie + 2, ssid.ssid_len);
  193. } else {
  194. ssid.ssid_len = hapd->conf->ssid.ssid_len;
  195. os_memcpy(ssid.ssid, hapd->conf->ssid.ssid, ssid.ssid_len);
  196. }
  197. while ((ie = get_ie(pos, len, WLAN_EID_MEASURE_REQUEST))) {
  198. if (ie[1] < 3)
  199. break;
  200. wpa_printf(MSG_DEBUG,
  201. "Neighbor report request, measure type %u",
  202. ie[4]);
  203. switch (ie[4]) { /* Measurement Type */
  204. case MEASURE_TYPE_LCI:
  205. lci = ie[2]; /* Measurement Token */
  206. lci_max_age = hostapd_parse_location_lci_req_age(ie + 2,
  207. ie[1]);
  208. break;
  209. case MEASURE_TYPE_LOCATION_CIVIC:
  210. civic = ie[2]; /* Measurement token */
  211. break;
  212. }
  213. pos = ie + ie[1] + 2;
  214. len = end - pos;
  215. }
  216. hostapd_send_nei_report_resp(hapd, mgmt->sa, token, &ssid, lci, civic,
  217. lci_max_age);
  218. }
  219. void hostapd_handle_radio_measurement(struct hostapd_data *hapd,
  220. const u8 *buf, size_t len)
  221. {
  222. const struct ieee80211_mgmt *mgmt = (const struct ieee80211_mgmt *) buf;
  223. /*
  224. * Check for enough bytes: header + (1B)Category + (1B)Action +
  225. * (1B)Dialog Token.
  226. */
  227. if (len < IEEE80211_HDRLEN + 3)
  228. return;
  229. wpa_printf(MSG_DEBUG, "Radio measurement frame, action %u from " MACSTR,
  230. mgmt->u.action.u.rrm.action, MAC2STR(mgmt->sa));
  231. switch (mgmt->u.action.u.rrm.action) {
  232. case WLAN_RRM_RADIO_MEASUREMENT_REPORT:
  233. hostapd_handle_radio_msmt_report(hapd, buf, len);
  234. break;
  235. case WLAN_RRM_NEIGHBOR_REPORT_REQUEST:
  236. hostapd_handle_nei_report_req(hapd, buf, len);
  237. break;
  238. default:
  239. wpa_printf(MSG_DEBUG, "RRM action %u is not supported",
  240. mgmt->u.action.u.rrm.action);
  241. break;
  242. }
  243. }
  244. int hostapd_send_lci_req(struct hostapd_data *hapd, const u8 *addr)
  245. {
  246. struct wpabuf *buf;
  247. struct sta_info *sta = ap_get_sta(hapd, addr);
  248. int ret;
  249. if (!sta) {
  250. wpa_printf(MSG_INFO,
  251. "Request LCI: Destination address is not in station list");
  252. return -1;
  253. }
  254. if (!(sta->flags & WLAN_STA_AUTHORIZED)) {
  255. wpa_printf(MSG_INFO,
  256. "Request LCI: Destination address is not connected");
  257. return -1;
  258. }
  259. if (!(sta->rrm_enabled_capa[1] & WLAN_RRM_CAPS_LCI_MEASUREMENT)) {
  260. wpa_printf(MSG_INFO,
  261. "Request LCI: Station does not support LCI in RRM");
  262. return -1;
  263. }
  264. if (hapd->lci_req_active) {
  265. wpa_printf(MSG_DEBUG,
  266. "Request LCI: LCI request is already in process, overriding");
  267. hapd->lci_req_active = 0;
  268. eloop_cancel_timeout(hostapd_lci_rep_timeout_handler, hapd,
  269. NULL);
  270. }
  271. /* Measurement request (5) + Measurement element with LCI (10) */
  272. buf = wpabuf_alloc(5 + 10);
  273. if (!buf)
  274. return -1;
  275. hapd->lci_req_token++;
  276. /* For wraparounds - the token must be nonzero */
  277. if (!hapd->lci_req_token)
  278. hapd->lci_req_token++;
  279. wpabuf_put_u8(buf, WLAN_ACTION_RADIO_MEASUREMENT);
  280. wpabuf_put_u8(buf, WLAN_RRM_RADIO_MEASUREMENT_REQUEST);
  281. wpabuf_put_u8(buf, hapd->lci_req_token);
  282. wpabuf_put_le16(buf, 0); /* Number of repetitions */
  283. wpabuf_put_u8(buf, WLAN_EID_MEASURE_REQUEST);
  284. wpabuf_put_u8(buf, 3 + 1 + 4);
  285. wpabuf_put_u8(buf, 1); /* Measurement Token */
  286. /*
  287. * Parallel and Enable bits are 0, Duration, Request, and Report are
  288. * reserved.
  289. */
  290. wpabuf_put_u8(buf, 0);
  291. wpabuf_put_u8(buf, MEASURE_TYPE_LCI);
  292. wpabuf_put_u8(buf, LOCATION_SUBJECT_REMOTE);
  293. wpabuf_put_u8(buf, LCI_REQ_SUBELEM_MAX_AGE);
  294. wpabuf_put_u8(buf, 2);
  295. wpabuf_put_le16(buf, 0xffff);
  296. ret = hostapd_drv_send_action(hapd, hapd->iface->freq, 0, addr,
  297. wpabuf_head(buf), wpabuf_len(buf));
  298. wpabuf_free(buf);
  299. if (ret)
  300. return ret;
  301. hapd->lci_req_active = 1;
  302. eloop_register_timeout(HOSTAPD_RRM_REQUEST_TIMEOUT, 0,
  303. hostapd_lci_rep_timeout_handler, hapd, NULL);
  304. return 0;
  305. }
  306. void hostapd_clean_rrm(struct hostapd_data *hapd)
  307. {
  308. hostpad_free_neighbor_db(hapd);
  309. eloop_cancel_timeout(hostapd_lci_rep_timeout_handler, hapd, NULL);
  310. hapd->lci_req_active = 0;
  311. }