xref: /openbmc/linux/drivers/net/wireless/ath/ath11k/ce.c (revision 4f52ec65)
1d5c65159SKalle Valo // SPDX-License-Identifier: BSD-3-Clause-Clear
2d5c65159SKalle Valo /*
3d5c65159SKalle Valo  * Copyright (c) 2018-2019 The Linux Foundation. All rights reserved.
433b67a4bSManikanta Pubbisetty  * Copyright (c) 2021, Qualcomm Innovation Center, Inc. All rights reserved.
5d5c65159SKalle Valo  */
6d5c65159SKalle Valo 
7d5c65159SKalle Valo #include "dp_rx.h"
8d5c65159SKalle Valo #include "debug.h"
9c4eacabeSGovind Singh #include "hif.h"
10d5c65159SKalle Valo 
11e3396b8bSCarl Huang const struct ce_attr ath11k_host_ce_config_ipq8074[] = {
12d5c65159SKalle Valo 	/* CE0: host->target HTC control and raw streams */
13d5c65159SKalle Valo 	{
14d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS,
15d5c65159SKalle Valo 		.src_nentries = 16,
16d5c65159SKalle Valo 		.src_sz_max = 2048,
17d5c65159SKalle Valo 		.dest_nentries = 0,
18f951380aSP Praneesh 		.send_cb = ath11k_htc_tx_completion_handler,
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,
45f951380aSP Praneesh 		.send_cb = ath11k_htc_tx_completion_handler,
46d5c65159SKalle Valo 	},
47d5c65159SKalle Valo 
48d5c65159SKalle Valo 	/* CE4: host->target HTT */
49d5c65159SKalle Valo 	{
50d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS | CE_ATTR_DIS_INTR,
51d5c65159SKalle Valo 		.src_nentries = 2048,
52d5c65159SKalle Valo 		.src_sz_max = 256,
53d5c65159SKalle Valo 		.dest_nentries = 0,
54d5c65159SKalle Valo 	},
55d5c65159SKalle Valo 
56d5c65159SKalle Valo 	/* CE5: target->host pktlog */
57d5c65159SKalle Valo 	{
58d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS,
59d5c65159SKalle Valo 		.src_nentries = 0,
60d5c65159SKalle Valo 		.src_sz_max = 2048,
61d5c65159SKalle Valo 		.dest_nentries = 512,
62d5c65159SKalle Valo 		.recv_cb = ath11k_dp_htt_htc_t2h_msg_handler,
63d5c65159SKalle Valo 	},
64d5c65159SKalle Valo 
65d5c65159SKalle Valo 	/* CE6: target autonomous hif_memcpy */
66d5c65159SKalle Valo 	{
67d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS | CE_ATTR_DIS_INTR,
68d5c65159SKalle Valo 		.src_nentries = 0,
69d5c65159SKalle Valo 		.src_sz_max = 0,
70d5c65159SKalle Valo 		.dest_nentries = 0,
71d5c65159SKalle Valo 	},
72d5c65159SKalle Valo 
73d5c65159SKalle Valo 	/* CE7: host->target WMI (mac1) */
74d5c65159SKalle Valo 	{
75d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS,
76d5c65159SKalle Valo 		.src_nentries = 32,
77d5c65159SKalle Valo 		.src_sz_max = 2048,
78d5c65159SKalle Valo 		.dest_nentries = 0,
79f951380aSP Praneesh 		.send_cb = ath11k_htc_tx_completion_handler,
80d5c65159SKalle Valo 	},
81d5c65159SKalle Valo 
82d5c65159SKalle Valo 	/* CE8: target autonomous hif_memcpy */
83d5c65159SKalle Valo 	{
842c5545bfSP Praneesh 		.flags = CE_ATTR_FLAGS | CE_ATTR_DIS_INTR,
85d5c65159SKalle Valo 		.src_nentries = 0,
86d5c65159SKalle Valo 		.src_sz_max = 0,
87d5c65159SKalle Valo 		.dest_nentries = 0,
88d5c65159SKalle Valo 	},
89d5c65159SKalle Valo 
90d5c65159SKalle Valo 	/* CE9: host->target WMI (mac2) */
91d5c65159SKalle Valo 	{
92d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS,
93d5c65159SKalle Valo 		.src_nentries = 32,
94d5c65159SKalle Valo 		.src_sz_max = 2048,
95d5c65159SKalle Valo 		.dest_nentries = 0,
96f951380aSP Praneesh 		.send_cb = ath11k_htc_tx_completion_handler,
97d5c65159SKalle Valo 	},
98d5c65159SKalle Valo 
99d5c65159SKalle Valo 	/* CE10: target->host HTT */
100d5c65159SKalle Valo 	{
101d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS,
102d5c65159SKalle Valo 		.src_nentries = 0,
103d5c65159SKalle Valo 		.src_sz_max = 2048,
104d5c65159SKalle Valo 		.dest_nentries = 512,
105d5c65159SKalle Valo 		.recv_cb = ath11k_htc_rx_completion_handler,
106d5c65159SKalle Valo 	},
107d5c65159SKalle Valo 
108d5c65159SKalle Valo 	/* CE11: Not used */
109d5c65159SKalle Valo 	{
110d5c65159SKalle Valo 		.flags = CE_ATTR_FLAGS,
111d5c65159SKalle Valo 		.src_nentries = 0,
112d5c65159SKalle Valo 		.src_sz_max = 0,
113d5c65159SKalle Valo 		.dest_nentries = 0,
114d5c65159SKalle Valo 	},
115d5c65159SKalle Valo };
116d5c65159SKalle Valo 
117e3396b8bSCarl Huang const struct ce_attr ath11k_host_ce_config_qca6390[] = {
118e3396b8bSCarl Huang 	/* CE0: host->target HTC control and raw streams */
119e3396b8bSCarl Huang 	{
120e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS,
121e3396b8bSCarl Huang 		.src_nentries = 16,
122e3396b8bSCarl Huang 		.src_sz_max = 2048,
123e3396b8bSCarl Huang 		.dest_nentries = 0,
124e3396b8bSCarl Huang 	},
125e3396b8bSCarl Huang 
126e3396b8bSCarl Huang 	/* CE1: target->host HTT + HTC control */
127e3396b8bSCarl Huang 	{
128e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS,
129e3396b8bSCarl Huang 		.src_nentries = 0,
130e3396b8bSCarl Huang 		.src_sz_max = 2048,
131e3396b8bSCarl Huang 		.dest_nentries = 512,
132e3396b8bSCarl Huang 		.recv_cb = ath11k_htc_rx_completion_handler,
133e3396b8bSCarl Huang 	},
134e3396b8bSCarl Huang 
135e3396b8bSCarl Huang 	/* CE2: target->host WMI */
136e3396b8bSCarl Huang 	{
137e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS,
138e3396b8bSCarl Huang 		.src_nentries = 0,
139e3396b8bSCarl Huang 		.src_sz_max = 2048,
140e3396b8bSCarl Huang 		.dest_nentries = 512,
141e3396b8bSCarl Huang 		.recv_cb = ath11k_htc_rx_completion_handler,
142e3396b8bSCarl Huang 	},
143e3396b8bSCarl Huang 
144e3396b8bSCarl Huang 	/* CE3: host->target WMI (mac0) */
145e3396b8bSCarl Huang 	{
146e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS,
147e3396b8bSCarl Huang 		.src_nentries = 32,
148e3396b8bSCarl Huang 		.src_sz_max = 2048,
149e3396b8bSCarl Huang 		.dest_nentries = 0,
150f951380aSP Praneesh 		.send_cb = ath11k_htc_tx_completion_handler,
151e3396b8bSCarl Huang 	},
152e3396b8bSCarl Huang 
153e3396b8bSCarl Huang 	/* CE4: host->target HTT */
154e3396b8bSCarl Huang 	{
155e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS | CE_ATTR_DIS_INTR,
156e3396b8bSCarl Huang 		.src_nentries = 2048,
157e3396b8bSCarl Huang 		.src_sz_max = 256,
158e3396b8bSCarl Huang 		.dest_nentries = 0,
159e3396b8bSCarl Huang 	},
160e3396b8bSCarl Huang 
161e3396b8bSCarl Huang 	/* CE5: target->host pktlog */
162e3396b8bSCarl Huang 	{
163e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS,
164e3396b8bSCarl Huang 		.src_nentries = 0,
165e3396b8bSCarl Huang 		.src_sz_max = 2048,
166e3396b8bSCarl Huang 		.dest_nentries = 512,
167e3396b8bSCarl Huang 		.recv_cb = ath11k_dp_htt_htc_t2h_msg_handler,
168e3396b8bSCarl Huang 	},
169e3396b8bSCarl Huang 
170e3396b8bSCarl Huang 	/* CE6: target autonomous hif_memcpy */
171e3396b8bSCarl Huang 	{
172e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS | CE_ATTR_DIS_INTR,
173e3396b8bSCarl Huang 		.src_nentries = 0,
174e3396b8bSCarl Huang 		.src_sz_max = 0,
175e3396b8bSCarl Huang 		.dest_nentries = 0,
176e3396b8bSCarl Huang 	},
177e3396b8bSCarl Huang 
178e3396b8bSCarl Huang 	/* CE7: host->target WMI (mac1) */
179e3396b8bSCarl Huang 	{
180e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS,
181e3396b8bSCarl Huang 		.src_nentries = 32,
182e3396b8bSCarl Huang 		.src_sz_max = 2048,
183e3396b8bSCarl Huang 		.dest_nentries = 0,
184f951380aSP Praneesh 		.send_cb = ath11k_htc_tx_completion_handler,
185e3396b8bSCarl Huang 	},
186e3396b8bSCarl Huang 
187e3396b8bSCarl Huang 	/* CE8: target autonomous hif_memcpy */
188e3396b8bSCarl Huang 	{
189e3396b8bSCarl Huang 		.flags = CE_ATTR_FLAGS,
190e3396b8bSCarl Huang 		.src_nentries = 0,
191e3396b8bSCarl Huang 		.src_sz_max = 0,
192e3396b8bSCarl Huang 		.dest_nentries = 0,
193e3396b8bSCarl Huang 	},
194e3396b8bSCarl Huang 
195e3396b8bSCarl Huang };
196e3396b8bSCarl Huang 
1976289ac2bSKarthikeyan Periyasamy const struct ce_attr ath11k_host_ce_config_qcn9074[] = {
1986289ac2bSKarthikeyan Periyasamy 	/* CE0: host->target HTC control and raw streams */
1996289ac2bSKarthikeyan Periyasamy 	{
2006289ac2bSKarthikeyan Periyasamy 		.flags = CE_ATTR_FLAGS,
2016289ac2bSKarthikeyan Periyasamy 		.src_nentries = 16,
2026289ac2bSKarthikeyan Periyasamy 		.src_sz_max = 2048,
2036289ac2bSKarthikeyan Periyasamy 		.dest_nentries = 0,
2046289ac2bSKarthikeyan Periyasamy 	},
2056289ac2bSKarthikeyan Periyasamy 
2066289ac2bSKarthikeyan Periyasamy 	/* CE1: target->host HTT + HTC control */
2076289ac2bSKarthikeyan Periyasamy 	{
2086289ac2bSKarthikeyan Periyasamy 		.flags = CE_ATTR_FLAGS,
2096289ac2bSKarthikeyan Periyasamy 		.src_nentries = 0,
2106289ac2bSKarthikeyan Periyasamy 		.src_sz_max = 2048,
2116289ac2bSKarthikeyan Periyasamy 		.dest_nentries = 512,
2126289ac2bSKarthikeyan Periyasamy 		.recv_cb = ath11k_htc_rx_completion_handler,
2136289ac2bSKarthikeyan Periyasamy 	},
2146289ac2bSKarthikeyan Periyasamy 
2156289ac2bSKarthikeyan Periyasamy 	/* CE2: target->host WMI */
2166289ac2bSKarthikeyan Periyasamy 	{
2176289ac2bSKarthikeyan Periyasamy 		.flags = CE_ATTR_FLAGS,
2186289ac2bSKarthikeyan Periyasamy 		.src_nentries = 0,
2196289ac2bSKarthikeyan Periyasamy 		.src_sz_max = 2048,
2206289ac2bSKarthikeyan Periyasamy 		.dest_nentries = 32,
2216289ac2bSKarthikeyan Periyasamy 		.recv_cb = ath11k_htc_rx_completion_handler,
2226289ac2bSKarthikeyan Periyasamy 	},
2236289ac2bSKarthikeyan Periyasamy 
2246289ac2bSKarthikeyan Periyasamy 	/* CE3: host->target WMI (mac0) */
2256289ac2bSKarthikeyan Periyasamy 	{
2266289ac2bSKarthikeyan Periyasamy 		.flags = CE_ATTR_FLAGS,
2276289ac2bSKarthikeyan Periyasamy 		.src_nentries = 32,
2286289ac2bSKarthikeyan Periyasamy 		.src_sz_max = 2048,
2296289ac2bSKarthikeyan Periyasamy 		.dest_nentries = 0,
230f951380aSP Praneesh 		.send_cb = ath11k_htc_tx_completion_handler,
2316289ac2bSKarthikeyan Periyasamy 	},
2326289ac2bSKarthikeyan Periyasamy 
2336289ac2bSKarthikeyan Periyasamy 	/* CE4: host->target HTT */
2346289ac2bSKarthikeyan Periyasamy 	{
2356289ac2bSKarthikeyan Periyasamy 		.flags = CE_ATTR_FLAGS | CE_ATTR_DIS_INTR,
2366289ac2bSKarthikeyan Periyasamy 		.src_nentries = 2048,
2376289ac2bSKarthikeyan Periyasamy 		.src_sz_max = 256,
2386289ac2bSKarthikeyan Periyasamy 		.dest_nentries = 0,
2396289ac2bSKarthikeyan Periyasamy 	},
2406289ac2bSKarthikeyan Periyasamy 
2416289ac2bSKarthikeyan Periyasamy 	/* CE5: target->host pktlog */
2426289ac2bSKarthikeyan Periyasamy 	{
2436289ac2bSKarthikeyan Periyasamy 		.flags = CE_ATTR_FLAGS,
2446289ac2bSKarthikeyan Periyasamy 		.src_nentries = 0,
2456289ac2bSKarthikeyan Periyasamy 		.src_sz_max = 2048,
2466289ac2bSKarthikeyan Periyasamy 		.dest_nentries = 512,
2476289ac2bSKarthikeyan Periyasamy 		.recv_cb = ath11k_dp_htt_htc_t2h_msg_handler,
2486289ac2bSKarthikeyan Periyasamy 	},
2496289ac2bSKarthikeyan Periyasamy };
2506289ac2bSKarthikeyan Periyasamy 
ath11k_ce_need_shadow_fix(int ce_id)2519b309970SCarl Huang static bool ath11k_ce_need_shadow_fix(int ce_id)
2529b309970SCarl Huang {
2533fecca0eSJeff Johnson 	/* only ce4 needs shadow workaround */
2549b309970SCarl Huang 	if (ce_id == 4)
2559b309970SCarl Huang 		return true;
2569b309970SCarl Huang 	return false;
2579b309970SCarl Huang }
2589b309970SCarl Huang 
ath11k_ce_stop_shadow_timers(struct ath11k_base * ab)259d1b0c338SCarl Huang void ath11k_ce_stop_shadow_timers(struct ath11k_base *ab)
2609b309970SCarl Huang {
2619b309970SCarl Huang 	int i;
2629b309970SCarl Huang 
2639b309970SCarl Huang 	if (!ab->hw_params.supports_shadow_regs)
2649b309970SCarl Huang 		return;
2659b309970SCarl Huang 
2669b309970SCarl Huang 	for (i = 0; i < ab->hw_params.ce_count; i++)
2679b309970SCarl Huang 		if (ath11k_ce_need_shadow_fix(i))
2689b309970SCarl Huang 			ath11k_dp_shadow_stop_timer(ab, &ab->ce.hp_timer[i]);
2699b309970SCarl Huang }
2709b309970SCarl Huang 
ath11k_ce_rx_buf_enqueue_pipe(struct ath11k_ce_pipe * pipe,struct sk_buff * skb,dma_addr_t paddr)271d5c65159SKalle Valo static int ath11k_ce_rx_buf_enqueue_pipe(struct ath11k_ce_pipe *pipe,
272d5c65159SKalle Valo 					 struct sk_buff *skb, dma_addr_t paddr)
273d5c65159SKalle Valo {
274d5c65159SKalle Valo 	struct ath11k_base *ab = pipe->ab;
275d5c65159SKalle Valo 	struct ath11k_ce_ring *ring = pipe->dest_ring;
276d5c65159SKalle Valo 	struct hal_srng *srng;
277d5c65159SKalle Valo 	unsigned int write_index;
278d5c65159SKalle Valo 	unsigned int nentries_mask = ring->nentries_mask;
279d5c65159SKalle Valo 	u32 *desc;
280d5c65159SKalle Valo 	int ret;
281d5c65159SKalle Valo 
282d5c65159SKalle Valo 	lockdep_assert_held(&ab->ce.ce_lock);
283d5c65159SKalle Valo 
284d5c65159SKalle Valo 	write_index = ring->write_index;
285d5c65159SKalle Valo 
286d5c65159SKalle Valo 	srng = &ab->hal.srng_list[ring->hal_ring_id];
287d5c65159SKalle Valo 
288d5c65159SKalle Valo 	spin_lock_bh(&srng->lock);
289d5c65159SKalle Valo 
290d5c65159SKalle Valo 	ath11k_hal_srng_access_begin(ab, srng);
291d5c65159SKalle Valo 
292d5c65159SKalle Valo 	if (unlikely(ath11k_hal_srng_src_num_free(ab, srng, false) < 1)) {
293d5c65159SKalle Valo 		ret = -ENOSPC;
294d5c65159SKalle Valo 		goto exit;
295d5c65159SKalle Valo 	}
296d5c65159SKalle Valo 
297d5c65159SKalle Valo 	desc = ath11k_hal_srng_src_get_next_entry(ab, srng);
298d5c65159SKalle Valo 	if (!desc) {
299d5c65159SKalle Valo 		ret = -ENOSPC;
300d5c65159SKalle Valo 		goto exit;
301d5c65159SKalle Valo 	}
302d5c65159SKalle Valo 
303d5c65159SKalle Valo 	ath11k_hal_ce_dst_set_desc(desc, paddr);
304d5c65159SKalle Valo 
305d5c65159SKalle Valo 	ring->skb[write_index] = skb;
306d5c65159SKalle Valo 	write_index = CE_RING_IDX_INCR(nentries_mask, write_index);
307d5c65159SKalle Valo 	ring->write_index = write_index;
308d5c65159SKalle Valo 
309d5c65159SKalle Valo 	pipe->rx_buf_needed--;
310d5c65159SKalle Valo 
311d5c65159SKalle Valo 	ret = 0;
312d5c65159SKalle Valo exit:
313d5c65159SKalle Valo 	ath11k_hal_srng_access_end(ab, srng);
314d5c65159SKalle Valo 
315d5c65159SKalle Valo 	spin_unlock_bh(&srng->lock);
316d5c65159SKalle Valo 
317d5c65159SKalle Valo 	return ret;
318d5c65159SKalle Valo }
319d5c65159SKalle Valo 
ath11k_ce_rx_post_pipe(struct ath11k_ce_pipe * pipe)320d5c65159SKalle Valo static int ath11k_ce_rx_post_pipe(struct ath11k_ce_pipe *pipe)
321d5c65159SKalle Valo {
322d5c65159SKalle Valo 	struct ath11k_base *ab = pipe->ab;
323d5c65159SKalle Valo 	struct sk_buff *skb;
324d5c65159SKalle Valo 	dma_addr_t paddr;
325d5c65159SKalle Valo 	int ret = 0;
326d5c65159SKalle Valo 
327d5c65159SKalle Valo 	if (!(pipe->dest_ring || pipe->status_ring))
328d5c65159SKalle Valo 		return 0;
329d5c65159SKalle Valo 
330d5c65159SKalle Valo 	spin_lock_bh(&ab->ce.ce_lock);
331d5c65159SKalle Valo 	while (pipe->rx_buf_needed) {
332d5c65159SKalle Valo 		skb = dev_alloc_skb(pipe->buf_sz);
333d5c65159SKalle Valo 		if (!skb) {
334d5c65159SKalle Valo 			ret = -ENOMEM;
335d5c65159SKalle Valo 			goto exit;
336d5c65159SKalle Valo 		}
337d5c65159SKalle Valo 
338d5c65159SKalle Valo 		WARN_ON_ONCE(!IS_ALIGNED((unsigned long)skb->data, 4));
339d5c65159SKalle Valo 
340d5c65159SKalle Valo 		paddr = dma_map_single(ab->dev, skb->data,
341d5c65159SKalle Valo 				       skb->len + skb_tailroom(skb),
342d5c65159SKalle Valo 				       DMA_FROM_DEVICE);
343d5c65159SKalle Valo 		if (unlikely(dma_mapping_error(ab->dev, paddr))) {
344d5c65159SKalle Valo 			ath11k_warn(ab, "failed to dma map ce rx buf\n");
345d5c65159SKalle Valo 			dev_kfree_skb_any(skb);
346d5c65159SKalle Valo 			ret = -EIO;
347d5c65159SKalle Valo 			goto exit;
348d5c65159SKalle Valo 		}
349d5c65159SKalle Valo 
350d5c65159SKalle Valo 		ATH11K_SKB_RXCB(skb)->paddr = paddr;
351d5c65159SKalle Valo 
352d5c65159SKalle Valo 		ret = ath11k_ce_rx_buf_enqueue_pipe(pipe, skb, paddr);
353d5c65159SKalle Valo 
354d5c65159SKalle Valo 		if (ret) {
355d5c65159SKalle Valo 			ath11k_warn(ab, "failed to enqueue rx buf: %d\n", ret);
356d5c65159SKalle Valo 			dma_unmap_single(ab->dev, paddr,
357d5c65159SKalle Valo 					 skb->len + skb_tailroom(skb),
358d5c65159SKalle Valo 					 DMA_FROM_DEVICE);
359d5c65159SKalle Valo 			dev_kfree_skb_any(skb);
360d5c65159SKalle Valo 			goto exit;
361d5c65159SKalle Valo 		}
362d5c65159SKalle Valo 	}
363d5c65159SKalle Valo 
364d5c65159SKalle Valo exit:
365d5c65159SKalle Valo 	spin_unlock_bh(&ab->ce.ce_lock);
366d5c65159SKalle Valo 	return ret;
367d5c65159SKalle Valo }
368d5c65159SKalle Valo 
ath11k_ce_completed_recv_next(struct ath11k_ce_pipe * pipe,struct sk_buff ** skb,int * nbytes)369d5c65159SKalle Valo static int ath11k_ce_completed_recv_next(struct ath11k_ce_pipe *pipe,
370d5c65159SKalle Valo 					 struct sk_buff **skb, int *nbytes)
371d5c65159SKalle Valo {
372d5c65159SKalle Valo 	struct ath11k_base *ab = pipe->ab;
373d5c65159SKalle Valo 	struct hal_srng *srng;
374d5c65159SKalle Valo 	unsigned int sw_index;
375d5c65159SKalle Valo 	unsigned int nentries_mask;
376d5c65159SKalle Valo 	u32 *desc;
377d5c65159SKalle Valo 	int ret = 0;
378d5c65159SKalle Valo 
379d5c65159SKalle Valo 	spin_lock_bh(&ab->ce.ce_lock);
380d5c65159SKalle Valo 
381d5c65159SKalle Valo 	sw_index = pipe->dest_ring->sw_index;
382d5c65159SKalle Valo 	nentries_mask = pipe->dest_ring->nentries_mask;
383d5c65159SKalle Valo 
384d5c65159SKalle Valo 	srng = &ab->hal.srng_list[pipe->status_ring->hal_ring_id];
385d5c65159SKalle Valo 
386d5c65159SKalle Valo 	spin_lock_bh(&srng->lock);
387d5c65159SKalle Valo 
388d5c65159SKalle Valo 	ath11k_hal_srng_access_begin(ab, srng);
389d5c65159SKalle Valo 
390d5c65159SKalle Valo 	desc = ath11k_hal_srng_dst_get_next_entry(ab, srng);
391d5c65159SKalle Valo 	if (!desc) {
392d5c65159SKalle Valo 		ret = -EIO;
393d5c65159SKalle Valo 		goto err;
394d5c65159SKalle Valo 	}
395d5c65159SKalle Valo 
396d5c65159SKalle Valo 	*nbytes = ath11k_hal_ce_dst_status_get_length(desc);
397d5c65159SKalle Valo 	if (*nbytes == 0) {
398d5c65159SKalle Valo 		ret = -EIO;
399d5c65159SKalle Valo 		goto err;
400d5c65159SKalle Valo 	}
401d5c65159SKalle Valo 
402d5c65159SKalle Valo 	*skb = pipe->dest_ring->skb[sw_index];
403d5c65159SKalle Valo 	pipe->dest_ring->skb[sw_index] = NULL;
404d5c65159SKalle Valo 
405d5c65159SKalle Valo 	sw_index = CE_RING_IDX_INCR(nentries_mask, sw_index);
406d5c65159SKalle Valo 	pipe->dest_ring->sw_index = sw_index;
407d5c65159SKalle Valo 
408d5c65159SKalle Valo 	pipe->rx_buf_needed++;
409d5c65159SKalle Valo err:
410d5c65159SKalle Valo 	ath11k_hal_srng_access_end(ab, srng);
411d5c65159SKalle Valo 
412d5c65159SKalle Valo 	spin_unlock_bh(&srng->lock);
413d5c65159SKalle Valo 
414d5c65159SKalle Valo 	spin_unlock_bh(&ab->ce.ce_lock);
415d5c65159SKalle Valo 
416d5c65159SKalle Valo 	return ret;
417d5c65159SKalle Valo }
418d5c65159SKalle Valo 
ath11k_ce_recv_process_cb(struct ath11k_ce_pipe * pipe)419d5c65159SKalle Valo static void ath11k_ce_recv_process_cb(struct ath11k_ce_pipe *pipe)
420d5c65159SKalle Valo {
421d5c65159SKalle Valo 	struct ath11k_base *ab = pipe->ab;
422d5c65159SKalle Valo 	struct sk_buff *skb;
423d5c65159SKalle Valo 	struct sk_buff_head list;
424d5c65159SKalle Valo 	unsigned int nbytes, max_nbytes;
425d5c65159SKalle Valo 	int ret;
426d5c65159SKalle Valo 
427d5c65159SKalle Valo 	__skb_queue_head_init(&list);
428d5c65159SKalle Valo 	while (ath11k_ce_completed_recv_next(pipe, &skb, &nbytes) == 0) {
429d5c65159SKalle Valo 		max_nbytes = skb->len + skb_tailroom(skb);
430d5c65159SKalle Valo 		dma_unmap_single(ab->dev, ATH11K_SKB_RXCB(skb)->paddr,
431d5c65159SKalle Valo 				 max_nbytes, DMA_FROM_DEVICE);
432d5c65159SKalle Valo 
433d5c65159SKalle Valo 		if (unlikely(max_nbytes < nbytes)) {
434d5c65159SKalle Valo 			ath11k_warn(ab, "rxed more than expected (nbytes %d, max %d)",
435d5c65159SKalle Valo 				    nbytes, max_nbytes);
436d5c65159SKalle Valo 			dev_kfree_skb_any(skb);
437d5c65159SKalle Valo 			continue;
438d5c65159SKalle Valo 		}
439d5c65159SKalle Valo 
440d5c65159SKalle Valo 		skb_put(skb, nbytes);
441d5c65159SKalle Valo 		__skb_queue_tail(&list, skb);
442d5c65159SKalle Valo 	}
443d5c65159SKalle Valo 
444d5c65159SKalle Valo 	while ((skb = __skb_dequeue(&list))) {
445*4f52ec65SKalle Valo 		ath11k_dbg(ab, ATH11K_DBG_CE, "rx ce pipe %d len %d\n",
446d5c65159SKalle Valo 			   pipe->pipe_num, skb->len);
447d5c65159SKalle Valo 		pipe->recv_cb(ab, skb);
448d5c65159SKalle Valo 	}
449d5c65159SKalle Valo 
450d5c65159SKalle Valo 	ret = ath11k_ce_rx_post_pipe(pipe);
451d5c65159SKalle Valo 	if (ret && ret != -ENOSPC) {
452d5c65159SKalle Valo 		ath11k_warn(ab, "failed to post rx buf to pipe: %d err: %d\n",
453d5c65159SKalle Valo 			    pipe->pipe_num, ret);
454d5c65159SKalle Valo 		mod_timer(&ab->rx_replenish_retry,
455d5c65159SKalle Valo 			  jiffies + ATH11K_CE_RX_POST_RETRY_JIFFIES);
456d5c65159SKalle Valo 	}
457d5c65159SKalle Valo }
458d5c65159SKalle Valo 
ath11k_ce_completed_send_next(struct ath11k_ce_pipe * pipe)459d5c65159SKalle Valo static struct sk_buff *ath11k_ce_completed_send_next(struct ath11k_ce_pipe *pipe)
460d5c65159SKalle Valo {
461d5c65159SKalle Valo 	struct ath11k_base *ab = pipe->ab;
462d5c65159SKalle Valo 	struct hal_srng *srng;
463d5c65159SKalle Valo 	unsigned int sw_index;
464d5c65159SKalle Valo 	unsigned int nentries_mask;
465d5c65159SKalle Valo 	struct sk_buff *skb;
466d5c65159SKalle Valo 	u32 *desc;
467d5c65159SKalle Valo 
468d5c65159SKalle Valo 	spin_lock_bh(&ab->ce.ce_lock);
469d5c65159SKalle Valo 
470d5c65159SKalle Valo 	sw_index = pipe->src_ring->sw_index;
471d5c65159SKalle Valo 	nentries_mask = pipe->src_ring->nentries_mask;
472d5c65159SKalle Valo 
473d5c65159SKalle Valo 	srng = &ab->hal.srng_list[pipe->src_ring->hal_ring_id];
474d5c65159SKalle Valo 
475d5c65159SKalle Valo 	spin_lock_bh(&srng->lock);
476d5c65159SKalle Valo 
477d5c65159SKalle Valo 	ath11k_hal_srng_access_begin(ab, srng);
478d5c65159SKalle Valo 
479d5c65159SKalle Valo 	desc = ath11k_hal_srng_src_reap_next(ab, srng);
480d5c65159SKalle Valo 	if (!desc) {
481d5c65159SKalle Valo 		skb = ERR_PTR(-EIO);
482d5c65159SKalle Valo 		goto err_unlock;
483d5c65159SKalle Valo 	}
484d5c65159SKalle Valo 
485d5c65159SKalle Valo 	skb = pipe->src_ring->skb[sw_index];
486d5c65159SKalle Valo 
487d5c65159SKalle Valo 	pipe->src_ring->skb[sw_index] = NULL;
488d5c65159SKalle Valo 
489d5c65159SKalle Valo 	sw_index = CE_RING_IDX_INCR(nentries_mask, sw_index);
490d5c65159SKalle Valo 	pipe->src_ring->sw_index = sw_index;
491d5c65159SKalle Valo 
492d5c65159SKalle Valo err_unlock:
493d5c65159SKalle Valo 	spin_unlock_bh(&srng->lock);
494d5c65159SKalle Valo 
495d5c65159SKalle Valo 	spin_unlock_bh(&ab->ce.ce_lock);
496d5c65159SKalle Valo 
497d5c65159SKalle Valo 	return skb;
498d5c65159SKalle Valo }
499d5c65159SKalle Valo 
ath11k_ce_tx_process_cb(struct ath11k_ce_pipe * pipe)500f951380aSP Praneesh static void ath11k_ce_tx_process_cb(struct ath11k_ce_pipe *pipe)
501d5c65159SKalle Valo {
502d5c65159SKalle Valo 	struct ath11k_base *ab = pipe->ab;
503d5c65159SKalle Valo 	struct sk_buff *skb;
504f951380aSP Praneesh 	struct sk_buff_head list;
505d5c65159SKalle Valo 
506f951380aSP Praneesh 	__skb_queue_head_init(&list);
507d5c65159SKalle Valo 	while (!IS_ERR(skb = ath11k_ce_completed_send_next(pipe))) {
508d5c65159SKalle Valo 		if (!skb)
509d5c65159SKalle Valo 			continue;
510d5c65159SKalle Valo 
511d5c65159SKalle Valo 		dma_unmap_single(ab->dev, ATH11K_SKB_CB(skb)->paddr, skb->len,
512d5c65159SKalle Valo 				 DMA_TO_DEVICE);
513f951380aSP Praneesh 
514f951380aSP Praneesh 		if ((!pipe->send_cb) || ab->hw_params.credit_flow) {
515d5c65159SKalle Valo 			dev_kfree_skb_any(skb);
516f951380aSP Praneesh 			continue;
517f951380aSP Praneesh 		}
518f951380aSP Praneesh 
519f951380aSP Praneesh 		__skb_queue_tail(&list, skb);
520f951380aSP Praneesh 	}
521f951380aSP Praneesh 
522f951380aSP Praneesh 	while ((skb = __skb_dequeue(&list))) {
523*4f52ec65SKalle Valo 		ath11k_dbg(ab, ATH11K_DBG_CE, "tx ce pipe %d len %d\n",
524f951380aSP Praneesh 			   pipe->pipe_num, skb->len);
525f951380aSP Praneesh 		pipe->send_cb(ab, skb);
526d5c65159SKalle Valo 	}
527d5c65159SKalle Valo }
528d5c65159SKalle Valo 
ath11k_ce_srng_msi_ring_params_setup(struct ath11k_base * ab,u32 ce_id,struct hal_srng_params * ring_params)529c4eacabeSGovind Singh static void ath11k_ce_srng_msi_ring_params_setup(struct ath11k_base *ab, u32 ce_id,
530c4eacabeSGovind Singh 						 struct hal_srng_params *ring_params)
531c4eacabeSGovind Singh {
532c4eacabeSGovind Singh 	u32 msi_data_start;
5336289ac2bSKarthikeyan Periyasamy 	u32 msi_data_count, msi_data_idx;
534c4eacabeSGovind Singh 	u32 msi_irq_start;
535c4eacabeSGovind Singh 	u32 addr_lo;
536c4eacabeSGovind Singh 	u32 addr_hi;
537c4eacabeSGovind Singh 	int ret;
538c4eacabeSGovind Singh 
539c4eacabeSGovind Singh 	ret = ath11k_get_user_msi_vector(ab, "CE",
540c4eacabeSGovind Singh 					 &msi_data_count, &msi_data_start,
541c4eacabeSGovind Singh 					 &msi_irq_start);
542c4eacabeSGovind Singh 
543c4eacabeSGovind Singh 	if (ret)
544c4eacabeSGovind Singh 		return;
545c4eacabeSGovind Singh 
546c4eacabeSGovind Singh 	ath11k_get_msi_address(ab, &addr_lo, &addr_hi);
5476289ac2bSKarthikeyan Periyasamy 	ath11k_get_ce_msi_idx(ab, ce_id, &msi_data_idx);
548c4eacabeSGovind Singh 
549c4eacabeSGovind Singh 	ring_params->msi_addr = addr_lo;
550c4eacabeSGovind Singh 	ring_params->msi_addr |= (dma_addr_t)(((uint64_t)addr_hi) << 32);
5516289ac2bSKarthikeyan Periyasamy 	ring_params->msi_data = (msi_data_idx % msi_data_count) + msi_data_start;
552c4eacabeSGovind Singh 	ring_params->flags |= HAL_SRNG_FLAGS_MSI_INTR;
553c4eacabeSGovind Singh }
554c4eacabeSGovind Singh 
ath11k_ce_init_ring(struct ath11k_base * ab,struct ath11k_ce_ring * ce_ring,int ce_id,enum hal_ring_type type)555d5c65159SKalle Valo static int ath11k_ce_init_ring(struct ath11k_base *ab,
556d5c65159SKalle Valo 			       struct ath11k_ce_ring *ce_ring,
557d5c65159SKalle Valo 			       int ce_id, enum hal_ring_type type)
558d5c65159SKalle Valo {
559d5c65159SKalle Valo 	struct hal_srng_params params = { 0 };
560d5c65159SKalle Valo 	int ret;
561d5c65159SKalle Valo 
562d5c65159SKalle Valo 	params.ring_base_paddr = ce_ring->base_addr_ce_space;
563d5c65159SKalle Valo 	params.ring_base_vaddr = ce_ring->base_addr_owner_space;
564d5c65159SKalle Valo 	params.num_entries = ce_ring->nentries;
565d5c65159SKalle Valo 
5661a05ed37SCarl Huang 	if (!(CE_ATTR_DIS_INTR & ab->hw_params.host_ce_config[ce_id].flags))
5671a05ed37SCarl Huang 		ath11k_ce_srng_msi_ring_params_setup(ab, ce_id, &params);
5681a05ed37SCarl Huang 
569d5c65159SKalle Valo 	switch (type) {
570d5c65159SKalle Valo 	case HAL_CE_SRC:
5716e5e9f59SKalle Valo 		if (!(CE_ATTR_DIS_INTR & ab->hw_params.host_ce_config[ce_id].flags))
572d5c65159SKalle Valo 			params.intr_batch_cntr_thres_entries = 1;
573d5c65159SKalle Valo 		break;
574d5c65159SKalle Valo 	case HAL_CE_DST:
5756e5e9f59SKalle Valo 		params.max_buffer_len = ab->hw_params.host_ce_config[ce_id].src_sz_max;
5766e5e9f59SKalle Valo 		if (!(ab->hw_params.host_ce_config[ce_id].flags & CE_ATTR_DIS_INTR)) {
577d5c65159SKalle Valo 			params.intr_timer_thres_us = 1024;
578d5c65159SKalle Valo 			params.flags |= HAL_SRNG_FLAGS_LOW_THRESH_INTR_EN;
579d5c65159SKalle Valo 			params.low_threshold = ce_ring->nentries - 3;
580d5c65159SKalle Valo 		}
581d5c65159SKalle Valo 		break;
582d5c65159SKalle Valo 	case HAL_CE_DST_STATUS:
5836e5e9f59SKalle Valo 		if (!(ab->hw_params.host_ce_config[ce_id].flags & CE_ATTR_DIS_INTR)) {
584d5c65159SKalle Valo 			params.intr_batch_cntr_thres_entries = 1;
585d5c65159SKalle Valo 			params.intr_timer_thres_us = 0x1000;
586d5c65159SKalle Valo 		}
587d5c65159SKalle Valo 		break;
588d5c65159SKalle Valo 	default:
589d5c65159SKalle Valo 		ath11k_warn(ab, "Invalid CE ring type %d\n", type);
590d5c65159SKalle Valo 		return -EINVAL;
591d5c65159SKalle Valo 	}
592d5c65159SKalle Valo 
593d5c65159SKalle Valo 	/* TODO: Init other params needed by HAL to init the ring */
594d5c65159SKalle Valo 
595d5c65159SKalle Valo 	ret = ath11k_hal_srng_setup(ab, type, ce_id, 0, &params);
596d5c65159SKalle Valo 	if (ret < 0) {
597d5c65159SKalle Valo 		ath11k_warn(ab, "failed to setup srng: %d ring_id %d\n",
598d5c65159SKalle Valo 			    ret, ce_id);
599d5c65159SKalle Valo 		return ret;
600d5c65159SKalle Valo 	}
601c4eacabeSGovind Singh 
602d5c65159SKalle Valo 	ce_ring->hal_ring_id = ret;
603d5c65159SKalle Valo 
6049b309970SCarl Huang 	if (ab->hw_params.supports_shadow_regs &&
6059b309970SCarl Huang 	    ath11k_ce_need_shadow_fix(ce_id))
6069b309970SCarl Huang 		ath11k_dp_shadow_init_timer(ab, &ab->ce.hp_timer[ce_id],
6079b309970SCarl Huang 					    ATH11K_SHADOW_CTRL_TIMER_INTERVAL,
6089b309970SCarl Huang 					    ce_ring->hal_ring_id);
6099b309970SCarl Huang 
610d5c65159SKalle Valo 	return 0;
611d5c65159SKalle Valo }
612d5c65159SKalle Valo 
613d5c65159SKalle Valo static struct ath11k_ce_ring *
ath11k_ce_alloc_ring(struct ath11k_base * ab,int nentries,int desc_sz)614d5c65159SKalle Valo ath11k_ce_alloc_ring(struct ath11k_base *ab, int nentries, int desc_sz)
615d5c65159SKalle Valo {
616d5c65159SKalle Valo 	struct ath11k_ce_ring *ce_ring;
617d5c65159SKalle Valo 	dma_addr_t base_addr;
618d5c65159SKalle Valo 
619d5c65159SKalle Valo 	ce_ring = kzalloc(struct_size(ce_ring, skb, nentries), GFP_KERNEL);
620d5c65159SKalle Valo 	if (ce_ring == NULL)
621d5c65159SKalle Valo 		return ERR_PTR(-ENOMEM);
622d5c65159SKalle Valo 
623d5c65159SKalle Valo 	ce_ring->nentries = nentries;
624d5c65159SKalle Valo 	ce_ring->nentries_mask = nentries - 1;
625d5c65159SKalle Valo 
626d5c65159SKalle Valo 	/* Legacy platforms that do not support cache
627d5c65159SKalle Valo 	 * coherent DMA are unsupported
628d5c65159SKalle Valo 	 */
629d5c65159SKalle Valo 	ce_ring->base_addr_owner_space_unaligned =
630d5c65159SKalle Valo 		dma_alloc_coherent(ab->dev,
631d5c65159SKalle Valo 				   nentries * desc_sz + CE_DESC_RING_ALIGN,
632d5c65159SKalle Valo 				   &base_addr, GFP_KERNEL);
633d5c65159SKalle Valo 	if (!ce_ring->base_addr_owner_space_unaligned) {
634d5c65159SKalle Valo 		kfree(ce_ring);
635d5c65159SKalle Valo 		return ERR_PTR(-ENOMEM);
636d5c65159SKalle Valo 	}
637d5c65159SKalle Valo 
638d5c65159SKalle Valo 	ce_ring->base_addr_ce_space_unaligned = base_addr;
639d5c65159SKalle Valo 
640d5c65159SKalle Valo 	ce_ring->base_addr_owner_space = PTR_ALIGN(
641d5c65159SKalle Valo 			ce_ring->base_addr_owner_space_unaligned,
642d5c65159SKalle Valo 			CE_DESC_RING_ALIGN);
643d5c65159SKalle Valo 	ce_ring->base_addr_ce_space = ALIGN(
644d5c65159SKalle Valo 			ce_ring->base_addr_ce_space_unaligned,
645d5c65159SKalle Valo 			CE_DESC_RING_ALIGN);
646d5c65159SKalle Valo 
647d5c65159SKalle Valo 	return ce_ring;
648d5c65159SKalle Valo }
649d5c65159SKalle Valo 
ath11k_ce_alloc_pipe(struct ath11k_base * ab,int ce_id)650d5c65159SKalle Valo static int ath11k_ce_alloc_pipe(struct ath11k_base *ab, int ce_id)
651d5c65159SKalle Valo {
652d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe = &ab->ce.ce_pipe[ce_id];
6536e5e9f59SKalle Valo 	const struct ce_attr *attr = &ab->hw_params.host_ce_config[ce_id];
654c76fa846SDan Carpenter 	struct ath11k_ce_ring *ring;
655d5c65159SKalle Valo 	int nentries;
656d5c65159SKalle Valo 	int desc_sz;
657d5c65159SKalle Valo 
658d5c65159SKalle Valo 	pipe->attr_flags = attr->flags;
659d5c65159SKalle Valo 
660d5c65159SKalle Valo 	if (attr->src_nentries) {
661f951380aSP Praneesh 		pipe->send_cb = attr->send_cb;
662d5c65159SKalle Valo 		nentries = roundup_pow_of_two(attr->src_nentries);
663d5c65159SKalle Valo 		desc_sz = ath11k_hal_ce_get_desc_size(HAL_CE_DESC_SRC);
664c76fa846SDan Carpenter 		ring = ath11k_ce_alloc_ring(ab, nentries, desc_sz);
665c76fa846SDan Carpenter 		if (IS_ERR(ring))
666c76fa846SDan Carpenter 			return PTR_ERR(ring);
667c76fa846SDan Carpenter 		pipe->src_ring = ring;
668d5c65159SKalle Valo 	}
669d5c65159SKalle Valo 
670d5c65159SKalle Valo 	if (attr->dest_nentries) {
671d5c65159SKalle Valo 		pipe->recv_cb = attr->recv_cb;
672d5c65159SKalle Valo 		nentries = roundup_pow_of_two(attr->dest_nentries);
673d5c65159SKalle Valo 		desc_sz = ath11k_hal_ce_get_desc_size(HAL_CE_DESC_DST);
674c76fa846SDan Carpenter 		ring = ath11k_ce_alloc_ring(ab, nentries, desc_sz);
675c76fa846SDan Carpenter 		if (IS_ERR(ring))
676c76fa846SDan Carpenter 			return PTR_ERR(ring);
677c76fa846SDan Carpenter 		pipe->dest_ring = ring;
678d5c65159SKalle Valo 
679d5c65159SKalle Valo 		desc_sz = ath11k_hal_ce_get_desc_size(HAL_CE_DESC_DST_STATUS);
680c76fa846SDan Carpenter 		ring = ath11k_ce_alloc_ring(ab, nentries, desc_sz);
681c76fa846SDan Carpenter 		if (IS_ERR(ring))
682c76fa846SDan Carpenter 			return PTR_ERR(ring);
683c76fa846SDan Carpenter 		pipe->status_ring = ring;
684d5c65159SKalle Valo 	}
685d5c65159SKalle Valo 
686d5c65159SKalle Valo 	return 0;
687d5c65159SKalle Valo }
688d5c65159SKalle Valo 
ath11k_ce_per_engine_service(struct ath11k_base * ab,u16 ce_id)689d5c65159SKalle Valo void ath11k_ce_per_engine_service(struct ath11k_base *ab, u16 ce_id)
690d5c65159SKalle Valo {
691d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe = &ab->ce.ce_pipe[ce_id];
692f951380aSP Praneesh 	const struct ce_attr *attr = &ab->hw_params.host_ce_config[ce_id];
693d5c65159SKalle Valo 
694f951380aSP Praneesh 	if (attr->src_nentries)
695f951380aSP Praneesh 		ath11k_ce_tx_process_cb(pipe);
696d5c65159SKalle Valo 
697d5c65159SKalle Valo 	if (pipe->recv_cb)
698d5c65159SKalle Valo 		ath11k_ce_recv_process_cb(pipe);
699d5c65159SKalle Valo }
700d5c65159SKalle Valo 
ath11k_ce_poll_send_completed(struct ath11k_base * ab,u8 pipe_id)701d5c65159SKalle Valo void ath11k_ce_poll_send_completed(struct ath11k_base *ab, u8 pipe_id)
702d5c65159SKalle Valo {
703d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe = &ab->ce.ce_pipe[pipe_id];
704f951380aSP Praneesh 	const struct ce_attr *attr =  &ab->hw_params.host_ce_config[pipe_id];
705d5c65159SKalle Valo 
706f951380aSP Praneesh 	if ((pipe->attr_flags & CE_ATTR_DIS_INTR) && attr->src_nentries)
707f951380aSP Praneesh 		ath11k_ce_tx_process_cb(pipe);
708d5c65159SKalle Valo }
7092c3960c2SGovind Singh EXPORT_SYMBOL(ath11k_ce_per_engine_service);
710d5c65159SKalle Valo 
ath11k_ce_send(struct ath11k_base * ab,struct sk_buff * skb,u8 pipe_id,u16 transfer_id)711d5c65159SKalle Valo int ath11k_ce_send(struct ath11k_base *ab, struct sk_buff *skb, u8 pipe_id,
712d5c65159SKalle Valo 		   u16 transfer_id)
713d5c65159SKalle Valo {
714d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe = &ab->ce.ce_pipe[pipe_id];
715d5c65159SKalle Valo 	struct hal_srng *srng;
716d5c65159SKalle Valo 	u32 *desc;
717d5c65159SKalle Valo 	unsigned int write_index, sw_index;
718d5c65159SKalle Valo 	unsigned int nentries_mask;
719d5c65159SKalle Valo 	int ret = 0;
720d5c65159SKalle Valo 	u8 byte_swap_data = 0;
721d5c65159SKalle Valo 	int num_used;
722d5c65159SKalle Valo 
723d5c65159SKalle Valo 	/* Check if some entries could be regained by handling tx completion if
724d5c65159SKalle Valo 	 * the CE has interrupts disabled and the used entries is more than the
725d5c65159SKalle Valo 	 * defined usage threshold.
726d5c65159SKalle Valo 	 */
727d5c65159SKalle Valo 	if (pipe->attr_flags & CE_ATTR_DIS_INTR) {
728d5c65159SKalle Valo 		spin_lock_bh(&ab->ce.ce_lock);
729d5c65159SKalle Valo 		write_index = pipe->src_ring->write_index;
730d5c65159SKalle Valo 
731d5c65159SKalle Valo 		sw_index = pipe->src_ring->sw_index;
732d5c65159SKalle Valo 
733d5c65159SKalle Valo 		if (write_index >= sw_index)
734d5c65159SKalle Valo 			num_used = write_index - sw_index;
735d5c65159SKalle Valo 		else
736d5c65159SKalle Valo 			num_used = pipe->src_ring->nentries - sw_index +
737d5c65159SKalle Valo 				   write_index;
738d5c65159SKalle Valo 
739d5c65159SKalle Valo 		spin_unlock_bh(&ab->ce.ce_lock);
740d5c65159SKalle Valo 
741d5c65159SKalle Valo 		if (num_used > ATH11K_CE_USAGE_THRESHOLD)
742d5c65159SKalle Valo 			ath11k_ce_poll_send_completed(ab, pipe->pipe_num);
743d5c65159SKalle Valo 	}
744d5c65159SKalle Valo 
745d5c65159SKalle Valo 	if (test_bit(ATH11K_FLAG_CRASH_FLUSH, &ab->dev_flags))
746d5c65159SKalle Valo 		return -ESHUTDOWN;
747d5c65159SKalle Valo 
748d5c65159SKalle Valo 	spin_lock_bh(&ab->ce.ce_lock);
749d5c65159SKalle Valo 
750d5c65159SKalle Valo 	write_index = pipe->src_ring->write_index;
751d5c65159SKalle Valo 	nentries_mask = pipe->src_ring->nentries_mask;
752d5c65159SKalle Valo 
753d5c65159SKalle Valo 	srng = &ab->hal.srng_list[pipe->src_ring->hal_ring_id];
754d5c65159SKalle Valo 
755d5c65159SKalle Valo 	spin_lock_bh(&srng->lock);
756d5c65159SKalle Valo 
757d5c65159SKalle Valo 	ath11k_hal_srng_access_begin(ab, srng);
758d5c65159SKalle Valo 
759d5c65159SKalle Valo 	if (unlikely(ath11k_hal_srng_src_num_free(ab, srng, false) < 1)) {
760d5c65159SKalle Valo 		ath11k_hal_srng_access_end(ab, srng);
761d5c65159SKalle Valo 		ret = -ENOBUFS;
762d5c65159SKalle Valo 		goto err_unlock;
763d5c65159SKalle Valo 	}
764d5c65159SKalle Valo 
765d5c65159SKalle Valo 	desc = ath11k_hal_srng_src_get_next_reaped(ab, srng);
766d5c65159SKalle Valo 	if (!desc) {
767d5c65159SKalle Valo 		ath11k_hal_srng_access_end(ab, srng);
768d5c65159SKalle Valo 		ret = -ENOBUFS;
769d5c65159SKalle Valo 		goto err_unlock;
770d5c65159SKalle Valo 	}
771d5c65159SKalle Valo 
772d5c65159SKalle Valo 	if (pipe->attr_flags & CE_ATTR_BYTE_SWAP_DATA)
773d5c65159SKalle Valo 		byte_swap_data = 1;
774d5c65159SKalle Valo 
775d5c65159SKalle Valo 	ath11k_hal_ce_src_set_desc(desc, ATH11K_SKB_CB(skb)->paddr,
776d5c65159SKalle Valo 				   skb->len, transfer_id, byte_swap_data);
777d5c65159SKalle Valo 
778d5c65159SKalle Valo 	pipe->src_ring->skb[write_index] = skb;
779d5c65159SKalle Valo 	pipe->src_ring->write_index = CE_RING_IDX_INCR(nentries_mask,
780d5c65159SKalle Valo 						       write_index);
781d5c65159SKalle Valo 
782d5c65159SKalle Valo 	ath11k_hal_srng_access_end(ab, srng);
783d5c65159SKalle Valo 
7849b309970SCarl Huang 	if (ath11k_ce_need_shadow_fix(pipe_id))
7859b309970SCarl Huang 		ath11k_dp_shadow_start_timer(ab, srng, &ab->ce.hp_timer[pipe_id]);
7869b309970SCarl Huang 
787d5c65159SKalle Valo 	spin_unlock_bh(&srng->lock);
788d5c65159SKalle Valo 
789d5c65159SKalle Valo 	spin_unlock_bh(&ab->ce.ce_lock);
790d5c65159SKalle Valo 
791d5c65159SKalle Valo 	return 0;
792d5c65159SKalle Valo 
793d5c65159SKalle Valo err_unlock:
794d5c65159SKalle Valo 	spin_unlock_bh(&srng->lock);
795d5c65159SKalle Valo 
796d5c65159SKalle Valo 	spin_unlock_bh(&ab->ce.ce_lock);
797d5c65159SKalle Valo 
798d5c65159SKalle Valo 	return ret;
799d5c65159SKalle Valo }
800d5c65159SKalle Valo 
ath11k_ce_rx_pipe_cleanup(struct ath11k_ce_pipe * pipe)801d5c65159SKalle Valo static void ath11k_ce_rx_pipe_cleanup(struct ath11k_ce_pipe *pipe)
802d5c65159SKalle Valo {
803d5c65159SKalle Valo 	struct ath11k_base *ab = pipe->ab;
804d5c65159SKalle Valo 	struct ath11k_ce_ring *ring = pipe->dest_ring;
805d5c65159SKalle Valo 	struct sk_buff *skb;
806d5c65159SKalle Valo 	int i;
807d5c65159SKalle Valo 
808d5c65159SKalle Valo 	if (!(ring && pipe->buf_sz))
809d5c65159SKalle Valo 		return;
810d5c65159SKalle Valo 
811d5c65159SKalle Valo 	for (i = 0; i < ring->nentries; i++) {
812d5c65159SKalle Valo 		skb = ring->skb[i];
813d5c65159SKalle Valo 		if (!skb)
814d5c65159SKalle Valo 			continue;
815d5c65159SKalle Valo 
816d5c65159SKalle Valo 		ring->skb[i] = NULL;
817d5c65159SKalle Valo 		dma_unmap_single(ab->dev, ATH11K_SKB_RXCB(skb)->paddr,
818d5c65159SKalle Valo 				 skb->len + skb_tailroom(skb), DMA_FROM_DEVICE);
819d5c65159SKalle Valo 		dev_kfree_skb_any(skb);
820d5c65159SKalle Valo 	}
821d5c65159SKalle Valo }
822d5c65159SKalle Valo 
ath11k_ce_shadow_config(struct ath11k_base * ab)823e838c14aSCarl Huang static void ath11k_ce_shadow_config(struct ath11k_base *ab)
824e838c14aSCarl Huang {
825e838c14aSCarl Huang 	int i;
826e838c14aSCarl Huang 
827e838c14aSCarl Huang 	for (i = 0; i < ab->hw_params.ce_count; i++) {
828e838c14aSCarl Huang 		if (ab->hw_params.host_ce_config[i].src_nentries)
829e838c14aSCarl Huang 			ath11k_hal_srng_update_shadow_config(ab,
830e838c14aSCarl Huang 							     HAL_CE_SRC, i);
831e838c14aSCarl Huang 
832e838c14aSCarl Huang 		if (ab->hw_params.host_ce_config[i].dest_nentries) {
833e838c14aSCarl Huang 			ath11k_hal_srng_update_shadow_config(ab,
834e838c14aSCarl Huang 							     HAL_CE_DST, i);
835e838c14aSCarl Huang 
836e838c14aSCarl Huang 			ath11k_hal_srng_update_shadow_config(ab,
837e838c14aSCarl Huang 							     HAL_CE_DST_STATUS, i);
838e838c14aSCarl Huang 		}
839e838c14aSCarl Huang 	}
840e838c14aSCarl Huang }
841e838c14aSCarl Huang 
ath11k_ce_get_shadow_config(struct ath11k_base * ab,u32 ** shadow_cfg,u32 * shadow_cfg_len)842e838c14aSCarl Huang void ath11k_ce_get_shadow_config(struct ath11k_base *ab,
843e838c14aSCarl Huang 				 u32 **shadow_cfg, u32 *shadow_cfg_len)
844e838c14aSCarl Huang {
845e838c14aSCarl Huang 	if (!ab->hw_params.supports_shadow_regs)
846e838c14aSCarl Huang 		return;
847e838c14aSCarl Huang 
848e838c14aSCarl Huang 	ath11k_hal_srng_get_shadow_config(ab, shadow_cfg, shadow_cfg_len);
849e838c14aSCarl Huang 
850e838c14aSCarl Huang 	/* shadow is already configured */
851e838c14aSCarl Huang 	if (*shadow_cfg_len)
852e838c14aSCarl Huang 		return;
853e838c14aSCarl Huang 
854e838c14aSCarl Huang 	/* shadow isn't configured yet, configure now.
855e838c14aSCarl Huang 	 * non-CE srngs are configured firstly, then
856e838c14aSCarl Huang 	 * all CE srngs.
857e838c14aSCarl Huang 	 */
858e838c14aSCarl Huang 	ath11k_hal_srng_shadow_config(ab);
859e838c14aSCarl Huang 	ath11k_ce_shadow_config(ab);
860e838c14aSCarl Huang 
861e838c14aSCarl Huang 	/* get the shadow configuration */
862e838c14aSCarl Huang 	ath11k_hal_srng_get_shadow_config(ab, shadow_cfg, shadow_cfg_len);
863e838c14aSCarl Huang }
864e838c14aSCarl Huang EXPORT_SYMBOL(ath11k_ce_get_shadow_config);
865e838c14aSCarl Huang 
ath11k_ce_cleanup_pipes(struct ath11k_base * ab)866d5c65159SKalle Valo void ath11k_ce_cleanup_pipes(struct ath11k_base *ab)
867d5c65159SKalle Valo {
868d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe;
869d5c65159SKalle Valo 	int pipe_num;
870d5c65159SKalle Valo 
8719b309970SCarl Huang 	ath11k_ce_stop_shadow_timers(ab);
8729b309970SCarl Huang 
873d9d4b5f3SKalle Valo 	for (pipe_num = 0; pipe_num < ab->hw_params.ce_count; pipe_num++) {
874d5c65159SKalle Valo 		pipe = &ab->ce.ce_pipe[pipe_num];
875d5c65159SKalle Valo 		ath11k_ce_rx_pipe_cleanup(pipe);
876d5c65159SKalle Valo 
877d5c65159SKalle Valo 		/* Cleanup any src CE's which have interrupts disabled */
878d5c65159SKalle Valo 		ath11k_ce_poll_send_completed(ab, pipe_num);
879d5c65159SKalle Valo 
880d5c65159SKalle Valo 		/* NOTE: Should we also clean up tx buffer in all pipes? */
881d5c65159SKalle Valo 	}
882d5c65159SKalle Valo }
8837f4beda2SGovind Singh EXPORT_SYMBOL(ath11k_ce_cleanup_pipes);
884d5c65159SKalle Valo 
ath11k_ce_rx_post_buf(struct ath11k_base * ab)885d5c65159SKalle Valo void ath11k_ce_rx_post_buf(struct ath11k_base *ab)
886d5c65159SKalle Valo {
887d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe;
888d5c65159SKalle Valo 	int i;
889d5c65159SKalle Valo 	int ret;
890d5c65159SKalle Valo 
891d9d4b5f3SKalle Valo 	for (i = 0; i < ab->hw_params.ce_count; i++) {
892d5c65159SKalle Valo 		pipe = &ab->ce.ce_pipe[i];
893d5c65159SKalle Valo 		ret = ath11k_ce_rx_post_pipe(pipe);
894d5c65159SKalle Valo 		if (ret) {
895d5c65159SKalle Valo 			if (ret == -ENOSPC)
896d5c65159SKalle Valo 				continue;
897d5c65159SKalle Valo 
898d5c65159SKalle Valo 			ath11k_warn(ab, "failed to post rx buf to pipe: %d err: %d\n",
899d5c65159SKalle Valo 				    i, ret);
900d5c65159SKalle Valo 			mod_timer(&ab->rx_replenish_retry,
901d5c65159SKalle Valo 				  jiffies + ATH11K_CE_RX_POST_RETRY_JIFFIES);
902d5c65159SKalle Valo 
903d5c65159SKalle Valo 			return;
904d5c65159SKalle Valo 		}
905d5c65159SKalle Valo 	}
906d5c65159SKalle Valo }
9072c3960c2SGovind Singh EXPORT_SYMBOL(ath11k_ce_rx_post_buf);
908d5c65159SKalle Valo 
ath11k_ce_rx_replenish_retry(struct timer_list * t)909d5c65159SKalle Valo void ath11k_ce_rx_replenish_retry(struct timer_list *t)
910d5c65159SKalle Valo {
911d5c65159SKalle Valo 	struct ath11k_base *ab = from_timer(ab, t, rx_replenish_retry);
912d5c65159SKalle Valo 
913d5c65159SKalle Valo 	ath11k_ce_rx_post_buf(ab);
914d5c65159SKalle Valo }
915d5c65159SKalle Valo 
ath11k_ce_init_pipes(struct ath11k_base * ab)916d5c65159SKalle Valo int ath11k_ce_init_pipes(struct ath11k_base *ab)
917d5c65159SKalle Valo {
918d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe;
919d5c65159SKalle Valo 	int i;
920d5c65159SKalle Valo 	int ret;
921d5c65159SKalle Valo 
922d9d4b5f3SKalle Valo 	for (i = 0; i < ab->hw_params.ce_count; i++) {
923d5c65159SKalle Valo 		pipe = &ab->ce.ce_pipe[i];
924d5c65159SKalle Valo 
925d5c65159SKalle Valo 		if (pipe->src_ring) {
926d5c65159SKalle Valo 			ret = ath11k_ce_init_ring(ab, pipe->src_ring, i,
927d5c65159SKalle Valo 						  HAL_CE_SRC);
928d5c65159SKalle Valo 			if (ret) {
929d5c65159SKalle Valo 				ath11k_warn(ab, "failed to init src ring: %d\n",
930d5c65159SKalle Valo 					    ret);
931d5c65159SKalle Valo 				/* Should we clear any partial init */
932d5c65159SKalle Valo 				return ret;
933d5c65159SKalle Valo 			}
934d5c65159SKalle Valo 
935d5c65159SKalle Valo 			pipe->src_ring->write_index = 0;
936d5c65159SKalle Valo 			pipe->src_ring->sw_index = 0;
937d5c65159SKalle Valo 		}
938d5c65159SKalle Valo 
939d5c65159SKalle Valo 		if (pipe->dest_ring) {
940d5c65159SKalle Valo 			ret = ath11k_ce_init_ring(ab, pipe->dest_ring, i,
941d5c65159SKalle Valo 						  HAL_CE_DST);
942d5c65159SKalle Valo 			if (ret) {
943d5c65159SKalle Valo 				ath11k_warn(ab, "failed to init dest ring: %d\n",
944d5c65159SKalle Valo 					    ret);
945d5c65159SKalle Valo 				/* Should we clear any partial init */
946d5c65159SKalle Valo 				return ret;
947d5c65159SKalle Valo 			}
948d5c65159SKalle Valo 
949d5c65159SKalle Valo 			pipe->rx_buf_needed = pipe->dest_ring->nentries ?
950d5c65159SKalle Valo 					      pipe->dest_ring->nentries - 2 : 0;
951d5c65159SKalle Valo 
952d5c65159SKalle Valo 			pipe->dest_ring->write_index = 0;
953d5c65159SKalle Valo 			pipe->dest_ring->sw_index = 0;
954d5c65159SKalle Valo 		}
955d5c65159SKalle Valo 
956d5c65159SKalle Valo 		if (pipe->status_ring) {
957d5c65159SKalle Valo 			ret = ath11k_ce_init_ring(ab, pipe->status_ring, i,
958d5c65159SKalle Valo 						  HAL_CE_DST_STATUS);
959d5c65159SKalle Valo 			if (ret) {
960d5c65159SKalle Valo 				ath11k_warn(ab, "failed to init dest status ing: %d\n",
961d5c65159SKalle Valo 					    ret);
962d5c65159SKalle Valo 				/* Should we clear any partial init */
963d5c65159SKalle Valo 				return ret;
964d5c65159SKalle Valo 			}
965d5c65159SKalle Valo 
966d5c65159SKalle Valo 			pipe->status_ring->write_index = 0;
967d5c65159SKalle Valo 			pipe->status_ring->sw_index = 0;
968d5c65159SKalle Valo 		}
969d5c65159SKalle Valo 	}
970d5c65159SKalle Valo 
971d5c65159SKalle Valo 	return 0;
972d5c65159SKalle Valo }
973d5c65159SKalle Valo 
ath11k_ce_free_pipes(struct ath11k_base * ab)974d5c65159SKalle Valo void ath11k_ce_free_pipes(struct ath11k_base *ab)
975d5c65159SKalle Valo {
976d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe;
97731aeaf54SKarthikeyan Periyasamy 	struct ath11k_ce_ring *ce_ring;
978d5c65159SKalle Valo 	int desc_sz;
979d5c65159SKalle Valo 	int i;
980d5c65159SKalle Valo 
981d9d4b5f3SKalle Valo 	for (i = 0; i < ab->hw_params.ce_count; i++) {
982d5c65159SKalle Valo 		pipe = &ab->ce.ce_pipe[i];
983d5c65159SKalle Valo 
9849b309970SCarl Huang 		if (ath11k_ce_need_shadow_fix(i))
9859b309970SCarl Huang 			ath11k_dp_shadow_stop_timer(ab, &ab->ce.hp_timer[i]);
9869b309970SCarl Huang 
987d5c65159SKalle Valo 		if (pipe->src_ring) {
988d5c65159SKalle Valo 			desc_sz = ath11k_hal_ce_get_desc_size(HAL_CE_DESC_SRC);
98931aeaf54SKarthikeyan Periyasamy 			ce_ring = pipe->src_ring;
990d5c65159SKalle Valo 			dma_free_coherent(ab->dev,
991d5c65159SKalle Valo 					  pipe->src_ring->nentries * desc_sz +
992d5c65159SKalle Valo 					  CE_DESC_RING_ALIGN,
99331aeaf54SKarthikeyan Periyasamy 					  ce_ring->base_addr_owner_space_unaligned,
99431aeaf54SKarthikeyan Periyasamy 					  ce_ring->base_addr_ce_space_unaligned);
995d5c65159SKalle Valo 			kfree(pipe->src_ring);
996d5c65159SKalle Valo 			pipe->src_ring = NULL;
997d5c65159SKalle Valo 		}
998d5c65159SKalle Valo 
999d5c65159SKalle Valo 		if (pipe->dest_ring) {
1000d5c65159SKalle Valo 			desc_sz = ath11k_hal_ce_get_desc_size(HAL_CE_DESC_DST);
100131aeaf54SKarthikeyan Periyasamy 			ce_ring = pipe->dest_ring;
1002d5c65159SKalle Valo 			dma_free_coherent(ab->dev,
1003d5c65159SKalle Valo 					  pipe->dest_ring->nentries * desc_sz +
1004d5c65159SKalle Valo 					  CE_DESC_RING_ALIGN,
100531aeaf54SKarthikeyan Periyasamy 					  ce_ring->base_addr_owner_space_unaligned,
100631aeaf54SKarthikeyan Periyasamy 					  ce_ring->base_addr_ce_space_unaligned);
1007d5c65159SKalle Valo 			kfree(pipe->dest_ring);
1008d5c65159SKalle Valo 			pipe->dest_ring = NULL;
1009d5c65159SKalle Valo 		}
1010d5c65159SKalle Valo 
1011d5c65159SKalle Valo 		if (pipe->status_ring) {
1012d5c65159SKalle Valo 			desc_sz =
1013d5c65159SKalle Valo 			  ath11k_hal_ce_get_desc_size(HAL_CE_DESC_DST_STATUS);
101431aeaf54SKarthikeyan Periyasamy 			ce_ring = pipe->status_ring;
1015d5c65159SKalle Valo 			dma_free_coherent(ab->dev,
1016d5c65159SKalle Valo 					  pipe->status_ring->nentries * desc_sz +
1017d5c65159SKalle Valo 					  CE_DESC_RING_ALIGN,
101831aeaf54SKarthikeyan Periyasamy 					  ce_ring->base_addr_owner_space_unaligned,
101931aeaf54SKarthikeyan Periyasamy 					  ce_ring->base_addr_ce_space_unaligned);
1020d5c65159SKalle Valo 			kfree(pipe->status_ring);
1021d5c65159SKalle Valo 			pipe->status_ring = NULL;
1022d5c65159SKalle Valo 		}
1023d5c65159SKalle Valo 	}
1024d5c65159SKalle Valo }
10256e0355afSGovind Singh EXPORT_SYMBOL(ath11k_ce_free_pipes);
1026d5c65159SKalle Valo 
ath11k_ce_alloc_pipes(struct ath11k_base * ab)1027d5c65159SKalle Valo int ath11k_ce_alloc_pipes(struct ath11k_base *ab)
1028d5c65159SKalle Valo {
1029d5c65159SKalle Valo 	struct ath11k_ce_pipe *pipe;
1030d5c65159SKalle Valo 	int i;
1031d5c65159SKalle Valo 	int ret;
1032d5c65159SKalle Valo 	const struct ce_attr *attr;
1033d5c65159SKalle Valo 
1034d5c65159SKalle Valo 	spin_lock_init(&ab->ce.ce_lock);
1035d5c65159SKalle Valo 
1036d9d4b5f3SKalle Valo 	for (i = 0; i < ab->hw_params.ce_count; i++) {
10376e5e9f59SKalle Valo 		attr = &ab->hw_params.host_ce_config[i];
1038d5c65159SKalle Valo 		pipe = &ab->ce.ce_pipe[i];
1039d5c65159SKalle Valo 		pipe->pipe_num = i;
1040d5c65159SKalle Valo 		pipe->ab = ab;
1041d5c65159SKalle Valo 		pipe->buf_sz = attr->src_sz_max;
1042d5c65159SKalle Valo 
1043d5c65159SKalle Valo 		ret = ath11k_ce_alloc_pipe(ab, i);
1044d5c65159SKalle Valo 		if (ret) {
10453fecca0eSJeff Johnson 			/* Free any partial successful allocation */
1046d5c65159SKalle Valo 			ath11k_ce_free_pipes(ab);
1047d5c65159SKalle Valo 			return ret;
1048d5c65159SKalle Valo 		}
1049d5c65159SKalle Valo 	}
1050d5c65159SKalle Valo 
1051d5c65159SKalle Valo 	return 0;
1052d5c65159SKalle Valo }
10537f4beda2SGovind Singh EXPORT_SYMBOL(ath11k_ce_alloc_pipes);
1054d5c65159SKalle Valo 
1055d5c65159SKalle Valo /* For Big Endian Host, Copy Engine byte_swap is enabled
1056d5c65159SKalle Valo  * When Copy Engine does byte_swap, need to byte swap again for the
1057d5c65159SKalle Valo  * Host to get/put buffer content in the correct byte order
1058d5c65159SKalle Valo  */
ath11k_ce_byte_swap(void * mem,u32 len)1059d5c65159SKalle Valo void ath11k_ce_byte_swap(void *mem, u32 len)
1060d5c65159SKalle Valo {
1061d5c65159SKalle Valo 	int i;
1062d5c65159SKalle Valo 
1063d5c65159SKalle Valo 	if (IS_ENABLED(CONFIG_CPU_BIG_ENDIAN)) {
1064d5c65159SKalle Valo 		if (!mem)
1065d5c65159SKalle Valo 			return;
1066d5c65159SKalle Valo 
1067d5c65159SKalle Valo 		for (i = 0; i < (len / 4); i++) {
1068d5c65159SKalle Valo 			*(u32 *)mem = swab32(*(u32 *)mem);
1069d5c65159SKalle Valo 			mem += 4;
1070d5c65159SKalle Valo 		}
1071d5c65159SKalle Valo 	}
1072d5c65159SKalle Valo }
1073d5c65159SKalle Valo 
ath11k_ce_get_attr_flags(struct ath11k_base * ab,int ce_id)1074e3396b8bSCarl Huang int ath11k_ce_get_attr_flags(struct ath11k_base *ab, int ce_id)
1075d5c65159SKalle Valo {
1076d9d4b5f3SKalle Valo 	if (ce_id >= ab->hw_params.ce_count)
1077d5c65159SKalle Valo 		return -EINVAL;
1078d5c65159SKalle Valo 
10796e5e9f59SKalle Valo 	return ab->hw_params.host_ce_config[ce_id].flags;
1080d5c65159SKalle Valo }
10816e0355afSGovind Singh EXPORT_SYMBOL(ath11k_ce_get_attr_flags);
1082