rrm.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624
  1. /*
  2. * wpa_supplicant - Radio Measurements
  3. * Copyright (c) 2003-2016, Jouni Malinen <j@w1.fi>
  4. *
  5. * This software may be distributed under the terms of the BSD license.
  6. * See README for more details.
  7. */
  8. #include "includes.h"
  9. #include "utils/common.h"
  10. #include "utils/eloop.h"
  11. #include "common/ieee802_11_common.h"
  12. #include "wpa_supplicant_i.h"
  13. #include "driver_i.h"
  14. #include "bss.h"
  15. static void wpas_rrm_neighbor_rep_timeout_handler(void *data, void *user_ctx)
  16. {
  17. struct rrm_data *rrm = data;
  18. if (!rrm->notify_neighbor_rep) {
  19. wpa_printf(MSG_ERROR,
  20. "RRM: Unexpected neighbor report timeout");
  21. return;
  22. }
  23. wpa_printf(MSG_DEBUG, "RRM: Notifying neighbor report - NONE");
  24. rrm->notify_neighbor_rep(rrm->neighbor_rep_cb_ctx, NULL);
  25. rrm->notify_neighbor_rep = NULL;
  26. rrm->neighbor_rep_cb_ctx = NULL;
  27. }
  28. /*
  29. * wpas_rrm_reset - Clear and reset all RRM data in wpa_supplicant
  30. * @wpa_s: Pointer to wpa_supplicant
  31. */
  32. void wpas_rrm_reset(struct wpa_supplicant *wpa_s)
  33. {
  34. wpa_s->rrm.rrm_used = 0;
  35. eloop_cancel_timeout(wpas_rrm_neighbor_rep_timeout_handler, &wpa_s->rrm,
  36. NULL);
  37. if (wpa_s->rrm.notify_neighbor_rep)
  38. wpas_rrm_neighbor_rep_timeout_handler(&wpa_s->rrm, NULL);
  39. wpa_s->rrm.next_neighbor_rep_token = 1;
  40. }
  41. /*
  42. * wpas_rrm_process_neighbor_rep - Handle incoming neighbor report
  43. * @wpa_s: Pointer to wpa_supplicant
  44. * @report: Neighbor report buffer, prefixed by a 1-byte dialog token
  45. * @report_len: Length of neighbor report buffer
  46. */
  47. void wpas_rrm_process_neighbor_rep(struct wpa_supplicant *wpa_s,
  48. const u8 *report, size_t report_len)
  49. {
  50. struct wpabuf *neighbor_rep;
  51. wpa_hexdump(MSG_DEBUG, "RRM: New Neighbor Report", report, report_len);
  52. if (report_len < 1)
  53. return;
  54. if (report[0] != wpa_s->rrm.next_neighbor_rep_token - 1) {
  55. wpa_printf(MSG_DEBUG,
  56. "RRM: Discarding neighbor report with token %d (expected %d)",
  57. report[0], wpa_s->rrm.next_neighbor_rep_token - 1);
  58. return;
  59. }
  60. eloop_cancel_timeout(wpas_rrm_neighbor_rep_timeout_handler, &wpa_s->rrm,
  61. NULL);
  62. if (!wpa_s->rrm.notify_neighbor_rep) {
  63. wpa_printf(MSG_ERROR, "RRM: Unexpected neighbor report");
  64. return;
  65. }
  66. /* skipping the first byte, which is only an id (dialog token) */
  67. neighbor_rep = wpabuf_alloc(report_len - 1);
  68. if (neighbor_rep == NULL)
  69. return;
  70. wpabuf_put_data(neighbor_rep, report + 1, report_len - 1);
  71. wpa_printf(MSG_DEBUG, "RRM: Notifying neighbor report (token = %d)",
  72. report[0]);
  73. wpa_s->rrm.notify_neighbor_rep(wpa_s->rrm.neighbor_rep_cb_ctx,
  74. neighbor_rep);
  75. wpa_s->rrm.notify_neighbor_rep = NULL;
  76. wpa_s->rrm.neighbor_rep_cb_ctx = NULL;
  77. }
  78. #if defined(__CYGWIN__) || defined(CONFIG_NATIVE_WINDOWS)
  79. /* Workaround different, undefined for Windows, error codes used here */
  80. #define ENOTCONN -1
  81. #define EOPNOTSUPP -1
  82. #define ECANCELED -1
  83. #endif
  84. /* Measurement Request element + Location Subject + Maximum Age subelement */
  85. #define MEASURE_REQUEST_LCI_LEN (3 + 1 + 4)
  86. /* Measurement Request element + Location Civic Request */
  87. #define MEASURE_REQUEST_CIVIC_LEN (3 + 5)
  88. /**
  89. * wpas_rrm_send_neighbor_rep_request - Request a neighbor report from our AP
  90. * @wpa_s: Pointer to wpa_supplicant
  91. * @ssid: if not null, this is sent in the request. Otherwise, no SSID IE
  92. * is sent in the request.
  93. * @lci: if set, neighbor request will include LCI request
  94. * @civic: if set, neighbor request will include civic location request
  95. * @cb: Callback function to be called once the requested report arrives, or
  96. * timed out after RRM_NEIGHBOR_REPORT_TIMEOUT seconds.
  97. * In the former case, 'neighbor_rep' is a newly allocated wpabuf, and it's
  98. * the requester's responsibility to free it.
  99. * In the latter case NULL will be sent in 'neighbor_rep'.
  100. * @cb_ctx: Context value to send the callback function
  101. * Returns: 0 in case of success, negative error code otherwise
  102. *
  103. * In case there is a previous request which has not been answered yet, the
  104. * new request fails. The caller may retry after RRM_NEIGHBOR_REPORT_TIMEOUT.
  105. * Request must contain a callback function.
  106. */
  107. int wpas_rrm_send_neighbor_rep_request(struct wpa_supplicant *wpa_s,
  108. const struct wpa_ssid_value *ssid,
  109. int lci, int civic,
  110. void (*cb)(void *ctx,
  111. struct wpabuf *neighbor_rep),
  112. void *cb_ctx)
  113. {
  114. struct wpabuf *buf;
  115. const u8 *rrm_ie;
  116. if (wpa_s->wpa_state != WPA_COMPLETED || wpa_s->current_ssid == NULL) {
  117. wpa_printf(MSG_DEBUG, "RRM: No connection, no RRM.");
  118. return -ENOTCONN;
  119. }
  120. if (!wpa_s->rrm.rrm_used) {
  121. wpa_printf(MSG_DEBUG, "RRM: No RRM in current connection.");
  122. return -EOPNOTSUPP;
  123. }
  124. rrm_ie = wpa_bss_get_ie(wpa_s->current_bss,
  125. WLAN_EID_RRM_ENABLED_CAPABILITIES);
  126. if (!rrm_ie || !(wpa_s->current_bss->caps & IEEE80211_CAP_RRM) ||
  127. !(rrm_ie[2] & WLAN_RRM_CAPS_NEIGHBOR_REPORT)) {
  128. wpa_printf(MSG_DEBUG,
  129. "RRM: No network support for Neighbor Report.");
  130. return -EOPNOTSUPP;
  131. }
  132. if (!cb) {
  133. wpa_printf(MSG_DEBUG,
  134. "RRM: Neighbor Report request must provide a callback.");
  135. return -EINVAL;
  136. }
  137. /* Refuse if there's a live request */
  138. if (wpa_s->rrm.notify_neighbor_rep) {
  139. wpa_printf(MSG_DEBUG,
  140. "RRM: Currently handling previous Neighbor Report.");
  141. return -EBUSY;
  142. }
  143. /* 3 = action category + action code + dialog token */
  144. buf = wpabuf_alloc(3 + (ssid ? 2 + ssid->ssid_len : 0) +
  145. (lci ? 2 + MEASURE_REQUEST_LCI_LEN : 0) +
  146. (civic ? 2 + MEASURE_REQUEST_CIVIC_LEN : 0));
  147. if (buf == NULL) {
  148. wpa_printf(MSG_DEBUG,
  149. "RRM: Failed to allocate Neighbor Report Request");
  150. return -ENOMEM;
  151. }
  152. wpa_printf(MSG_DEBUG, "RRM: Neighbor report request (for %s), token=%d",
  153. (ssid ? wpa_ssid_txt(ssid->ssid, ssid->ssid_len) : ""),
  154. wpa_s->rrm.next_neighbor_rep_token);
  155. wpabuf_put_u8(buf, WLAN_ACTION_RADIO_MEASUREMENT);
  156. wpabuf_put_u8(buf, WLAN_RRM_NEIGHBOR_REPORT_REQUEST);
  157. wpabuf_put_u8(buf, wpa_s->rrm.next_neighbor_rep_token);
  158. if (ssid) {
  159. wpabuf_put_u8(buf, WLAN_EID_SSID);
  160. wpabuf_put_u8(buf, ssid->ssid_len);
  161. wpabuf_put_data(buf, ssid->ssid, ssid->ssid_len);
  162. }
  163. if (lci) {
  164. /* IEEE P802.11-REVmc/D5.0 9.4.2.21 */
  165. wpabuf_put_u8(buf, WLAN_EID_MEASURE_REQUEST);
  166. wpabuf_put_u8(buf, MEASURE_REQUEST_LCI_LEN);
  167. /*
  168. * Measurement token; nonzero number that is unique among the
  169. * Measurement Request elements in a particular frame.
  170. */
  171. wpabuf_put_u8(buf, 1); /* Measurement Token */
  172. /*
  173. * Parallel, Enable, Request, and Report bits are 0, Duration is
  174. * reserved.
  175. */
  176. wpabuf_put_u8(buf, 0); /* Measurement Request Mode */
  177. wpabuf_put_u8(buf, MEASURE_TYPE_LCI); /* Measurement Type */
  178. /* IEEE P802.11-REVmc/D5.0 9.4.2.21.10 - LCI request */
  179. /* Location Subject */
  180. wpabuf_put_u8(buf, LOCATION_SUBJECT_REMOTE);
  181. /* Optional Subelements */
  182. /*
  183. * IEEE P802.11-REVmc/D5.0 Figure 9-170
  184. * The Maximum Age subelement is required, otherwise the AP can
  185. * send only data that was determined after receiving the
  186. * request. Setting it here to unlimited age.
  187. */
  188. wpabuf_put_u8(buf, LCI_REQ_SUBELEM_MAX_AGE);
  189. wpabuf_put_u8(buf, 2);
  190. wpabuf_put_le16(buf, 0xffff);
  191. }
  192. if (civic) {
  193. /* IEEE P802.11-REVmc/D5.0 9.4.2.21 */
  194. wpabuf_put_u8(buf, WLAN_EID_MEASURE_REQUEST);
  195. wpabuf_put_u8(buf, MEASURE_REQUEST_CIVIC_LEN);
  196. /*
  197. * Measurement token; nonzero number that is unique among the
  198. * Measurement Request elements in a particular frame.
  199. */
  200. wpabuf_put_u8(buf, 2); /* Measurement Token */
  201. /*
  202. * Parallel, Enable, Request, and Report bits are 0, Duration is
  203. * reserved.
  204. */
  205. wpabuf_put_u8(buf, 0); /* Measurement Request Mode */
  206. /* Measurement Type */
  207. wpabuf_put_u8(buf, MEASURE_TYPE_LOCATION_CIVIC);
  208. /* IEEE P802.11-REVmc/D5.0 9.4.2.21.14:
  209. * Location Civic request */
  210. /* Location Subject */
  211. wpabuf_put_u8(buf, LOCATION_SUBJECT_REMOTE);
  212. wpabuf_put_u8(buf, 0); /* Civic Location Type: IETF RFC 4776 */
  213. /* Location Service Interval Units: Seconds */
  214. wpabuf_put_u8(buf, 0);
  215. /* Location Service Interval: 0 - Only one report is requested
  216. */
  217. wpabuf_put_le16(buf, 0);
  218. /* No optional subelements */
  219. }
  220. wpa_s->rrm.next_neighbor_rep_token++;
  221. if (wpa_drv_send_action(wpa_s, wpa_s->assoc_freq, 0, wpa_s->bssid,
  222. wpa_s->own_addr, wpa_s->bssid,
  223. wpabuf_head(buf), wpabuf_len(buf), 0) < 0) {
  224. wpa_printf(MSG_DEBUG,
  225. "RRM: Failed to send Neighbor Report Request");
  226. wpabuf_free(buf);
  227. return -ECANCELED;
  228. }
  229. wpa_s->rrm.neighbor_rep_cb_ctx = cb_ctx;
  230. wpa_s->rrm.notify_neighbor_rep = cb;
  231. eloop_register_timeout(RRM_NEIGHBOR_REPORT_TIMEOUT, 0,
  232. wpas_rrm_neighbor_rep_timeout_handler,
  233. &wpa_s->rrm, NULL);
  234. wpabuf_free(buf);
  235. return 0;
  236. }
  237. static int wpas_rrm_report_elem(struct wpabuf *buf, u8 token, u8 mode, u8 type,
  238. const u8 *data, size_t data_len)
  239. {
  240. if (wpabuf_tailroom(buf) < 5 + data_len)
  241. return -1;
  242. wpabuf_put_u8(buf, WLAN_EID_MEASURE_REPORT);
  243. wpabuf_put_u8(buf, 3 + data_len);
  244. wpabuf_put_u8(buf, token);
  245. wpabuf_put_u8(buf, mode);
  246. wpabuf_put_u8(buf, type);
  247. if (data_len)
  248. wpabuf_put_data(buf, data, data_len);
  249. return 0;
  250. }
  251. static struct wpabuf * wpas_rrm_build_lci_report(struct wpa_supplicant *wpa_s,
  252. const u8 *request, size_t len,
  253. struct wpabuf *report)
  254. {
  255. u8 token, type, subject;
  256. u16 max_age = 0;
  257. struct os_reltime t, diff;
  258. unsigned long diff_l;
  259. u8 *ptoken;
  260. const u8 *subelem;
  261. if (!wpa_s->lci || len < 3 + 4)
  262. return report;
  263. token = *request++;
  264. /* Measurement request mode isn't used */
  265. request++;
  266. type = *request++;
  267. subject = *request++;
  268. len -= 4;
  269. wpa_printf(MSG_DEBUG,
  270. "Measurement request token %u type %u location subject %u",
  271. token, type, subject);
  272. if (type != MEASURE_TYPE_LCI || subject != LOCATION_SUBJECT_REMOTE) {
  273. wpa_printf(MSG_INFO,
  274. "Not building LCI report - bad type or location subject");
  275. return report;
  276. }
  277. /* Subelements are formatted exactly like elements */
  278. subelem = get_ie(request, len, LCI_REQ_SUBELEM_MAX_AGE);
  279. if (subelem && subelem[1] == 2)
  280. max_age = WPA_GET_LE16(subelem + 2);
  281. if (os_get_reltime(&t))
  282. return report;
  283. os_reltime_sub(&t, &wpa_s->lci_time, &diff);
  284. /* LCI age is calculated in 10th of a second units. */
  285. diff_l = diff.sec * 10 + diff.usec / 100000;
  286. if (max_age != 0xffff && max_age < diff_l)
  287. return report;
  288. if (wpabuf_resize(&report, 2 + wpabuf_len(wpa_s->lci)))
  289. return report;
  290. wpabuf_put_u8(report, WLAN_EID_MEASURE_REPORT);
  291. wpabuf_put_u8(report, wpabuf_len(wpa_s->lci));
  292. /* We'll override user's measurement token */
  293. ptoken = wpabuf_put(report, 0);
  294. wpabuf_put_buf(report, wpa_s->lci);
  295. *ptoken = token;
  296. return report;
  297. }
  298. static void wpas_rrm_send_msr_report_mpdu(struct wpa_supplicant *wpa_s,
  299. const u8 *data, size_t len)
  300. {
  301. struct wpabuf *report = wpabuf_alloc(len + 3);
  302. if (!report)
  303. return;
  304. wpabuf_put_u8(report, WLAN_ACTION_RADIO_MEASUREMENT);
  305. wpabuf_put_u8(report, WLAN_RRM_RADIO_MEASUREMENT_REPORT);
  306. wpabuf_put_u8(report, wpa_s->rrm.token);
  307. wpabuf_put_data(report, data, len);
  308. if (wpa_drv_send_action(wpa_s, wpa_s->assoc_freq, 0, wpa_s->bssid,
  309. wpa_s->own_addr, wpa_s->bssid,
  310. wpabuf_head(report), wpabuf_len(report), 0)) {
  311. wpa_printf(MSG_ERROR,
  312. "RRM: Radio measurement report failed: Sending Action frame failed");
  313. }
  314. wpabuf_free(report);
  315. }
  316. static void wpas_rrm_send_msr_report(struct wpa_supplicant *wpa_s,
  317. struct wpabuf *buf)
  318. {
  319. int len = wpabuf_len(buf);
  320. const u8 *pos = wpabuf_head_u8(buf), *next = pos;
  321. #define MPDU_REPORT_LEN (int) (IEEE80211_MAX_MMPDU_SIZE - IEEE80211_HDRLEN - 3)
  322. while (len) {
  323. int send_len = (len > MPDU_REPORT_LEN) ? next - pos : len;
  324. if (send_len == len ||
  325. (send_len + next[1] + 2) > MPDU_REPORT_LEN) {
  326. wpas_rrm_send_msr_report_mpdu(wpa_s, pos, send_len);
  327. len -= send_len;
  328. pos = next;
  329. }
  330. next += next[1] + 2;
  331. }
  332. #undef MPDU_REPORT_LEN
  333. }
  334. static int
  335. wpas_rrm_handle_msr_req_element(
  336. struct wpa_supplicant *wpa_s,
  337. const struct rrm_measurement_request_element *req,
  338. struct wpabuf **buf)
  339. {
  340. wpa_printf(MSG_DEBUG, "Measurement request type %d token %d",
  341. req->type, req->token);
  342. if (req->mode & MEASUREMENT_REQUEST_MODE_ENABLE) {
  343. /* Enable bit is not supported for now */
  344. wpa_printf(MSG_DEBUG, "RRM: Enable bit not supported, ignore");
  345. return 0;
  346. }
  347. if ((req->mode & MEASUREMENT_REQUEST_MODE_PARALLEL) &&
  348. req->type > MEASURE_TYPE_RPI_HIST) {
  349. /* Parallel measurements are not supported for now */
  350. wpa_printf(MSG_DEBUG,
  351. "RRM: Parallel measurements are not supported, reject");
  352. goto reject;
  353. }
  354. switch (req->type) {
  355. case MEASURE_TYPE_LCI:
  356. *buf = wpas_rrm_build_lci_report(wpa_s, &req->token, req->len,
  357. *buf);
  358. return 0;
  359. default:
  360. wpa_printf(MSG_INFO,
  361. "RRM: Unsupported radio measurement type %u",
  362. req->type);
  363. break;
  364. }
  365. reject:
  366. if (wpabuf_resize(buf, sizeof(struct rrm_measurement_report_element))) {
  367. wpa_printf(MSG_DEBUG, "RRM: Memory allocation failed");
  368. return -1;
  369. }
  370. if (wpas_rrm_report_elem(*buf, req->token,
  371. MEASUREMENT_REPORT_MODE_REJECT_INCAPABLE,
  372. req->type, NULL, 0) < 0) {
  373. wpa_printf(MSG_DEBUG, "RRM: Failed to add report element");
  374. return -1;
  375. }
  376. return 0;
  377. }
  378. static struct wpabuf *
  379. wpas_rrm_process_msr_req_elems(struct wpa_supplicant *wpa_s, const u8 *pos,
  380. size_t len)
  381. {
  382. struct wpabuf *buf = NULL;
  383. while (len) {
  384. const struct rrm_measurement_request_element *req;
  385. int res;
  386. if (len < 2) {
  387. wpa_printf(MSG_DEBUG, "RRM: Truncated element");
  388. goto out;
  389. }
  390. req = (const struct rrm_measurement_request_element *) pos;
  391. if (req->eid != WLAN_EID_MEASURE_REQUEST) {
  392. wpa_printf(MSG_DEBUG,
  393. "RRM: Expected Measurement Request element, but EID is %u",
  394. req->eid);
  395. goto out;
  396. }
  397. if (req->len < 3) {
  398. wpa_printf(MSG_DEBUG, "RRM: Element length too short");
  399. goto out;
  400. }
  401. if (req->len > len - 2) {
  402. wpa_printf(MSG_DEBUG, "RRM: Element length too long");
  403. goto out;
  404. }
  405. res = wpas_rrm_handle_msr_req_element(wpa_s, req, &buf);
  406. if (res < 0)
  407. goto out;
  408. pos += req->len + 2;
  409. len -= req->len + 2;
  410. }
  411. return buf;
  412. out:
  413. wpabuf_free(buf);
  414. return NULL;
  415. }
  416. void wpas_rrm_handle_radio_measurement_request(struct wpa_supplicant *wpa_s,
  417. const u8 *src,
  418. const u8 *frame, size_t len)
  419. {
  420. struct wpabuf *report;
  421. if (wpa_s->wpa_state != WPA_COMPLETED) {
  422. wpa_printf(MSG_INFO,
  423. "RRM: Ignoring radio measurement request: Not associated");
  424. return;
  425. }
  426. if (!wpa_s->rrm.rrm_used) {
  427. wpa_printf(MSG_INFO,
  428. "RRM: Ignoring radio measurement request: Not RRM network");
  429. return;
  430. }
  431. if (len < 3) {
  432. wpa_printf(MSG_INFO,
  433. "RRM: Ignoring too short radio measurement request");
  434. return;
  435. }
  436. wpa_s->rrm.token = *frame;
  437. /* Number of repetitions is not supported */
  438. report = wpas_rrm_process_msr_req_elems(wpa_s, frame + 3, len - 3);
  439. if (!report)
  440. return;
  441. wpas_rrm_send_msr_report(wpa_s, report);
  442. wpabuf_free(report);
  443. }
  444. void wpas_rrm_handle_link_measurement_request(struct wpa_supplicant *wpa_s,
  445. const u8 *src,
  446. const u8 *frame, size_t len,
  447. int rssi)
  448. {
  449. struct wpabuf *buf;
  450. const struct rrm_link_measurement_request *req;
  451. struct rrm_link_measurement_report report;
  452. if (wpa_s->wpa_state != WPA_COMPLETED) {
  453. wpa_printf(MSG_INFO,
  454. "RRM: Ignoring link measurement request. Not associated");
  455. return;
  456. }
  457. if (!wpa_s->rrm.rrm_used) {
  458. wpa_printf(MSG_INFO,
  459. "RRM: Ignoring link measurement request. Not RRM network");
  460. return;
  461. }
  462. if (!(wpa_s->drv_rrm_flags & WPA_DRIVER_FLAGS_TX_POWER_INSERTION)) {
  463. wpa_printf(MSG_INFO,
  464. "RRM: Measurement report failed. TX power insertion not supported");
  465. return;
  466. }
  467. req = (const struct rrm_link_measurement_request *) frame;
  468. if (len < sizeof(*req)) {
  469. wpa_printf(MSG_INFO,
  470. "RRM: Link measurement report failed. Request too short");
  471. return;
  472. }
  473. os_memset(&report, 0, sizeof(report));
  474. report.tpc.eid = WLAN_EID_TPC_REPORT;
  475. report.tpc.len = 2;
  476. report.rsni = 255; /* 255 indicates that RSNI is not available */
  477. report.dialog_token = req->dialog_token;
  478. /*
  479. * It's possible to estimate RCPI based on RSSI in dBm. This
  480. * calculation will not reflect the correct value for high rates,
  481. * but it's good enough for Action frames which are transmitted
  482. * with up to 24 Mbps rates.
  483. */
  484. if (!rssi)
  485. report.rcpi = 255; /* not available */
  486. else if (rssi < -110)
  487. report.rcpi = 0;
  488. else if (rssi > 0)
  489. report.rcpi = 220;
  490. else
  491. report.rcpi = (rssi + 110) * 2;
  492. /* action_category + action_code */
  493. buf = wpabuf_alloc(2 + sizeof(report));
  494. if (buf == NULL) {
  495. wpa_printf(MSG_ERROR,
  496. "RRM: Link measurement report failed. Buffer allocation failed");
  497. return;
  498. }
  499. wpabuf_put_u8(buf, WLAN_ACTION_RADIO_MEASUREMENT);
  500. wpabuf_put_u8(buf, WLAN_RRM_LINK_MEASUREMENT_REPORT);
  501. wpabuf_put_data(buf, &report, sizeof(report));
  502. wpa_hexdump(MSG_DEBUG, "RRM: Link measurement report:",
  503. wpabuf_head(buf), wpabuf_len(buf));
  504. if (wpa_drv_send_action(wpa_s, wpa_s->assoc_freq, 0, src,
  505. wpa_s->own_addr, wpa_s->bssid,
  506. wpabuf_head(buf), wpabuf_len(buf), 0)) {
  507. wpa_printf(MSG_ERROR,
  508. "RRM: Link measurement report failed. Send action failed");
  509. }
  510. wpabuf_free(buf);
  511. }