| /* |
| * Copyright (c) 2014, 2016-2017 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. |
| */ |
| |
| /*======================================================================== |
| |
| \file epping_main.c |
| |
| \brief WLAN End Point Ping test tool implementation |
| |
| ========================================================================*/ |
| |
| /*-------------------------------------------------------------------------- |
| Include Files |
| ------------------------------------------------------------------------*/ |
| #include <wlan_hdd_includes.h> |
| #include <vos_api.h> |
| #include <vos_sched.h> |
| #include <linux/etherdevice.h> |
| #include <linux/firmware.h> |
| #include <wcnss_api.h> |
| #include <wlan_hdd_tx_rx.h> |
| #include <wniApi.h> |
| #include <wlan_nlink_srv.h> |
| #include <wlan_hdd_cfg.h> |
| #include <wlan_ptt_sock_svc.h> |
| #include <wlan_hdd_wowl.h> |
| #include <wlan_hdd_misc.h> |
| #include <wlan_hdd_wext.h> |
| #include <linux/wireless.h> |
| #include <net/cfg80211.h> |
| #include <linux/rtnetlink.h> |
| #include <linux/semaphore.h> |
| #include <linux/ctype.h> |
| #include <wlan_hdd_hostapd.h> |
| #include <wlan_hdd_softap_tx_rx.h> |
| #include "epping_main.h" |
| #include "epping_internal.h" |
| #include "epping_test.h" |
| |
| #define AR6000_MAX_RX_BUFFERS 16 |
| #define AR6000_BUFFER_SIZE 1664 |
| #define AR6000_MIN_HEAD_ROOM 64 |
| |
| static bool enb_rx_dump = 0; |
| |
| #ifdef HIF_SDIO |
| void epping_refill(void *ctx, HTC_ENDPOINT_ID Endpoint) |
| { |
| epping_context_t *pEpping_ctx = (epping_context_t *)ctx; |
| void *osBuf; |
| int RxBuffers; |
| int buffersToRefill; |
| HTC_PACKET *pPacket; |
| HTC_PACKET_QUEUE queue; |
| |
| buffersToRefill = (int)AR6000_MAX_RX_BUFFERS - |
| HTCGetNumRecvBuffers(pEpping_ctx->HTCHandle, Endpoint); |
| |
| if (buffersToRefill <= 0) { |
| /* fast return, nothing to fill */ |
| return; |
| } |
| |
| INIT_HTC_PACKET_QUEUE(&queue); |
| |
| EPPING_LOG(VOS_TRACE_LEVEL_INFO, |
| "%s: providing htc with %d buffers at eid=%d\n", |
| __func__, buffersToRefill, Endpoint); |
| |
| for (RxBuffers = 0; RxBuffers < buffersToRefill; RxBuffers++) { |
| osBuf = adf_nbuf_alloc(NULL, AR6000_BUFFER_SIZE, |
| AR6000_MIN_HEAD_ROOM, 4, FALSE); |
| if (NULL == osBuf) { |
| break; |
| } |
| /* the HTC packet wrapper is at the head of the reserved area |
| * in the skb */ |
| pPacket = (HTC_PACKET *)(A_NETBUF_HEAD(osBuf)); |
| /* set re-fill info */ |
| SET_HTC_PACKET_INFO_RX_REFILL(pPacket,osBuf, |
| adf_nbuf_data(osBuf), |
| AR6000_BUFFER_SIZE,Endpoint); |
| SET_HTC_PACKET_NET_BUF_CONTEXT(pPacket,osBuf); |
| /* add to queue */ |
| HTC_PACKET_ENQUEUE(&queue,pPacket); |
| } |
| |
| if (!HTC_QUEUE_EMPTY(&queue)) { |
| /* add packets */ |
| HTCAddReceivePktMultiple(pEpping_ctx->HTCHandle, &queue); |
| } |
| } |
| #endif /* HIF_SDIO */ |
| |
| void epping_rx(void *ctx, HTC_PACKET *pPacket) |
| { |
| epping_context_t *pEpping_ctx = (epping_context_t *)ctx; |
| epping_adapter_t *pAdapter = pEpping_ctx->epping_adapter; |
| struct net_device* dev = pAdapter->dev; |
| A_STATUS status = pPacket->Status; |
| HTC_ENDPOINT_ID eid = pPacket->Endpoint; |
| struct sk_buff *pktSkb = (struct sk_buff *)pPacket->pPktContext; |
| |
| EPPING_LOG(VOS_TRACE_LEVEL_INFO, |
| "%s: pAdapter = 0x%pK eid=%d, skb=0x%pK, data=0x%pK, len=0x%x status:%d", |
| __func__, pAdapter, eid, pktSkb, pPacket->pBuffer, |
| pPacket->ActualLength, status); |
| |
| if (status != A_OK) { |
| if (status != A_ECANCELED) { |
| printk("%s: RX ERR (%d) \n", __func__, status); |
| } |
| adf_nbuf_free(pktSkb); |
| return; |
| } |
| |
| /* deliver to up layer */ |
| if (pktSkb) |
| { |
| EPPING_HEADER *eppingHdr = (EPPING_HEADER *)adf_nbuf_data(pktSkb); |
| if (EPPING_ALIGNMENT_PAD > 0) { |
| A_NETBUF_PULL(pktSkb, EPPING_ALIGNMENT_PAD); |
| } |
| if (enb_rx_dump) |
| epping_hex_dump((void *)adf_nbuf_data(pktSkb), |
| pktSkb->len, __func__); |
| pktSkb->dev = dev; |
| if ((pktSkb->dev->flags & IFF_UP) == IFF_UP) { |
| pktSkb->protocol = eth_type_trans(pktSkb, pktSkb->dev); |
| ++pAdapter->stats.rx_packets; |
| pAdapter->stats.rx_bytes += pktSkb->len; |
| if (pEpping_ctx->kperf[eid] == true) { |
| switch (eppingHdr->Cmd_h) { |
| case EPPING_CMD_CONT_RX_STOP: |
| EPPING_LOG(VOS_TRACE_LEVEL_FATAL, |
| "%s: RXPERF: EID = %d, num_pkts_received = %u\n", |
| __func__, eid, pEpping_ctx->kperf_num_rx_recv[eid]); |
| OS_MEMCPY(eppingHdr->CmdBuffer_t, |
| &pEpping_ctx->kperf_num_rx_recv[eid], |
| sizeof(unsigned int)); |
| epping_set_kperf_flag(pAdapter, eid, false); |
| adf_net_buf_debug_release_skb(pktSkb); |
| netif_rx_ni(pktSkb); |
| break; |
| case 0: /* RXPERF hard code 0 in FW */ |
| adf_nbuf_free(pktSkb); |
| pEpping_ctx->kperf_num_rx_recv[eid]++; |
| if ((pAdapter->stats.rx_packets % EPPING_STATS_LOG_COUNT) == 0) { |
| EPPING_LOG(VOS_TRACE_LEVEL_FATAL, "%s: total_rx_pkts = %lu", |
| __func__, pAdapter->stats.rx_packets); |
| } |
| break; |
| case EPPING_CMD_CAPTURE_RECV_CNT: |
| epping_set_kperf_flag(pAdapter, eid, false); |
| adf_net_buf_debug_release_skb(pktSkb); |
| netif_rx_ni(pktSkb); |
| break; |
| default: |
| adf_net_buf_debug_release_skb(pktSkb); |
| netif_rx_ni(pktSkb); |
| pEpping_ctx->kperf_num_rx_recv[eid]++; |
| if ((pAdapter->stats.rx_packets % EPPING_STATS_LOG_COUNT) == 0) { |
| EPPING_LOG(VOS_TRACE_LEVEL_FATAL, "%s: total_rx_pkts = %lu", |
| __func__, pAdapter->stats.rx_packets); |
| } |
| break; |
| } |
| } else { |
| adf_net_buf_debug_release_skb(pktSkb); |
| netif_rx_ni(pktSkb); |
| if ((pAdapter->stats.rx_packets % EPPING_STATS_LOG_COUNT) == 0) { |
| EPPING_LOG(VOS_TRACE_LEVEL_FATAL, "%s: total_rx_pkts = %lu", |
| __func__, pAdapter->stats.rx_packets); |
| } |
| } |
| } else { |
| ++pAdapter->stats.rx_dropped; |
| adf_nbuf_free(pktSkb); |
| } |
| } |
| } |