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 int
  252. wpas_rrm_build_lci_report(struct wpa_supplicant *wpa_s,
  253. const struct rrm_measurement_request_element *req,
  254. struct wpabuf **buf)
  255. {
  256. u8 subject;
  257. u16 max_age = 0;
  258. struct os_reltime t, diff;
  259. unsigned long diff_l;
  260. const u8 *subelem;
  261. const u8 *request = req->variable;
  262. size_t len = req->len - 3;
  263. if (len < 4)
  264. return -1;
  265. if (!wpa_s->lci)
  266. goto reject;
  267. subject = *request++;
  268. len--;
  269. wpa_printf(MSG_DEBUG, "Measurement request location subject=%u",
  270. subject);
  271. if (subject != LOCATION_SUBJECT_REMOTE) {
  272. wpa_printf(MSG_INFO,
  273. "Not building LCI report - bad location subject");
  274. return 0;
  275. }
  276. /* Subelements are formatted exactly like elements */
  277. subelem = get_ie(request, len, LCI_REQ_SUBELEM_MAX_AGE);
  278. if (subelem && subelem[1] == 2)
  279. max_age = WPA_GET_LE16(subelem + 2);
  280. if (os_get_reltime(&t))
  281. goto reject;
  282. os_reltime_sub(&t, &wpa_s->lci_time, &diff);
  283. /* LCI age is calculated in 10th of a second units. */
  284. diff_l = diff.sec * 10 + diff.usec / 100000;
  285. if (max_age != 0xffff && max_age < diff_l)
  286. goto reject;
  287. if (wpabuf_resize(buf, 5 + wpabuf_len(wpa_s->lci)))
  288. return -1;
  289. if (wpas_rrm_report_elem(*buf, req->token,
  290. MEASUREMENT_REPORT_MODE_ACCEPT, req->type,
  291. wpabuf_head_u8(wpa_s->lci),
  292. wpabuf_len(wpa_s->lci)) < 0) {
  293. wpa_printf(MSG_DEBUG, "Failed to add LCI report element");
  294. return -1;
  295. }
  296. return 0;
  297. reject:
  298. if (wpabuf_resize(buf, sizeof(struct rrm_measurement_report_element))) {
  299. wpa_printf(MSG_DEBUG, "RRM: Memory allocation failed");
  300. return -1;
  301. }
  302. if (wpas_rrm_report_elem(*buf, req->token,
  303. MEASUREMENT_REPORT_MODE_REJECT_INCAPABLE,
  304. req->type, NULL, 0) < 0) {
  305. wpa_printf(MSG_DEBUG, "RRM: Failed to add report element");
  306. return -1;
  307. }
  308. return 0;
  309. }
  310. static void wpas_rrm_send_msr_report_mpdu(struct wpa_supplicant *wpa_s,
  311. const u8 *data, size_t len)
  312. {
  313. struct wpabuf *report = wpabuf_alloc(len + 3);
  314. if (!report)
  315. return;
  316. wpabuf_put_u8(report, WLAN_ACTION_RADIO_MEASUREMENT);
  317. wpabuf_put_u8(report, WLAN_RRM_RADIO_MEASUREMENT_REPORT);
  318. wpabuf_put_u8(report, wpa_s->rrm.token);
  319. wpabuf_put_data(report, data, len);
  320. if (wpa_drv_send_action(wpa_s, wpa_s->assoc_freq, 0, wpa_s->bssid,
  321. wpa_s->own_addr, wpa_s->bssid,
  322. wpabuf_head(report), wpabuf_len(report), 0)) {
  323. wpa_printf(MSG_ERROR,
  324. "RRM: Radio measurement report failed: Sending Action frame failed");
  325. }
  326. wpabuf_free(report);
  327. }
  328. static void wpas_rrm_send_msr_report(struct wpa_supplicant *wpa_s,
  329. struct wpabuf *buf)
  330. {
  331. int len = wpabuf_len(buf);
  332. const u8 *pos = wpabuf_head_u8(buf), *next = pos;
  333. #define MPDU_REPORT_LEN (int) (IEEE80211_MAX_MMPDU_SIZE - IEEE80211_HDRLEN - 3)
  334. while (len) {
  335. int send_len = (len > MPDU_REPORT_LEN) ? next - pos : len;
  336. if (send_len == len ||
  337. (send_len + next[1] + 2) > MPDU_REPORT_LEN) {
  338. wpas_rrm_send_msr_report_mpdu(wpa_s, pos, send_len);
  339. len -= send_len;
  340. pos = next;
  341. }
  342. next += next[1] + 2;
  343. }
  344. #undef MPDU_REPORT_LEN
  345. }
  346. static int
  347. wpas_rrm_handle_msr_req_element(
  348. struct wpa_supplicant *wpa_s,
  349. const struct rrm_measurement_request_element *req,
  350. struct wpabuf **buf)
  351. {
  352. wpa_printf(MSG_DEBUG, "Measurement request type %d token %d",
  353. req->type, req->token);
  354. if (req->mode & MEASUREMENT_REQUEST_MODE_ENABLE) {
  355. /* Enable bit is not supported for now */
  356. wpa_printf(MSG_DEBUG, "RRM: Enable bit not supported, ignore");
  357. return 0;
  358. }
  359. if ((req->mode & MEASUREMENT_REQUEST_MODE_PARALLEL) &&
  360. req->type > MEASURE_TYPE_RPI_HIST) {
  361. /* Parallel measurements are not supported for now */
  362. wpa_printf(MSG_DEBUG,
  363. "RRM: Parallel measurements are not supported, reject");
  364. goto reject;
  365. }
  366. switch (req->type) {
  367. case MEASURE_TYPE_LCI:
  368. return wpas_rrm_build_lci_report(wpa_s, req, buf);
  369. default:
  370. wpa_printf(MSG_INFO,
  371. "RRM: Unsupported radio measurement type %u",
  372. req->type);
  373. break;
  374. }
  375. reject:
  376. if (wpabuf_resize(buf, sizeof(struct rrm_measurement_report_element))) {
  377. wpa_printf(MSG_DEBUG, "RRM: Memory allocation failed");
  378. return -1;
  379. }
  380. if (wpas_rrm_report_elem(*buf, req->token,
  381. MEASUREMENT_REPORT_MODE_REJECT_INCAPABLE,
  382. req->type, NULL, 0) < 0) {
  383. wpa_printf(MSG_DEBUG, "RRM: Failed to add report element");
  384. return -1;
  385. }
  386. return 0;
  387. }
  388. static struct wpabuf *
  389. wpas_rrm_process_msr_req_elems(struct wpa_supplicant *wpa_s, const u8 *pos,
  390. size_t len)
  391. {
  392. struct wpabuf *buf = NULL;
  393. while (len) {
  394. const struct rrm_measurement_request_element *req;
  395. int res;
  396. if (len < 2) {
  397. wpa_printf(MSG_DEBUG, "RRM: Truncated element");
  398. goto out;
  399. }
  400. req = (const struct rrm_measurement_request_element *) pos;
  401. if (req->eid != WLAN_EID_MEASURE_REQUEST) {
  402. wpa_printf(MSG_DEBUG,
  403. "RRM: Expected Measurement Request element, but EID is %u",
  404. req->eid);
  405. goto out;
  406. }
  407. if (req->len < 3) {
  408. wpa_printf(MSG_DEBUG, "RRM: Element length too short");
  409. goto out;
  410. }
  411. if (req->len > len - 2) {
  412. wpa_printf(MSG_DEBUG, "RRM: Element length too long");
  413. goto out;
  414. }
  415. res = wpas_rrm_handle_msr_req_element(wpa_s, req, &buf);
  416. if (res < 0)
  417. goto out;
  418. pos += req->len + 2;
  419. len -= req->len + 2;
  420. }
  421. return buf;
  422. out:
  423. wpabuf_free(buf);
  424. return NULL;
  425. }
  426. void wpas_rrm_handle_radio_measurement_request(struct wpa_supplicant *wpa_s,
  427. const u8 *src,
  428. const u8 *frame, size_t len)
  429. {
  430. struct wpabuf *report;
  431. if (wpa_s->wpa_state != WPA_COMPLETED) {
  432. wpa_printf(MSG_INFO,
  433. "RRM: Ignoring radio measurement request: Not associated");
  434. return;
  435. }
  436. if (!wpa_s->rrm.rrm_used) {
  437. wpa_printf(MSG_INFO,
  438. "RRM: Ignoring radio measurement request: Not RRM network");
  439. return;
  440. }
  441. if (len < 3) {
  442. wpa_printf(MSG_INFO,
  443. "RRM: Ignoring too short radio measurement request");
  444. return;
  445. }
  446. wpa_s->rrm.token = *frame;
  447. /* Number of repetitions is not supported */
  448. report = wpas_rrm_process_msr_req_elems(wpa_s, frame + 3, len - 3);
  449. if (!report)
  450. return;
  451. wpas_rrm_send_msr_report(wpa_s, report);
  452. wpabuf_free(report);
  453. }
  454. void wpas_rrm_handle_link_measurement_request(struct wpa_supplicant *wpa_s,
  455. const u8 *src,
  456. const u8 *frame, size_t len,
  457. int rssi)
  458. {
  459. struct wpabuf *buf;
  460. const struct rrm_link_measurement_request *req;
  461. struct rrm_link_measurement_report report;
  462. if (wpa_s->wpa_state != WPA_COMPLETED) {
  463. wpa_printf(MSG_INFO,
  464. "RRM: Ignoring link measurement request. Not associated");
  465. return;
  466. }
  467. if (!wpa_s->rrm.rrm_used) {
  468. wpa_printf(MSG_INFO,
  469. "RRM: Ignoring link measurement request. Not RRM network");
  470. return;
  471. }
  472. if (!(wpa_s->drv_rrm_flags & WPA_DRIVER_FLAGS_TX_POWER_INSERTION)) {
  473. wpa_printf(MSG_INFO,
  474. "RRM: Measurement report failed. TX power insertion not supported");
  475. return;
  476. }
  477. req = (const struct rrm_link_measurement_request *) frame;
  478. if (len < sizeof(*req)) {
  479. wpa_printf(MSG_INFO,
  480. "RRM: Link measurement report failed. Request too short");
  481. return;
  482. }
  483. os_memset(&report, 0, sizeof(report));
  484. report.tpc.eid = WLAN_EID_TPC_REPORT;
  485. report.tpc.len = 2;
  486. report.rsni = 255; /* 255 indicates that RSNI is not available */
  487. report.dialog_token = req->dialog_token;
  488. report.rcpi = rssi_to_rcpi(rssi);
  489. /* action_category + action_code */
  490. buf = wpabuf_alloc(2 + sizeof(report));
  491. if (buf == NULL) {
  492. wpa_printf(MSG_ERROR,
  493. "RRM: Link measurement report failed. Buffer allocation failed");
  494. return;
  495. }
  496. wpabuf_put_u8(buf, WLAN_ACTION_RADIO_MEASUREMENT);
  497. wpabuf_put_u8(buf, WLAN_RRM_LINK_MEASUREMENT_REPORT);
  498. wpabuf_put_data(buf, &report, sizeof(report));
  499. wpa_hexdump(MSG_DEBUG, "RRM: Link measurement report:",
  500. wpabuf_head(buf), wpabuf_len(buf));
  501. if (wpa_drv_send_action(wpa_s, wpa_s->assoc_freq, 0, src,
  502. wpa_s->own_addr, wpa_s->bssid,
  503. wpabuf_head(buf), wpabuf_len(buf), 0)) {
  504. wpa_printf(MSG_ERROR,
  505. "RRM: Link measurement report failed. Send action failed");
  506. }
  507. wpabuf_free(buf);
  508. }