|
@@ -833,17 +833,19 @@ void htt_tx_desc_display(void *tx_desc)
|
|
|
#endif
|
|
|
|
|
|
#ifdef IPA_OFFLOAD
|
|
|
+#ifdef QCA_WIFI_2_0
|
|
|
/**
|
|
|
- * htt_tx_ipa_uc_attach() - attach htt ipa uc tx resource
|
|
|
+ * htt_tx_ipa_uc_wdi_tx_buf_alloc() - Alloc WDI TX buffers
|
|
|
* @pdev: htt context
|
|
|
- * @uc_tx_buf_sz: single tx buffer size
|
|
|
- * @uc_tx_buf_cnt: total tx buffer count
|
|
|
- * @uc_tx_partition_base: tx buffer partition start
|
|
|
+ * @uc_tx_buf_sz: TX buffer size
|
|
|
+ * @uc_tx_buf_cnt: TX Buffer count
|
|
|
+ * @uc_tx_partition_base: IPA UC TX partition base value
|
|
|
+ *
|
|
|
+ * Allocate WDI TX buffers. Also note Rome supports only WDI 1.0.
|
|
|
*
|
|
|
* Return: 0 success
|
|
|
- * ENOBUFS No memory fail
|
|
|
*/
|
|
|
-int htt_tx_ipa_uc_attach(struct htt_pdev_t *pdev,
|
|
|
+int htt_tx_ipa_uc_wdi_tx_buf_alloc(struct htt_pdev_t *pdev,
|
|
|
unsigned int uc_tx_buf_sz,
|
|
|
unsigned int uc_tx_buf_cnt,
|
|
|
unsigned int uc_tx_partition_base)
|
|
@@ -853,52 +855,68 @@ int htt_tx_ipa_uc_attach(struct htt_pdev_t *pdev,
|
|
|
cdf_dma_addr_t buffer_paddr;
|
|
|
uint32_t *header_ptr;
|
|
|
uint32_t *ring_vaddr;
|
|
|
- int return_code = 0;
|
|
|
- unsigned int tx_comp_ring_size;
|
|
|
+#define IPA_UC_TX_BUF_FRAG_DESC_OFFSET 16
|
|
|
+#define IPA_UC_TX_BUF_FRAG_HDR_OFFSET 32
|
|
|
|
|
|
- /* Allocate CE Write Index WORD */
|
|
|
- pdev->ipa_uc_tx_rsc.tx_ce_idx.vaddr =
|
|
|
- cdf_os_mem_alloc_consistent(
|
|
|
- pdev->osdev,
|
|
|
- 4,
|
|
|
- &pdev->ipa_uc_tx_rsc.tx_ce_idx.paddr,
|
|
|
- cdf_get_dma_mem_context(
|
|
|
- (&pdev->ipa_uc_tx_rsc.tx_ce_idx),
|
|
|
- memctx));
|
|
|
- if (!pdev->ipa_uc_tx_rsc.tx_ce_idx.vaddr) {
|
|
|
- cdf_print("%s: CE Write Index WORD alloc fail", __func__);
|
|
|
- return -ENOBUFS;
|
|
|
- }
|
|
|
+ ring_vaddr = pdev->ipa_uc_tx_rsc.tx_comp_base.vaddr;
|
|
|
+ /* Allocate TX buffers as many as possible */
|
|
|
+ for (tx_buffer_count = 0;
|
|
|
+ tx_buffer_count < (uc_tx_buf_cnt - 1); tx_buffer_count++) {
|
|
|
+ buffer_vaddr = cdf_nbuf_alloc(pdev->osdev,
|
|
|
+ uc_tx_buf_sz, 0, 4, false);
|
|
|
+ if (!buffer_vaddr) {
|
|
|
+ cdf_print("%s: TX BUF alloc fail, loop index: %d",
|
|
|
+ __func__, tx_buffer_count);
|
|
|
+ return tx_buffer_count;
|
|
|
+ }
|
|
|
|
|
|
- /* Allocate TX COMP Ring */
|
|
|
- tx_comp_ring_size = uc_tx_buf_cnt * sizeof(cdf_nbuf_t);
|
|
|
- pdev->ipa_uc_tx_rsc.tx_comp_base.vaddr =
|
|
|
- cdf_os_mem_alloc_consistent(
|
|
|
- pdev->osdev,
|
|
|
- tx_comp_ring_size,
|
|
|
- &pdev->ipa_uc_tx_rsc.tx_comp_base.paddr,
|
|
|
- cdf_get_dma_mem_context((&pdev->ipa_uc_tx_rsc.
|
|
|
- tx_comp_base),
|
|
|
- memctx));
|
|
|
- if (!pdev->ipa_uc_tx_rsc.tx_comp_base.vaddr) {
|
|
|
- cdf_print("%s: TX COMP ring alloc fail", __func__);
|
|
|
- return_code = -ENOBUFS;
|
|
|
- goto free_tx_ce_idx;
|
|
|
- }
|
|
|
+ /* Init buffer */
|
|
|
+ cdf_mem_zero(cdf_nbuf_data(buffer_vaddr), uc_tx_buf_sz);
|
|
|
+ header_ptr = (uint32_t *) cdf_nbuf_data(buffer_vaddr);
|
|
|
|
|
|
- cdf_mem_zero(pdev->ipa_uc_tx_rsc.tx_comp_base.vaddr, tx_comp_ring_size);
|
|
|
+ /* HTT control header */
|
|
|
+ *header_ptr = HTT_IPA_UC_OFFLOAD_TX_HEADER_DEFAULT;
|
|
|
+ header_ptr++;
|
|
|
|
|
|
- /* Allocate TX BUF vAddress Storage */
|
|
|
- pdev->ipa_uc_tx_rsc.tx_buf_pool_vaddr_strg =
|
|
|
- (cdf_nbuf_t *) cdf_mem_malloc(uc_tx_buf_cnt *
|
|
|
- sizeof(cdf_nbuf_t));
|
|
|
- if (!pdev->ipa_uc_tx_rsc.tx_buf_pool_vaddr_strg) {
|
|
|
- cdf_print("%s: TX BUF POOL vaddr storage alloc fail", __func__);
|
|
|
- return_code = -ENOBUFS;
|
|
|
- goto free_tx_comp_base;
|
|
|
+ /* PKT ID */
|
|
|
+ *header_ptr |= ((uint16_t) uc_tx_partition_base +
|
|
|
+ tx_buffer_count) << 16;
|
|
|
+
|
|
|
+ cdf_nbuf_map(pdev->osdev, buffer_vaddr, CDF_DMA_BIDIRECTIONAL);
|
|
|
+ buffer_paddr = cdf_nbuf_get_frag_paddr_lo(buffer_vaddr, 0);
|
|
|
+ header_ptr++;
|
|
|
+ *header_ptr = (uint32_t) (buffer_paddr +
|
|
|
+ IPA_UC_TX_BUF_FRAG_DESC_OFFSET);
|
|
|
+ header_ptr++;
|
|
|
+ *header_ptr = 0xFFFFFFFF;
|
|
|
+
|
|
|
+ /* FRAG Header */
|
|
|
+ header_ptr++;
|
|
|
+ *header_ptr = buffer_paddr + IPA_UC_TX_BUF_FRAG_HDR_OFFSET;
|
|
|
+
|
|
|
+ *ring_vaddr = buffer_paddr;
|
|
|
+ pdev->ipa_uc_tx_rsc.tx_buf_pool_vaddr_strg[tx_buffer_count] =
|
|
|
+ buffer_vaddr;
|
|
|
+ /* Memory barrier to ensure actual value updated */
|
|
|
+
|
|
|
+ ring_vaddr++;
|
|
|
}
|
|
|
- cdf_mem_zero(pdev->ipa_uc_tx_rsc.tx_buf_pool_vaddr_strg,
|
|
|
- uc_tx_buf_cnt * sizeof(cdf_nbuf_t));
|
|
|
+ return tx_buffer_count;
|
|
|
+}
|
|
|
+#else
|
|
|
+int htt_tx_ipa_uc_wdi_tx_buf_alloc(struct htt_pdev_t *pdev,
|
|
|
+ unsigned int uc_tx_buf_sz,
|
|
|
+ unsigned int uc_tx_buf_cnt,
|
|
|
+ unsigned int uc_tx_partition_base)
|
|
|
+{
|
|
|
+ unsigned int tx_buffer_count;
|
|
|
+ cdf_nbuf_t buffer_vaddr;
|
|
|
+ uint32_t buffer_paddr;
|
|
|
+ uint32_t *header_ptr;
|
|
|
+ uint32_t *ring_vaddr;
|
|
|
+#define IPA_UC_TX_BUF_FRAG_DESC_OFFSET 20
|
|
|
+#define IPA_UC_TX_BUF_FRAG_HDR_OFFSET 64
|
|
|
+#define IPA_UC_TX_BUF_TSO_HDR_SIZE 6
|
|
|
|
|
|
ring_vaddr = pdev->ipa_uc_tx_rsc.tx_comp_base.vaddr;
|
|
|
/* Allocate TX buffers as many as possible */
|
|
@@ -909,7 +927,7 @@ int htt_tx_ipa_uc_attach(struct htt_pdev_t *pdev,
|
|
|
if (!buffer_vaddr) {
|
|
|
cdf_print("%s: TX BUF alloc fail, loop index: %d",
|
|
|
__func__, tx_buffer_count);
|
|
|
- return 0;
|
|
|
+ return tx_buffer_count;
|
|
|
}
|
|
|
|
|
|
/* Init buffer */
|
|
@@ -930,7 +948,8 @@ int htt_tx_ipa_uc_attach(struct htt_pdev_t *pdev,
|
|
|
|
|
|
/* Frag Desc Pointer */
|
|
|
/* 64bits descriptor, Low 32bits */
|
|
|
- *header_ptr = (uint32_t) (buffer_paddr + 20);
|
|
|
+ *header_ptr = (uint32_t) (buffer_paddr +
|
|
|
+ IPA_UC_TX_BUF_FRAG_DESC_OFFSET);
|
|
|
header_ptr++;
|
|
|
|
|
|
/* 64bits descriptor, high 32bits */
|
|
@@ -943,8 +962,8 @@ int htt_tx_ipa_uc_attach(struct htt_pdev_t *pdev,
|
|
|
|
|
|
/* FRAG Header */
|
|
|
/* 6 words TSO header */
|
|
|
- header_ptr += 6;
|
|
|
- *header_ptr = buffer_paddr + 64;
|
|
|
+ header_ptr += IPA_UC_TX_BUF_TSO_HDR_SIZE;
|
|
|
+ *header_ptr = buffer_paddr + IPA_UC_TX_BUF_FRAG_HDR_OFFSET;
|
|
|
|
|
|
*ring_vaddr = buffer_paddr;
|
|
|
pdev->ipa_uc_tx_rsc.tx_buf_pool_vaddr_strg[tx_buffer_count] =
|
|
@@ -953,8 +972,75 @@ int htt_tx_ipa_uc_attach(struct htt_pdev_t *pdev,
|
|
|
|
|
|
ring_vaddr += 2;
|
|
|
}
|
|
|
+ return tx_buffer_count;
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
+/**
|
|
|
+ * htt_tx_ipa_uc_attach() - attach htt ipa uc tx resource
|
|
|
+ * @pdev: htt context
|
|
|
+ * @uc_tx_buf_sz: single tx buffer size
|
|
|
+ * @uc_tx_buf_cnt: total tx buffer count
|
|
|
+ * @uc_tx_partition_base: tx buffer partition start
|
|
|
+ *
|
|
|
+ * Return: 0 success
|
|
|
+ * ENOBUFS No memory fail
|
|
|
+ */
|
|
|
+int htt_tx_ipa_uc_attach(struct htt_pdev_t *pdev,
|
|
|
+ unsigned int uc_tx_buf_sz,
|
|
|
+ unsigned int uc_tx_buf_cnt,
|
|
|
+ unsigned int uc_tx_partition_base)
|
|
|
+{
|
|
|
+ int return_code = 0;
|
|
|
+ unsigned int tx_comp_ring_size;
|
|
|
+
|
|
|
+ /* Allocate CE Write Index WORD */
|
|
|
+ pdev->ipa_uc_tx_rsc.tx_ce_idx.vaddr =
|
|
|
+ cdf_os_mem_alloc_consistent(
|
|
|
+ pdev->osdev,
|
|
|
+ 4,
|
|
|
+ &pdev->ipa_uc_tx_rsc.tx_ce_idx.paddr,
|
|
|
+ cdf_get_dma_mem_context(
|
|
|
+ (&pdev->ipa_uc_tx_rsc.tx_ce_idx),
|
|
|
+ memctx));
|
|
|
+ if (!pdev->ipa_uc_tx_rsc.tx_ce_idx.vaddr) {
|
|
|
+ cdf_print("%s: CE Write Index WORD alloc fail", __func__);
|
|
|
+ return -ENOBUFS;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Allocate TX COMP Ring */
|
|
|
+ tx_comp_ring_size = uc_tx_buf_cnt * sizeof(cdf_nbuf_t);
|
|
|
+ pdev->ipa_uc_tx_rsc.tx_comp_base.vaddr =
|
|
|
+ cdf_os_mem_alloc_consistent(
|
|
|
+ pdev->osdev,
|
|
|
+ tx_comp_ring_size,
|
|
|
+ &pdev->ipa_uc_tx_rsc.tx_comp_base.paddr,
|
|
|
+ cdf_get_dma_mem_context((&pdev->ipa_uc_tx_rsc.
|
|
|
+ tx_comp_base),
|
|
|
+ memctx));
|
|
|
+ if (!pdev->ipa_uc_tx_rsc.tx_comp_base.vaddr) {
|
|
|
+ cdf_print("%s: TX COMP ring alloc fail", __func__);
|
|
|
+ return_code = -ENOBUFS;
|
|
|
+ goto free_tx_ce_idx;
|
|
|
+ }
|
|
|
+
|
|
|
+ cdf_mem_zero(pdev->ipa_uc_tx_rsc.tx_comp_base.vaddr, tx_comp_ring_size);
|
|
|
+
|
|
|
+ /* Allocate TX BUF vAddress Storage */
|
|
|
+ pdev->ipa_uc_tx_rsc.tx_buf_pool_vaddr_strg =
|
|
|
+ (cdf_nbuf_t *) cdf_mem_malloc(uc_tx_buf_cnt *
|
|
|
+ sizeof(cdf_nbuf_t));
|
|
|
+ if (!pdev->ipa_uc_tx_rsc.tx_buf_pool_vaddr_strg) {
|
|
|
+ cdf_print("%s: TX BUF POOL vaddr storage alloc fail", __func__);
|
|
|
+ return_code = -ENOBUFS;
|
|
|
+ goto free_tx_comp_base;
|
|
|
+ }
|
|
|
+ cdf_mem_zero(pdev->ipa_uc_tx_rsc.tx_buf_pool_vaddr_strg,
|
|
|
+ uc_tx_buf_cnt * sizeof(cdf_nbuf_t));
|
|
|
+
|
|
|
+ pdev->ipa_uc_tx_rsc.alloc_tx_buf_cnt = htt_tx_ipa_uc_wdi_tx_buf_alloc(
|
|
|
+ pdev, uc_tx_buf_sz, uc_tx_buf_cnt, uc_tx_partition_base);
|
|
|
|
|
|
- pdev->ipa_uc_tx_rsc.alloc_tx_buf_cnt = tx_buffer_count;
|
|
|
|
|
|
return 0;
|
|
|
|