xref: /openbmc/linux/drivers/net/wireless/ath/ath11k/ce.c (revision e3396b8bddd2ea822f9390f1ba49c22d769a7534)
1d5c65159SKalle Valo // SPDX-License-Identifier: BSD-3-Clause-Clear
2d5c65159SKalle Valo /*
3d5c65159SKalle Valo  * Copyright (c) 2018-2019 The Linux Foundation. All rights reserved.
4d5c65159SKalle Valo  */
5d5c65159SKalle Valo 
6d5c65159SKalle Valo #include "dp_rx.h"
7d5c65159SKalle Valo #include "debug.h"
8c4eacabeSGovind Singh #include "hif.h"
9d5c65159SKalle Valo 
10*e3396b8bSCarl Huang #define host_ce_config_wlan ab->hw_params.host_ce_config
11*e3396b8bSCarl Huang 
12*e3396b8bSCarl Huang const struct ce_attr ath11k_host_ce_config_ipq8074[] = {
13d5c65159SKalle Valo 	/* CE0: host->target HTC control and raw streams */
14d5c65159SKalle Valo 	{
15d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS,
16d5c65159SKalle Valo 		.src_nentries = 16,
17d5c65159SKalle Valo 		.src_sz_max = 2048,
18d5c65159SKalle Valo 		.dest_nentries = 0,
19d5c65159SKalle Valo 	},
20d5c65159SKalle Valo 
21d5c65159SKalle Valo 	/* CE1: target->host HTT + HTC control */
22d5c65159SKalle Valo 	{
23d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS,
24d5c65159SKalle Valo 		.src_nentries = 0,
25d5c65159SKalle Valo 		.src_sz_max = 2048,
26d5c65159SKalle Valo 		.dest_nentries = 512,
27d5c65159SKalle Valo 		.recv_cb = ath11k_htc_rx_completion_handler,
28d5c65159SKalle Valo 	},
29d5c65159SKalle Valo 
30d5c65159SKalle Valo 	/* CE2: target->host WMI */
31d5c65159SKalle Valo 	{
32d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS,
33d5c65159SKalle Valo 		.src_nentries = 0,
34d5c65159SKalle Valo 		.src_sz_max = 2048,
35d5c65159SKalle Valo 		.dest_nentries = 512,
36d5c65159SKalle Valo 		.recv_cb = ath11k_htc_rx_completion_handler,
37d5c65159SKalle Valo 	},
38d5c65159SKalle Valo 
39d5c65159SKalle Valo 	/* CE3: host->target WMI (mac0) */
40d5c65159SKalle Valo 	{
41d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS,
42d5c65159SKalle Valo 		.src_nentries = 32,
43d5c65159SKalle Valo 		.src_sz_max = 2048,
44d5c65159SKalle Valo 		.dest_nentries = 0,
45d5c65159SKalle Valo 	},
46d5c65159SKalle Valo 
47d5c65159SKalle Valo 	/* CE4: host->target HTT */
48d5c65159SKalle Valo 	{
49d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS | CE_ATTR_DIS_INTR,
50d5c65159SKalle Valo 		.src_nentries = 2048,
51d5c65159SKalle Valo 		.src_sz_max = 256,
52d5c65159SKalle Valo 		.dest_nentries = 0,
53d5c65159SKalle Valo 	},
54d5c65159SKalle Valo 
55d5c65159SKalle Valo 	/* CE5: target->host pktlog */
56d5c65159SKalle Valo 	{
57d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS,
58d5c65159SKalle Valo 		.src_nentries = 0,
59d5c65159SKalle Valo 		.src_sz_max = 2048,
60d5c65159SKalle Valo 		.dest_nentries = 512,
61d5c65159SKalle Valo 		.recv_cb = ath11k_dp_htt_htc_t2h_msg_handler,
62d5c65159SKalle Valo 	},
63d5c65159SKalle Valo 
64d5c65159SKalle Valo 	/* CE6: target autonomous hif_memcpy */
65d5c65159SKalle Valo 	{
66d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS | CE_ATTR_DIS_INTR,
67d5c65159SKalle Valo 		.src_nentries = 0,
68d5c65159SKalle Valo 		.src_sz_max = 0,
69d5c65159SKalle Valo 		.dest_nentries = 0,
70d5c65159SKalle Valo 	},
71d5c65159SKalle Valo 
72d5c65159SKalle Valo 	/* CE7: host->target WMI (mac1) */
73d5c65159SKalle Valo 	{
74d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS,
75d5c65159SKalle Valo 		.src_nentries = 32,
76d5c65159SKalle Valo 		.src_sz_max = 2048,
77d5c65159SKalle Valo 		.dest_nentries = 0,
78d5c65159SKalle Valo 	},
79d5c65159SKalle Valo 
80d5c65159SKalle Valo 	/* CE8: target autonomous hif_memcpy */
81d5c65159SKalle Valo 	{
82d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS,
83d5c65159SKalle Valo 		.src_nentries = 0,
84d5c65159SKalle Valo 		.src_sz_max = 0,
85d5c65159SKalle Valo 		.dest_nentries = 0,
86d5c65159SKalle Valo 	},
87d5c65159SKalle Valo 
88d5c65159SKalle Valo 	/* CE9: host->target WMI (mac2) */
89d5c65159SKalle Valo 	{
90d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS,
91d5c65159SKalle Valo 		.src_nentries = 32,
92d5c65159SKalle Valo 		.src_sz_max = 2048,
93d5c65159SKalle Valo 		.dest_nentries = 0,
94d5c65159SKalle Valo 	},
95d5c65159SKalle Valo 
96d5c65159SKalle Valo 	/* CE10: target->host HTT */
97d5c65159SKalle Valo 	{
98d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS,
99d5c65159SKalle Valo 		.src_nentries = 0,
100d5c65159SKalle Valo 		.src_sz_max = 2048,
101d5c65159SKalle Valo 		.dest_nentries = 512,
102d5c65159SKalle Valo 		.recv_cb = ath11k_htc_rx_completion_handler,
103d5c65159SKalle Valo 	},
104d5c65159SKalle Valo 
105d5c65159SKalle Valo 	/* CE11: Not used */
106d5c65159SKalle Valo 	{
107d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS,
108d5c65159SKalle Valo 		.src_nentries = 0,
109d5c65159SKalle Valo 		.src_sz_max = 0,
110d5c65159SKalle Valo 		.dest_nentries = 0,
111d5c65159SKalle Valo 	},
112d5c65159SKalle Valo };
113d5c65159SKalle Valo 
114*e3396b8bSCarl Huang const struct ce_attr ath11k_host_ce_config_qca6390[] = {
115*e3396b8bSCarl Huang 	/* CE0: host->target HTC control and raw streams */
116*e3396b8bSCarl Huang 	{
117*e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS,
118*e3396b8bSCarl Huang 		.src_nentries = 16,
119*e3396b8bSCarl Huang 		.src_sz_max = 2048,
120*e3396b8bSCarl Huang 		.dest_nentries = 0,
121*e3396b8bSCarl Huang 	},
122*e3396b8bSCarl Huang 
123*e3396b8bSCarl Huang 	/* CE1: target->host HTT + HTC control */
124*e3396b8bSCarl Huang 	{
125*e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS,
126*e3396b8bSCarl Huang 		.src_nentries = 0,
127*e3396b8bSCarl Huang 		.src_sz_max = 2048,
128*e3396b8bSCarl Huang 		.dest_nentries = 512,
129*e3396b8bSCarl Huang 		.recv_cb = ath11k_htc_rx_completion_handler,
130*e3396b8bSCarl Huang 	},
131*e3396b8bSCarl Huang 
132*e3396b8bSCarl Huang 	/* CE2: target->host WMI */
133*e3396b8bSCarl Huang 	{
134*e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS,
135*e3396b8bSCarl Huang 		.src_nentries = 0,
136*e3396b8bSCarl Huang 		.src_sz_max = 2048,
137*e3396b8bSCarl Huang 		.dest_nentries = 512,
138*e3396b8bSCarl Huang 		.recv_cb = ath11k_htc_rx_completion_handler,
139*e3396b8bSCarl Huang 	},
140*e3396b8bSCarl Huang 
141*e3396b8bSCarl Huang 	/* CE3: host->target WMI (mac0) */
142*e3396b8bSCarl Huang 	{
143*e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS,
144*e3396b8bSCarl Huang 		.src_nentries = 32,
145*e3396b8bSCarl Huang 		.src_sz_max = 2048,
146*e3396b8bSCarl Huang 		.dest_nentries = 0,
147*e3396b8bSCarl Huang 	},
148*e3396b8bSCarl Huang 
149*e3396b8bSCarl Huang 	/* CE4: host->target HTT */
150*e3396b8bSCarl Huang 	{
151*e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS | CE_ATTR_DIS_INTR,
152*e3396b8bSCarl Huang 		.src_nentries = 2048,
153*e3396b8bSCarl Huang 		.src_sz_max = 256,
154*e3396b8bSCarl Huang 		.dest_nentries = 0,
155*e3396b8bSCarl Huang 	},
156*e3396b8bSCarl Huang 
157*e3396b8bSCarl Huang 	/* CE5: target->host pktlog */
158*e3396b8bSCarl Huang 	{
159*e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS,
160*e3396b8bSCarl Huang 		.src_nentries = 0,
161*e3396b8bSCarl Huang 		.src_sz_max = 2048,
162*e3396b8bSCarl Huang 		.dest_nentries = 512,
163*e3396b8bSCarl Huang 		.recv_cb = ath11k_dp_htt_htc_t2h_msg_handler,
164*e3396b8bSCarl Huang 	},
165*e3396b8bSCarl Huang 
166*e3396b8bSCarl Huang 	/* CE6: target autonomous hif_memcpy */
167*e3396b8bSCarl Huang 	{
168*e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS | CE_ATTR_DIS_INTR,
169*e3396b8bSCarl Huang 		.src_nentries = 0,
170*e3396b8bSCarl Huang 		.src_sz_max = 0,
171*e3396b8bSCarl Huang 		.dest_nentries = 0,
172*e3396b8bSCarl Huang 	},
173*e3396b8bSCarl Huang 
174*e3396b8bSCarl Huang 	/* CE7: host->target WMI (mac1) */
175*e3396b8bSCarl Huang 	{
176*e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS,
177*e3396b8bSCarl Huang 		.src_nentries = 32,
178*e3396b8bSCarl Huang 		.src_sz_max = 2048,
179*e3396b8bSCarl Huang 		.dest_nentries = 0,
180*e3396b8bSCarl Huang 	},
181*e3396b8bSCarl Huang 
182*e3396b8bSCarl Huang 	/* CE8: target autonomous hif_memcpy */
183*e3396b8bSCarl Huang 	{
184*e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS,
185*e3396b8bSCarl Huang 		.src_nentries = 0,
186*e3396b8bSCarl Huang 		.src_sz_max = 0,
187*e3396b8bSCarl Huang 		.dest_nentries = 0,
188*e3396b8bSCarl Huang 	},
189*e3396b8bSCarl Huang 
190*e3396b8bSCarl Huang };
191*e3396b8bSCarl Huang 
192d5c65159SKalle Valo static int ath11k_ce_rx_buf_enqueue_pipe(struct ath11k_ce_pipe *pipe,
193d5c65159SKalle Valo 					 struct sk_buff *skb, dma_addr_t paddr)
194d5c65159SKalle Valo {
195d5c65159SKalle Valo 	struct ath11k_base *ab = pipe->ab;
196d5c65159SKalle Valo 	struct ath11k_ce_ring *ring = pipe->dest_ring;
197d5c65159SKalle Valo 	struct hal_srng *srng;
198d5c65159SKalle Valo 	unsigned int write_index;
199d5c65159SKalle Valo 	unsigned int nentries_mask = ring->nentries_mask;
200d5c65159SKalle Valo 	u32 *desc;
201d5c65159SKalle Valo 	int ret;
202d5c65159SKalle Valo 
203d5c65159SKalle Valo 	lockdep_assert_held(&ab->ce.ce_lock);
204d5c65159SKalle Valo 
205d5c65159SKalle Valo 	write_index = ring->write_index;
206d5c65159SKalle Valo 
207d5c65159SKalle Valo 	srng = &ab->hal.srng_list[ring->hal_ring_id];
208d5c65159SKalle Valo 
209d5c65159SKalle Valo 	spin_lock_bh(&srng->lock);
210d5c65159SKalle Valo 
211d5c65159SKalle Valo 	ath11k_hal_srng_access_begin(ab, srng);
212d5c65159SKalle Valo 
213d5c65159SKalle Valo 	if (unlikely(ath11k_hal_srng_src_num_free(ab, srng, false) < 1)) {
214d5c65159SKalle Valo 		ret = -ENOSPC;
215d5c65159SKalle Valo 		goto exit;
216d5c65159SKalle Valo 	}
217d5c65159SKalle Valo 
218d5c65159SKalle Valo 	desc = ath11k_hal_srng_src_get_next_entry(ab, srng);
219d5c65159SKalle Valo 	if (!desc) {
220d5c65159SKalle Valo 		ret = -ENOSPC;
221d5c65159SKalle Valo 		goto exit;
222d5c65159SKalle Valo 	}
223d5c65159SKalle Valo 
224d5c65159SKalle Valo 	ath11k_hal_ce_dst_set_desc(desc, paddr);
225d5c65159SKalle Valo 
226d5c65159SKalle Valo 	ring->skb[write_index] = skb;
227d5c65159SKalle Valo 	write_index = CE_RING_IDX_INCR(nentries_mask, write_index);
228d5c65159SKalle Valo 	ring->write_index = write_index;
229d5c65159SKalle Valo 
230d5c65159SKalle Valo 	pipe->rx_buf_needed--;
231d5c65159SKalle Valo 
232d5c65159SKalle Valo 	ret = 0;
233d5c65159SKalle Valo exit:
234d5c65159SKalle Valo 	ath11k_hal_srng_access_end(ab, srng);
235d5c65159SKalle Valo 
236d5c65159SKalle Valo 	spin_unlock_bh(&srng->lock);
237d5c65159SKalle Valo 
238d5c65159SKalle Valo 	return ret;
239d5c65159SKalle Valo }
240d5c65159SKalle Valo 
241d5c65159SKalle Valo static int ath11k_ce_rx_post_pipe(struct ath11k_ce_pipe *pipe)
242d5c65159SKalle Valo {
243d5c65159SKalle Valo 	struct ath11k_base *ab = pipe->ab;
244d5c65159SKalle Valo 	struct sk_buff *skb;
245d5c65159SKalle Valo 	dma_addr_t paddr;
246d5c65159SKalle Valo 	int ret = 0;
247d5c65159SKalle Valo 
248d5c65159SKalle Valo 	if (!(pipe->dest_ring || pipe->status_ring))
249d5c65159SKalle Valo 		return 0;
250d5c65159SKalle Valo 
251d5c65159SKalle Valo 	spin_lock_bh(&ab->ce.ce_lock);
252d5c65159SKalle Valo 	while (pipe->rx_buf_needed) {
253d5c65159SKalle Valo 		skb = dev_alloc_skb(pipe->buf_sz);
254d5c65159SKalle Valo 		if (!skb) {
255d5c65159SKalle Valo 			ret = -ENOMEM;
256d5c65159SKalle Valo 			goto exit;
257d5c65159SKalle Valo 		}
258d5c65159SKalle Valo 
259d5c65159SKalle Valo 		WARN_ON_ONCE(!IS_ALIGNED((unsigned long)skb->data, 4));
260d5c65159SKalle Valo 
261d5c65159SKalle Valo 		paddr = dma_map_single(ab->dev, skb->data,
262d5c65159SKalle Valo 				       skb->len + skb_tailroom(skb),
263d5c65159SKalle Valo 				       DMA_FROM_DEVICE);
264d5c65159SKalle Valo 		if (unlikely(dma_mapping_error(ab->dev, paddr))) {
265d5c65159SKalle Valo 			ath11k_warn(ab, "failed to dma map ce rx buf\n");
266d5c65159SKalle Valo 			dev_kfree_skb_any(skb);
267d5c65159SKalle Valo 			ret = -EIO;
268d5c65159SKalle Valo 			goto exit;
269d5c65159SKalle Valo 		}
270d5c65159SKalle Valo 
271d5c65159SKalle Valo 		ATH11K_SKB_RXCB(skb)->paddr = paddr;
272d5c65159SKalle Valo 
273d5c65159SKalle Valo 		ret = ath11k_ce_rx_buf_enqueue_pipe(pipe, skb, paddr);
274d5c65159SKalle Valo 
275d5c65159SKalle Valo 		if (ret) {
276d5c65159SKalle Valo 			ath11k_warn(ab, "failed to enqueue rx buf: %d\n", ret);
277d5c65159SKalle Valo 			dma_unmap_single(ab->dev, paddr,
278d5c65159SKalle Valo 					 skb->len + skb_tailroom(skb),
279d5c65159SKalle Valo 					 DMA_FROM_DEVICE);
280d5c65159SKalle Valo 			dev_kfree_skb_any(skb);
281d5c65159SKalle Valo 			goto exit;
282d5c65159SKalle Valo 		}
283d5c65159SKalle Valo 	}
284d5c65159SKalle Valo 
285d5c65159SKalle Valo exit:
286d5c65159SKalle Valo 	spin_unlock_bh(&ab->ce.ce_lock);
287d5c65159SKalle Valo 	return ret;
288d5c65159SKalle Valo }
289d5c65159SKalle Valo 
290d5c65159SKalle Valo static int ath11k_ce_completed_recv_next(struct ath11k_ce_pipe *pipe,
291d5c65159SKalle Valo 					 struct sk_buff **skb, int *nbytes)
292d5c65159SKalle Valo {
293d5c65159SKalle Valo 	struct ath11k_base *ab = pipe->ab;
294d5c65159SKalle Valo 	struct hal_srng *srng;
295d5c65159SKalle Valo 	unsigned int sw_index;
296d5c65159SKalle Valo 	unsigned int nentries_mask;
297d5c65159SKalle Valo 	u32 *desc;
298d5c65159SKalle Valo 	int ret = 0;
299d5c65159SKalle Valo 
300d5c65159SKalle Valo 	spin_lock_bh(&ab->ce.ce_lock);
301d5c65159SKalle Valo 
302d5c65159SKalle Valo 	sw_index = pipe->dest_ring->sw_index;
303d5c65159SKalle Valo 	nentries_mask = pipe->dest_ring->nentries_mask;
304d5c65159SKalle Valo 
305d5c65159SKalle Valo 	srng = &ab->hal.srng_list[pipe->status_ring->hal_ring_id];
306d5c65159SKalle Valo 
307d5c65159SKalle Valo 	spin_lock_bh(&srng->lock);
308d5c65159SKalle Valo 
309d5c65159SKalle Valo 	ath11k_hal_srng_access_begin(ab, srng);
310d5c65159SKalle Valo 
311d5c65159SKalle Valo 	desc = ath11k_hal_srng_dst_get_next_entry(ab, srng);
312d5c65159SKalle Valo 	if (!desc) {
313d5c65159SKalle Valo 		ret = -EIO;
314d5c65159SKalle Valo 		goto err;
315d5c65159SKalle Valo 	}
316d5c65159SKalle Valo 
317d5c65159SKalle Valo 	*nbytes = ath11k_hal_ce_dst_status_get_length(desc);
318d5c65159SKalle Valo 	if (*nbytes == 0) {
319d5c65159SKalle Valo 		ret = -EIO;
320d5c65159SKalle Valo 		goto err;
321d5c65159SKalle Valo 	}
322d5c65159SKalle Valo 
323d5c65159SKalle Valo 	*skb = pipe->dest_ring->skb[sw_index];
324d5c65159SKalle Valo 	pipe->dest_ring->skb[sw_index] = NULL;
325d5c65159SKalle Valo 
326d5c65159SKalle Valo 	sw_index = CE_RING_IDX_INCR(nentries_mask, sw_index);
327d5c65159SKalle Valo 	pipe->dest_ring->sw_index = sw_index;
328d5c65159SKalle Valo 
329d5c65159SKalle Valo 	pipe->rx_buf_needed++;
330d5c65159SKalle Valo err:
331d5c65159SKalle Valo 	ath11k_hal_srng_access_end(ab, srng);
332d5c65159SKalle Valo 
333d5c65159SKalle Valo 	spin_unlock_bh(&srng->lock);
334d5c65159SKalle Valo 
335d5c65159SKalle Valo 	spin_unlock_bh(&ab->ce.ce_lock);
336d5c65159SKalle Valo 
337d5c65159SKalle Valo 	return ret;
338d5c65159SKalle Valo }
339d5c65159SKalle Valo 
340d5c65159SKalle Valo static void ath11k_ce_recv_process_cb(struct ath11k_ce_pipe *pipe)
341d5c65159SKalle Valo {
342d5c65159SKalle Valo 	struct ath11k_base *ab = pipe->ab;
343d5c65159SKalle Valo 	struct sk_buff *skb;
344d5c65159SKalle Valo 	struct sk_buff_head list;
345d5c65159SKalle Valo 	unsigned int nbytes, max_nbytes;
346d5c65159SKalle Valo 	int ret;
347d5c65159SKalle Valo 
348d5c65159SKalle Valo 	__skb_queue_head_init(&list);
349d5c65159SKalle Valo 	while (ath11k_ce_completed_recv_next(pipe, &skb, &nbytes) == 0) {
350d5c65159SKalle Valo 		max_nbytes = skb->len + skb_tailroom(skb);
351d5c65159SKalle Valo 		dma_unmap_single(ab->dev, ATH11K_SKB_RXCB(skb)->paddr,
352d5c65159SKalle Valo 				 max_nbytes, DMA_FROM_DEVICE);
353d5c65159SKalle Valo 
354d5c65159SKalle Valo 		if (unlikely(max_nbytes < nbytes)) {
355d5c65159SKalle Valo 			ath11k_warn(ab, "rxed more than expected (nbytes %d, max %d)",
356d5c65159SKalle Valo 				    nbytes, max_nbytes);
357d5c65159SKalle Valo 			dev_kfree_skb_any(skb);
358d5c65159SKalle Valo 			continue;
359d5c65159SKalle Valo 		}
360d5c65159SKalle Valo 
361d5c65159SKalle Valo 		skb_put(skb, nbytes);
362d5c65159SKalle Valo 		__skb_queue_tail(&list, skb);
363d5c65159SKalle Valo 	}
364d5c65159SKalle Valo 
365d5c65159SKalle Valo 	while ((skb = __skb_dequeue(&list))) {
366d5c65159SKalle Valo 		ath11k_dbg(ab, ATH11K_DBG_AHB, "rx ce pipe %d len %d\n",
367d5c65159SKalle Valo 			   pipe->pipe_num, skb->len);
368d5c65159SKalle Valo 		pipe->recv_cb(ab, skb);
369d5c65159SKalle Valo 	}
370d5c65159SKalle Valo 
371d5c65159SKalle Valo 	ret = ath11k_ce_rx_post_pipe(pipe);
372d5c65159SKalle Valo 	if (ret && ret != -ENOSPC) {
373d5c65159SKalle Valo 		ath11k_warn(ab, "failed to post rx buf to pipe: %d err: %d\n",
374d5c65159SKalle Valo 			    pipe->pipe_num, ret);
375d5c65159SKalle Valo 		mod_timer(&ab->rx_replenish_retry,
376d5c65159SKalle Valo 			  jiffies + ATH11K_CE_RX_POST_RETRY_JIFFIES);
377d5c65159SKalle Valo 	}
378d5c65159SKalle Valo }
379d5c65159SKalle Valo 
380d5c65159SKalle Valo static struct sk_buff *ath11k_ce_completed_send_next(struct ath11k_ce_pipe *pipe)
381d5c65159SKalle Valo {
382d5c65159SKalle Valo 	struct ath11k_base *ab = pipe->ab;
383d5c65159SKalle Valo 	struct hal_srng *srng;
384d5c65159SKalle Valo 	unsigned int sw_index;
385d5c65159SKalle Valo 	unsigned int nentries_mask;
386d5c65159SKalle Valo 	struct sk_buff *skb;
387d5c65159SKalle Valo 	u32 *desc;
388d5c65159SKalle Valo 
389d5c65159SKalle Valo 	spin_lock_bh(&ab->ce.ce_lock);
390d5c65159SKalle Valo 
391d5c65159SKalle Valo 	sw_index = pipe->src_ring->sw_index;
392d5c65159SKalle Valo 	nentries_mask = pipe->src_ring->nentries_mask;
393d5c65159SKalle Valo 
394d5c65159SKalle Valo 	srng = &ab->hal.srng_list[pipe->src_ring->hal_ring_id];
395d5c65159SKalle Valo 
396d5c65159SKalle Valo 	spin_lock_bh(&srng->lock);
397d5c65159SKalle Valo 
398d5c65159SKalle Valo 	ath11k_hal_srng_access_begin(ab, srng);
399d5c65159SKalle Valo 
400d5c65159SKalle Valo 	desc = ath11k_hal_srng_src_reap_next(ab, srng);
401d5c65159SKalle Valo 	if (!desc) {
402d5c65159SKalle Valo 		skb = ERR_PTR(-EIO);
403d5c65159SKalle Valo 		goto err_unlock;
404d5c65159SKalle Valo 	}
405d5c65159SKalle Valo 
406d5c65159SKalle Valo 	skb = pipe->src_ring->skb[sw_index];
407d5c65159SKalle Valo 
408d5c65159SKalle Valo 	pipe->src_ring->skb[sw_index] = NULL;
409d5c65159SKalle Valo 
410d5c65159SKalle Valo 	sw_index = CE_RING_IDX_INCR(nentries_mask, sw_index);
411d5c65159SKalle Valo 	pipe->src_ring->sw_index = sw_index;
412d5c65159SKalle Valo 
413d5c65159SKalle Valo err_unlock:
414d5c65159SKalle Valo 	spin_unlock_bh(&srng->lock);
415d5c65159SKalle Valo 
416d5c65159SKalle Valo 	spin_unlock_bh(&ab->ce.ce_lock);
417d5c65159SKalle Valo 
418d5c65159SKalle Valo 	return skb;
419d5c65159SKalle Valo }
420d5c65159SKalle Valo 
421d5c65159SKalle Valo static void ath11k_ce_send_done_cb(struct ath11k_ce_pipe *pipe)
422d5c65159SKalle Valo {
423d5c65159SKalle Valo 	struct ath11k_base *ab = pipe->ab;
424d5c65159SKalle Valo 	struct sk_buff *skb;
425d5c65159SKalle Valo 
426d5c65159SKalle Valo 	while (!IS_ERR(skb = ath11k_ce_completed_send_next(pipe))) {
427d5c65159SKalle Valo 		if (!skb)
428d5c65159SKalle Valo 			continue;
429d5c65159SKalle Valo 
430d5c65159SKalle Valo 		dma_unmap_single(ab->dev, ATH11K_SKB_CB(skb)->paddr, skb->len,
431d5c65159SKalle Valo 				 DMA_TO_DEVICE);
432d5c65159SKalle Valo 		dev_kfree_skb_any(skb);
433d5c65159SKalle Valo 	}
434d5c65159SKalle Valo }
435d5c65159SKalle Valo 
436c4eacabeSGovind Singh static void ath11k_ce_srng_msi_ring_params_setup(struct ath11k_base *ab, u32 ce_id,
437c4eacabeSGovind Singh 						 struct hal_srng_params *ring_params)
438c4eacabeSGovind Singh {
439c4eacabeSGovind Singh 	u32 msi_data_start;
440c4eacabeSGovind Singh 	u32 msi_data_count;
441c4eacabeSGovind Singh 	u32 msi_irq_start;
442c4eacabeSGovind Singh 	u32 addr_lo;
443c4eacabeSGovind Singh 	u32 addr_hi;
444c4eacabeSGovind Singh 	int ret;
445c4eacabeSGovind Singh 
446c4eacabeSGovind Singh 	ret = ath11k_get_user_msi_vector(ab, "CE",
447c4eacabeSGovind Singh 					 &msi_data_count, &msi_data_start,
448c4eacabeSGovind Singh 					 &msi_irq_start);
449c4eacabeSGovind Singh 
450c4eacabeSGovind Singh 	if (ret)
451c4eacabeSGovind Singh 		return;
452c4eacabeSGovind Singh 
453c4eacabeSGovind Singh 	ath11k_get_msi_address(ab, &addr_lo, &addr_hi);
454c4eacabeSGovind Singh 
455c4eacabeSGovind Singh 	ring_params->msi_addr = addr_lo;
456c4eacabeSGovind Singh 	ring_params->msi_addr |= (dma_addr_t)(((uint64_t)addr_hi) << 32);
457c4eacabeSGovind Singh 	ring_params->msi_data = (ce_id % msi_data_count) + msi_data_start;
458c4eacabeSGovind Singh 	ring_params->flags |= HAL_SRNG_FLAGS_MSI_INTR;
459c4eacabeSGovind Singh }
460c4eacabeSGovind Singh 
461d5c65159SKalle Valo static int ath11k_ce_init_ring(struct ath11k_base *ab,
462d5c65159SKalle Valo 			       struct ath11k_ce_ring *ce_ring,
463d5c65159SKalle Valo 			       int ce_id, enum hal_ring_type type)
464d5c65159SKalle Valo {
465d5c65159SKalle Valo 	struct hal_srng_params params = { 0 };
466d5c65159SKalle Valo 	int ret;
467d5c65159SKalle Valo 
468d5c65159SKalle Valo 	params.ring_base_paddr = ce_ring->base_addr_ce_space;
469d5c65159SKalle Valo 	params.ring_base_vaddr = ce_ring->base_addr_owner_space;
470d5c65159SKalle Valo 	params.num_entries = ce_ring->nentries;
471d5c65159SKalle Valo 
472d5c65159SKalle Valo 	switch (type) {
473d5c65159SKalle Valo 	case HAL_CE_SRC:
474d5c65159SKalle Valo 		if (!(CE_ATTR_DIS_INTR & host_ce_config_wlan[ce_id].flags))
475d5c65159SKalle Valo 			params.intr_batch_cntr_thres_entries = 1;
476d5c65159SKalle Valo 		break;
477d5c65159SKalle Valo 	case HAL_CE_DST:
478d5c65159SKalle Valo 		params.max_buffer_len = host_ce_config_wlan[ce_id].src_sz_max;
479d5c65159SKalle Valo 		if (!(host_ce_config_wlan[ce_id].flags & CE_ATTR_DIS_INTR)) {
480d5c65159SKalle Valo 			params.intr_timer_thres_us = 1024;
481d5c65159SKalle Valo 			params.flags |= HAL_SRNG_FLAGS_LOW_THRESH_INTR_EN;
482d5c65159SKalle Valo 			params.low_threshold = ce_ring->nentries - 3;
483d5c65159SKalle Valo 		}
484d5c65159SKalle Valo 		break;
485d5c65159SKalle Valo 	case HAL_CE_DST_STATUS:
486d5c65159SKalle Valo 		if (!(host_ce_config_wlan[ce_id].flags & CE_ATTR_DIS_INTR)) {
487d5c65159SKalle Valo 			params.intr_batch_cntr_thres_entries = 1;
488d5c65159SKalle Valo 			params.intr_timer_thres_us = 0x1000;
489d5c65159SKalle Valo 		}
490d5c65159SKalle Valo 		break;
491d5c65159SKalle Valo 	default:
492d5c65159SKalle Valo 		ath11k_warn(ab, "Invalid CE ring type %d\n", type);
493d5c65159SKalle Valo 		return -EINVAL;
494d5c65159SKalle Valo 	}
495d5c65159SKalle Valo 
496d5c65159SKalle Valo 	/* TODO: Init other params needed by HAL to init the ring */
497d5c65159SKalle Valo 
498d5c65159SKalle Valo 	ret = ath11k_hal_srng_setup(ab, type, ce_id, 0, &params);
499d5c65159SKalle Valo 	if (ret < 0) {
500d5c65159SKalle Valo 		ath11k_warn(ab, "failed to setup srng: %d ring_id %d\n",
501d5c65159SKalle Valo 			    ret, ce_id);
502d5c65159SKalle Valo 		return ret;
503d5c65159SKalle Valo 	}
504c4eacabeSGovind Singh 
505c4eacabeSGovind Singh 	if (!(CE_ATTR_DIS_INTR & host_ce_config_wlan[ce_id].flags))
506c4eacabeSGovind Singh 		ath11k_ce_srng_msi_ring_params_setup(ab, ce_id, &params);
507c4eacabeSGovind Singh 
508d5c65159SKalle Valo 	ce_ring->hal_ring_id = ret;
509d5c65159SKalle Valo 
510d5c65159SKalle Valo 	return 0;
511d5c65159SKalle Valo }
512d5c65159SKalle Valo 
513d5c65159SKalle Valo static struct ath11k_ce_ring *
514d5c65159SKalle Valo ath11k_ce_alloc_ring(struct ath11k_base *ab, int nentries, int desc_sz)
515d5c65159SKalle Valo {
516d5c65159SKalle Valo 	struct ath11k_ce_ring *ce_ring;
517d5c65159SKalle Valo 	dma_addr_t base_addr;
518d5c65159SKalle Valo 
519d5c65159SKalle Valo 	ce_ring = kzalloc(struct_size(ce_ring, skb, nentries), GFP_KERNEL);
520d5c65159SKalle Valo 	if (ce_ring == NULL)
521d5c65159SKalle Valo 		return ERR_PTR(-ENOMEM);
522d5c65159SKalle Valo 
523d5c65159SKalle Valo 	ce_ring->nentries = nentries;
524d5c65159SKalle Valo 	ce_ring->nentries_mask = nentries - 1;
525d5c65159SKalle Valo 
526d5c65159SKalle Valo 	/* Legacy platforms that do not support cache
527d5c65159SKalle Valo 	 * coherent DMA are unsupported
528d5c65159SKalle Valo 	 */
529d5c65159SKalle Valo 	ce_ring->base_addr_owner_space_unaligned =
530d5c65159SKalle Valo 		dma_alloc_coherent(ab->dev,
531d5c65159SKalle Valo 				   nentries * desc_sz + CE_DESC_RING_ALIGN,
532d5c65159SKalle Valo 				   &base_addr, GFP_KERNEL);
533d5c65159SKalle Valo 	if (!ce_ring->base_addr_owner_space_unaligned) {
534d5c65159SKalle Valo 		kfree(ce_ring);
535d5c65159SKalle Valo 		return ERR_PTR(-ENOMEM);
536d5c65159SKalle Valo 	}
537d5c65159SKalle Valo 
538d5c65159SKalle Valo 	ce_ring->base_addr_ce_space_unaligned = base_addr;
539d5c65159SKalle Valo 
540d5c65159SKalle Valo 	ce_ring->base_addr_owner_space = PTR_ALIGN(
541d5c65159SKalle Valo 			ce_ring->base_addr_owner_space_unaligned,
542d5c65159SKalle Valo 			CE_DESC_RING_ALIGN);
543d5c65159SKalle Valo 	ce_ring->base_addr_ce_space = ALIGN(
544d5c65159SKalle Valo 			ce_ring->base_addr_ce_space_unaligned,
545d5c65159SKalle Valo 			CE_DESC_RING_ALIGN);
546d5c65159SKalle Valo 
547d5c65159SKalle Valo 	return ce_ring;
548d5c65159SKalle Valo }
549d5c65159SKalle Valo 
550d5c65159SKalle Valo static int ath11k_ce_alloc_pipe(struct ath11k_base *ab, int ce_id)
551d5c65159SKalle Valo {
552d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe = &ab->ce.ce_pipe[ce_id];
553d5c65159SKalle Valo 	const struct ce_attr *attr = &host_ce_config_wlan[ce_id];
554c76fa846SDan Carpenter 	struct ath11k_ce_ring *ring;
555d5c65159SKalle Valo 	int nentries;
556d5c65159SKalle Valo 	int desc_sz;
557d5c65159SKalle Valo 
558d5c65159SKalle Valo 	pipe->attr_flags = attr->flags;
559d5c65159SKalle Valo 
560d5c65159SKalle Valo 	if (attr->src_nentries) {
561d5c65159SKalle Valo 		pipe->send_cb = ath11k_ce_send_done_cb;
562d5c65159SKalle Valo 		nentries = roundup_pow_of_two(attr->src_nentries);
563d5c65159SKalle Valo 		desc_sz = ath11k_hal_ce_get_desc_size(HAL_CE_DESC_SRC);
564c76fa846SDan Carpenter 		ring = ath11k_ce_alloc_ring(ab, nentries, desc_sz);
565c76fa846SDan Carpenter 		if (IS_ERR(ring))
566c76fa846SDan Carpenter 			return PTR_ERR(ring);
567c76fa846SDan Carpenter 		pipe->src_ring = ring;
568d5c65159SKalle Valo 	}
569d5c65159SKalle Valo 
570d5c65159SKalle Valo 	if (attr->dest_nentries) {
571d5c65159SKalle Valo 		pipe->recv_cb = attr->recv_cb;
572d5c65159SKalle Valo 		nentries = roundup_pow_of_two(attr->dest_nentries);
573d5c65159SKalle Valo 		desc_sz = ath11k_hal_ce_get_desc_size(HAL_CE_DESC_DST);
574c76fa846SDan Carpenter 		ring = ath11k_ce_alloc_ring(ab, nentries, desc_sz);
575c76fa846SDan Carpenter 		if (IS_ERR(ring))
576c76fa846SDan Carpenter 			return PTR_ERR(ring);
577c76fa846SDan Carpenter 		pipe->dest_ring = ring;
578d5c65159SKalle Valo 
579d5c65159SKalle Valo 		desc_sz = ath11k_hal_ce_get_desc_size(HAL_CE_DESC_DST_STATUS);
580c76fa846SDan Carpenter 		ring = ath11k_ce_alloc_ring(ab, nentries, desc_sz);
581c76fa846SDan Carpenter 		if (IS_ERR(ring))
582c76fa846SDan Carpenter 			return PTR_ERR(ring);
583c76fa846SDan Carpenter 		pipe->status_ring = ring;
584d5c65159SKalle Valo 	}
585d5c65159SKalle Valo 
586d5c65159SKalle Valo 	return 0;
587d5c65159SKalle Valo }
588d5c65159SKalle Valo 
589d5c65159SKalle Valo void ath11k_ce_per_engine_service(struct ath11k_base *ab, u16 ce_id)
590d5c65159SKalle Valo {
591d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe = &ab->ce.ce_pipe[ce_id];
592d5c65159SKalle Valo 
593d5c65159SKalle Valo 	if (pipe->send_cb)
594d5c65159SKalle Valo 		pipe->send_cb(pipe);
595d5c65159SKalle Valo 
596d5c65159SKalle Valo 	if (pipe->recv_cb)
597d5c65159SKalle Valo 		ath11k_ce_recv_process_cb(pipe);
598d5c65159SKalle Valo }
599d5c65159SKalle Valo 
600d5c65159SKalle Valo void ath11k_ce_poll_send_completed(struct ath11k_base *ab, u8 pipe_id)
601d5c65159SKalle Valo {
602d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe = &ab->ce.ce_pipe[pipe_id];
603d5c65159SKalle Valo 
604d5c65159SKalle Valo 	if ((pipe->attr_flags & CE_ATTR_DIS_INTR) && pipe->send_cb)
605d5c65159SKalle Valo 		pipe->send_cb(pipe);
606d5c65159SKalle Valo }
6072c3960c2SGovind Singh EXPORT_SYMBOL(ath11k_ce_per_engine_service);
608d5c65159SKalle Valo 
609d5c65159SKalle Valo int ath11k_ce_send(struct ath11k_base *ab, struct sk_buff *skb, u8 pipe_id,
610d5c65159SKalle Valo 		   u16 transfer_id)
611d5c65159SKalle Valo {
612d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe = &ab->ce.ce_pipe[pipe_id];
613d5c65159SKalle Valo 	struct hal_srng *srng;
614d5c65159SKalle Valo 	u32 *desc;
615d5c65159SKalle Valo 	unsigned int write_index, sw_index;
616d5c65159SKalle Valo 	unsigned int nentries_mask;
617d5c65159SKalle Valo 	int ret = 0;
618d5c65159SKalle Valo 	u8 byte_swap_data = 0;
619d5c65159SKalle Valo 	int num_used;
620d5c65159SKalle Valo 
621d5c65159SKalle Valo 	/* Check if some entries could be regained by handling tx completion if
622d5c65159SKalle Valo 	 * the CE has interrupts disabled and the used entries is more than the
623d5c65159SKalle Valo 	 * defined usage threshold.
624d5c65159SKalle Valo 	 */
625d5c65159SKalle Valo 	if (pipe->attr_flags & CE_ATTR_DIS_INTR) {
626d5c65159SKalle Valo 		spin_lock_bh(&ab->ce.ce_lock);
627d5c65159SKalle Valo 		write_index = pipe->src_ring->write_index;
628d5c65159SKalle Valo 
629d5c65159SKalle Valo 		sw_index = pipe->src_ring->sw_index;
630d5c65159SKalle Valo 
631d5c65159SKalle Valo 		if (write_index >= sw_index)
632d5c65159SKalle Valo 			num_used = write_index - sw_index;
633d5c65159SKalle Valo 		else
634d5c65159SKalle Valo 			num_used = pipe->src_ring->nentries - sw_index +
635d5c65159SKalle Valo 				   write_index;
636d5c65159SKalle Valo 
637d5c65159SKalle Valo 		spin_unlock_bh(&ab->ce.ce_lock);
638d5c65159SKalle Valo 
639d5c65159SKalle Valo 		if (num_used > ATH11K_CE_USAGE_THRESHOLD)
640d5c65159SKalle Valo 			ath11k_ce_poll_send_completed(ab, pipe->pipe_num);
641d5c65159SKalle Valo 	}
642d5c65159SKalle Valo 
643d5c65159SKalle Valo 	if (test_bit(ATH11K_FLAG_CRASH_FLUSH, &ab->dev_flags))
644d5c65159SKalle Valo 		return -ESHUTDOWN;
645d5c65159SKalle Valo 
646d5c65159SKalle Valo 	spin_lock_bh(&ab->ce.ce_lock);
647d5c65159SKalle Valo 
648d5c65159SKalle Valo 	write_index = pipe->src_ring->write_index;
649d5c65159SKalle Valo 	nentries_mask = pipe->src_ring->nentries_mask;
650d5c65159SKalle Valo 
651d5c65159SKalle Valo 	srng = &ab->hal.srng_list[pipe->src_ring->hal_ring_id];
652d5c65159SKalle Valo 
653d5c65159SKalle Valo 	spin_lock_bh(&srng->lock);
654d5c65159SKalle Valo 
655d5c65159SKalle Valo 	ath11k_hal_srng_access_begin(ab, srng);
656d5c65159SKalle Valo 
657d5c65159SKalle Valo 	if (unlikely(ath11k_hal_srng_src_num_free(ab, srng, false) < 1)) {
658d5c65159SKalle Valo 		ath11k_hal_srng_access_end(ab, srng);
659d5c65159SKalle Valo 		ret = -ENOBUFS;
660d5c65159SKalle Valo 		goto err_unlock;
661d5c65159SKalle Valo 	}
662d5c65159SKalle Valo 
663d5c65159SKalle Valo 	desc = ath11k_hal_srng_src_get_next_reaped(ab, srng);
664d5c65159SKalle Valo 	if (!desc) {
665d5c65159SKalle Valo 		ath11k_hal_srng_access_end(ab, srng);
666d5c65159SKalle Valo 		ret = -ENOBUFS;
667d5c65159SKalle Valo 		goto err_unlock;
668d5c65159SKalle Valo 	}
669d5c65159SKalle Valo 
670d5c65159SKalle Valo 	if (pipe->attr_flags & CE_ATTR_BYTE_SWAP_DATA)
671d5c65159SKalle Valo 		byte_swap_data = 1;
672d5c65159SKalle Valo 
673d5c65159SKalle Valo 	ath11k_hal_ce_src_set_desc(desc, ATH11K_SKB_CB(skb)->paddr,
674d5c65159SKalle Valo 				   skb->len, transfer_id, byte_swap_data);
675d5c65159SKalle Valo 
676d5c65159SKalle Valo 	pipe->src_ring->skb[write_index] = skb;
677d5c65159SKalle Valo 	pipe->src_ring->write_index = CE_RING_IDX_INCR(nentries_mask,
678d5c65159SKalle Valo 						       write_index);
679d5c65159SKalle Valo 
680d5c65159SKalle Valo 	ath11k_hal_srng_access_end(ab, srng);
681d5c65159SKalle Valo 
682d5c65159SKalle Valo 	spin_unlock_bh(&srng->lock);
683d5c65159SKalle Valo 
684d5c65159SKalle Valo 	spin_unlock_bh(&ab->ce.ce_lock);
685d5c65159SKalle Valo 
686d5c65159SKalle Valo 	return 0;
687d5c65159SKalle Valo 
688d5c65159SKalle Valo err_unlock:
689d5c65159SKalle Valo 	spin_unlock_bh(&srng->lock);
690d5c65159SKalle Valo 
691d5c65159SKalle Valo 	spin_unlock_bh(&ab->ce.ce_lock);
692d5c65159SKalle Valo 
693d5c65159SKalle Valo 	return ret;
694d5c65159SKalle Valo }
695d5c65159SKalle Valo 
696d5c65159SKalle Valo static void ath11k_ce_rx_pipe_cleanup(struct ath11k_ce_pipe *pipe)
697d5c65159SKalle Valo {
698d5c65159SKalle Valo 	struct ath11k_base *ab = pipe->ab;
699d5c65159SKalle Valo 	struct ath11k_ce_ring *ring = pipe->dest_ring;
700d5c65159SKalle Valo 	struct sk_buff *skb;
701d5c65159SKalle Valo 	int i;
702d5c65159SKalle Valo 
703d5c65159SKalle Valo 	if (!(ring && pipe->buf_sz))
704d5c65159SKalle Valo 		return;
705d5c65159SKalle Valo 
706d5c65159SKalle Valo 	for (i = 0; i < ring->nentries; i++) {
707d5c65159SKalle Valo 		skb = ring->skb[i];
708d5c65159SKalle Valo 		if (!skb)
709d5c65159SKalle Valo 			continue;
710d5c65159SKalle Valo 
711d5c65159SKalle Valo 		ring->skb[i] = NULL;
712d5c65159SKalle Valo 		dma_unmap_single(ab->dev, ATH11K_SKB_RXCB(skb)->paddr,
713d5c65159SKalle Valo 				 skb->len + skb_tailroom(skb), DMA_FROM_DEVICE);
714d5c65159SKalle Valo 		dev_kfree_skb_any(skb);
715d5c65159SKalle Valo 	}
716d5c65159SKalle Valo }
717d5c65159SKalle Valo 
718d5c65159SKalle Valo void ath11k_ce_cleanup_pipes(struct ath11k_base *ab)
719d5c65159SKalle Valo {
720d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe;
721d5c65159SKalle Valo 	int pipe_num;
722d5c65159SKalle Valo 
723d5c65159SKalle Valo 	for (pipe_num = 0; pipe_num < CE_COUNT; pipe_num++) {
724d5c65159SKalle Valo 		pipe = &ab->ce.ce_pipe[pipe_num];
725d5c65159SKalle Valo 		ath11k_ce_rx_pipe_cleanup(pipe);
726d5c65159SKalle Valo 
727d5c65159SKalle Valo 		/* Cleanup any src CE's which have interrupts disabled */
728d5c65159SKalle Valo 		ath11k_ce_poll_send_completed(ab, pipe_num);
729d5c65159SKalle Valo 
730d5c65159SKalle Valo 		/* NOTE: Should we also clean up tx buffer in all pipes? */
731d5c65159SKalle Valo 	}
732d5c65159SKalle Valo }
7337f4beda2SGovind Singh EXPORT_SYMBOL(ath11k_ce_cleanup_pipes);
734d5c65159SKalle Valo 
735d5c65159SKalle Valo void ath11k_ce_rx_post_buf(struct ath11k_base *ab)
736d5c65159SKalle Valo {
737d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe;
738d5c65159SKalle Valo 	int i;
739d5c65159SKalle Valo 	int ret;
740d5c65159SKalle Valo 
741d5c65159SKalle Valo 	for (i = 0; i < CE_COUNT; i++) {
742d5c65159SKalle Valo 		pipe = &ab->ce.ce_pipe[i];
743d5c65159SKalle Valo 		ret = ath11k_ce_rx_post_pipe(pipe);
744d5c65159SKalle Valo 		if (ret) {
745d5c65159SKalle Valo 			if (ret == -ENOSPC)
746d5c65159SKalle Valo 				continue;
747d5c65159SKalle Valo 
748d5c65159SKalle Valo 			ath11k_warn(ab, "failed to post rx buf to pipe: %d err: %d\n",
749d5c65159SKalle Valo 				    i, ret);
750d5c65159SKalle Valo 			mod_timer(&ab->rx_replenish_retry,
751d5c65159SKalle Valo 				  jiffies + ATH11K_CE_RX_POST_RETRY_JIFFIES);
752d5c65159SKalle Valo 
753d5c65159SKalle Valo 			return;
754d5c65159SKalle Valo 		}
755d5c65159SKalle Valo 	}
756d5c65159SKalle Valo }
7572c3960c2SGovind Singh EXPORT_SYMBOL(ath11k_ce_rx_post_buf);
758d5c65159SKalle Valo 
759d5c65159SKalle Valo void ath11k_ce_rx_replenish_retry(struct timer_list *t)
760d5c65159SKalle Valo {
761d5c65159SKalle Valo 	struct ath11k_base *ab = from_timer(ab, t, rx_replenish_retry);
762d5c65159SKalle Valo 
763d5c65159SKalle Valo 	ath11k_ce_rx_post_buf(ab);
764d5c65159SKalle Valo }
765d5c65159SKalle Valo 
766d5c65159SKalle Valo int ath11k_ce_init_pipes(struct ath11k_base *ab)
767d5c65159SKalle Valo {
768d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe;
769d5c65159SKalle Valo 	int i;
770d5c65159SKalle Valo 	int ret;
771d5c65159SKalle Valo 
772d5c65159SKalle Valo 	for (i = 0; i < CE_COUNT; i++) {
773d5c65159SKalle Valo 		pipe = &ab->ce.ce_pipe[i];
774d5c65159SKalle Valo 
775d5c65159SKalle Valo 		if (pipe->src_ring) {
776d5c65159SKalle Valo 			ret = ath11k_ce_init_ring(ab, pipe->src_ring, i,
777d5c65159SKalle Valo 						  HAL_CE_SRC);
778d5c65159SKalle Valo 			if (ret) {
779d5c65159SKalle Valo 				ath11k_warn(ab, "failed to init src ring: %d\n",
780d5c65159SKalle Valo 					    ret);
781d5c65159SKalle Valo 				/* Should we clear any partial init */
782d5c65159SKalle Valo 				return ret;
783d5c65159SKalle Valo 			}
784d5c65159SKalle Valo 
785d5c65159SKalle Valo 			pipe->src_ring->write_index = 0;
786d5c65159SKalle Valo 			pipe->src_ring->sw_index = 0;
787d5c65159SKalle Valo 		}
788d5c65159SKalle Valo 
789d5c65159SKalle Valo 		if (pipe->dest_ring) {
790d5c65159SKalle Valo 			ret = ath11k_ce_init_ring(ab, pipe->dest_ring, i,
791d5c65159SKalle Valo 						  HAL_CE_DST);
792d5c65159SKalle Valo 			if (ret) {
793d5c65159SKalle Valo 				ath11k_warn(ab, "failed to init dest ring: %d\n",
794d5c65159SKalle Valo 					    ret);
795d5c65159SKalle Valo 				/* Should we clear any partial init */
796d5c65159SKalle Valo 				return ret;
797d5c65159SKalle Valo 			}
798d5c65159SKalle Valo 
799d5c65159SKalle Valo 			pipe->rx_buf_needed = pipe->dest_ring->nentries ?
800d5c65159SKalle Valo 					      pipe->dest_ring->nentries - 2 : 0;
801d5c65159SKalle Valo 
802d5c65159SKalle Valo 			pipe->dest_ring->write_index = 0;
803d5c65159SKalle Valo 			pipe->dest_ring->sw_index = 0;
804d5c65159SKalle Valo 		}
805d5c65159SKalle Valo 
806d5c65159SKalle Valo 		if (pipe->status_ring) {
807d5c65159SKalle Valo 			ret = ath11k_ce_init_ring(ab, pipe->status_ring, i,
808d5c65159SKalle Valo 						  HAL_CE_DST_STATUS);
809d5c65159SKalle Valo 			if (ret) {
810d5c65159SKalle Valo 				ath11k_warn(ab, "failed to init dest status ing: %d\n",
811d5c65159SKalle Valo 					    ret);
812d5c65159SKalle Valo 				/* Should we clear any partial init */
813d5c65159SKalle Valo 				return ret;
814d5c65159SKalle Valo 			}
815d5c65159SKalle Valo 
816d5c65159SKalle Valo 			pipe->status_ring->write_index = 0;
817d5c65159SKalle Valo 			pipe->status_ring->sw_index = 0;
818d5c65159SKalle Valo 		}
819d5c65159SKalle Valo 	}
820d5c65159SKalle Valo 
821d5c65159SKalle Valo 	return 0;
822d5c65159SKalle Valo }
823d5c65159SKalle Valo 
824d5c65159SKalle Valo void ath11k_ce_free_pipes(struct ath11k_base *ab)
825d5c65159SKalle Valo {
826d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe;
827d5c65159SKalle Valo 	int desc_sz;
828d5c65159SKalle Valo 	int i;
829d5c65159SKalle Valo 
830d5c65159SKalle Valo 	for (i = 0; i < CE_COUNT; i++) {
831d5c65159SKalle Valo 		pipe = &ab->ce.ce_pipe[i];
832d5c65159SKalle Valo 
833d5c65159SKalle Valo 		if (pipe->src_ring) {
834d5c65159SKalle Valo 			desc_sz = ath11k_hal_ce_get_desc_size(HAL_CE_DESC_SRC);
835d5c65159SKalle Valo 			dma_free_coherent(ab->dev,
836d5c65159SKalle Valo 					  pipe->src_ring->nentries * desc_sz +
837d5c65159SKalle Valo 					  CE_DESC_RING_ALIGN,
838d5c65159SKalle Valo 					  pipe->src_ring->base_addr_owner_space,
839d5c65159SKalle Valo 					  pipe->src_ring->base_addr_ce_space);
840d5c65159SKalle Valo 			kfree(pipe->src_ring);
841d5c65159SKalle Valo 			pipe->src_ring = NULL;
842d5c65159SKalle Valo 		}
843d5c65159SKalle Valo 
844d5c65159SKalle Valo 		if (pipe->dest_ring) {
845d5c65159SKalle Valo 			desc_sz = ath11k_hal_ce_get_desc_size(HAL_CE_DESC_DST);
846d5c65159SKalle Valo 			dma_free_coherent(ab->dev,
847d5c65159SKalle Valo 					  pipe->dest_ring->nentries * desc_sz +
848d5c65159SKalle Valo 					  CE_DESC_RING_ALIGN,
849d5c65159SKalle Valo 					  pipe->dest_ring->base_addr_owner_space,
850d5c65159SKalle Valo 					  pipe->dest_ring->base_addr_ce_space);
851d5c65159SKalle Valo 			kfree(pipe->dest_ring);
852d5c65159SKalle Valo 			pipe->dest_ring = NULL;
853d5c65159SKalle Valo 		}
854d5c65159SKalle Valo 
855d5c65159SKalle Valo 		if (pipe->status_ring) {
856d5c65159SKalle Valo 			desc_sz =
857d5c65159SKalle Valo 			  ath11k_hal_ce_get_desc_size(HAL_CE_DESC_DST_STATUS);
858d5c65159SKalle Valo 			dma_free_coherent(ab->dev,
859d5c65159SKalle Valo 					  pipe->status_ring->nentries * desc_sz +
860d5c65159SKalle Valo 					  CE_DESC_RING_ALIGN,
861d5c65159SKalle Valo 					  pipe->status_ring->base_addr_owner_space,
862d5c65159SKalle Valo 					  pipe->status_ring->base_addr_ce_space);
863d5c65159SKalle Valo 			kfree(pipe->status_ring);
864d5c65159SKalle Valo 			pipe->status_ring = NULL;
865d5c65159SKalle Valo 		}
866d5c65159SKalle Valo 	}
867d5c65159SKalle Valo }
8686e0355afSGovind Singh EXPORT_SYMBOL(ath11k_ce_free_pipes);
869d5c65159SKalle Valo 
870d5c65159SKalle Valo int ath11k_ce_alloc_pipes(struct ath11k_base *ab)
871d5c65159SKalle Valo {
872d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe;
873d5c65159SKalle Valo 	int i;
874d5c65159SKalle Valo 	int ret;
875d5c65159SKalle Valo 	const struct ce_attr *attr;
876d5c65159SKalle Valo 
877d5c65159SKalle Valo 	spin_lock_init(&ab->ce.ce_lock);
878d5c65159SKalle Valo 
879d5c65159SKalle Valo 	for (i = 0; i < CE_COUNT; i++) {
880d5c65159SKalle Valo 		attr = &host_ce_config_wlan[i];
881d5c65159SKalle Valo 		pipe = &ab->ce.ce_pipe[i];
882d5c65159SKalle Valo 		pipe->pipe_num = i;
883d5c65159SKalle Valo 		pipe->ab = ab;
884d5c65159SKalle Valo 		pipe->buf_sz = attr->src_sz_max;
885d5c65159SKalle Valo 
886d5c65159SKalle Valo 		ret = ath11k_ce_alloc_pipe(ab, i);
887d5c65159SKalle Valo 		if (ret) {
888d5c65159SKalle Valo 			/* Free any parial successful allocation */
889d5c65159SKalle Valo 			ath11k_ce_free_pipes(ab);
890d5c65159SKalle Valo 			return ret;
891d5c65159SKalle Valo 		}
892d5c65159SKalle Valo 	}
893d5c65159SKalle Valo 
894d5c65159SKalle Valo 	return 0;
895d5c65159SKalle Valo }
8967f4beda2SGovind Singh EXPORT_SYMBOL(ath11k_ce_alloc_pipes);
897d5c65159SKalle Valo 
898d5c65159SKalle Valo /* For Big Endian Host, Copy Engine byte_swap is enabled
899d5c65159SKalle Valo  * When Copy Engine does byte_swap, need to byte swap again for the
900d5c65159SKalle Valo  * Host to get/put buffer content in the correct byte order
901d5c65159SKalle Valo  */
902d5c65159SKalle Valo void ath11k_ce_byte_swap(void *mem, u32 len)
903d5c65159SKalle Valo {
904d5c65159SKalle Valo 	int i;
905d5c65159SKalle Valo 
906d5c65159SKalle Valo 	if (IS_ENABLED(CONFIG_CPU_BIG_ENDIAN)) {
907d5c65159SKalle Valo 		if (!mem)
908d5c65159SKalle Valo 			return;
909d5c65159SKalle Valo 
910d5c65159SKalle Valo 		for (i = 0; i < (len / 4); i++) {
911d5c65159SKalle Valo 			*(u32 *)mem = swab32(*(u32 *)mem);
912d5c65159SKalle Valo 			mem += 4;
913d5c65159SKalle Valo 		}
914d5c65159SKalle Valo 	}
915d5c65159SKalle Valo }
916d5c65159SKalle Valo 
917*e3396b8bSCarl Huang int ath11k_ce_get_attr_flags(struct ath11k_base *ab, int ce_id)
918d5c65159SKalle Valo {
919d5c65159SKalle Valo 	if (ce_id >= CE_COUNT)
920d5c65159SKalle Valo 		return -EINVAL;
921d5c65159SKalle Valo 
922d5c65159SKalle Valo 	return host_ce_config_wlan[ce_id].flags;
923d5c65159SKalle Valo }
9246e0355afSGovind Singh EXPORT_SYMBOL(ath11k_ce_get_attr_flags);
925