123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350 |
- /*
- * Copyright (c) 2011, 2013-2015 The Linux Foundation. All rights reserved.
- *
- * Previously licensed under the ISC license by Qualcomm Atheros, Inc.
- *
- *
- * Permission to use, copy, modify, and/or distribute this software for
- * any purpose with or without fee is hereby granted, provided that the
- * above copyright notice and this permission notice appear in all
- * copies.
- *
- * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
- * WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
- * WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
- * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
- * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
- * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
- * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
- * PERFORMANCE OF THIS SOFTWARE.
- */
- /*
- * This file was originally distributed by Qualcomm Atheros, Inc.
- * under proprietary terms before Copyright ownership was assigned
- * to the Linux Foundation.
- */
- #include <cdf_nbuf.h> /* cdf_nbuf_t */
- #include <ol_htt_rx_api.h> /* htt_rx_pn_t, etc. */
- #include <ol_ctrl_txrx_api.h> /* ol_rx_err */
- #include <ol_txrx_internal.h> /* ol_rx_mpdu_list_next */
- #include <ol_txrx_types.h> /* ol_txrx_vdev_t, etc. */
- #include <ol_rx_pn.h> /* our own defs */
- #include <ol_rx_fwd.h> /* ol_rx_fwd_check */
- #include <ol_rx.h> /* ol_rx_deliver */
- /* add the MSDUs from this MPDU to the list of good frames */
- #define ADD_MPDU_TO_LIST(head, tail, mpdu, mpdu_tail) do { \
- if (!head) { \
- head = mpdu; \
- } else { \
- cdf_nbuf_set_next(tail, mpdu); \
- } \
- tail = mpdu_tail; \
- } while (0)
- int ol_rx_pn_cmp24(union htt_rx_pn_t *new_pn,
- union htt_rx_pn_t *old_pn, int is_unicast, int opmode)
- {
- int rc = ((new_pn->pn24 & 0xffffff) <= (old_pn->pn24 & 0xffffff));
- return rc;
- }
- int ol_rx_pn_cmp48(union htt_rx_pn_t *new_pn,
- union htt_rx_pn_t *old_pn, int is_unicast, int opmode)
- {
- int rc = ((new_pn->pn48 & 0xffffffffffffULL) <=
- (old_pn->pn48 & 0xffffffffffffULL));
- return rc;
- }
- int ol_rx_pn_wapi_cmp(union htt_rx_pn_t *new_pn,
- union htt_rx_pn_t *old_pn, int is_unicast, int opmode)
- {
- int pn_is_replay = 0;
- if (new_pn->pn128[1] == old_pn->pn128[1])
- pn_is_replay = (new_pn->pn128[0] <= old_pn->pn128[0]);
- else
- pn_is_replay = (new_pn->pn128[1] < old_pn->pn128[1]);
- if (is_unicast) {
- if (opmode == wlan_op_mode_ap)
- pn_is_replay |= ((new_pn->pn128[0] & 0x1ULL) != 0);
- else
- pn_is_replay |= ((new_pn->pn128[0] & 0x1ULL) != 1);
- }
- return pn_is_replay;
- }
- cdf_nbuf_t
- ol_rx_pn_check_base(struct ol_txrx_vdev_t *vdev,
- struct ol_txrx_peer_t *peer,
- unsigned tid, cdf_nbuf_t msdu_list)
- {
- struct ol_txrx_pdev_t *pdev = vdev->pdev;
- union htt_rx_pn_t *last_pn;
- cdf_nbuf_t out_list_head = NULL;
- cdf_nbuf_t out_list_tail = NULL;
- cdf_nbuf_t mpdu;
- int index; /* unicast vs. multicast */
- int pn_len;
- void *rx_desc;
- int last_pn_valid;
- /* Make sure host pn check is not redundant */
- if ((cdf_atomic_read(&peer->fw_pn_check)) ||
- (vdev->opmode == wlan_op_mode_ibss)) {
- return msdu_list;
- }
- /* First, check whether the PN check applies */
- rx_desc = htt_rx_msdu_desc_retrieve(pdev->htt_pdev, msdu_list);
- cdf_assert(htt_rx_msdu_has_wlan_mcast_flag(pdev->htt_pdev, rx_desc));
- index = htt_rx_msdu_is_wlan_mcast(pdev->htt_pdev, rx_desc) ?
- txrx_sec_mcast : txrx_sec_ucast;
- pn_len = pdev->rx_pn[peer->security[index].sec_type].len;
- if (pn_len == 0)
- return msdu_list;
- last_pn_valid = peer->tids_last_pn_valid[tid];
- last_pn = &peer->tids_last_pn[tid];
- mpdu = msdu_list;
- while (mpdu) {
- cdf_nbuf_t mpdu_tail, next_mpdu;
- union htt_rx_pn_t new_pn;
- int pn_is_replay = 0;
- rx_desc = htt_rx_msdu_desc_retrieve(pdev->htt_pdev, mpdu);
- /*
- * Find the last MSDU within this MPDU, and
- * the find the first MSDU within the next MPDU.
- */
- ol_rx_mpdu_list_next(pdev, mpdu, &mpdu_tail, &next_mpdu);
- /* Don't check the PN replay for non-encrypted frames */
- if (!htt_rx_mpdu_is_encrypted(pdev->htt_pdev, rx_desc)) {
- ADD_MPDU_TO_LIST(out_list_head, out_list_tail, mpdu,
- mpdu_tail);
- mpdu = next_mpdu;
- continue;
- }
- /* retrieve PN from rx descriptor */
- htt_rx_mpdu_desc_pn(pdev->htt_pdev, rx_desc, &new_pn, pn_len);
- /* if there was no prior PN, there's nothing to check */
- if (last_pn_valid) {
- pn_is_replay =
- pdev->rx_pn[peer->security[index].sec_type].
- cmp(&new_pn, last_pn, index == txrx_sec_ucast,
- vdev->opmode);
- } else {
- last_pn_valid = peer->tids_last_pn_valid[tid] = 1;
- }
- if (pn_is_replay) {
- cdf_nbuf_t msdu;
- static uint32_t last_pncheck_print_time /* = 0 */;
- int log_level;
- uint32_t current_time_ms;
- /*
- * This MPDU failed the PN check:
- * 1. notify the control SW of the PN failure
- * (so countermeasures can be taken, if necessary)
- * 2. Discard all the MSDUs from this MPDU.
- */
- msdu = mpdu;
- current_time_ms =
- cdf_system_ticks_to_msecs(cdf_system_ticks());
- if (TXRX_PN_CHECK_FAILURE_PRINT_PERIOD_MS <
- (current_time_ms - last_pncheck_print_time)) {
- last_pncheck_print_time = current_time_ms;
- log_level = TXRX_PRINT_LEVEL_WARN;
- } else {
- log_level = TXRX_PRINT_LEVEL_INFO2;
- }
- TXRX_PRINT(log_level,
- "PN check failed - TID %d, peer %p "
- "(%02x:%02x:%02x:%02x:%02x:%02x) %s\n"
- " old PN (u64 x2)= 0x%08llx %08llx (LSBs = %lld)\n"
- " new PN (u64 x2)= 0x%08llx %08llx (LSBs = %lld)\n"
- " new seq num = %d\n",
- tid, peer,
- peer->mac_addr.raw[0], peer->mac_addr.raw[1],
- peer->mac_addr.raw[2], peer->mac_addr.raw[3],
- peer->mac_addr.raw[4], peer->mac_addr.raw[5],
- (index ==
- txrx_sec_ucast) ? "ucast" : "mcast",
- last_pn->pn128[1], last_pn->pn128[0],
- last_pn->pn128[0] & 0xffffffffffffULL,
- new_pn.pn128[1], new_pn.pn128[0],
- new_pn.pn128[0] & 0xffffffffffffULL,
- htt_rx_mpdu_desc_seq_num(pdev->htt_pdev,
- rx_desc));
- #if defined(ENABLE_RX_PN_TRACE)
- ol_rx_pn_trace_display(pdev, 1);
- #endif /* ENABLE_RX_PN_TRACE */
- ol_rx_err(pdev->ctrl_pdev,
- vdev->vdev_id, peer->mac_addr.raw, tid,
- htt_rx_mpdu_desc_tsf32(pdev->htt_pdev,
- rx_desc), OL_RX_ERR_PN,
- mpdu, NULL, 0);
- /* free all MSDUs within this MPDU */
- do {
- cdf_nbuf_t next_msdu;
- OL_RX_ERR_STATISTICS_1(pdev, vdev, peer,
- rx_desc, OL_RX_ERR_PN);
- next_msdu = cdf_nbuf_next(msdu);
- htt_rx_desc_frame_free(pdev->htt_pdev, msdu);
- if (msdu == mpdu_tail)
- break;
- else
- msdu = next_msdu;
- } while (1);
- } else {
- ADD_MPDU_TO_LIST(out_list_head, out_list_tail, mpdu,
- mpdu_tail);
- /*
- * Remember the new PN.
- * For simplicity, just do 2 64-bit word copies to
- * cover the worst case (WAPI), regardless of the length
- * of the PN.
- * This is more efficient than doing a conditional
- * branch to copy only the relevant portion.
- */
- last_pn->pn128[0] = new_pn.pn128[0];
- last_pn->pn128[1] = new_pn.pn128[1];
- OL_RX_PN_TRACE_ADD(pdev, peer, tid, rx_desc);
- }
- mpdu = next_mpdu;
- }
- /* make sure the list is null-terminated */
- if (out_list_tail)
- cdf_nbuf_set_next(out_list_tail, NULL);
- return out_list_head;
- }
- void
- ol_rx_pn_check(struct ol_txrx_vdev_t *vdev,
- struct ol_txrx_peer_t *peer, unsigned tid, cdf_nbuf_t msdu_list)
- {
- msdu_list = ol_rx_pn_check_base(vdev, peer, tid, msdu_list);
- ol_rx_fwd_check(vdev, peer, tid, msdu_list);
- }
- void
- ol_rx_pn_check_only(struct ol_txrx_vdev_t *vdev,
- struct ol_txrx_peer_t *peer,
- unsigned tid, cdf_nbuf_t msdu_list)
- {
- msdu_list = ol_rx_pn_check_base(vdev, peer, tid, msdu_list);
- ol_rx_deliver(vdev, peer, tid, msdu_list);
- }
- #if defined(ENABLE_RX_PN_TRACE)
- A_STATUS ol_rx_pn_trace_attach(ol_txrx_pdev_handle pdev)
- {
- int num_elems;
- num_elems = 1 << TXRX_RX_PN_TRACE_SIZE_LOG2;
- pdev->rx_pn_trace.idx = 0;
- pdev->rx_pn_trace.cnt = 0;
- pdev->rx_pn_trace.mask = num_elems - 1;
- pdev->rx_pn_trace.data =
- cdf_mem_malloc(sizeof(*pdev->rx_pn_trace.data) * num_elems);
- if (!pdev->rx_pn_trace.data)
- return A_NO_MEMORY;
- return A_OK;
- }
- void ol_rx_pn_trace_detach(ol_txrx_pdev_handle pdev)
- {
- cdf_mem_free(pdev->rx_pn_trace.data);
- }
- void
- ol_rx_pn_trace_add(struct ol_txrx_pdev_t *pdev,
- struct ol_txrx_peer_t *peer, uint16_t tid, void *rx_desc)
- {
- uint32_t idx = pdev->rx_pn_trace.idx;
- union htt_rx_pn_t pn;
- uint32_t pn32;
- uint16_t seq_num;
- uint8_t unicast;
- htt_rx_mpdu_desc_pn(pdev->htt_pdev, rx_desc, &pn, 48);
- pn32 = pn.pn48 & 0xffffffff;
- seq_num = htt_rx_mpdu_desc_seq_num(pdev->htt_pdev, rx_desc);
- unicast = !htt_rx_msdu_is_wlan_mcast(pdev->htt_pdev, rx_desc);
- pdev->rx_pn_trace.data[idx].peer = peer;
- pdev->rx_pn_trace.data[idx].tid = tid;
- pdev->rx_pn_trace.data[idx].seq_num = seq_num;
- pdev->rx_pn_trace.data[idx].unicast = unicast;
- pdev->rx_pn_trace.data[idx].pn32 = pn32;
- pdev->rx_pn_trace.cnt++;
- idx++;
- pdev->rx_pn_trace.idx = idx & pdev->rx_pn_trace.mask;
- }
- void ol_rx_pn_trace_display(ol_txrx_pdev_handle pdev, int just_once)
- {
- static int print_count /* = 0 */;
- uint32_t i, start, end;
- uint64_t cnt;
- int elems;
- int limit = 0; /* move this to the arg list? */
- if (print_count != 0 && just_once)
- return;
- print_count++;
- end = pdev->rx_pn_trace.idx;
- if (pdev->rx_pn_trace.cnt <= pdev->rx_pn_trace.mask) {
- /* trace log has not yet wrapped around - start at the top */
- start = 0;
- cnt = 0;
- } else {
- start = end;
- cnt = pdev->rx_pn_trace.cnt - (pdev->rx_pn_trace.mask + 1);
- }
- elems = (end - 1 - start) & pdev->rx_pn_trace.mask;
- if (limit > 0 && elems > limit) {
- int delta;
- delta = elems - limit;
- start += delta;
- start &= pdev->rx_pn_trace.mask;
- cnt += delta;
- }
- i = start;
- CDF_TRACE(CDF_MODULE_ID_TXRX, CDF_TRACE_LEVEL_INFO,
- " seq PN");
- CDF_TRACE(CDF_MODULE_ID_TXRX, CDF_TRACE_LEVEL_INFO,
- " count idx peer tid uni num LSBs");
- do {
- CDF_TRACE(CDF_MODULE_ID_TXRX, CDF_TRACE_LEVEL_INFO,
- " %6lld %4d %p %2d %d %4d %8d",
- cnt, i,
- pdev->rx_pn_trace.data[i].peer,
- pdev->rx_pn_trace.data[i].tid,
- pdev->rx_pn_trace.data[i].unicast,
- pdev->rx_pn_trace.data[i].seq_num,
- pdev->rx_pn_trace.data[i].pn32);
- cnt++;
- i++;
- i &= pdev->rx_pn_trace.mask;
- } while (i != end);
- }
- #endif /* ENABLE_RX_PN_TRACE */
|