ath10k: enable pktlog for WCN3990 target
authorGovind Singh <govinds@codeaurora.org>
Mon, 3 Sep 2018 17:08:01 +0000 (20:08 +0300)
committerKalle Valo <kvalo@codeaurora.org>
Thu, 6 Sep 2018 15:55:32 +0000 (18:55 +0300)
WCN3990 target uses separate htc service for pktlog.
Add pktlog service request and support for pktlog
rx path handling.

Testing:
    Tested on WCN3990 and QCA6174 HW.
    Tested FW: WLAN.HL.2.0-01192-QCAHLSWMTPLZ-1,
               WLAN.RM.4.4.1-00109-QCARMSWPZ-1

Signed-off-by: Govind Singh <govinds@codeaurora.org>
Signed-off-by: Kalle Valo <kvalo@codeaurora.org>
drivers/net/wireless/ath/ath10k/htc.c
drivers/net/wireless/ath/ath10k/pci.c
drivers/net/wireless/ath/ath10k/snoc.c

index 331b8d558791d3c36d605669cee6658fac22f937..4e70ff2f78ae3badb7b1eacd28887dfdfcf2d6aa 100644 (file)
@@ -803,8 +803,11 @@ setup:
                                                ep->service_id,
                                                &ep->ul_pipe_id,
                                                &ep->dl_pipe_id);
-       if (status)
+       if (status) {
+               ath10k_warn(ar, "unsupported HTC service id: %d\n",
+                           ep->service_id);
                return status;
+       }
 
        ath10k_dbg(ar, ATH10K_DBG_BOOT,
                   "boot htc service '%s' ul pipe %d dl pipe %d eid %d ready\n",
@@ -838,6 +841,56 @@ struct sk_buff *ath10k_htc_alloc_skb(struct ath10k *ar, int size)
        return skb;
 }
 
+static void ath10k_htc_pktlog_process_rx(struct ath10k *ar, struct sk_buff *skb)
+{
+       trace_ath10k_htt_pktlog(ar, skb->data, skb->len);
+       dev_kfree_skb_any(skb);
+}
+
+static int ath10k_htc_pktlog_connect(struct ath10k *ar)
+{
+       struct ath10k_htc_svc_conn_resp conn_resp;
+       struct ath10k_htc_svc_conn_req conn_req;
+       int status;
+
+       memset(&conn_req, 0, sizeof(conn_req));
+       memset(&conn_resp, 0, sizeof(conn_resp));
+
+       conn_req.ep_ops.ep_tx_complete = NULL;
+       conn_req.ep_ops.ep_rx_complete = ath10k_htc_pktlog_process_rx;
+       conn_req.ep_ops.ep_tx_credits = NULL;
+
+       /* connect to control service */
+       conn_req.service_id = ATH10K_HTC_SVC_ID_HTT_LOG_MSG;
+       status = ath10k_htc_connect_service(&ar->htc, &conn_req, &conn_resp);
+       if (status) {
+               ath10k_warn(ar, "failed to connect to PKTLOG service: %d\n",
+                           status);
+               return status;
+       }
+
+       return 0;
+}
+
+static bool ath10k_htc_pktlog_svc_supported(struct ath10k *ar)
+{
+       u8 ul_pipe_id;
+       u8 dl_pipe_id;
+       int status;
+
+       status = ath10k_hif_map_service_to_pipe(ar, ATH10K_HTC_SVC_ID_HTT_LOG_MSG,
+                                               &ul_pipe_id,
+                                               &dl_pipe_id);
+       if (status) {
+               ath10k_warn(ar, "unsupported HTC service id: %d\n",
+                           ATH10K_HTC_SVC_ID_HTT_LOG_MSG);
+
+               return false;
+       }
+
+       return true;
+}
+
 int ath10k_htc_start(struct ath10k_htc *htc)
 {
        struct ath10k *ar = htc->ar;
@@ -871,6 +924,14 @@ int ath10k_htc_start(struct ath10k_htc *htc)
                return status;
        }
 
+       if (ath10k_htc_pktlog_svc_supported(ar)) {
+               status = ath10k_htc_pktlog_connect(ar);
+               if (status) {
+                       ath10k_err(ar, "failed to connect to pktlog: %d\n", status);
+                       return status;
+               }
+       }
+
        return 0;
 }
 
index 7122e6aedc1e3a540bf1aabfc083c02a92cc748d..7db4a79156248cbbf6ec4c3eb1e7f0cb15f91510 100644 (file)
@@ -1839,7 +1839,7 @@ int ath10k_pci_hif_map_service_to_pipe(struct ath10k *ar, u16 service_id,
                }
        }
 
-       if (WARN_ON(!ul_set || !dl_set))
+       if (!ul_set || !dl_set)
                return -ENOENT;
 
        return 0;
index fa1843a7e0fdaaec3e74a96d9a8b359a921c9746..eabf4229d929928176681f43c296ea8597b86427 100644 (file)
@@ -62,6 +62,7 @@ static void ath10k_snoc_htt_tx_cb(struct ath10k_ce_pipe *ce_state);
 static void ath10k_snoc_htc_rx_cb(struct ath10k_ce_pipe *ce_state);
 static void ath10k_snoc_htt_rx_cb(struct ath10k_ce_pipe *ce_state);
 static void ath10k_snoc_htt_htc_rx_cb(struct ath10k_ce_pipe *ce_state);
+static void ath10k_snoc_pktlog_rx_cb(struct ath10k_ce_pipe *ce_state);
 
 static const struct ath10k_snoc_drv_priv drv_priv = {
        .hw_rev = ATH10K_HW_WCN3990,
@@ -171,7 +172,7 @@ static struct ce_attr host_ce_config_wlan[] = {
                .src_nentries = 0,
                .src_sz_max = 2048,
                .dest_nentries = 512,
-               .recv_cb = ath10k_snoc_htt_htc_rx_cb,
+               .recv_cb = ath10k_snoc_pktlog_rx_cb,
        },
 };
 
@@ -436,6 +437,14 @@ static void ath10k_snoc_htt_htc_rx_cb(struct ath10k_ce_pipe *ce_state)
        ath10k_snoc_process_rx_cb(ce_state, ath10k_htc_rx_completion_handler);
 }
 
+/* Called by lower (CE) layer when data is received from the Target.
+ * WCN3990 firmware uses separate CE(CE11) to transfer pktlog data.
+ */
+static void ath10k_snoc_pktlog_rx_cb(struct ath10k_ce_pipe *ce_state)
+{
+       ath10k_snoc_process_rx_cb(ce_state, ath10k_htc_rx_completion_handler);
+}
+
 static void ath10k_snoc_htt_rx_deliver(struct ath10k *ar, struct sk_buff *skb)
 {
        skb_pull(skb, sizeof(struct ath10k_htc_hdr));
@@ -616,7 +625,7 @@ static int ath10k_snoc_hif_map_service_to_pipe(struct ath10k *ar,
                }
        }
 
-       if (WARN_ON(!ul_set || !dl_set))
+       if (!ul_set || !dl_set)
                return -ENOENT;
 
        return 0;