peerstats.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468
  1. /*
  2. * Copyright (c) 2019 The Linux Foundation. All rights reserved.
  3. *
  4. * Permission to use, copy, modify, and/or distribute this software for
  5. * any purpose with or without fee is hereby granted, provided that the
  6. * above copyright notice and this permission notice appear in all
  7. * copies.
  8. *
  9. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
  10. * WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
  11. * WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
  12. * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
  13. * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
  14. * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
  15. * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
  16. * PERFORMANCE OF THIS SOFTWARE.
  17. */
  18. /**
  19. * file: peer rate statitistics application
  20. * This file provides framework to display peer rate statistics
  21. */
  22. #include <netinet/in.h>
  23. #include <arpa/inet.h>
  24. #include <qcatools_lib.h>
  25. #include <dp_rate_stats_pub.h>
  26. #ifndef min
  27. #define min(x, y) ((x) < (y) ? (x) : (y))
  28. #else
  29. #error confilicting defs of min
  30. #endif
  31. #define PRINT(fmt, ...) \
  32. do { \
  33. printf(fmt, ##__VA_ARGS__); \
  34. printf("\n"); \
  35. } while (0)
  36. char interface[IFNAMSIZ];
  37. static void dp_peer_rx_rate_stats_print(uint8_t *peer_mac,
  38. uint64_t peer_cookie,
  39. void *buffer,
  40. uint32_t buffer_len)
  41. {
  42. int i = 0;
  43. struct wlan_rx_rate_stats *rx_stats;
  44. uint8_t is_lithium;
  45. uint8_t chain, max_chain, bw, max_bw;
  46. struct wlan_rx_rate_stats *tmp_rx_stats;;
  47. rx_stats = tmp_rx_stats = (struct wlan_rx_rate_stats *)buffer;
  48. PRINT("\n......................................");
  49. PRINT("......................................");
  50. PRINT("PEER %02hhx:%02hhx:%02hhx:%02hhx:%02hhx:%02hhx\n",
  51. peer_mac[0],
  52. peer_mac[1],
  53. peer_mac[2],
  54. peer_mac[3],
  55. peer_mac[4],
  56. peer_mac[5]);
  57. PRINT("\tpeer cookie: %016lx\n", (peer_cookie & 0xFFFFFFFF00000000)
  58. >> WLANSTATS_PEER_COOKIE_LSB);
  59. is_lithium = (peer_cookie & WLANSTATS_COOKIE_PLATFORM_OFFSET)
  60. >> WLANSTATS_PEER_COOKIE_LSB;
  61. if (is_lithium) {
  62. max_chain = 8;
  63. max_bw = 8;
  64. } else {
  65. max_chain = 4;
  66. max_bw = 4;
  67. }
  68. PRINT("\n..............................................");
  69. PRINT("................................");
  70. PRINT("................................................");
  71. PRINT(".................................\n");
  72. PRINT("\tRx statistics:");
  73. PRINT(" %10s | %10s | %10s | %10s | %10s | %10s|",
  74. "rate",
  75. "rix",
  76. "bytes",
  77. "msdus",
  78. "mpdus",
  79. "ppdus");
  80. PRINT(" %10s | %10s | %10s |",
  81. "retries",
  82. "sgi",
  83. "rssi\n");
  84. for (i = 0; i < WLANSTATS_CACHE_SIZE; i++) {
  85. if (rx_stats->rix != INVALID_CACHE_IDX) {
  86. PRINT(" %10u | %10u | %10u | %10u | %10u | %10u |",
  87. rx_stats->rate,
  88. GET_DP_PEER_STATS_RIX(rx_stats->rix),
  89. rx_stats->num_bytes,
  90. rx_stats->num_msdus,
  91. rx_stats->num_mpdus,
  92. rx_stats->num_ppdus);
  93. PRINT(" %10u | %10u | %10lu |\n",
  94. rx_stats->num_retries,
  95. rx_stats->num_sgi,
  96. rx_stats->avg_rssi);
  97. }
  98. rx_stats = rx_stats + 1;
  99. }
  100. if (is_lithium) {
  101. PRINT("\n %10s | %10s | %10s | %10s | %10s |",
  102. "rate",
  103. "rssi 1 p20",
  104. "rssi 1 e20",
  105. "rssi 1 e40 low20",
  106. "rssi 1 e40 high20");
  107. PRINT("\n | %10s | %10s | %10s | %10s |",
  108. "rssi 1 ext80 low20",
  109. "rssi 1 ext80 low_high20",
  110. "rssi 1 ext80 high_low20",
  111. "rssi 1 ext80 high20");
  112. PRINT("\n | %10s | %10s | %10s | %10s |",
  113. "rssi 2 p20",
  114. "rssi 2 e20",
  115. "rssi 2 e40 low20",
  116. "rssi 2 e40 high20");
  117. PRINT("\n | %10s | %10s | %10s | %10s |",
  118. "rssi 2 ext80 low20",
  119. "rssi 2 ext80 low_high20",
  120. "rssi 2 ext80 high_low20",
  121. "rssi 2 ext80 high20");
  122. PRINT("\n | %10s | %10s | %10s | %10s |",
  123. "rssi 3 p20",
  124. "rssi 3 e20",
  125. "rssi 3 e40 low20",
  126. "rssi 3 e40 high20");
  127. PRINT("\n | %10s | %10s | %10s | %10s |",
  128. "rssi 3 ext80 low20",
  129. "rssi 3 ext80 low_high20",
  130. "rssi 3 ext80 high_low20",
  131. "rssi 3 ext80 high20");
  132. PRINT("\n | %10s | %10s | %10s | %10s |",
  133. "rssi 4 p20",
  134. "rssi 4 e20",
  135. "rssi 4 e40 low20",
  136. "rssi 4 e40 high20");
  137. PRINT("\n | %10s | %10s | %10s | %10s |",
  138. "rssi 4 ext80 low20",
  139. "rssi 4 ext80 low_high20",
  140. "rssi 4 ext80 high_low20",
  141. "rssi 4 ext80 high20");
  142. PRINT("\n | %10s | %10s | %10s | %10s |",
  143. "rssi 5 p20",
  144. "rssi 5 e20",
  145. "rssi 5 e40 low20",
  146. "rssi 5 e40 high20");
  147. PRINT("\n | %10s | %10s | %10s | %10s |",
  148. "rssi 5 ext80 low20",
  149. "rssi 5 ext80 low_high20",
  150. "rssi 5 ext80 high_low20",
  151. "rssi 5 ext80 high20");
  152. PRINT("\n | %10s | %10s | %10s | %10s |",
  153. "rssi 6 p20",
  154. "rssi 6 e20",
  155. "rssi 6 e40 low20",
  156. "rssi 6 e40 high20");
  157. PRINT("\n | %10s | %10s | %10s | %10s |",
  158. "rssi 6 ext80 low20",
  159. "rssi 6 ext80 low_high20",
  160. "rssi 6 ext80 high_low20",
  161. "rssi 6 ext80 high20");
  162. PRINT("\n | %10s | %10s | %10s | %10s |",
  163. "rssi 7 p20",
  164. "rssi 7 e20",
  165. "rssi 7 e40 low20",
  166. "rssi 7 e40 high20");
  167. PRINT("\n | %10s | %10s | %10s | %10s |",
  168. "rssi 7 ext80 low20",
  169. "rssi 7 ext80 low_high20",
  170. "rssi 7 ext80 high_low20",
  171. "rssi 7 ext80 high20");
  172. PRINT("\n | %10s | %10s | %10s | %10s |",
  173. "rssi 8 p20",
  174. "rssi 8 e20",
  175. "rssi 8 e40 low20",
  176. "rssi 8 e40 high20");
  177. PRINT("\n | %10s | %10s | %10s | %10s |\n\n\n",
  178. "rssi 8 ext80 low20",
  179. "rssi 8 ext80 low_high20",
  180. "rssi 8 ext80 high_low20",
  181. "rssi 8 ext80 high20");
  182. } else {
  183. PRINT("\n %10s | %10s | %10s | %10s | %10s |",
  184. "rate",
  185. "rssi 1 p20",
  186. "rssi 1 e20",
  187. "rssi 1 e40",
  188. "rssi 1 e80");
  189. PRINT(" | %10s | %10s | %10s | %10s |",
  190. "rssi 2 p20",
  191. "rssi 2 e20",
  192. "rssi 2 e40",
  193. "rssi 2 e80");
  194. PRINT(" | %10s | %10s | %10s | %10s |",
  195. "rssi 3 p20",
  196. "rssi 3 e20",
  197. "rssi 3 e40",
  198. "rssi 3 e80");
  199. PRINT(" | %10s | %10s | %10s | %10s |\n\n\n",
  200. "rssi 4 p20",
  201. "rssi 4 e20",
  202. "rssi 4 e40",
  203. "rssi 4 e80");
  204. }
  205. for (i = 0; i < WLANSTATS_CACHE_SIZE; i++) {
  206. if (tmp_rx_stats->rix != INVALID_CACHE_IDX) {
  207. printf(" %10u |", tmp_rx_stats->rate);
  208. for (chain = 0; chain < max_chain; chain++) {
  209. for (bw = 0; bw < max_bw; bw++) {
  210. printf(" %10lu |",
  211. tmp_rx_stats->avg_rssi_ant[chain][bw]);
  212. }
  213. printf(" \n\t ");
  214. }
  215. PRINT(" ");
  216. }
  217. tmp_rx_stats = tmp_rx_stats + 1;
  218. }
  219. }
  220. static void
  221. dp_peer_tx_sojourn_stats_print(uint8_t *peer_mac,
  222. uint64_t peer_cookie,
  223. struct wlan_tx_sojourn_stats *sojourn_stats)
  224. {
  225. uint8_t tid;
  226. PRINT("\n..........................................");
  227. PRINT("....................................");
  228. PRINT("....................................");
  229. PRINT(".........................................\n");
  230. PRINT("PEER %02hhx:%02hhx:%02hhx:%02hhx%02hhx:%02hhx\n",
  231. peer_mac[0],
  232. peer_mac[1],
  233. peer_mac[2],
  234. peer_mac[3],
  235. peer_mac[4],
  236. peer_mac[5]);
  237. PRINT("\tPEER Cookie: %016lx\n", peer_cookie);
  238. PRINT("\n...........................................");
  239. PRINT("...................................");
  240. PRINT("..................................");
  241. PRINT("............................................");
  242. PRINT("\n\tSojourn statistics:\n");
  243. PRINT("\t\t%10s %10s %20s %20s\n", "tid", "ave", "sum", "num");
  244. for (tid = 0; tid < WLAN_DATA_TID_MAX; tid++) {
  245. /* change sum_sojourn_msdu data type to u64 */
  246. PRINT("\t\t%10d %10lu %20u %20u\n",
  247. tid,
  248. sojourn_stats->avg_sojourn_msdu[tid],
  249. sojourn_stats->sum_sojourn_msdu[tid],
  250. sojourn_stats->num_msdus[tid]);
  251. }
  252. PRINT("sizeof(avg): %lu", sizeof(sojourn_stats->avg_sojourn_msdu[tid]));
  253. PRINT("\n...........................................");
  254. PRINT("...................................");
  255. PRINT("...................................");
  256. PRINT("...........................................\n");
  257. }
  258. static void dp_peer_tx_rate_stats_print(uint8_t *peer_mac,
  259. uint64_t peer_cookie,
  260. void *buffer,
  261. uint32_t buffer_len)
  262. {
  263. int i = 0;
  264. struct wlan_tx_rate_stats *tx_stats;
  265. struct wlan_tx_sojourn_stats *sojourn_stats;
  266. if (buffer_len < (WLANSTATS_CACHE_SIZE *
  267. sizeof(struct wlan_tx_rate_stats))
  268. + sizeof(struct wlan_tx_sojourn_stats)) {
  269. PRINT("invalid buffer len, return");
  270. return;
  271. }
  272. tx_stats = (struct wlan_tx_rate_stats *)buffer;
  273. PRINT("\n...........................................");
  274. PRINT("...................................");
  275. PRINT("...................................");
  276. PRINT("...........................................\n");
  277. PRINT("PEER %02hhx:%02hhx:%02hhx:%02hhx%02hhx:%02hhx\n\n",
  278. peer_mac[0],
  279. peer_mac[1],
  280. peer_mac[2],
  281. peer_mac[3],
  282. peer_mac[4],
  283. peer_mac[5]);
  284. PRINT("\tPEER Cookie: %016lx", peer_cookie);
  285. PRINT("\n...........................................");
  286. PRINT("...................................");
  287. PRINT("...................................");
  288. PRINT("...........................................\n");
  289. PRINT("\tTx statistics:\n");
  290. PRINT("\t\t%10s | %10s | %10s | %10s | %10s",
  291. "rate",
  292. "rix",
  293. "attempts",
  294. "success",
  295. "ppdus");
  296. PRINT(" %10s | %10s | %10s |",
  297. "msdus",
  298. "bytes",
  299. "retries\n");
  300. for (i = 0; i < WLANSTATS_CACHE_SIZE; i++) {
  301. if (tx_stats->rix != INVALID_CACHE_IDX) {
  302. PRINT("\t\t%10u | %10u | %10u | %10u | %10u\n",
  303. tx_stats->rate,
  304. GET_DP_PEER_STATS_RIX(tx_stats->rix),
  305. tx_stats->mpdu_attempts,
  306. tx_stats->mpdu_success,
  307. tx_stats->num_ppdus);
  308. PRINT(" %10u | %10u | %10lu |\n",
  309. tx_stats->num_msdus,
  310. tx_stats->num_bytes,
  311. tx_stats->num_retries);
  312. }
  313. tx_stats = tx_stats + 1;
  314. }
  315. sojourn_stats = (struct wlan_tx_sojourn_stats *)((uint8_t *)buffer
  316. + (WLANSTATS_CACHE_SIZE *
  317. sizeof(struct wlan_tx_rate_stats)));
  318. dp_peer_tx_sojourn_stats_print(peer_mac, peer_cookie, sojourn_stats);
  319. return;
  320. }
  321. static void dp_peer_stats_handler(uint32_t cache_type,
  322. uint8_t *peer_mac,
  323. uint64_t peer_cookie,
  324. void *buffer,
  325. uint32_t buffer_len)
  326. {
  327. switch (cache_type) {
  328. case DP_PEER_RX_RATE_STATS:
  329. dp_peer_rx_rate_stats_print(peer_mac, peer_cookie,
  330. buffer, buffer_len);
  331. break;
  332. case DP_PEER_TX_RATE_STATS:
  333. dp_peer_tx_rate_stats_print(peer_mac, peer_cookie,
  334. buffer, buffer_len);
  335. break;
  336. }
  337. }
  338. static void
  339. dp_peer_stats_event_callback(char *ifname,
  340. uint32_t cmdid,
  341. uint8_t *data,
  342. size_t len)
  343. {
  344. struct nlattr *tb_array[QCA_WLAN_VENDOR_ATTR_PEER_STATS_CACHE_MAX + 1];
  345. struct nlattr *tb;
  346. void *buffer = NULL;
  347. uint32_t buffer_len = 0;
  348. uint8_t *peer_mac;
  349. uint32_t cache_type;
  350. uint64_t peer_cookie;
  351. if (cmdid != QCA_NL80211_VENDOR_SUBCMD_PEER_STATS_CACHE_FLUSH) {
  352. /* ignore anyother events*/
  353. return;
  354. }
  355. if (strncmp(interface, ifname, sizeof(interface)) != 0) {
  356. /* ignore events for other interfaces*/
  357. return;
  358. }
  359. if (nla_parse(tb_array, QCA_WLAN_VENDOR_ATTR_PEER_STATS_CACHE_MAX,
  360. (struct nlattr *)data, len, NULL)) {
  361. PRINT("Invalid event\n");
  362. return;
  363. }
  364. tb = tb_array[QCA_WLAN_VENDOR_ATTR_PEER_STATS_CACHE_TYPE];
  365. if (!tb) {
  366. PRINT("Cache type in NULL, return");
  367. return;
  368. }
  369. cache_type = nla_get_u32(tb);
  370. tb = tb_array[QCA_WLAN_VENDOR_ATTR_PEER_STATS_CACHE_PEER_MAC];
  371. if (!tb) {
  372. PRINT("Peer mac addr is null, return");
  373. return;
  374. }
  375. peer_mac = (uint8_t *)nla_data(tb);
  376. tb = tb_array[QCA_WLAN_VENDOR_ATTR_PEER_STATS_CACHE_DATA];
  377. if (tb) {
  378. buffer = (void *)nla_data(tb);
  379. buffer_len = nla_len(tb);
  380. }
  381. tb = tb_array[QCA_WLAN_VENDOR_ATTR_PEER_STATS_CACHE_PEER_COOKIE];
  382. if (!tb) {
  383. PRINT("peer cookie attribute is null, return");
  384. return;
  385. }
  386. peer_cookie = nla_get_u64(tb);
  387. if (!buffer) {
  388. PRINT(" stats buffer is null, return");
  389. return;
  390. }
  391. dp_peer_stats_handler(cache_type, peer_mac, peer_cookie,
  392. buffer, buffer_len);
  393. }
  394. int main(int argc, char *argv[])
  395. {
  396. int err = 0;
  397. wifi_cfg80211_context cfg80211_ctxt;
  398. char *ifname;
  399. int num_msecs = 0;
  400. int status = 0;
  401. if (argc < 2) {
  402. fprintf(stderr, "Invalid commands args\n");
  403. return -EIO;
  404. }
  405. /* Reset the cfg80211 context to 0 if the application does not pass
  406. * custom private event and command sockets. In this case, the default
  407. * port is used for netlink communication.
  408. */
  409. memset(&cfg80211_ctxt, 0, sizeof(wifi_cfg80211_context));
  410. ifname = argv[1];
  411. memcpy(interface, ifname, sizeof(interface));
  412. cfg80211_ctxt.event_callback = dp_peer_stats_event_callback;
  413. err = wifi_init_nl80211(&cfg80211_ctxt);
  414. if (err) {
  415. fprintf(stderr, "unable to create NL socket\n");
  416. return -EIO;
  417. }
  418. /* Starting event thread to listen for responses*/
  419. if (wifi_nl80211_start_event_thread(&cfg80211_ctxt)) {
  420. fprintf(stderr, "Unable to setup nl80211 event thread\n");
  421. status = -EIO;
  422. goto cleanup;
  423. }
  424. while (true) {
  425. /*sleep for 1 ms*/
  426. usleep(1000);
  427. num_msecs++;
  428. }
  429. wifi_destroy_nl80211(&cfg80211_ctxt);
  430. return 0;
  431. cleanup:
  432. wifi_destroy_nl80211(&cfg80211_ctxt);
  433. return status;
  434. }