ol_rx_pn.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350
  1. /*
  2. * Copyright (c) 2011, 2013-2015 The Linux Foundation. All rights reserved.
  3. *
  4. * Previously licensed under the ISC license by Qualcomm Atheros, Inc.
  5. *
  6. *
  7. * Permission to use, copy, modify, and/or distribute this software for
  8. * any purpose with or without fee is hereby granted, provided that the
  9. * above copyright notice and this permission notice appear in all
  10. * copies.
  11. *
  12. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
  13. * WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
  14. * WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
  15. * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
  16. * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
  17. * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
  18. * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
  19. * PERFORMANCE OF THIS SOFTWARE.
  20. */
  21. /*
  22. * This file was originally distributed by Qualcomm Atheros, Inc.
  23. * under proprietary terms before Copyright ownership was assigned
  24. * to the Linux Foundation.
  25. */
  26. #include <cdf_nbuf.h> /* cdf_nbuf_t */
  27. #include <ol_htt_rx_api.h> /* htt_rx_pn_t, etc. */
  28. #include <ol_ctrl_txrx_api.h> /* ol_rx_err */
  29. #include <ol_txrx_internal.h> /* ol_rx_mpdu_list_next */
  30. #include <ol_txrx_types.h> /* ol_txrx_vdev_t, etc. */
  31. #include <ol_rx_pn.h> /* our own defs */
  32. #include <ol_rx_fwd.h> /* ol_rx_fwd_check */
  33. #include <ol_rx.h> /* ol_rx_deliver */
  34. /* add the MSDUs from this MPDU to the list of good frames */
  35. #define ADD_MPDU_TO_LIST(head, tail, mpdu, mpdu_tail) do { \
  36. if (!head) { \
  37. head = mpdu; \
  38. } else { \
  39. cdf_nbuf_set_next(tail, mpdu); \
  40. } \
  41. tail = mpdu_tail; \
  42. } while (0)
  43. int ol_rx_pn_cmp24(union htt_rx_pn_t *new_pn,
  44. union htt_rx_pn_t *old_pn, int is_unicast, int opmode)
  45. {
  46. int rc = ((new_pn->pn24 & 0xffffff) <= (old_pn->pn24 & 0xffffff));
  47. return rc;
  48. }
  49. int ol_rx_pn_cmp48(union htt_rx_pn_t *new_pn,
  50. union htt_rx_pn_t *old_pn, int is_unicast, int opmode)
  51. {
  52. int rc = ((new_pn->pn48 & 0xffffffffffffULL) <=
  53. (old_pn->pn48 & 0xffffffffffffULL));
  54. return rc;
  55. }
  56. int ol_rx_pn_wapi_cmp(union htt_rx_pn_t *new_pn,
  57. union htt_rx_pn_t *old_pn, int is_unicast, int opmode)
  58. {
  59. int pn_is_replay = 0;
  60. if (new_pn->pn128[1] == old_pn->pn128[1])
  61. pn_is_replay = (new_pn->pn128[0] <= old_pn->pn128[0]);
  62. else
  63. pn_is_replay = (new_pn->pn128[1] < old_pn->pn128[1]);
  64. if (is_unicast) {
  65. if (opmode == wlan_op_mode_ap)
  66. pn_is_replay |= ((new_pn->pn128[0] & 0x1ULL) != 0);
  67. else
  68. pn_is_replay |= ((new_pn->pn128[0] & 0x1ULL) != 1);
  69. }
  70. return pn_is_replay;
  71. }
  72. cdf_nbuf_t
  73. ol_rx_pn_check_base(struct ol_txrx_vdev_t *vdev,
  74. struct ol_txrx_peer_t *peer,
  75. unsigned tid, cdf_nbuf_t msdu_list)
  76. {
  77. struct ol_txrx_pdev_t *pdev = vdev->pdev;
  78. union htt_rx_pn_t *last_pn;
  79. cdf_nbuf_t out_list_head = NULL;
  80. cdf_nbuf_t out_list_tail = NULL;
  81. cdf_nbuf_t mpdu;
  82. int index; /* unicast vs. multicast */
  83. int pn_len;
  84. void *rx_desc;
  85. int last_pn_valid;
  86. /* Make sure host pn check is not redundant */
  87. if ((cdf_atomic_read(&peer->fw_pn_check)) ||
  88. (vdev->opmode == wlan_op_mode_ibss)) {
  89. return msdu_list;
  90. }
  91. /* First, check whether the PN check applies */
  92. rx_desc = htt_rx_msdu_desc_retrieve(pdev->htt_pdev, msdu_list);
  93. cdf_assert(htt_rx_msdu_has_wlan_mcast_flag(pdev->htt_pdev, rx_desc));
  94. index = htt_rx_msdu_is_wlan_mcast(pdev->htt_pdev, rx_desc) ?
  95. txrx_sec_mcast : txrx_sec_ucast;
  96. pn_len = pdev->rx_pn[peer->security[index].sec_type].len;
  97. if (pn_len == 0)
  98. return msdu_list;
  99. last_pn_valid = peer->tids_last_pn_valid[tid];
  100. last_pn = &peer->tids_last_pn[tid];
  101. mpdu = msdu_list;
  102. while (mpdu) {
  103. cdf_nbuf_t mpdu_tail, next_mpdu;
  104. union htt_rx_pn_t new_pn;
  105. int pn_is_replay = 0;
  106. rx_desc = htt_rx_msdu_desc_retrieve(pdev->htt_pdev, mpdu);
  107. /*
  108. * Find the last MSDU within this MPDU, and
  109. * the find the first MSDU within the next MPDU.
  110. */
  111. ol_rx_mpdu_list_next(pdev, mpdu, &mpdu_tail, &next_mpdu);
  112. /* Don't check the PN replay for non-encrypted frames */
  113. if (!htt_rx_mpdu_is_encrypted(pdev->htt_pdev, rx_desc)) {
  114. ADD_MPDU_TO_LIST(out_list_head, out_list_tail, mpdu,
  115. mpdu_tail);
  116. mpdu = next_mpdu;
  117. continue;
  118. }
  119. /* retrieve PN from rx descriptor */
  120. htt_rx_mpdu_desc_pn(pdev->htt_pdev, rx_desc, &new_pn, pn_len);
  121. /* if there was no prior PN, there's nothing to check */
  122. if (last_pn_valid) {
  123. pn_is_replay =
  124. pdev->rx_pn[peer->security[index].sec_type].
  125. cmp(&new_pn, last_pn, index == txrx_sec_ucast,
  126. vdev->opmode);
  127. } else {
  128. last_pn_valid = peer->tids_last_pn_valid[tid] = 1;
  129. }
  130. if (pn_is_replay) {
  131. cdf_nbuf_t msdu;
  132. static uint32_t last_pncheck_print_time /* = 0 */;
  133. int log_level;
  134. uint32_t current_time_ms;
  135. /*
  136. * This MPDU failed the PN check:
  137. * 1. notify the control SW of the PN failure
  138. * (so countermeasures can be taken, if necessary)
  139. * 2. Discard all the MSDUs from this MPDU.
  140. */
  141. msdu = mpdu;
  142. current_time_ms =
  143. cdf_system_ticks_to_msecs(cdf_system_ticks());
  144. if (TXRX_PN_CHECK_FAILURE_PRINT_PERIOD_MS <
  145. (current_time_ms - last_pncheck_print_time)) {
  146. last_pncheck_print_time = current_time_ms;
  147. log_level = TXRX_PRINT_LEVEL_WARN;
  148. } else {
  149. log_level = TXRX_PRINT_LEVEL_INFO2;
  150. }
  151. TXRX_PRINT(log_level,
  152. "PN check failed - TID %d, peer %p "
  153. "(%02x:%02x:%02x:%02x:%02x:%02x) %s\n"
  154. " old PN (u64 x2)= 0x%08llx %08llx (LSBs = %lld)\n"
  155. " new PN (u64 x2)= 0x%08llx %08llx (LSBs = %lld)\n"
  156. " new seq num = %d\n",
  157. tid, peer,
  158. peer->mac_addr.raw[0], peer->mac_addr.raw[1],
  159. peer->mac_addr.raw[2], peer->mac_addr.raw[3],
  160. peer->mac_addr.raw[4], peer->mac_addr.raw[5],
  161. (index ==
  162. txrx_sec_ucast) ? "ucast" : "mcast",
  163. last_pn->pn128[1], last_pn->pn128[0],
  164. last_pn->pn128[0] & 0xffffffffffffULL,
  165. new_pn.pn128[1], new_pn.pn128[0],
  166. new_pn.pn128[0] & 0xffffffffffffULL,
  167. htt_rx_mpdu_desc_seq_num(pdev->htt_pdev,
  168. rx_desc));
  169. #if defined(ENABLE_RX_PN_TRACE)
  170. ol_rx_pn_trace_display(pdev, 1);
  171. #endif /* ENABLE_RX_PN_TRACE */
  172. ol_rx_err(pdev->ctrl_pdev,
  173. vdev->vdev_id, peer->mac_addr.raw, tid,
  174. htt_rx_mpdu_desc_tsf32(pdev->htt_pdev,
  175. rx_desc), OL_RX_ERR_PN,
  176. mpdu, NULL, 0);
  177. /* free all MSDUs within this MPDU */
  178. do {
  179. cdf_nbuf_t next_msdu;
  180. OL_RX_ERR_STATISTICS_1(pdev, vdev, peer,
  181. rx_desc, OL_RX_ERR_PN);
  182. next_msdu = cdf_nbuf_next(msdu);
  183. htt_rx_desc_frame_free(pdev->htt_pdev, msdu);
  184. if (msdu == mpdu_tail)
  185. break;
  186. else
  187. msdu = next_msdu;
  188. } while (1);
  189. } else {
  190. ADD_MPDU_TO_LIST(out_list_head, out_list_tail, mpdu,
  191. mpdu_tail);
  192. /*
  193. * Remember the new PN.
  194. * For simplicity, just do 2 64-bit word copies to
  195. * cover the worst case (WAPI), regardless of the length
  196. * of the PN.
  197. * This is more efficient than doing a conditional
  198. * branch to copy only the relevant portion.
  199. */
  200. last_pn->pn128[0] = new_pn.pn128[0];
  201. last_pn->pn128[1] = new_pn.pn128[1];
  202. OL_RX_PN_TRACE_ADD(pdev, peer, tid, rx_desc);
  203. }
  204. mpdu = next_mpdu;
  205. }
  206. /* make sure the list is null-terminated */
  207. if (out_list_tail)
  208. cdf_nbuf_set_next(out_list_tail, NULL);
  209. return out_list_head;
  210. }
  211. void
  212. ol_rx_pn_check(struct ol_txrx_vdev_t *vdev,
  213. struct ol_txrx_peer_t *peer, unsigned tid, cdf_nbuf_t msdu_list)
  214. {
  215. msdu_list = ol_rx_pn_check_base(vdev, peer, tid, msdu_list);
  216. ol_rx_fwd_check(vdev, peer, tid, msdu_list);
  217. }
  218. void
  219. ol_rx_pn_check_only(struct ol_txrx_vdev_t *vdev,
  220. struct ol_txrx_peer_t *peer,
  221. unsigned tid, cdf_nbuf_t msdu_list)
  222. {
  223. msdu_list = ol_rx_pn_check_base(vdev, peer, tid, msdu_list);
  224. ol_rx_deliver(vdev, peer, tid, msdu_list);
  225. }
  226. #if defined(ENABLE_RX_PN_TRACE)
  227. A_STATUS ol_rx_pn_trace_attach(ol_txrx_pdev_handle pdev)
  228. {
  229. int num_elems;
  230. num_elems = 1 << TXRX_RX_PN_TRACE_SIZE_LOG2;
  231. pdev->rx_pn_trace.idx = 0;
  232. pdev->rx_pn_trace.cnt = 0;
  233. pdev->rx_pn_trace.mask = num_elems - 1;
  234. pdev->rx_pn_trace.data =
  235. cdf_mem_malloc(sizeof(*pdev->rx_pn_trace.data) * num_elems);
  236. if (!pdev->rx_pn_trace.data)
  237. return A_NO_MEMORY;
  238. return A_OK;
  239. }
  240. void ol_rx_pn_trace_detach(ol_txrx_pdev_handle pdev)
  241. {
  242. cdf_mem_free(pdev->rx_pn_trace.data);
  243. }
  244. void
  245. ol_rx_pn_trace_add(struct ol_txrx_pdev_t *pdev,
  246. struct ol_txrx_peer_t *peer, uint16_t tid, void *rx_desc)
  247. {
  248. uint32_t idx = pdev->rx_pn_trace.idx;
  249. union htt_rx_pn_t pn;
  250. uint32_t pn32;
  251. uint16_t seq_num;
  252. uint8_t unicast;
  253. htt_rx_mpdu_desc_pn(pdev->htt_pdev, rx_desc, &pn, 48);
  254. pn32 = pn.pn48 & 0xffffffff;
  255. seq_num = htt_rx_mpdu_desc_seq_num(pdev->htt_pdev, rx_desc);
  256. unicast = !htt_rx_msdu_is_wlan_mcast(pdev->htt_pdev, rx_desc);
  257. pdev->rx_pn_trace.data[idx].peer = peer;
  258. pdev->rx_pn_trace.data[idx].tid = tid;
  259. pdev->rx_pn_trace.data[idx].seq_num = seq_num;
  260. pdev->rx_pn_trace.data[idx].unicast = unicast;
  261. pdev->rx_pn_trace.data[idx].pn32 = pn32;
  262. pdev->rx_pn_trace.cnt++;
  263. idx++;
  264. pdev->rx_pn_trace.idx = idx & pdev->rx_pn_trace.mask;
  265. }
  266. void ol_rx_pn_trace_display(ol_txrx_pdev_handle pdev, int just_once)
  267. {
  268. static int print_count /* = 0 */;
  269. uint32_t i, start, end;
  270. uint64_t cnt;
  271. int elems;
  272. int limit = 0; /* move this to the arg list? */
  273. if (print_count != 0 && just_once)
  274. return;
  275. print_count++;
  276. end = pdev->rx_pn_trace.idx;
  277. if (pdev->rx_pn_trace.cnt <= pdev->rx_pn_trace.mask) {
  278. /* trace log has not yet wrapped around - start at the top */
  279. start = 0;
  280. cnt = 0;
  281. } else {
  282. start = end;
  283. cnt = pdev->rx_pn_trace.cnt - (pdev->rx_pn_trace.mask + 1);
  284. }
  285. elems = (end - 1 - start) & pdev->rx_pn_trace.mask;
  286. if (limit > 0 && elems > limit) {
  287. int delta;
  288. delta = elems - limit;
  289. start += delta;
  290. start &= pdev->rx_pn_trace.mask;
  291. cnt += delta;
  292. }
  293. i = start;
  294. CDF_TRACE(CDF_MODULE_ID_TXRX, CDF_TRACE_LEVEL_INFO,
  295. " seq PN");
  296. CDF_TRACE(CDF_MODULE_ID_TXRX, CDF_TRACE_LEVEL_INFO,
  297. " count idx peer tid uni num LSBs");
  298. do {
  299. CDF_TRACE(CDF_MODULE_ID_TXRX, CDF_TRACE_LEVEL_INFO,
  300. " %6lld %4d %p %2d %d %4d %8d",
  301. cnt, i,
  302. pdev->rx_pn_trace.data[i].peer,
  303. pdev->rx_pn_trace.data[i].tid,
  304. pdev->rx_pn_trace.data[i].unicast,
  305. pdev->rx_pn_trace.data[i].seq_num,
  306. pdev->rx_pn_trace.data[i].pn32);
  307. cnt++;
  308. i++;
  309. i &= pdev->rx_pn_trace.mask;
  310. } while (i != end);
  311. }
  312. #endif /* ENABLE_RX_PN_TRACE */