xref: /openbmc/linux/drivers/dma/ti/k3-udma-glue.c (revision 5b65781d06ea90ef2f8e51a13352c43c3daa8cdc)
1d7024191SGrygorii Strashko // SPDX-License-Identifier: GPL-2.0
2d7024191SGrygorii Strashko /*
3d7024191SGrygorii Strashko  * K3 NAVSS DMA glue interface
4d7024191SGrygorii Strashko  *
5d7024191SGrygorii Strashko  * Copyright (C) 2019 Texas Instruments Incorporated - http://www.ti.com
6d7024191SGrygorii Strashko  *
7d7024191SGrygorii Strashko  */
8d7024191SGrygorii Strashko 
9d7024191SGrygorii Strashko #include <linux/atomic.h>
10d7024191SGrygorii Strashko #include <linux/delay.h>
11d7024191SGrygorii Strashko #include <linux/dma-mapping.h>
12d7024191SGrygorii Strashko #include <linux/io.h>
13d7024191SGrygorii Strashko #include <linux/init.h>
14d7024191SGrygorii Strashko #include <linux/of.h>
15d7024191SGrygorii Strashko #include <linux/platform_device.h>
16d7024191SGrygorii Strashko #include <linux/soc/ti/k3-ringacc.h>
17d7024191SGrygorii Strashko #include <linux/dma/ti-cppi5.h>
18d7024191SGrygorii Strashko #include <linux/dma/k3-udma-glue.h>
19d7024191SGrygorii Strashko 
20d7024191SGrygorii Strashko #include "k3-udma.h"
21d7024191SGrygorii Strashko #include "k3-psil-priv.h"
22d7024191SGrygorii Strashko 
23d7024191SGrygorii Strashko struct k3_udma_glue_common {
24d7024191SGrygorii Strashko 	struct device *dev;
25*5b65781dSVignesh Raghavendra 	struct device chan_dev;
26d7024191SGrygorii Strashko 	struct udma_dev *udmax;
27d7024191SGrygorii Strashko 	const struct udma_tisci_rm *tisci_rm;
28d7024191SGrygorii Strashko 	struct k3_ringacc *ringacc;
29d7024191SGrygorii Strashko 	u32 src_thread;
30d7024191SGrygorii Strashko 	u32 dst_thread;
31d7024191SGrygorii Strashko 
32d7024191SGrygorii Strashko 	u32  hdesc_size;
33d7024191SGrygorii Strashko 	bool epib;
34d7024191SGrygorii Strashko 	u32  psdata_size;
35d7024191SGrygorii Strashko 	u32  swdata_size;
36*5b65781dSVignesh Raghavendra 	u32  atype_asel;
37*5b65781dSVignesh Raghavendra 	struct psil_endpoint_config *ep_config;
38d7024191SGrygorii Strashko };
39d7024191SGrygorii Strashko 
40d7024191SGrygorii Strashko struct k3_udma_glue_tx_channel {
41d7024191SGrygorii Strashko 	struct k3_udma_glue_common common;
42d7024191SGrygorii Strashko 
43d7024191SGrygorii Strashko 	struct udma_tchan *udma_tchanx;
44d7024191SGrygorii Strashko 	int udma_tchan_id;
45d7024191SGrygorii Strashko 
46d7024191SGrygorii Strashko 	struct k3_ring *ringtx;
47d7024191SGrygorii Strashko 	struct k3_ring *ringtxcq;
48d7024191SGrygorii Strashko 
49d7024191SGrygorii Strashko 	bool psil_paired;
50d7024191SGrygorii Strashko 
51d7024191SGrygorii Strashko 	int virq;
52d7024191SGrygorii Strashko 
53d7024191SGrygorii Strashko 	atomic_t free_pkts;
54d7024191SGrygorii Strashko 	bool tx_pause_on_err;
55d7024191SGrygorii Strashko 	bool tx_filt_einfo;
56d7024191SGrygorii Strashko 	bool tx_filt_pswords;
57d7024191SGrygorii Strashko 	bool tx_supr_tdpkt;
58*5b65781dSVignesh Raghavendra 
59*5b65781dSVignesh Raghavendra 	int udma_tflow_id;
60d7024191SGrygorii Strashko };
61d7024191SGrygorii Strashko 
62d7024191SGrygorii Strashko struct k3_udma_glue_rx_flow {
63d7024191SGrygorii Strashko 	struct udma_rflow *udma_rflow;
64d7024191SGrygorii Strashko 	int udma_rflow_id;
65d7024191SGrygorii Strashko 	struct k3_ring *ringrx;
66d7024191SGrygorii Strashko 	struct k3_ring *ringrxfdq;
67d7024191SGrygorii Strashko 
68d7024191SGrygorii Strashko 	int virq;
69d7024191SGrygorii Strashko };
70d7024191SGrygorii Strashko 
71d7024191SGrygorii Strashko struct k3_udma_glue_rx_channel {
72d7024191SGrygorii Strashko 	struct k3_udma_glue_common common;
73d7024191SGrygorii Strashko 
74d7024191SGrygorii Strashko 	struct udma_rchan *udma_rchanx;
75d7024191SGrygorii Strashko 	int udma_rchan_id;
76d7024191SGrygorii Strashko 	bool remote;
77d7024191SGrygorii Strashko 
78d7024191SGrygorii Strashko 	bool psil_paired;
79d7024191SGrygorii Strashko 
80d7024191SGrygorii Strashko 	u32  swdata_size;
81d7024191SGrygorii Strashko 	int  flow_id_base;
82d7024191SGrygorii Strashko 
83d7024191SGrygorii Strashko 	struct k3_udma_glue_rx_flow *flows;
84d7024191SGrygorii Strashko 	u32 flow_num;
85d7024191SGrygorii Strashko 	u32 flows_ready;
86d7024191SGrygorii Strashko };
87d7024191SGrygorii Strashko 
88*5b65781dSVignesh Raghavendra static void k3_udma_chan_dev_release(struct device *dev)
89*5b65781dSVignesh Raghavendra {
90*5b65781dSVignesh Raghavendra 	/* The struct containing the device is devm managed */
91*5b65781dSVignesh Raghavendra }
92*5b65781dSVignesh Raghavendra 
93*5b65781dSVignesh Raghavendra static struct class k3_udma_glue_devclass = {
94*5b65781dSVignesh Raghavendra 	.name		= "k3_udma_glue_chan",
95*5b65781dSVignesh Raghavendra 	.dev_release	= k3_udma_chan_dev_release,
96*5b65781dSVignesh Raghavendra };
97*5b65781dSVignesh Raghavendra 
98d7024191SGrygorii Strashko #define K3_UDMAX_TDOWN_TIMEOUT_US 1000
99d7024191SGrygorii Strashko 
100d7024191SGrygorii Strashko static int of_k3_udma_glue_parse(struct device_node *udmax_np,
101d7024191SGrygorii Strashko 				 struct k3_udma_glue_common *common)
102d7024191SGrygorii Strashko {
103d7024191SGrygorii Strashko 	common->udmax = of_xudma_dev_get(udmax_np, NULL);
104d7024191SGrygorii Strashko 	if (IS_ERR(common->udmax))
105d7024191SGrygorii Strashko 		return PTR_ERR(common->udmax);
106d7024191SGrygorii Strashko 
107aa8a4c4eSPeter Ujfalusi 	common->ringacc = xudma_get_ringacc(common->udmax);
108d7024191SGrygorii Strashko 	common->tisci_rm = xudma_dev_get_tisci_rm(common->udmax);
109d7024191SGrygorii Strashko 
110d7024191SGrygorii Strashko 	return 0;
111d7024191SGrygorii Strashko }
112d7024191SGrygorii Strashko 
113d7024191SGrygorii Strashko static int of_k3_udma_glue_parse_chn(struct device_node *chn_np,
114d7024191SGrygorii Strashko 		const char *name, struct k3_udma_glue_common *common,
115d7024191SGrygorii Strashko 		bool tx_chn)
116d7024191SGrygorii Strashko {
117d7024191SGrygorii Strashko 	struct of_phandle_args dma_spec;
118d7024191SGrygorii Strashko 	u32 thread_id;
119d7024191SGrygorii Strashko 	int ret = 0;
120d7024191SGrygorii Strashko 	int index;
121d7024191SGrygorii Strashko 
122d7024191SGrygorii Strashko 	if (unlikely(!name))
123d7024191SGrygorii Strashko 		return -EINVAL;
124d7024191SGrygorii Strashko 
125d7024191SGrygorii Strashko 	index = of_property_match_string(chn_np, "dma-names", name);
126d7024191SGrygorii Strashko 	if (index < 0)
127d7024191SGrygorii Strashko 		return index;
128d7024191SGrygorii Strashko 
129d7024191SGrygorii Strashko 	if (of_parse_phandle_with_args(chn_np, "dmas", "#dma-cells", index,
130d7024191SGrygorii Strashko 				       &dma_spec))
131d7024191SGrygorii Strashko 		return -ENOENT;
132d7024191SGrygorii Strashko 
133*5b65781dSVignesh Raghavendra 	ret = of_k3_udma_glue_parse(dma_spec.np, common);
134*5b65781dSVignesh Raghavendra 	if (ret)
135*5b65781dSVignesh Raghavendra 		goto out_put_spec;
136*5b65781dSVignesh Raghavendra 
137d7024191SGrygorii Strashko 	thread_id = dma_spec.args[0];
1380ebcf1a2SPeter Ujfalusi 	if (dma_spec.args_count == 2) {
139*5b65781dSVignesh Raghavendra 		if (dma_spec.args[1] > 2 && !xudma_is_pktdma(common->udmax)) {
1400ebcf1a2SPeter Ujfalusi 			dev_err(common->dev, "Invalid channel atype: %u\n",
1410ebcf1a2SPeter Ujfalusi 				dma_spec.args[1]);
1420ebcf1a2SPeter Ujfalusi 			ret = -EINVAL;
1430ebcf1a2SPeter Ujfalusi 			goto out_put_spec;
1440ebcf1a2SPeter Ujfalusi 		}
145*5b65781dSVignesh Raghavendra 		if (dma_spec.args[1] > 15 && xudma_is_pktdma(common->udmax)) {
146*5b65781dSVignesh Raghavendra 			dev_err(common->dev, "Invalid channel asel: %u\n",
147*5b65781dSVignesh Raghavendra 				dma_spec.args[1]);
148*5b65781dSVignesh Raghavendra 			ret = -EINVAL;
149*5b65781dSVignesh Raghavendra 			goto out_put_spec;
150*5b65781dSVignesh Raghavendra 		}
151*5b65781dSVignesh Raghavendra 
152*5b65781dSVignesh Raghavendra 		common->atype_asel = dma_spec.args[1];
1530ebcf1a2SPeter Ujfalusi 	}
154d7024191SGrygorii Strashko 
155d7024191SGrygorii Strashko 	if (tx_chn && !(thread_id & K3_PSIL_DST_THREAD_ID_OFFSET)) {
156d7024191SGrygorii Strashko 		ret = -EINVAL;
157d7024191SGrygorii Strashko 		goto out_put_spec;
158d7024191SGrygorii Strashko 	}
159d7024191SGrygorii Strashko 
160d7024191SGrygorii Strashko 	if (!tx_chn && (thread_id & K3_PSIL_DST_THREAD_ID_OFFSET)) {
161d7024191SGrygorii Strashko 		ret = -EINVAL;
162d7024191SGrygorii Strashko 		goto out_put_spec;
163d7024191SGrygorii Strashko 	}
164d7024191SGrygorii Strashko 
165d7024191SGrygorii Strashko 	/* get psil endpoint config */
166*5b65781dSVignesh Raghavendra 	common->ep_config = psil_get_ep_config(thread_id);
167*5b65781dSVignesh Raghavendra 	if (IS_ERR(common->ep_config)) {
168d7024191SGrygorii Strashko 		dev_err(common->dev,
169d7024191SGrygorii Strashko 			"No configuration for psi-l thread 0x%04x\n",
170d7024191SGrygorii Strashko 			thread_id);
171*5b65781dSVignesh Raghavendra 		ret = PTR_ERR(common->ep_config);
172d7024191SGrygorii Strashko 		goto out_put_spec;
173d7024191SGrygorii Strashko 	}
174d7024191SGrygorii Strashko 
175*5b65781dSVignesh Raghavendra 	common->epib = common->ep_config->needs_epib;
176*5b65781dSVignesh Raghavendra 	common->psdata_size = common->ep_config->psd_size;
177d7024191SGrygorii Strashko 
178d7024191SGrygorii Strashko 	if (tx_chn)
179d7024191SGrygorii Strashko 		common->dst_thread = thread_id;
180d7024191SGrygorii Strashko 	else
181d7024191SGrygorii Strashko 		common->src_thread = thread_id;
182d7024191SGrygorii Strashko 
183d7024191SGrygorii Strashko out_put_spec:
184d7024191SGrygorii Strashko 	of_node_put(dma_spec.np);
185d7024191SGrygorii Strashko 	return ret;
186d7024191SGrygorii Strashko };
187d7024191SGrygorii Strashko 
188d7024191SGrygorii Strashko static void k3_udma_glue_dump_tx_chn(struct k3_udma_glue_tx_channel *tx_chn)
189d7024191SGrygorii Strashko {
190d7024191SGrygorii Strashko 	struct device *dev = tx_chn->common.dev;
191d7024191SGrygorii Strashko 
192d7024191SGrygorii Strashko 	dev_dbg(dev, "dump_tx_chn:\n"
193d7024191SGrygorii Strashko 		"udma_tchan_id: %d\n"
194d7024191SGrygorii Strashko 		"src_thread: %08x\n"
195d7024191SGrygorii Strashko 		"dst_thread: %08x\n",
196d7024191SGrygorii Strashko 		tx_chn->udma_tchan_id,
197d7024191SGrygorii Strashko 		tx_chn->common.src_thread,
198d7024191SGrygorii Strashko 		tx_chn->common.dst_thread);
199d7024191SGrygorii Strashko }
200d7024191SGrygorii Strashko 
201d7024191SGrygorii Strashko static void k3_udma_glue_dump_tx_rt_chn(struct k3_udma_glue_tx_channel *chn,
202d7024191SGrygorii Strashko 					char *mark)
203d7024191SGrygorii Strashko {
204d7024191SGrygorii Strashko 	struct device *dev = chn->common.dev;
205d7024191SGrygorii Strashko 
206d7024191SGrygorii Strashko 	dev_dbg(dev, "=== dump ===> %s\n", mark);
207bc7e5523SPeter Ujfalusi 	dev_dbg(dev, "0x%08X: %08X\n", UDMA_CHAN_RT_CTL_REG,
208bc7e5523SPeter Ujfalusi 		xudma_tchanrt_read(chn->udma_tchanx, UDMA_CHAN_RT_CTL_REG));
209bc7e5523SPeter Ujfalusi 	dev_dbg(dev, "0x%08X: %08X\n", UDMA_CHAN_RT_PEER_RT_EN_REG,
210d7024191SGrygorii Strashko 		xudma_tchanrt_read(chn->udma_tchanx,
211bc7e5523SPeter Ujfalusi 				   UDMA_CHAN_RT_PEER_RT_EN_REG));
212bc7e5523SPeter Ujfalusi 	dev_dbg(dev, "0x%08X: %08X\n", UDMA_CHAN_RT_PCNT_REG,
213bc7e5523SPeter Ujfalusi 		xudma_tchanrt_read(chn->udma_tchanx, UDMA_CHAN_RT_PCNT_REG));
214bc7e5523SPeter Ujfalusi 	dev_dbg(dev, "0x%08X: %08X\n", UDMA_CHAN_RT_BCNT_REG,
215bc7e5523SPeter Ujfalusi 		xudma_tchanrt_read(chn->udma_tchanx, UDMA_CHAN_RT_BCNT_REG));
216bc7e5523SPeter Ujfalusi 	dev_dbg(dev, "0x%08X: %08X\n", UDMA_CHAN_RT_SBCNT_REG,
217bc7e5523SPeter Ujfalusi 		xudma_tchanrt_read(chn->udma_tchanx, UDMA_CHAN_RT_SBCNT_REG));
218d7024191SGrygorii Strashko }
219d7024191SGrygorii Strashko 
220d7024191SGrygorii Strashko static int k3_udma_glue_cfg_tx_chn(struct k3_udma_glue_tx_channel *tx_chn)
221d7024191SGrygorii Strashko {
222d7024191SGrygorii Strashko 	const struct udma_tisci_rm *tisci_rm = tx_chn->common.tisci_rm;
223d7024191SGrygorii Strashko 	struct ti_sci_msg_rm_udmap_tx_ch_cfg req;
224d7024191SGrygorii Strashko 
225d7024191SGrygorii Strashko 	memset(&req, 0, sizeof(req));
226d7024191SGrygorii Strashko 
227d7024191SGrygorii Strashko 	req.valid_params = TI_SCI_MSG_VALUE_RM_UDMAP_CH_PAUSE_ON_ERR_VALID |
228d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_CH_TX_FILT_EINFO_VALID |
229d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_CH_TX_FILT_PSWORDS_VALID |
230d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_CH_CHAN_TYPE_VALID |
231d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_CH_TX_SUPR_TDPKT_VALID |
232d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_CH_FETCH_SIZE_VALID |
2330ebcf1a2SPeter Ujfalusi 			TI_SCI_MSG_VALUE_RM_UDMAP_CH_CQ_QNUM_VALID |
2340ebcf1a2SPeter Ujfalusi 			TI_SCI_MSG_VALUE_RM_UDMAP_CH_ATYPE_VALID;
235d7024191SGrygorii Strashko 	req.nav_id = tisci_rm->tisci_dev_id;
236d7024191SGrygorii Strashko 	req.index = tx_chn->udma_tchan_id;
237d7024191SGrygorii Strashko 	if (tx_chn->tx_pause_on_err)
238d7024191SGrygorii Strashko 		req.tx_pause_on_err = 1;
239d7024191SGrygorii Strashko 	if (tx_chn->tx_filt_einfo)
240d7024191SGrygorii Strashko 		req.tx_filt_einfo = 1;
241d7024191SGrygorii Strashko 	if (tx_chn->tx_filt_pswords)
242d7024191SGrygorii Strashko 		req.tx_filt_pswords = 1;
243d7024191SGrygorii Strashko 	req.tx_chan_type = TI_SCI_RM_UDMAP_CHAN_TYPE_PKT_PBRR;
244d7024191SGrygorii Strashko 	if (tx_chn->tx_supr_tdpkt)
245d7024191SGrygorii Strashko 		req.tx_supr_tdpkt = 1;
246d7024191SGrygorii Strashko 	req.tx_fetch_size = tx_chn->common.hdesc_size >> 2;
247d7024191SGrygorii Strashko 	req.txcq_qnum = k3_ringacc_get_ring_id(tx_chn->ringtxcq);
248*5b65781dSVignesh Raghavendra 	req.tx_atype = tx_chn->common.atype_asel;
249d7024191SGrygorii Strashko 
250d7024191SGrygorii Strashko 	return tisci_rm->tisci_udmap_ops->tx_ch_cfg(tisci_rm->tisci, &req);
251d7024191SGrygorii Strashko }
252d7024191SGrygorii Strashko 
253d7024191SGrygorii Strashko struct k3_udma_glue_tx_channel *k3_udma_glue_request_tx_chn(struct device *dev,
254d7024191SGrygorii Strashko 		const char *name, struct k3_udma_glue_tx_channel_cfg *cfg)
255d7024191SGrygorii Strashko {
256d7024191SGrygorii Strashko 	struct k3_udma_glue_tx_channel *tx_chn;
257d7024191SGrygorii Strashko 	int ret;
258d7024191SGrygorii Strashko 
259d7024191SGrygorii Strashko 	tx_chn = devm_kzalloc(dev, sizeof(*tx_chn), GFP_KERNEL);
260d7024191SGrygorii Strashko 	if (!tx_chn)
261d7024191SGrygorii Strashko 		return ERR_PTR(-ENOMEM);
262d7024191SGrygorii Strashko 
263d7024191SGrygorii Strashko 	tx_chn->common.dev = dev;
264d7024191SGrygorii Strashko 	tx_chn->common.swdata_size = cfg->swdata_size;
265d7024191SGrygorii Strashko 	tx_chn->tx_pause_on_err = cfg->tx_pause_on_err;
266d7024191SGrygorii Strashko 	tx_chn->tx_filt_einfo = cfg->tx_filt_einfo;
267d7024191SGrygorii Strashko 	tx_chn->tx_filt_pswords = cfg->tx_filt_pswords;
268d7024191SGrygorii Strashko 	tx_chn->tx_supr_tdpkt = cfg->tx_supr_tdpkt;
269d7024191SGrygorii Strashko 
270d7024191SGrygorii Strashko 	/* parse of udmap channel */
271d7024191SGrygorii Strashko 	ret = of_k3_udma_glue_parse_chn(dev->of_node, name,
272d7024191SGrygorii Strashko 					&tx_chn->common, true);
273d7024191SGrygorii Strashko 	if (ret)
274d7024191SGrygorii Strashko 		goto err;
275d7024191SGrygorii Strashko 
276d7024191SGrygorii Strashko 	tx_chn->common.hdesc_size = cppi5_hdesc_calc_size(tx_chn->common.epib,
277d7024191SGrygorii Strashko 						tx_chn->common.psdata_size,
278d7024191SGrygorii Strashko 						tx_chn->common.swdata_size);
279d7024191SGrygorii Strashko 
280*5b65781dSVignesh Raghavendra 	if (xudma_is_pktdma(tx_chn->common.udmax))
281*5b65781dSVignesh Raghavendra 		tx_chn->udma_tchan_id = tx_chn->common.ep_config->mapped_channel_id;
282*5b65781dSVignesh Raghavendra 	else
283*5b65781dSVignesh Raghavendra 		tx_chn->udma_tchan_id = -1;
284*5b65781dSVignesh Raghavendra 
285d7024191SGrygorii Strashko 	/* request and cfg UDMAP TX channel */
286*5b65781dSVignesh Raghavendra 	tx_chn->udma_tchanx = xudma_tchan_get(tx_chn->common.udmax,
287*5b65781dSVignesh Raghavendra 					      tx_chn->udma_tchan_id);
288d7024191SGrygorii Strashko 	if (IS_ERR(tx_chn->udma_tchanx)) {
289d7024191SGrygorii Strashko 		ret = PTR_ERR(tx_chn->udma_tchanx);
290d7024191SGrygorii Strashko 		dev_err(dev, "UDMAX tchanx get err %d\n", ret);
291d7024191SGrygorii Strashko 		goto err;
292d7024191SGrygorii Strashko 	}
293d7024191SGrygorii Strashko 	tx_chn->udma_tchan_id = xudma_tchan_get_id(tx_chn->udma_tchanx);
294d7024191SGrygorii Strashko 
295*5b65781dSVignesh Raghavendra 	tx_chn->common.chan_dev.class = &k3_udma_glue_devclass;
296*5b65781dSVignesh Raghavendra 	tx_chn->common.chan_dev.parent = xudma_get_device(tx_chn->common.udmax);
297*5b65781dSVignesh Raghavendra 	dev_set_name(&tx_chn->common.chan_dev, "tchan%d-0x%04x",
298*5b65781dSVignesh Raghavendra 		     tx_chn->udma_tchan_id, tx_chn->common.dst_thread);
299*5b65781dSVignesh Raghavendra 	ret = device_register(&tx_chn->common.chan_dev);
300*5b65781dSVignesh Raghavendra 	if (ret) {
301*5b65781dSVignesh Raghavendra 		dev_err(dev, "Channel Device registration failed %d\n", ret);
302*5b65781dSVignesh Raghavendra 		tx_chn->common.chan_dev.parent = NULL;
303*5b65781dSVignesh Raghavendra 		goto err;
304*5b65781dSVignesh Raghavendra 	}
305*5b65781dSVignesh Raghavendra 
306*5b65781dSVignesh Raghavendra 	if (xudma_is_pktdma(tx_chn->common.udmax)) {
307*5b65781dSVignesh Raghavendra 		/* prepare the channel device as coherent */
308*5b65781dSVignesh Raghavendra 		tx_chn->common.chan_dev.dma_coherent = true;
309*5b65781dSVignesh Raghavendra 		dma_coerce_mask_and_coherent(&tx_chn->common.chan_dev,
310*5b65781dSVignesh Raghavendra 					     DMA_BIT_MASK(48));
311*5b65781dSVignesh Raghavendra 	}
312*5b65781dSVignesh Raghavendra 
313d7024191SGrygorii Strashko 	atomic_set(&tx_chn->free_pkts, cfg->txcq_cfg.size);
314d7024191SGrygorii Strashko 
315*5b65781dSVignesh Raghavendra 	if (xudma_is_pktdma(tx_chn->common.udmax))
316*5b65781dSVignesh Raghavendra 		tx_chn->udma_tflow_id = tx_chn->common.ep_config->default_flow_id;
317*5b65781dSVignesh Raghavendra 	else
318*5b65781dSVignesh Raghavendra 		tx_chn->udma_tflow_id = tx_chn->udma_tchan_id;
319*5b65781dSVignesh Raghavendra 
320d7024191SGrygorii Strashko 	/* request and cfg rings */
3214927b1abSPeter Ujfalusi 	ret =  k3_ringacc_request_rings_pair(tx_chn->common.ringacc,
322*5b65781dSVignesh Raghavendra 					     tx_chn->udma_tflow_id, -1,
3234927b1abSPeter Ujfalusi 					     &tx_chn->ringtx,
3244927b1abSPeter Ujfalusi 					     &tx_chn->ringtxcq);
3254927b1abSPeter Ujfalusi 	if (ret) {
3264927b1abSPeter Ujfalusi 		dev_err(dev, "Failed to get TX/TXCQ rings %d\n", ret);
327d7024191SGrygorii Strashko 		goto err;
328d7024191SGrygorii Strashko 	}
329d7024191SGrygorii Strashko 
330d553e2abSPeter Ujfalusi 	/* Set the dma_dev for the rings to be configured */
331d553e2abSPeter Ujfalusi 	cfg->tx_cfg.dma_dev = k3_udma_glue_tx_get_dma_device(tx_chn);
332d553e2abSPeter Ujfalusi 	cfg->txcq_cfg.dma_dev = cfg->tx_cfg.dma_dev;
333d553e2abSPeter Ujfalusi 
334*5b65781dSVignesh Raghavendra 	/* Set the ASEL value for DMA rings of PKTDMA */
335*5b65781dSVignesh Raghavendra 	if (xudma_is_pktdma(tx_chn->common.udmax)) {
336*5b65781dSVignesh Raghavendra 		cfg->tx_cfg.asel = tx_chn->common.atype_asel;
337*5b65781dSVignesh Raghavendra 		cfg->txcq_cfg.asel = tx_chn->common.atype_asel;
338*5b65781dSVignesh Raghavendra 	}
339*5b65781dSVignesh Raghavendra 
340d7024191SGrygorii Strashko 	ret = k3_ringacc_ring_cfg(tx_chn->ringtx, &cfg->tx_cfg);
341d7024191SGrygorii Strashko 	if (ret) {
342d7024191SGrygorii Strashko 		dev_err(dev, "Failed to cfg ringtx %d\n", ret);
343d7024191SGrygorii Strashko 		goto err;
344d7024191SGrygorii Strashko 	}
345d7024191SGrygorii Strashko 
346d7024191SGrygorii Strashko 	ret = k3_ringacc_ring_cfg(tx_chn->ringtxcq, &cfg->txcq_cfg);
347d7024191SGrygorii Strashko 	if (ret) {
348d7024191SGrygorii Strashko 		dev_err(dev, "Failed to cfg ringtx %d\n", ret);
349d7024191SGrygorii Strashko 		goto err;
350d7024191SGrygorii Strashko 	}
351d7024191SGrygorii Strashko 
352d7024191SGrygorii Strashko 	/* request and cfg psi-l */
353d7024191SGrygorii Strashko 	tx_chn->common.src_thread =
354d7024191SGrygorii Strashko 			xudma_dev_get_psil_base(tx_chn->common.udmax) +
355d7024191SGrygorii Strashko 			tx_chn->udma_tchan_id;
356d7024191SGrygorii Strashko 
357d7024191SGrygorii Strashko 	ret = k3_udma_glue_cfg_tx_chn(tx_chn);
358d7024191SGrygorii Strashko 	if (ret) {
359d7024191SGrygorii Strashko 		dev_err(dev, "Failed to cfg tchan %d\n", ret);
360d7024191SGrygorii Strashko 		goto err;
361d7024191SGrygorii Strashko 	}
362d7024191SGrygorii Strashko 
363d7024191SGrygorii Strashko 	k3_udma_glue_dump_tx_chn(tx_chn);
364d7024191SGrygorii Strashko 
365d7024191SGrygorii Strashko 	return tx_chn;
366d7024191SGrygorii Strashko 
367d7024191SGrygorii Strashko err:
368d7024191SGrygorii Strashko 	k3_udma_glue_release_tx_chn(tx_chn);
369d7024191SGrygorii Strashko 	return ERR_PTR(ret);
370d7024191SGrygorii Strashko }
371d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_request_tx_chn);
372d7024191SGrygorii Strashko 
373d7024191SGrygorii Strashko void k3_udma_glue_release_tx_chn(struct k3_udma_glue_tx_channel *tx_chn)
374d7024191SGrygorii Strashko {
375d7024191SGrygorii Strashko 	if (tx_chn->psil_paired) {
376d7024191SGrygorii Strashko 		xudma_navss_psil_unpair(tx_chn->common.udmax,
377d7024191SGrygorii Strashko 					tx_chn->common.src_thread,
378d7024191SGrygorii Strashko 					tx_chn->common.dst_thread);
379d7024191SGrygorii Strashko 		tx_chn->psil_paired = false;
380d7024191SGrygorii Strashko 	}
381d7024191SGrygorii Strashko 
382d7024191SGrygorii Strashko 	if (!IS_ERR_OR_NULL(tx_chn->udma_tchanx))
383d7024191SGrygorii Strashko 		xudma_tchan_put(tx_chn->common.udmax,
384d7024191SGrygorii Strashko 				tx_chn->udma_tchanx);
385d7024191SGrygorii Strashko 
386d7024191SGrygorii Strashko 	if (tx_chn->ringtxcq)
387d7024191SGrygorii Strashko 		k3_ringacc_ring_free(tx_chn->ringtxcq);
388d7024191SGrygorii Strashko 
389d7024191SGrygorii Strashko 	if (tx_chn->ringtx)
390d7024191SGrygorii Strashko 		k3_ringacc_ring_free(tx_chn->ringtx);
391*5b65781dSVignesh Raghavendra 
392*5b65781dSVignesh Raghavendra 	if (tx_chn->common.chan_dev.parent) {
393*5b65781dSVignesh Raghavendra 		device_unregister(&tx_chn->common.chan_dev);
394*5b65781dSVignesh Raghavendra 		tx_chn->common.chan_dev.parent = NULL;
395*5b65781dSVignesh Raghavendra 	}
396d7024191SGrygorii Strashko }
397d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_release_tx_chn);
398d7024191SGrygorii Strashko 
399d7024191SGrygorii Strashko int k3_udma_glue_push_tx_chn(struct k3_udma_glue_tx_channel *tx_chn,
400d7024191SGrygorii Strashko 			     struct cppi5_host_desc_t *desc_tx,
401d7024191SGrygorii Strashko 			     dma_addr_t desc_dma)
402d7024191SGrygorii Strashko {
403d7024191SGrygorii Strashko 	u32 ringtxcq_id;
404d7024191SGrygorii Strashko 
405d7024191SGrygorii Strashko 	if (!atomic_add_unless(&tx_chn->free_pkts, -1, 0))
406d7024191SGrygorii Strashko 		return -ENOMEM;
407d7024191SGrygorii Strashko 
408d7024191SGrygorii Strashko 	ringtxcq_id = k3_ringacc_get_ring_id(tx_chn->ringtxcq);
409d7024191SGrygorii Strashko 	cppi5_desc_set_retpolicy(&desc_tx->hdr, 0, ringtxcq_id);
410d7024191SGrygorii Strashko 
411d7024191SGrygorii Strashko 	return k3_ringacc_ring_push(tx_chn->ringtx, &desc_dma);
412d7024191SGrygorii Strashko }
413d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_push_tx_chn);
414d7024191SGrygorii Strashko 
415d7024191SGrygorii Strashko int k3_udma_glue_pop_tx_chn(struct k3_udma_glue_tx_channel *tx_chn,
416d7024191SGrygorii Strashko 			    dma_addr_t *desc_dma)
417d7024191SGrygorii Strashko {
418d7024191SGrygorii Strashko 	int ret;
419d7024191SGrygorii Strashko 
420d7024191SGrygorii Strashko 	ret = k3_ringacc_ring_pop(tx_chn->ringtxcq, desc_dma);
421d7024191SGrygorii Strashko 	if (!ret)
422d7024191SGrygorii Strashko 		atomic_inc(&tx_chn->free_pkts);
423d7024191SGrygorii Strashko 
424d7024191SGrygorii Strashko 	return ret;
425d7024191SGrygorii Strashko }
426d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_pop_tx_chn);
427d7024191SGrygorii Strashko 
428d7024191SGrygorii Strashko int k3_udma_glue_enable_tx_chn(struct k3_udma_glue_tx_channel *tx_chn)
429d7024191SGrygorii Strashko {
43069973b48SGrygorii Strashko 	int ret;
43169973b48SGrygorii Strashko 
43269973b48SGrygorii Strashko 	ret = xudma_navss_psil_pair(tx_chn->common.udmax,
43369973b48SGrygorii Strashko 				    tx_chn->common.src_thread,
43469973b48SGrygorii Strashko 				    tx_chn->common.dst_thread);
43569973b48SGrygorii Strashko 	if (ret) {
43669973b48SGrygorii Strashko 		dev_err(tx_chn->common.dev, "PSI-L request err %d\n", ret);
43769973b48SGrygorii Strashko 		return ret;
43869973b48SGrygorii Strashko 	}
43969973b48SGrygorii Strashko 
44069973b48SGrygorii Strashko 	tx_chn->psil_paired = true;
44169973b48SGrygorii Strashko 
442bc7e5523SPeter Ujfalusi 	xudma_tchanrt_write(tx_chn->udma_tchanx, UDMA_CHAN_RT_PEER_RT_EN_REG,
44352c74d3dSGrygorii Strashko 			    UDMA_PEER_RT_EN_ENABLE);
444d7024191SGrygorii Strashko 
445bc7e5523SPeter Ujfalusi 	xudma_tchanrt_write(tx_chn->udma_tchanx, UDMA_CHAN_RT_CTL_REG,
44652c74d3dSGrygorii Strashko 			    UDMA_CHAN_RT_CTL_EN);
447d7024191SGrygorii Strashko 
448d7024191SGrygorii Strashko 	k3_udma_glue_dump_tx_rt_chn(tx_chn, "txchn en");
449d7024191SGrygorii Strashko 	return 0;
450d7024191SGrygorii Strashko }
451d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_enable_tx_chn);
452d7024191SGrygorii Strashko 
453d7024191SGrygorii Strashko void k3_udma_glue_disable_tx_chn(struct k3_udma_glue_tx_channel *tx_chn)
454d7024191SGrygorii Strashko {
455d7024191SGrygorii Strashko 	k3_udma_glue_dump_tx_rt_chn(tx_chn, "txchn dis1");
456d7024191SGrygorii Strashko 
457bc7e5523SPeter Ujfalusi 	xudma_tchanrt_write(tx_chn->udma_tchanx, UDMA_CHAN_RT_CTL_REG, 0);
458d7024191SGrygorii Strashko 
459d7024191SGrygorii Strashko 	xudma_tchanrt_write(tx_chn->udma_tchanx,
460bc7e5523SPeter Ujfalusi 			    UDMA_CHAN_RT_PEER_RT_EN_REG, 0);
461d7024191SGrygorii Strashko 	k3_udma_glue_dump_tx_rt_chn(tx_chn, "txchn dis2");
46269973b48SGrygorii Strashko 
46369973b48SGrygorii Strashko 	if (tx_chn->psil_paired) {
46469973b48SGrygorii Strashko 		xudma_navss_psil_unpair(tx_chn->common.udmax,
46569973b48SGrygorii Strashko 					tx_chn->common.src_thread,
46669973b48SGrygorii Strashko 					tx_chn->common.dst_thread);
46769973b48SGrygorii Strashko 		tx_chn->psil_paired = false;
46869973b48SGrygorii Strashko 	}
469d7024191SGrygorii Strashko }
470d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_disable_tx_chn);
471d7024191SGrygorii Strashko 
472d7024191SGrygorii Strashko void k3_udma_glue_tdown_tx_chn(struct k3_udma_glue_tx_channel *tx_chn,
473d7024191SGrygorii Strashko 			       bool sync)
474d7024191SGrygorii Strashko {
475d7024191SGrygorii Strashko 	int i = 0;
476d7024191SGrygorii Strashko 	u32 val;
477d7024191SGrygorii Strashko 
478d7024191SGrygorii Strashko 	k3_udma_glue_dump_tx_rt_chn(tx_chn, "txchn tdown1");
479d7024191SGrygorii Strashko 
480bc7e5523SPeter Ujfalusi 	xudma_tchanrt_write(tx_chn->udma_tchanx, UDMA_CHAN_RT_CTL_REG,
481d7024191SGrygorii Strashko 			    UDMA_CHAN_RT_CTL_EN | UDMA_CHAN_RT_CTL_TDOWN);
482d7024191SGrygorii Strashko 
483bc7e5523SPeter Ujfalusi 	val = xudma_tchanrt_read(tx_chn->udma_tchanx, UDMA_CHAN_RT_CTL_REG);
484d7024191SGrygorii Strashko 
485d7024191SGrygorii Strashko 	while (sync && (val & UDMA_CHAN_RT_CTL_EN)) {
486d7024191SGrygorii Strashko 		val = xudma_tchanrt_read(tx_chn->udma_tchanx,
487bc7e5523SPeter Ujfalusi 					 UDMA_CHAN_RT_CTL_REG);
488d7024191SGrygorii Strashko 		udelay(1);
489d7024191SGrygorii Strashko 		if (i > K3_UDMAX_TDOWN_TIMEOUT_US) {
490d7024191SGrygorii Strashko 			dev_err(tx_chn->common.dev, "TX tdown timeout\n");
491d7024191SGrygorii Strashko 			break;
492d7024191SGrygorii Strashko 		}
493d7024191SGrygorii Strashko 		i++;
494d7024191SGrygorii Strashko 	}
495d7024191SGrygorii Strashko 
496d7024191SGrygorii Strashko 	val = xudma_tchanrt_read(tx_chn->udma_tchanx,
497bc7e5523SPeter Ujfalusi 				 UDMA_CHAN_RT_PEER_RT_EN_REG);
498d7024191SGrygorii Strashko 	if (sync && (val & UDMA_PEER_RT_EN_ENABLE))
499d7024191SGrygorii Strashko 		dev_err(tx_chn->common.dev, "TX tdown peer not stopped\n");
500d7024191SGrygorii Strashko 	k3_udma_glue_dump_tx_rt_chn(tx_chn, "txchn tdown2");
501d7024191SGrygorii Strashko }
502d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_tdown_tx_chn);
503d7024191SGrygorii Strashko 
504d7024191SGrygorii Strashko void k3_udma_glue_reset_tx_chn(struct k3_udma_glue_tx_channel *tx_chn,
505d7024191SGrygorii Strashko 			       void *data,
506d7024191SGrygorii Strashko 			       void (*cleanup)(void *data, dma_addr_t desc_dma))
507d7024191SGrygorii Strashko {
508*5b65781dSVignesh Raghavendra 	struct device *dev = tx_chn->common.dev;
509d7024191SGrygorii Strashko 	dma_addr_t desc_dma;
510d7024191SGrygorii Strashko 	int occ_tx, i, ret;
511d7024191SGrygorii Strashko 
512d7024191SGrygorii Strashko 	/*
513d7024191SGrygorii Strashko 	 * TXQ reset need to be special way as it is input for udma and its
514d7024191SGrygorii Strashko 	 * state cached by udma, so:
515d7024191SGrygorii Strashko 	 * 1) save TXQ occ
516d7024191SGrygorii Strashko 	 * 2) clean up TXQ and call callback .cleanup() for each desc
517d7024191SGrygorii Strashko 	 * 3) reset TXQ in a special way
518d7024191SGrygorii Strashko 	 */
519d7024191SGrygorii Strashko 	occ_tx = k3_ringacc_ring_get_occ(tx_chn->ringtx);
520*5b65781dSVignesh Raghavendra 	dev_dbg(dev, "TX reset occ_tx %u\n", occ_tx);
521d7024191SGrygorii Strashko 
522d7024191SGrygorii Strashko 	for (i = 0; i < occ_tx; i++) {
523d7024191SGrygorii Strashko 		ret = k3_ringacc_ring_pop(tx_chn->ringtx, &desc_dma);
524d7024191SGrygorii Strashko 		if (ret) {
525*5b65781dSVignesh Raghavendra 			if (ret != -ENODATA)
526*5b65781dSVignesh Raghavendra 				dev_err(dev, "TX reset pop %d\n", ret);
527d7024191SGrygorii Strashko 			break;
528d7024191SGrygorii Strashko 		}
529d7024191SGrygorii Strashko 		cleanup(data, desc_dma);
530d7024191SGrygorii Strashko 	}
531d7024191SGrygorii Strashko 
532*5b65781dSVignesh Raghavendra 	/* reset TXCQ as it is not input for udma - expected to be empty */
533*5b65781dSVignesh Raghavendra 	k3_ringacc_ring_reset(tx_chn->ringtxcq);
534d7024191SGrygorii Strashko 	k3_ringacc_ring_reset_dma(tx_chn->ringtx, occ_tx);
535d7024191SGrygorii Strashko }
536d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_reset_tx_chn);
537d7024191SGrygorii Strashko 
538d7024191SGrygorii Strashko u32 k3_udma_glue_tx_get_hdesc_size(struct k3_udma_glue_tx_channel *tx_chn)
539d7024191SGrygorii Strashko {
540d7024191SGrygorii Strashko 	return tx_chn->common.hdesc_size;
541d7024191SGrygorii Strashko }
542d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_tx_get_hdesc_size);
543d7024191SGrygorii Strashko 
544d7024191SGrygorii Strashko u32 k3_udma_glue_tx_get_txcq_id(struct k3_udma_glue_tx_channel *tx_chn)
545d7024191SGrygorii Strashko {
546d7024191SGrygorii Strashko 	return k3_ringacc_get_ring_id(tx_chn->ringtxcq);
547d7024191SGrygorii Strashko }
548d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_tx_get_txcq_id);
549d7024191SGrygorii Strashko 
550d7024191SGrygorii Strashko int k3_udma_glue_tx_get_irq(struct k3_udma_glue_tx_channel *tx_chn)
551d7024191SGrygorii Strashko {
552*5b65781dSVignesh Raghavendra 	if (xudma_is_pktdma(tx_chn->common.udmax)) {
553*5b65781dSVignesh Raghavendra 		tx_chn->virq = xudma_pktdma_tflow_get_irq(tx_chn->common.udmax,
554*5b65781dSVignesh Raghavendra 							  tx_chn->udma_tflow_id);
555*5b65781dSVignesh Raghavendra 	} else {
556d7024191SGrygorii Strashko 		tx_chn->virq = k3_ringacc_get_ring_irq_num(tx_chn->ringtxcq);
557*5b65781dSVignesh Raghavendra 	}
558d7024191SGrygorii Strashko 
559d7024191SGrygorii Strashko 	return tx_chn->virq;
560d7024191SGrygorii Strashko }
561d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_tx_get_irq);
562d7024191SGrygorii Strashko 
563426506a7SPeter Ujfalusi struct device *
564426506a7SPeter Ujfalusi 	k3_udma_glue_tx_get_dma_device(struct k3_udma_glue_tx_channel *tx_chn)
565426506a7SPeter Ujfalusi {
566*5b65781dSVignesh Raghavendra 	if (xudma_is_pktdma(tx_chn->common.udmax) &&
567*5b65781dSVignesh Raghavendra 	    (tx_chn->common.atype_asel == 14 || tx_chn->common.atype_asel == 15))
568*5b65781dSVignesh Raghavendra 		return &tx_chn->common.chan_dev;
569*5b65781dSVignesh Raghavendra 
570426506a7SPeter Ujfalusi 	return xudma_get_device(tx_chn->common.udmax);
571426506a7SPeter Ujfalusi }
572426506a7SPeter Ujfalusi EXPORT_SYMBOL_GPL(k3_udma_glue_tx_get_dma_device);
573426506a7SPeter Ujfalusi 
574*5b65781dSVignesh Raghavendra void k3_udma_glue_tx_dma_to_cppi5_addr(struct k3_udma_glue_tx_channel *tx_chn,
575*5b65781dSVignesh Raghavendra 				       dma_addr_t *addr)
576*5b65781dSVignesh Raghavendra {
577*5b65781dSVignesh Raghavendra 	if (!xudma_is_pktdma(tx_chn->common.udmax) ||
578*5b65781dSVignesh Raghavendra 	    !tx_chn->common.atype_asel)
579*5b65781dSVignesh Raghavendra 		return;
580*5b65781dSVignesh Raghavendra 
581*5b65781dSVignesh Raghavendra 	*addr |= (u64)tx_chn->common.atype_asel << K3_ADDRESS_ASEL_SHIFT;
582*5b65781dSVignesh Raghavendra }
583*5b65781dSVignesh Raghavendra EXPORT_SYMBOL_GPL(k3_udma_glue_tx_dma_to_cppi5_addr);
584*5b65781dSVignesh Raghavendra 
585*5b65781dSVignesh Raghavendra void k3_udma_glue_tx_cppi5_to_dma_addr(struct k3_udma_glue_tx_channel *tx_chn,
586*5b65781dSVignesh Raghavendra 				       dma_addr_t *addr)
587*5b65781dSVignesh Raghavendra {
588*5b65781dSVignesh Raghavendra 	if (!xudma_is_pktdma(tx_chn->common.udmax) ||
589*5b65781dSVignesh Raghavendra 	    !tx_chn->common.atype_asel)
590*5b65781dSVignesh Raghavendra 		return;
591*5b65781dSVignesh Raghavendra 
592*5b65781dSVignesh Raghavendra 	*addr &= (u64)GENMASK(K3_ADDRESS_ASEL_SHIFT - 1, 0);
593*5b65781dSVignesh Raghavendra }
594*5b65781dSVignesh Raghavendra EXPORT_SYMBOL_GPL(k3_udma_glue_tx_cppi5_to_dma_addr);
595*5b65781dSVignesh Raghavendra 
596d7024191SGrygorii Strashko static int k3_udma_glue_cfg_rx_chn(struct k3_udma_glue_rx_channel *rx_chn)
597d7024191SGrygorii Strashko {
598d7024191SGrygorii Strashko 	const struct udma_tisci_rm *tisci_rm = rx_chn->common.tisci_rm;
599d7024191SGrygorii Strashko 	struct ti_sci_msg_rm_udmap_rx_ch_cfg req;
600d7024191SGrygorii Strashko 	int ret;
601d7024191SGrygorii Strashko 
602d7024191SGrygorii Strashko 	memset(&req, 0, sizeof(req));
603d7024191SGrygorii Strashko 
604d7024191SGrygorii Strashko 	req.valid_params = TI_SCI_MSG_VALUE_RM_UDMAP_CH_FETCH_SIZE_VALID |
605d7024191SGrygorii Strashko 			   TI_SCI_MSG_VALUE_RM_UDMAP_CH_CQ_QNUM_VALID |
606d7024191SGrygorii Strashko 			   TI_SCI_MSG_VALUE_RM_UDMAP_CH_CHAN_TYPE_VALID |
6070ebcf1a2SPeter Ujfalusi 			   TI_SCI_MSG_VALUE_RM_UDMAP_CH_ATYPE_VALID;
608d7024191SGrygorii Strashko 
609d7024191SGrygorii Strashko 	req.nav_id = tisci_rm->tisci_dev_id;
610d7024191SGrygorii Strashko 	req.index = rx_chn->udma_rchan_id;
611d7024191SGrygorii Strashko 	req.rx_fetch_size = rx_chn->common.hdesc_size >> 2;
612d7024191SGrygorii Strashko 	/*
613d7024191SGrygorii Strashko 	 * TODO: we can't support rxcq_qnum/RCHAN[a]_RCQ cfg with current sysfw
614d7024191SGrygorii Strashko 	 * and udmax impl, so just configure it to invalid value.
615d7024191SGrygorii Strashko 	 * req.rxcq_qnum = k3_ringacc_get_ring_id(rx_chn->flows[0].ringrx);
616d7024191SGrygorii Strashko 	 */
617d7024191SGrygorii Strashko 	req.rxcq_qnum = 0xFFFF;
618*5b65781dSVignesh Raghavendra 	if (!xudma_is_pktdma(rx_chn->common.udmax) && rx_chn->flow_num &&
619*5b65781dSVignesh Raghavendra 	    rx_chn->flow_id_base != rx_chn->udma_rchan_id) {
620d7024191SGrygorii Strashko 		/* Default flow + extra ones */
621*5b65781dSVignesh Raghavendra 		req.valid_params |= TI_SCI_MSG_VALUE_RM_UDMAP_CH_RX_FLOWID_START_VALID |
622*5b65781dSVignesh Raghavendra 				    TI_SCI_MSG_VALUE_RM_UDMAP_CH_RX_FLOWID_CNT_VALID;
623d7024191SGrygorii Strashko 		req.flowid_start = rx_chn->flow_id_base;
624d7024191SGrygorii Strashko 		req.flowid_cnt = rx_chn->flow_num;
625d7024191SGrygorii Strashko 	}
626d7024191SGrygorii Strashko 	req.rx_chan_type = TI_SCI_RM_UDMAP_CHAN_TYPE_PKT_PBRR;
627*5b65781dSVignesh Raghavendra 	req.rx_atype = rx_chn->common.atype_asel;
628d7024191SGrygorii Strashko 
629d7024191SGrygorii Strashko 	ret = tisci_rm->tisci_udmap_ops->rx_ch_cfg(tisci_rm->tisci, &req);
630d7024191SGrygorii Strashko 	if (ret)
631d7024191SGrygorii Strashko 		dev_err(rx_chn->common.dev, "rchan%d cfg failed %d\n",
632d7024191SGrygorii Strashko 			rx_chn->udma_rchan_id, ret);
633d7024191SGrygorii Strashko 
634d7024191SGrygorii Strashko 	return ret;
635d7024191SGrygorii Strashko }
636d7024191SGrygorii Strashko 
637d7024191SGrygorii Strashko static void k3_udma_glue_release_rx_flow(struct k3_udma_glue_rx_channel *rx_chn,
638d7024191SGrygorii Strashko 					 u32 flow_num)
639d7024191SGrygorii Strashko {
640d7024191SGrygorii Strashko 	struct k3_udma_glue_rx_flow *flow = &rx_chn->flows[flow_num];
641d7024191SGrygorii Strashko 
642d7024191SGrygorii Strashko 	if (IS_ERR_OR_NULL(flow->udma_rflow))
643d7024191SGrygorii Strashko 		return;
644d7024191SGrygorii Strashko 
645d7024191SGrygorii Strashko 	if (flow->ringrxfdq)
646d7024191SGrygorii Strashko 		k3_ringacc_ring_free(flow->ringrxfdq);
647d7024191SGrygorii Strashko 
648d7024191SGrygorii Strashko 	if (flow->ringrx)
649d7024191SGrygorii Strashko 		k3_ringacc_ring_free(flow->ringrx);
650d7024191SGrygorii Strashko 
651d7024191SGrygorii Strashko 	xudma_rflow_put(rx_chn->common.udmax, flow->udma_rflow);
652d7024191SGrygorii Strashko 	flow->udma_rflow = NULL;
653d7024191SGrygorii Strashko 	rx_chn->flows_ready--;
654d7024191SGrygorii Strashko }
655d7024191SGrygorii Strashko 
656d7024191SGrygorii Strashko static int k3_udma_glue_cfg_rx_flow(struct k3_udma_glue_rx_channel *rx_chn,
657d7024191SGrygorii Strashko 				    u32 flow_idx,
658d7024191SGrygorii Strashko 				    struct k3_udma_glue_rx_flow_cfg *flow_cfg)
659d7024191SGrygorii Strashko {
660d7024191SGrygorii Strashko 	struct k3_udma_glue_rx_flow *flow = &rx_chn->flows[flow_idx];
661d7024191SGrygorii Strashko 	const struct udma_tisci_rm *tisci_rm = rx_chn->common.tisci_rm;
662d7024191SGrygorii Strashko 	struct device *dev = rx_chn->common.dev;
663d7024191SGrygorii Strashko 	struct ti_sci_msg_rm_udmap_flow_cfg req;
664d7024191SGrygorii Strashko 	int rx_ring_id;
665d7024191SGrygorii Strashko 	int rx_ringfdq_id;
666d7024191SGrygorii Strashko 	int ret = 0;
667d7024191SGrygorii Strashko 
668d7024191SGrygorii Strashko 	flow->udma_rflow = xudma_rflow_get(rx_chn->common.udmax,
669d7024191SGrygorii Strashko 					   flow->udma_rflow_id);
670d7024191SGrygorii Strashko 	if (IS_ERR(flow->udma_rflow)) {
671d7024191SGrygorii Strashko 		ret = PTR_ERR(flow->udma_rflow);
672d7024191SGrygorii Strashko 		dev_err(dev, "UDMAX rflow get err %d\n", ret);
673018af9beSChristophe JAILLET 		return ret;
674d7024191SGrygorii Strashko 	}
675d7024191SGrygorii Strashko 
676d7024191SGrygorii Strashko 	if (flow->udma_rflow_id != xudma_rflow_get_id(flow->udma_rflow)) {
677018af9beSChristophe JAILLET 		ret = -ENODEV;
678018af9beSChristophe JAILLET 		goto err_rflow_put;
679d7024191SGrygorii Strashko 	}
680d7024191SGrygorii Strashko 
681*5b65781dSVignesh Raghavendra 	if (xudma_is_pktdma(rx_chn->common.udmax)) {
682*5b65781dSVignesh Raghavendra 		rx_ringfdq_id = flow->udma_rflow_id +
683*5b65781dSVignesh Raghavendra 				xudma_get_rflow_ring_offset(rx_chn->common.udmax);
684*5b65781dSVignesh Raghavendra 		rx_ring_id = 0;
685*5b65781dSVignesh Raghavendra 	} else {
686*5b65781dSVignesh Raghavendra 		rx_ring_id = flow_cfg->ring_rxq_id;
687*5b65781dSVignesh Raghavendra 		rx_ringfdq_id = flow_cfg->ring_rxfdq0_id;
688*5b65781dSVignesh Raghavendra 	}
689*5b65781dSVignesh Raghavendra 
690d7024191SGrygorii Strashko 	/* request and cfg rings */
6914927b1abSPeter Ujfalusi 	ret =  k3_ringacc_request_rings_pair(rx_chn->common.ringacc,
692*5b65781dSVignesh Raghavendra 					     rx_ringfdq_id, rx_ring_id,
6934927b1abSPeter Ujfalusi 					     &flow->ringrxfdq,
6944927b1abSPeter Ujfalusi 					     &flow->ringrx);
6954927b1abSPeter Ujfalusi 	if (ret) {
6964927b1abSPeter Ujfalusi 		dev_err(dev, "Failed to get RX/RXFDQ rings %d\n", ret);
697018af9beSChristophe JAILLET 		goto err_rflow_put;
698d7024191SGrygorii Strashko 	}
699d7024191SGrygorii Strashko 
700d553e2abSPeter Ujfalusi 	/* Set the dma_dev for the rings to be configured */
701d553e2abSPeter Ujfalusi 	flow_cfg->rx_cfg.dma_dev = k3_udma_glue_rx_get_dma_device(rx_chn);
702d553e2abSPeter Ujfalusi 	flow_cfg->rxfdq_cfg.dma_dev = flow_cfg->rx_cfg.dma_dev;
703d553e2abSPeter Ujfalusi 
704*5b65781dSVignesh Raghavendra 	/* Set the ASEL value for DMA rings of PKTDMA */
705*5b65781dSVignesh Raghavendra 	if (xudma_is_pktdma(rx_chn->common.udmax)) {
706*5b65781dSVignesh Raghavendra 		flow_cfg->rx_cfg.asel = rx_chn->common.atype_asel;
707*5b65781dSVignesh Raghavendra 		flow_cfg->rxfdq_cfg.asel = rx_chn->common.atype_asel;
708*5b65781dSVignesh Raghavendra 	}
709*5b65781dSVignesh Raghavendra 
710d7024191SGrygorii Strashko 	ret = k3_ringacc_ring_cfg(flow->ringrx, &flow_cfg->rx_cfg);
711d7024191SGrygorii Strashko 	if (ret) {
712d7024191SGrygorii Strashko 		dev_err(dev, "Failed to cfg ringrx %d\n", ret);
713018af9beSChristophe JAILLET 		goto err_ringrxfdq_free;
714d7024191SGrygorii Strashko 	}
715d7024191SGrygorii Strashko 
716d7024191SGrygorii Strashko 	ret = k3_ringacc_ring_cfg(flow->ringrxfdq, &flow_cfg->rxfdq_cfg);
717d7024191SGrygorii Strashko 	if (ret) {
718d7024191SGrygorii Strashko 		dev_err(dev, "Failed to cfg ringrxfdq %d\n", ret);
719018af9beSChristophe JAILLET 		goto err_ringrxfdq_free;
720d7024191SGrygorii Strashko 	}
721d7024191SGrygorii Strashko 
722d7024191SGrygorii Strashko 	if (rx_chn->remote) {
723d7024191SGrygorii Strashko 		rx_ring_id = TI_SCI_RESOURCE_NULL;
724d7024191SGrygorii Strashko 		rx_ringfdq_id = TI_SCI_RESOURCE_NULL;
725d7024191SGrygorii Strashko 	} else {
726d7024191SGrygorii Strashko 		rx_ring_id = k3_ringacc_get_ring_id(flow->ringrx);
727d7024191SGrygorii Strashko 		rx_ringfdq_id = k3_ringacc_get_ring_id(flow->ringrxfdq);
728d7024191SGrygorii Strashko 	}
729d7024191SGrygorii Strashko 
730d7024191SGrygorii Strashko 	memset(&req, 0, sizeof(req));
731d7024191SGrygorii Strashko 
732d7024191SGrygorii Strashko 	req.valid_params =
733d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_EINFO_PRESENT_VALID |
734d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_PSINFO_PRESENT_VALID |
735d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_ERROR_HANDLING_VALID |
736d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_DESC_TYPE_VALID |
737d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_DEST_QNUM_VALID |
738d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_SRC_TAG_HI_SEL_VALID |
739d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_SRC_TAG_LO_SEL_VALID |
740d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_DEST_TAG_HI_SEL_VALID |
741d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_DEST_TAG_LO_SEL_VALID |
742d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_FDQ0_SZ0_QNUM_VALID |
743d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_FDQ1_QNUM_VALID |
744d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_FDQ2_QNUM_VALID |
745d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_FDQ3_QNUM_VALID;
746d7024191SGrygorii Strashko 	req.nav_id = tisci_rm->tisci_dev_id;
747d7024191SGrygorii Strashko 	req.flow_index = flow->udma_rflow_id;
748d7024191SGrygorii Strashko 	if (rx_chn->common.epib)
749d7024191SGrygorii Strashko 		req.rx_einfo_present = 1;
750d7024191SGrygorii Strashko 	if (rx_chn->common.psdata_size)
751d7024191SGrygorii Strashko 		req.rx_psinfo_present = 1;
752d7024191SGrygorii Strashko 	if (flow_cfg->rx_error_handling)
753d7024191SGrygorii Strashko 		req.rx_error_handling = 1;
754d7024191SGrygorii Strashko 	req.rx_desc_type = 0;
755d7024191SGrygorii Strashko 	req.rx_dest_qnum = rx_ring_id;
756d7024191SGrygorii Strashko 	req.rx_src_tag_hi_sel = 0;
757d7024191SGrygorii Strashko 	req.rx_src_tag_lo_sel = flow_cfg->src_tag_lo_sel;
758d7024191SGrygorii Strashko 	req.rx_dest_tag_hi_sel = 0;
759d7024191SGrygorii Strashko 	req.rx_dest_tag_lo_sel = 0;
760d7024191SGrygorii Strashko 	req.rx_fdq0_sz0_qnum = rx_ringfdq_id;
761d7024191SGrygorii Strashko 	req.rx_fdq1_qnum = rx_ringfdq_id;
762d7024191SGrygorii Strashko 	req.rx_fdq2_qnum = rx_ringfdq_id;
763d7024191SGrygorii Strashko 	req.rx_fdq3_qnum = rx_ringfdq_id;
764d7024191SGrygorii Strashko 
765d7024191SGrygorii Strashko 	ret = tisci_rm->tisci_udmap_ops->rx_flow_cfg(tisci_rm->tisci, &req);
766d7024191SGrygorii Strashko 	if (ret) {
767d7024191SGrygorii Strashko 		dev_err(dev, "flow%d config failed: %d\n", flow->udma_rflow_id,
768d7024191SGrygorii Strashko 			ret);
769018af9beSChristophe JAILLET 		goto err_ringrxfdq_free;
770d7024191SGrygorii Strashko 	}
771d7024191SGrygorii Strashko 
772d7024191SGrygorii Strashko 	rx_chn->flows_ready++;
773d7024191SGrygorii Strashko 	dev_dbg(dev, "flow%d config done. ready:%d\n",
774d7024191SGrygorii Strashko 		flow->udma_rflow_id, rx_chn->flows_ready);
775d7024191SGrygorii Strashko 
776d7024191SGrygorii Strashko 	return 0;
777018af9beSChristophe JAILLET 
778018af9beSChristophe JAILLET err_ringrxfdq_free:
779018af9beSChristophe JAILLET 	k3_ringacc_ring_free(flow->ringrxfdq);
780018af9beSChristophe JAILLET 	k3_ringacc_ring_free(flow->ringrx);
781018af9beSChristophe JAILLET 
782018af9beSChristophe JAILLET err_rflow_put:
783018af9beSChristophe JAILLET 	xudma_rflow_put(rx_chn->common.udmax, flow->udma_rflow);
784018af9beSChristophe JAILLET 	flow->udma_rflow = NULL;
785018af9beSChristophe JAILLET 
786d7024191SGrygorii Strashko 	return ret;
787d7024191SGrygorii Strashko }
788d7024191SGrygorii Strashko 
789d7024191SGrygorii Strashko static void k3_udma_glue_dump_rx_chn(struct k3_udma_glue_rx_channel *chn)
790d7024191SGrygorii Strashko {
791d7024191SGrygorii Strashko 	struct device *dev = chn->common.dev;
792d7024191SGrygorii Strashko 
793d7024191SGrygorii Strashko 	dev_dbg(dev, "dump_rx_chn:\n"
794d7024191SGrygorii Strashko 		"udma_rchan_id: %d\n"
795d7024191SGrygorii Strashko 		"src_thread: %08x\n"
796d7024191SGrygorii Strashko 		"dst_thread: %08x\n"
797d7024191SGrygorii Strashko 		"epib: %d\n"
798d7024191SGrygorii Strashko 		"hdesc_size: %u\n"
799d7024191SGrygorii Strashko 		"psdata_size: %u\n"
800d7024191SGrygorii Strashko 		"swdata_size: %u\n"
801d7024191SGrygorii Strashko 		"flow_id_base: %d\n"
802d7024191SGrygorii Strashko 		"flow_num: %d\n",
803d7024191SGrygorii Strashko 		chn->udma_rchan_id,
804d7024191SGrygorii Strashko 		chn->common.src_thread,
805d7024191SGrygorii Strashko 		chn->common.dst_thread,
806d7024191SGrygorii Strashko 		chn->common.epib,
807d7024191SGrygorii Strashko 		chn->common.hdesc_size,
808d7024191SGrygorii Strashko 		chn->common.psdata_size,
809d7024191SGrygorii Strashko 		chn->common.swdata_size,
810d7024191SGrygorii Strashko 		chn->flow_id_base,
811d7024191SGrygorii Strashko 		chn->flow_num);
812d7024191SGrygorii Strashko }
813d7024191SGrygorii Strashko 
814d7024191SGrygorii Strashko static void k3_udma_glue_dump_rx_rt_chn(struct k3_udma_glue_rx_channel *chn,
815d7024191SGrygorii Strashko 					char *mark)
816d7024191SGrygorii Strashko {
817d7024191SGrygorii Strashko 	struct device *dev = chn->common.dev;
818d7024191SGrygorii Strashko 
819d7024191SGrygorii Strashko 	dev_dbg(dev, "=== dump ===> %s\n", mark);
820d7024191SGrygorii Strashko 
821bc7e5523SPeter Ujfalusi 	dev_dbg(dev, "0x%08X: %08X\n", UDMA_CHAN_RT_CTL_REG,
822bc7e5523SPeter Ujfalusi 		xudma_rchanrt_read(chn->udma_rchanx, UDMA_CHAN_RT_CTL_REG));
823bc7e5523SPeter Ujfalusi 	dev_dbg(dev, "0x%08X: %08X\n", UDMA_CHAN_RT_PEER_RT_EN_REG,
824d7024191SGrygorii Strashko 		xudma_rchanrt_read(chn->udma_rchanx,
825bc7e5523SPeter Ujfalusi 				   UDMA_CHAN_RT_PEER_RT_EN_REG));
826bc7e5523SPeter Ujfalusi 	dev_dbg(dev, "0x%08X: %08X\n", UDMA_CHAN_RT_PCNT_REG,
827bc7e5523SPeter Ujfalusi 		xudma_rchanrt_read(chn->udma_rchanx, UDMA_CHAN_RT_PCNT_REG));
828bc7e5523SPeter Ujfalusi 	dev_dbg(dev, "0x%08X: %08X\n", UDMA_CHAN_RT_BCNT_REG,
829bc7e5523SPeter Ujfalusi 		xudma_rchanrt_read(chn->udma_rchanx, UDMA_CHAN_RT_BCNT_REG));
830bc7e5523SPeter Ujfalusi 	dev_dbg(dev, "0x%08X: %08X\n", UDMA_CHAN_RT_SBCNT_REG,
831bc7e5523SPeter Ujfalusi 		xudma_rchanrt_read(chn->udma_rchanx, UDMA_CHAN_RT_SBCNT_REG));
832d7024191SGrygorii Strashko }
833d7024191SGrygorii Strashko 
834d7024191SGrygorii Strashko static int
835d7024191SGrygorii Strashko k3_udma_glue_allocate_rx_flows(struct k3_udma_glue_rx_channel *rx_chn,
836d7024191SGrygorii Strashko 			       struct k3_udma_glue_rx_channel_cfg *cfg)
837d7024191SGrygorii Strashko {
838d7024191SGrygorii Strashko 	int ret;
839d7024191SGrygorii Strashko 
840d7024191SGrygorii Strashko 	/* default rflow */
841d7024191SGrygorii Strashko 	if (cfg->flow_id_use_rxchan_id)
842d7024191SGrygorii Strashko 		return 0;
843d7024191SGrygorii Strashko 
844d7024191SGrygorii Strashko 	/* not a GP rflows */
845d7024191SGrygorii Strashko 	if (rx_chn->flow_id_base != -1 &&
846d7024191SGrygorii Strashko 	    !xudma_rflow_is_gp(rx_chn->common.udmax, rx_chn->flow_id_base))
847d7024191SGrygorii Strashko 		return 0;
848d7024191SGrygorii Strashko 
849d7024191SGrygorii Strashko 	/* Allocate range of GP rflows */
850d7024191SGrygorii Strashko 	ret = xudma_alloc_gp_rflow_range(rx_chn->common.udmax,
851d7024191SGrygorii Strashko 					 rx_chn->flow_id_base,
852d7024191SGrygorii Strashko 					 rx_chn->flow_num);
853d7024191SGrygorii Strashko 	if (ret < 0) {
854d7024191SGrygorii Strashko 		dev_err(rx_chn->common.dev, "UDMAX reserve_rflow %d cnt:%d err: %d\n",
855d7024191SGrygorii Strashko 			rx_chn->flow_id_base, rx_chn->flow_num, ret);
856d7024191SGrygorii Strashko 		return ret;
857d7024191SGrygorii Strashko 	}
858d7024191SGrygorii Strashko 	rx_chn->flow_id_base = ret;
859d7024191SGrygorii Strashko 
860d7024191SGrygorii Strashko 	return 0;
861d7024191SGrygorii Strashko }
862d7024191SGrygorii Strashko 
863d7024191SGrygorii Strashko static struct k3_udma_glue_rx_channel *
864d7024191SGrygorii Strashko k3_udma_glue_request_rx_chn_priv(struct device *dev, const char *name,
865d7024191SGrygorii Strashko 				 struct k3_udma_glue_rx_channel_cfg *cfg)
866d7024191SGrygorii Strashko {
867d7024191SGrygorii Strashko 	struct k3_udma_glue_rx_channel *rx_chn;
868*5b65781dSVignesh Raghavendra 	struct psil_endpoint_config *ep_cfg;
869d7024191SGrygorii Strashko 	int ret, i;
870d7024191SGrygorii Strashko 
871d7024191SGrygorii Strashko 	if (cfg->flow_id_num <= 0)
872d7024191SGrygorii Strashko 		return ERR_PTR(-EINVAL);
873d7024191SGrygorii Strashko 
874d7024191SGrygorii Strashko 	if (cfg->flow_id_num != 1 &&
875d7024191SGrygorii Strashko 	    (cfg->def_flow_cfg || cfg->flow_id_use_rxchan_id))
876d7024191SGrygorii Strashko 		return ERR_PTR(-EINVAL);
877d7024191SGrygorii Strashko 
878d7024191SGrygorii Strashko 	rx_chn = devm_kzalloc(dev, sizeof(*rx_chn), GFP_KERNEL);
879d7024191SGrygorii Strashko 	if (!rx_chn)
880d7024191SGrygorii Strashko 		return ERR_PTR(-ENOMEM);
881d7024191SGrygorii Strashko 
882d7024191SGrygorii Strashko 	rx_chn->common.dev = dev;
883d7024191SGrygorii Strashko 	rx_chn->common.swdata_size = cfg->swdata_size;
884d7024191SGrygorii Strashko 	rx_chn->remote = false;
885d7024191SGrygorii Strashko 
886d7024191SGrygorii Strashko 	/* parse of udmap channel */
887d7024191SGrygorii Strashko 	ret = of_k3_udma_glue_parse_chn(dev->of_node, name,
888d7024191SGrygorii Strashko 					&rx_chn->common, false);
889d7024191SGrygorii Strashko 	if (ret)
890d7024191SGrygorii Strashko 		goto err;
891d7024191SGrygorii Strashko 
892d7024191SGrygorii Strashko 	rx_chn->common.hdesc_size = cppi5_hdesc_calc_size(rx_chn->common.epib,
893d7024191SGrygorii Strashko 						rx_chn->common.psdata_size,
894d7024191SGrygorii Strashko 						rx_chn->common.swdata_size);
895d7024191SGrygorii Strashko 
896*5b65781dSVignesh Raghavendra 	ep_cfg = rx_chn->common.ep_config;
897*5b65781dSVignesh Raghavendra 
898*5b65781dSVignesh Raghavendra 	if (xudma_is_pktdma(rx_chn->common.udmax))
899*5b65781dSVignesh Raghavendra 		rx_chn->udma_rchan_id = ep_cfg->mapped_channel_id;
900*5b65781dSVignesh Raghavendra 	else
901*5b65781dSVignesh Raghavendra 		rx_chn->udma_rchan_id = -1;
902*5b65781dSVignesh Raghavendra 
903d7024191SGrygorii Strashko 	/* request and cfg UDMAP RX channel */
904*5b65781dSVignesh Raghavendra 	rx_chn->udma_rchanx = xudma_rchan_get(rx_chn->common.udmax,
905*5b65781dSVignesh Raghavendra 					      rx_chn->udma_rchan_id);
906d7024191SGrygorii Strashko 	if (IS_ERR(rx_chn->udma_rchanx)) {
907d7024191SGrygorii Strashko 		ret = PTR_ERR(rx_chn->udma_rchanx);
908d7024191SGrygorii Strashko 		dev_err(dev, "UDMAX rchanx get err %d\n", ret);
909d7024191SGrygorii Strashko 		goto err;
910d7024191SGrygorii Strashko 	}
911d7024191SGrygorii Strashko 	rx_chn->udma_rchan_id = xudma_rchan_get_id(rx_chn->udma_rchanx);
912d7024191SGrygorii Strashko 
913*5b65781dSVignesh Raghavendra 	rx_chn->common.chan_dev.class = &k3_udma_glue_devclass;
914*5b65781dSVignesh Raghavendra 	rx_chn->common.chan_dev.parent = xudma_get_device(rx_chn->common.udmax);
915*5b65781dSVignesh Raghavendra 	dev_set_name(&rx_chn->common.chan_dev, "rchan%d-0x%04x",
916*5b65781dSVignesh Raghavendra 		     rx_chn->udma_rchan_id, rx_chn->common.src_thread);
917*5b65781dSVignesh Raghavendra 	ret = device_register(&rx_chn->common.chan_dev);
918*5b65781dSVignesh Raghavendra 	if (ret) {
919*5b65781dSVignesh Raghavendra 		dev_err(dev, "Channel Device registration failed %d\n", ret);
920*5b65781dSVignesh Raghavendra 		rx_chn->common.chan_dev.parent = NULL;
921*5b65781dSVignesh Raghavendra 		goto err;
922*5b65781dSVignesh Raghavendra 	}
923*5b65781dSVignesh Raghavendra 
924*5b65781dSVignesh Raghavendra 	if (xudma_is_pktdma(rx_chn->common.udmax)) {
925*5b65781dSVignesh Raghavendra 		/* prepare the channel device as coherent */
926*5b65781dSVignesh Raghavendra 		rx_chn->common.chan_dev.dma_coherent = true;
927*5b65781dSVignesh Raghavendra 		dma_coerce_mask_and_coherent(&rx_chn->common.chan_dev,
928*5b65781dSVignesh Raghavendra 					     DMA_BIT_MASK(48));
929*5b65781dSVignesh Raghavendra 	}
930*5b65781dSVignesh Raghavendra 
931*5b65781dSVignesh Raghavendra 	if (xudma_is_pktdma(rx_chn->common.udmax)) {
932*5b65781dSVignesh Raghavendra 		int flow_start = cfg->flow_id_base;
933*5b65781dSVignesh Raghavendra 		int flow_end;
934*5b65781dSVignesh Raghavendra 
935*5b65781dSVignesh Raghavendra 		if (flow_start == -1)
936*5b65781dSVignesh Raghavendra 			flow_start = ep_cfg->flow_start;
937*5b65781dSVignesh Raghavendra 
938*5b65781dSVignesh Raghavendra 		flow_end = flow_start + cfg->flow_id_num - 1;
939*5b65781dSVignesh Raghavendra 		if (flow_start < ep_cfg->flow_start ||
940*5b65781dSVignesh Raghavendra 		    flow_end > (ep_cfg->flow_start + ep_cfg->flow_num - 1)) {
941*5b65781dSVignesh Raghavendra 			dev_err(dev, "Invalid flow range requested\n");
942*5b65781dSVignesh Raghavendra 			ret = -EINVAL;
943*5b65781dSVignesh Raghavendra 			goto err;
944*5b65781dSVignesh Raghavendra 		}
945*5b65781dSVignesh Raghavendra 		rx_chn->flow_id_base = flow_start;
946*5b65781dSVignesh Raghavendra 	} else {
947d7024191SGrygorii Strashko 		rx_chn->flow_id_base = cfg->flow_id_base;
948d7024191SGrygorii Strashko 
949d7024191SGrygorii Strashko 		/* Use RX channel id as flow id: target dev can't generate flow_id */
950d7024191SGrygorii Strashko 		if (cfg->flow_id_use_rxchan_id)
951d7024191SGrygorii Strashko 			rx_chn->flow_id_base = rx_chn->udma_rchan_id;
952*5b65781dSVignesh Raghavendra 	}
953*5b65781dSVignesh Raghavendra 
954*5b65781dSVignesh Raghavendra 	rx_chn->flow_num = cfg->flow_id_num;
955d7024191SGrygorii Strashko 
956d7024191SGrygorii Strashko 	rx_chn->flows = devm_kcalloc(dev, rx_chn->flow_num,
957d7024191SGrygorii Strashko 				     sizeof(*rx_chn->flows), GFP_KERNEL);
958d7024191SGrygorii Strashko 	if (!rx_chn->flows) {
959d7024191SGrygorii Strashko 		ret = -ENOMEM;
960d7024191SGrygorii Strashko 		goto err;
961d7024191SGrygorii Strashko 	}
962d7024191SGrygorii Strashko 
963d7024191SGrygorii Strashko 	ret = k3_udma_glue_allocate_rx_flows(rx_chn, cfg);
964d7024191SGrygorii Strashko 	if (ret)
965d7024191SGrygorii Strashko 		goto err;
966d7024191SGrygorii Strashko 
967d7024191SGrygorii Strashko 	for (i = 0; i < rx_chn->flow_num; i++)
968d7024191SGrygorii Strashko 		rx_chn->flows[i].udma_rflow_id = rx_chn->flow_id_base + i;
969d7024191SGrygorii Strashko 
970d7024191SGrygorii Strashko 	/* request and cfg psi-l */
971d7024191SGrygorii Strashko 	rx_chn->common.dst_thread =
972d7024191SGrygorii Strashko 			xudma_dev_get_psil_base(rx_chn->common.udmax) +
973d7024191SGrygorii Strashko 			rx_chn->udma_rchan_id;
974d7024191SGrygorii Strashko 
975d7024191SGrygorii Strashko 	ret = k3_udma_glue_cfg_rx_chn(rx_chn);
976d7024191SGrygorii Strashko 	if (ret) {
977d7024191SGrygorii Strashko 		dev_err(dev, "Failed to cfg rchan %d\n", ret);
978d7024191SGrygorii Strashko 		goto err;
979d7024191SGrygorii Strashko 	}
980d7024191SGrygorii Strashko 
981d7024191SGrygorii Strashko 	/* init default RX flow only if flow_num = 1 */
982d7024191SGrygorii Strashko 	if (cfg->def_flow_cfg) {
983d7024191SGrygorii Strashko 		ret = k3_udma_glue_cfg_rx_flow(rx_chn, 0, cfg->def_flow_cfg);
984d7024191SGrygorii Strashko 		if (ret)
985d7024191SGrygorii Strashko 			goto err;
986d7024191SGrygorii Strashko 	}
987d7024191SGrygorii Strashko 
988d7024191SGrygorii Strashko 	k3_udma_glue_dump_rx_chn(rx_chn);
989d7024191SGrygorii Strashko 
990d7024191SGrygorii Strashko 	return rx_chn;
991d7024191SGrygorii Strashko 
992d7024191SGrygorii Strashko err:
993d7024191SGrygorii Strashko 	k3_udma_glue_release_rx_chn(rx_chn);
994d7024191SGrygorii Strashko 	return ERR_PTR(ret);
995d7024191SGrygorii Strashko }
996d7024191SGrygorii Strashko 
997d7024191SGrygorii Strashko static struct k3_udma_glue_rx_channel *
998d7024191SGrygorii Strashko k3_udma_glue_request_remote_rx_chn(struct device *dev, const char *name,
999d7024191SGrygorii Strashko 				   struct k3_udma_glue_rx_channel_cfg *cfg)
1000d7024191SGrygorii Strashko {
1001d7024191SGrygorii Strashko 	struct k3_udma_glue_rx_channel *rx_chn;
1002d7024191SGrygorii Strashko 	int ret, i;
1003d7024191SGrygorii Strashko 
1004d7024191SGrygorii Strashko 	if (cfg->flow_id_num <= 0 ||
1005d7024191SGrygorii Strashko 	    cfg->flow_id_use_rxchan_id ||
1006d7024191SGrygorii Strashko 	    cfg->def_flow_cfg ||
1007d7024191SGrygorii Strashko 	    cfg->flow_id_base < 0)
1008d7024191SGrygorii Strashko 		return ERR_PTR(-EINVAL);
1009d7024191SGrygorii Strashko 
1010d7024191SGrygorii Strashko 	/*
1011d7024191SGrygorii Strashko 	 * Remote RX channel is under control of Remote CPU core, so
1012d7024191SGrygorii Strashko 	 * Linux can only request and manipulate by dedicated RX flows
1013d7024191SGrygorii Strashko 	 */
1014d7024191SGrygorii Strashko 
1015d7024191SGrygorii Strashko 	rx_chn = devm_kzalloc(dev, sizeof(*rx_chn), GFP_KERNEL);
1016d7024191SGrygorii Strashko 	if (!rx_chn)
1017d7024191SGrygorii Strashko 		return ERR_PTR(-ENOMEM);
1018d7024191SGrygorii Strashko 
1019d7024191SGrygorii Strashko 	rx_chn->common.dev = dev;
1020d7024191SGrygorii Strashko 	rx_chn->common.swdata_size = cfg->swdata_size;
1021d7024191SGrygorii Strashko 	rx_chn->remote = true;
1022d7024191SGrygorii Strashko 	rx_chn->udma_rchan_id = -1;
1023d7024191SGrygorii Strashko 	rx_chn->flow_num = cfg->flow_id_num;
1024d7024191SGrygorii Strashko 	rx_chn->flow_id_base = cfg->flow_id_base;
1025d7024191SGrygorii Strashko 	rx_chn->psil_paired = false;
1026d7024191SGrygorii Strashko 
1027d7024191SGrygorii Strashko 	/* parse of udmap channel */
1028d7024191SGrygorii Strashko 	ret = of_k3_udma_glue_parse_chn(dev->of_node, name,
1029d7024191SGrygorii Strashko 					&rx_chn->common, false);
1030d7024191SGrygorii Strashko 	if (ret)
1031d7024191SGrygorii Strashko 		goto err;
1032d7024191SGrygorii Strashko 
1033d7024191SGrygorii Strashko 	rx_chn->common.hdesc_size = cppi5_hdesc_calc_size(rx_chn->common.epib,
1034d7024191SGrygorii Strashko 						rx_chn->common.psdata_size,
1035d7024191SGrygorii Strashko 						rx_chn->common.swdata_size);
1036d7024191SGrygorii Strashko 
1037d7024191SGrygorii Strashko 	rx_chn->flows = devm_kcalloc(dev, rx_chn->flow_num,
1038d7024191SGrygorii Strashko 				     sizeof(*rx_chn->flows), GFP_KERNEL);
1039d7024191SGrygorii Strashko 	if (!rx_chn->flows) {
1040d7024191SGrygorii Strashko 		ret = -ENOMEM;
1041d7024191SGrygorii Strashko 		goto err;
1042d7024191SGrygorii Strashko 	}
1043d7024191SGrygorii Strashko 
1044*5b65781dSVignesh Raghavendra 	rx_chn->common.chan_dev.class = &k3_udma_glue_devclass;
1045*5b65781dSVignesh Raghavendra 	rx_chn->common.chan_dev.parent = xudma_get_device(rx_chn->common.udmax);
1046*5b65781dSVignesh Raghavendra 	dev_set_name(&rx_chn->common.chan_dev, "rchan_remote-0x%04x",
1047*5b65781dSVignesh Raghavendra 		     rx_chn->common.src_thread);
1048*5b65781dSVignesh Raghavendra 	ret = device_register(&rx_chn->common.chan_dev);
1049*5b65781dSVignesh Raghavendra 	if (ret) {
1050*5b65781dSVignesh Raghavendra 		dev_err(dev, "Channel Device registration failed %d\n", ret);
1051*5b65781dSVignesh Raghavendra 		rx_chn->common.chan_dev.parent = NULL;
1052*5b65781dSVignesh Raghavendra 		goto err;
1053*5b65781dSVignesh Raghavendra 	}
1054*5b65781dSVignesh Raghavendra 
1055*5b65781dSVignesh Raghavendra 	if (xudma_is_pktdma(rx_chn->common.udmax)) {
1056*5b65781dSVignesh Raghavendra 		/* prepare the channel device as coherent */
1057*5b65781dSVignesh Raghavendra 		rx_chn->common.chan_dev.dma_coherent = true;
1058*5b65781dSVignesh Raghavendra 		dma_coerce_mask_and_coherent(&rx_chn->common.chan_dev,
1059*5b65781dSVignesh Raghavendra 					     DMA_BIT_MASK(48));
1060*5b65781dSVignesh Raghavendra 	}
1061*5b65781dSVignesh Raghavendra 
1062d7024191SGrygorii Strashko 	ret = k3_udma_glue_allocate_rx_flows(rx_chn, cfg);
1063d7024191SGrygorii Strashko 	if (ret)
1064d7024191SGrygorii Strashko 		goto err;
1065d7024191SGrygorii Strashko 
1066d7024191SGrygorii Strashko 	for (i = 0; i < rx_chn->flow_num; i++)
1067d7024191SGrygorii Strashko 		rx_chn->flows[i].udma_rflow_id = rx_chn->flow_id_base + i;
1068d7024191SGrygorii Strashko 
1069d7024191SGrygorii Strashko 	k3_udma_glue_dump_rx_chn(rx_chn);
1070d7024191SGrygorii Strashko 
1071d7024191SGrygorii Strashko 	return rx_chn;
1072d7024191SGrygorii Strashko 
1073d7024191SGrygorii Strashko err:
1074d7024191SGrygorii Strashko 	k3_udma_glue_release_rx_chn(rx_chn);
1075d7024191SGrygorii Strashko 	return ERR_PTR(ret);
1076d7024191SGrygorii Strashko }
1077d7024191SGrygorii Strashko 
1078d7024191SGrygorii Strashko struct k3_udma_glue_rx_channel *
1079d7024191SGrygorii Strashko k3_udma_glue_request_rx_chn(struct device *dev, const char *name,
1080d7024191SGrygorii Strashko 			    struct k3_udma_glue_rx_channel_cfg *cfg)
1081d7024191SGrygorii Strashko {
1082d7024191SGrygorii Strashko 	if (cfg->remote)
1083d7024191SGrygorii Strashko 		return k3_udma_glue_request_remote_rx_chn(dev, name, cfg);
1084d7024191SGrygorii Strashko 	else
1085d7024191SGrygorii Strashko 		return k3_udma_glue_request_rx_chn_priv(dev, name, cfg);
1086d7024191SGrygorii Strashko }
1087d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_request_rx_chn);
1088d7024191SGrygorii Strashko 
1089d7024191SGrygorii Strashko void k3_udma_glue_release_rx_chn(struct k3_udma_glue_rx_channel *rx_chn)
1090d7024191SGrygorii Strashko {
1091d7024191SGrygorii Strashko 	int i;
1092d7024191SGrygorii Strashko 
1093d7024191SGrygorii Strashko 	if (IS_ERR_OR_NULL(rx_chn->common.udmax))
1094d7024191SGrygorii Strashko 		return;
1095d7024191SGrygorii Strashko 
1096d7024191SGrygorii Strashko 	if (rx_chn->psil_paired) {
1097d7024191SGrygorii Strashko 		xudma_navss_psil_unpair(rx_chn->common.udmax,
1098d7024191SGrygorii Strashko 					rx_chn->common.src_thread,
1099d7024191SGrygorii Strashko 					rx_chn->common.dst_thread);
1100d7024191SGrygorii Strashko 		rx_chn->psil_paired = false;
1101d7024191SGrygorii Strashko 	}
1102d7024191SGrygorii Strashko 
1103d7024191SGrygorii Strashko 	for (i = 0; i < rx_chn->flow_num; i++)
1104d7024191SGrygorii Strashko 		k3_udma_glue_release_rx_flow(rx_chn, i);
1105d7024191SGrygorii Strashko 
1106d7024191SGrygorii Strashko 	if (xudma_rflow_is_gp(rx_chn->common.udmax, rx_chn->flow_id_base))
1107d7024191SGrygorii Strashko 		xudma_free_gp_rflow_range(rx_chn->common.udmax,
1108d7024191SGrygorii Strashko 					  rx_chn->flow_id_base,
1109d7024191SGrygorii Strashko 					  rx_chn->flow_num);
1110d7024191SGrygorii Strashko 
1111d7024191SGrygorii Strashko 	if (!IS_ERR_OR_NULL(rx_chn->udma_rchanx))
1112d7024191SGrygorii Strashko 		xudma_rchan_put(rx_chn->common.udmax,
1113d7024191SGrygorii Strashko 				rx_chn->udma_rchanx);
1114*5b65781dSVignesh Raghavendra 
1115*5b65781dSVignesh Raghavendra 	if (rx_chn->common.chan_dev.parent) {
1116*5b65781dSVignesh Raghavendra 		device_unregister(&rx_chn->common.chan_dev);
1117*5b65781dSVignesh Raghavendra 		rx_chn->common.chan_dev.parent = NULL;
1118*5b65781dSVignesh Raghavendra 	}
1119d7024191SGrygorii Strashko }
1120d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_release_rx_chn);
1121d7024191SGrygorii Strashko 
1122d7024191SGrygorii Strashko int k3_udma_glue_rx_flow_init(struct k3_udma_glue_rx_channel *rx_chn,
1123d7024191SGrygorii Strashko 			      u32 flow_idx,
1124d7024191SGrygorii Strashko 			      struct k3_udma_glue_rx_flow_cfg *flow_cfg)
1125d7024191SGrygorii Strashko {
1126d7024191SGrygorii Strashko 	if (flow_idx >= rx_chn->flow_num)
1127d7024191SGrygorii Strashko 		return -EINVAL;
1128d7024191SGrygorii Strashko 
1129d7024191SGrygorii Strashko 	return k3_udma_glue_cfg_rx_flow(rx_chn, flow_idx, flow_cfg);
1130d7024191SGrygorii Strashko }
1131d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_rx_flow_init);
1132d7024191SGrygorii Strashko 
1133d7024191SGrygorii Strashko u32 k3_udma_glue_rx_flow_get_fdq_id(struct k3_udma_glue_rx_channel *rx_chn,
1134d7024191SGrygorii Strashko 				    u32 flow_idx)
1135d7024191SGrygorii Strashko {
1136d7024191SGrygorii Strashko 	struct k3_udma_glue_rx_flow *flow;
1137d7024191SGrygorii Strashko 
1138d7024191SGrygorii Strashko 	if (flow_idx >= rx_chn->flow_num)
1139d7024191SGrygorii Strashko 		return -EINVAL;
1140d7024191SGrygorii Strashko 
1141d7024191SGrygorii Strashko 	flow = &rx_chn->flows[flow_idx];
1142d7024191SGrygorii Strashko 
1143d7024191SGrygorii Strashko 	return k3_ringacc_get_ring_id(flow->ringrxfdq);
1144d7024191SGrygorii Strashko }
1145d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_rx_flow_get_fdq_id);
1146d7024191SGrygorii Strashko 
1147d7024191SGrygorii Strashko u32 k3_udma_glue_rx_get_flow_id_base(struct k3_udma_glue_rx_channel *rx_chn)
1148d7024191SGrygorii Strashko {
1149d7024191SGrygorii Strashko 	return rx_chn->flow_id_base;
1150d7024191SGrygorii Strashko }
1151d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_rx_get_flow_id_base);
1152d7024191SGrygorii Strashko 
1153d7024191SGrygorii Strashko int k3_udma_glue_rx_flow_enable(struct k3_udma_glue_rx_channel *rx_chn,
1154d7024191SGrygorii Strashko 				u32 flow_idx)
1155d7024191SGrygorii Strashko {
1156d7024191SGrygorii Strashko 	struct k3_udma_glue_rx_flow *flow = &rx_chn->flows[flow_idx];
1157d7024191SGrygorii Strashko 	const struct udma_tisci_rm *tisci_rm = rx_chn->common.tisci_rm;
1158d7024191SGrygorii Strashko 	struct device *dev = rx_chn->common.dev;
1159d7024191SGrygorii Strashko 	struct ti_sci_msg_rm_udmap_flow_cfg req;
1160d7024191SGrygorii Strashko 	int rx_ring_id;
1161d7024191SGrygorii Strashko 	int rx_ringfdq_id;
1162d7024191SGrygorii Strashko 	int ret = 0;
1163d7024191SGrygorii Strashko 
1164d7024191SGrygorii Strashko 	if (!rx_chn->remote)
1165d7024191SGrygorii Strashko 		return -EINVAL;
1166d7024191SGrygorii Strashko 
1167d7024191SGrygorii Strashko 	rx_ring_id = k3_ringacc_get_ring_id(flow->ringrx);
1168d7024191SGrygorii Strashko 	rx_ringfdq_id = k3_ringacc_get_ring_id(flow->ringrxfdq);
1169d7024191SGrygorii Strashko 
1170d7024191SGrygorii Strashko 	memset(&req, 0, sizeof(req));
1171d7024191SGrygorii Strashko 
1172d7024191SGrygorii Strashko 	req.valid_params =
1173d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_DEST_QNUM_VALID |
1174d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_FDQ0_SZ0_QNUM_VALID |
1175d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_FDQ1_QNUM_VALID |
1176d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_FDQ2_QNUM_VALID |
1177d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_FDQ3_QNUM_VALID;
1178d7024191SGrygorii Strashko 	req.nav_id = tisci_rm->tisci_dev_id;
1179d7024191SGrygorii Strashko 	req.flow_index = flow->udma_rflow_id;
1180d7024191SGrygorii Strashko 	req.rx_dest_qnum = rx_ring_id;
1181d7024191SGrygorii Strashko 	req.rx_fdq0_sz0_qnum = rx_ringfdq_id;
1182d7024191SGrygorii Strashko 	req.rx_fdq1_qnum = rx_ringfdq_id;
1183d7024191SGrygorii Strashko 	req.rx_fdq2_qnum = rx_ringfdq_id;
1184d7024191SGrygorii Strashko 	req.rx_fdq3_qnum = rx_ringfdq_id;
1185d7024191SGrygorii Strashko 
1186d7024191SGrygorii Strashko 	ret = tisci_rm->tisci_udmap_ops->rx_flow_cfg(tisci_rm->tisci, &req);
1187d7024191SGrygorii Strashko 	if (ret) {
1188d7024191SGrygorii Strashko 		dev_err(dev, "flow%d enable failed: %d\n", flow->udma_rflow_id,
1189d7024191SGrygorii Strashko 			ret);
1190d7024191SGrygorii Strashko 	}
1191d7024191SGrygorii Strashko 
1192d7024191SGrygorii Strashko 	return ret;
1193d7024191SGrygorii Strashko }
1194d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_rx_flow_enable);
1195d7024191SGrygorii Strashko 
1196d7024191SGrygorii Strashko int k3_udma_glue_rx_flow_disable(struct k3_udma_glue_rx_channel *rx_chn,
1197d7024191SGrygorii Strashko 				 u32 flow_idx)
1198d7024191SGrygorii Strashko {
1199d7024191SGrygorii Strashko 	struct k3_udma_glue_rx_flow *flow = &rx_chn->flows[flow_idx];
1200d7024191SGrygorii Strashko 	const struct udma_tisci_rm *tisci_rm = rx_chn->common.tisci_rm;
1201d7024191SGrygorii Strashko 	struct device *dev = rx_chn->common.dev;
1202d7024191SGrygorii Strashko 	struct ti_sci_msg_rm_udmap_flow_cfg req;
1203d7024191SGrygorii Strashko 	int ret = 0;
1204d7024191SGrygorii Strashko 
1205d7024191SGrygorii Strashko 	if (!rx_chn->remote)
1206d7024191SGrygorii Strashko 		return -EINVAL;
1207d7024191SGrygorii Strashko 
1208d7024191SGrygorii Strashko 	memset(&req, 0, sizeof(req));
1209d7024191SGrygorii Strashko 	req.valid_params =
1210d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_DEST_QNUM_VALID |
1211d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_FDQ0_SZ0_QNUM_VALID |
1212d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_FDQ1_QNUM_VALID |
1213d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_FDQ2_QNUM_VALID |
1214d7024191SGrygorii Strashko 			TI_SCI_MSG_VALUE_RM_UDMAP_FLOW_FDQ3_QNUM_VALID;
1215d7024191SGrygorii Strashko 	req.nav_id = tisci_rm->tisci_dev_id;
1216d7024191SGrygorii Strashko 	req.flow_index = flow->udma_rflow_id;
1217d7024191SGrygorii Strashko 	req.rx_dest_qnum = TI_SCI_RESOURCE_NULL;
1218d7024191SGrygorii Strashko 	req.rx_fdq0_sz0_qnum = TI_SCI_RESOURCE_NULL;
1219d7024191SGrygorii Strashko 	req.rx_fdq1_qnum = TI_SCI_RESOURCE_NULL;
1220d7024191SGrygorii Strashko 	req.rx_fdq2_qnum = TI_SCI_RESOURCE_NULL;
1221d7024191SGrygorii Strashko 	req.rx_fdq3_qnum = TI_SCI_RESOURCE_NULL;
1222d7024191SGrygorii Strashko 
1223d7024191SGrygorii Strashko 	ret = tisci_rm->tisci_udmap_ops->rx_flow_cfg(tisci_rm->tisci, &req);
1224d7024191SGrygorii Strashko 	if (ret) {
1225d7024191SGrygorii Strashko 		dev_err(dev, "flow%d disable failed: %d\n", flow->udma_rflow_id,
1226d7024191SGrygorii Strashko 			ret);
1227d7024191SGrygorii Strashko 	}
1228d7024191SGrygorii Strashko 
1229d7024191SGrygorii Strashko 	return ret;
1230d7024191SGrygorii Strashko }
1231d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_rx_flow_disable);
1232d7024191SGrygorii Strashko 
1233d7024191SGrygorii Strashko int k3_udma_glue_enable_rx_chn(struct k3_udma_glue_rx_channel *rx_chn)
1234d7024191SGrygorii Strashko {
123569973b48SGrygorii Strashko 	int ret;
123669973b48SGrygorii Strashko 
1237d7024191SGrygorii Strashko 	if (rx_chn->remote)
1238d7024191SGrygorii Strashko 		return -EINVAL;
1239d7024191SGrygorii Strashko 
1240d7024191SGrygorii Strashko 	if (rx_chn->flows_ready < rx_chn->flow_num)
1241d7024191SGrygorii Strashko 		return -EINVAL;
1242d7024191SGrygorii Strashko 
124369973b48SGrygorii Strashko 	ret = xudma_navss_psil_pair(rx_chn->common.udmax,
124469973b48SGrygorii Strashko 				    rx_chn->common.src_thread,
124569973b48SGrygorii Strashko 				    rx_chn->common.dst_thread);
124669973b48SGrygorii Strashko 	if (ret) {
124769973b48SGrygorii Strashko 		dev_err(rx_chn->common.dev, "PSI-L request err %d\n", ret);
124869973b48SGrygorii Strashko 		return ret;
124969973b48SGrygorii Strashko 	}
125069973b48SGrygorii Strashko 
125169973b48SGrygorii Strashko 	rx_chn->psil_paired = true;
125269973b48SGrygorii Strashko 
1253bc7e5523SPeter Ujfalusi 	xudma_rchanrt_write(rx_chn->udma_rchanx, UDMA_CHAN_RT_CTL_REG,
125452c74d3dSGrygorii Strashko 			    UDMA_CHAN_RT_CTL_EN);
1255d7024191SGrygorii Strashko 
1256bc7e5523SPeter Ujfalusi 	xudma_rchanrt_write(rx_chn->udma_rchanx, UDMA_CHAN_RT_PEER_RT_EN_REG,
1257d7024191SGrygorii Strashko 			    UDMA_PEER_RT_EN_ENABLE);
1258d7024191SGrygorii Strashko 
1259d7024191SGrygorii Strashko 	k3_udma_glue_dump_rx_rt_chn(rx_chn, "rxrt en");
1260d7024191SGrygorii Strashko 	return 0;
1261d7024191SGrygorii Strashko }
1262d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_enable_rx_chn);
1263d7024191SGrygorii Strashko 
1264d7024191SGrygorii Strashko void k3_udma_glue_disable_rx_chn(struct k3_udma_glue_rx_channel *rx_chn)
1265d7024191SGrygorii Strashko {
1266d7024191SGrygorii Strashko 	k3_udma_glue_dump_rx_rt_chn(rx_chn, "rxrt dis1");
1267d7024191SGrygorii Strashko 
1268d7024191SGrygorii Strashko 	xudma_rchanrt_write(rx_chn->udma_rchanx,
1269bc7e5523SPeter Ujfalusi 			    UDMA_CHAN_RT_PEER_RT_EN_REG, 0);
1270bc7e5523SPeter Ujfalusi 	xudma_rchanrt_write(rx_chn->udma_rchanx, UDMA_CHAN_RT_CTL_REG, 0);
1271d7024191SGrygorii Strashko 
1272d7024191SGrygorii Strashko 	k3_udma_glue_dump_rx_rt_chn(rx_chn, "rxrt dis2");
127369973b48SGrygorii Strashko 
127469973b48SGrygorii Strashko 	if (rx_chn->psil_paired) {
127569973b48SGrygorii Strashko 		xudma_navss_psil_unpair(rx_chn->common.udmax,
127669973b48SGrygorii Strashko 					rx_chn->common.src_thread,
127769973b48SGrygorii Strashko 					rx_chn->common.dst_thread);
127869973b48SGrygorii Strashko 		rx_chn->psil_paired = false;
127969973b48SGrygorii Strashko 	}
1280d7024191SGrygorii Strashko }
1281d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_disable_rx_chn);
1282d7024191SGrygorii Strashko 
1283d7024191SGrygorii Strashko void k3_udma_glue_tdown_rx_chn(struct k3_udma_glue_rx_channel *rx_chn,
1284d7024191SGrygorii Strashko 			       bool sync)
1285d7024191SGrygorii Strashko {
1286d7024191SGrygorii Strashko 	int i = 0;
1287d7024191SGrygorii Strashko 	u32 val;
1288d7024191SGrygorii Strashko 
1289d7024191SGrygorii Strashko 	if (rx_chn->remote)
1290d7024191SGrygorii Strashko 		return;
1291d7024191SGrygorii Strashko 
1292d7024191SGrygorii Strashko 	k3_udma_glue_dump_rx_rt_chn(rx_chn, "rxrt tdown1");
1293d7024191SGrygorii Strashko 
1294bc7e5523SPeter Ujfalusi 	xudma_rchanrt_write(rx_chn->udma_rchanx, UDMA_CHAN_RT_PEER_RT_EN_REG,
1295d7024191SGrygorii Strashko 			    UDMA_PEER_RT_EN_ENABLE | UDMA_PEER_RT_EN_TEARDOWN);
1296d7024191SGrygorii Strashko 
1297bc7e5523SPeter Ujfalusi 	val = xudma_rchanrt_read(rx_chn->udma_rchanx, UDMA_CHAN_RT_CTL_REG);
1298d7024191SGrygorii Strashko 
1299d7024191SGrygorii Strashko 	while (sync && (val & UDMA_CHAN_RT_CTL_EN)) {
1300d7024191SGrygorii Strashko 		val = xudma_rchanrt_read(rx_chn->udma_rchanx,
1301bc7e5523SPeter Ujfalusi 					 UDMA_CHAN_RT_CTL_REG);
1302d7024191SGrygorii Strashko 		udelay(1);
1303d7024191SGrygorii Strashko 		if (i > K3_UDMAX_TDOWN_TIMEOUT_US) {
1304d7024191SGrygorii Strashko 			dev_err(rx_chn->common.dev, "RX tdown timeout\n");
1305d7024191SGrygorii Strashko 			break;
1306d7024191SGrygorii Strashko 		}
1307d7024191SGrygorii Strashko 		i++;
1308d7024191SGrygorii Strashko 	}
1309d7024191SGrygorii Strashko 
1310d7024191SGrygorii Strashko 	val = xudma_rchanrt_read(rx_chn->udma_rchanx,
1311bc7e5523SPeter Ujfalusi 				 UDMA_CHAN_RT_PEER_RT_EN_REG);
1312d7024191SGrygorii Strashko 	if (sync && (val & UDMA_PEER_RT_EN_ENABLE))
1313d7024191SGrygorii Strashko 		dev_err(rx_chn->common.dev, "TX tdown peer not stopped\n");
1314d7024191SGrygorii Strashko 	k3_udma_glue_dump_rx_rt_chn(rx_chn, "rxrt tdown2");
1315d7024191SGrygorii Strashko }
1316d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_tdown_rx_chn);
1317d7024191SGrygorii Strashko 
1318d7024191SGrygorii Strashko void k3_udma_glue_reset_rx_chn(struct k3_udma_glue_rx_channel *rx_chn,
1319d7024191SGrygorii Strashko 		u32 flow_num, void *data,
1320d7024191SGrygorii Strashko 		void (*cleanup)(void *data, dma_addr_t desc_dma), bool skip_fdq)
1321d7024191SGrygorii Strashko {
1322d7024191SGrygorii Strashko 	struct k3_udma_glue_rx_flow *flow = &rx_chn->flows[flow_num];
1323d7024191SGrygorii Strashko 	struct device *dev = rx_chn->common.dev;
1324d7024191SGrygorii Strashko 	dma_addr_t desc_dma;
1325d7024191SGrygorii Strashko 	int occ_rx, i, ret;
1326d7024191SGrygorii Strashko 
1327d7024191SGrygorii Strashko 	/* reset RXCQ as it is not input for udma - expected to be empty */
1328d7024191SGrygorii Strashko 	occ_rx = k3_ringacc_ring_get_occ(flow->ringrx);
1329d7024191SGrygorii Strashko 	dev_dbg(dev, "RX reset flow %u occ_rx %u\n", flow_num, occ_rx);
1330d7024191SGrygorii Strashko 
1331d7024191SGrygorii Strashko 	/* Skip RX FDQ in case one FDQ is used for the set of flows */
1332d7024191SGrygorii Strashko 	if (skip_fdq)
1333*5b65781dSVignesh Raghavendra 		goto do_reset;
1334d7024191SGrygorii Strashko 
1335d7024191SGrygorii Strashko 	/*
1336d7024191SGrygorii Strashko 	 * RX FDQ reset need to be special way as it is input for udma and its
1337d7024191SGrygorii Strashko 	 * state cached by udma, so:
1338d7024191SGrygorii Strashko 	 * 1) save RX FDQ occ
1339d7024191SGrygorii Strashko 	 * 2) clean up RX FDQ and call callback .cleanup() for each desc
1340d7024191SGrygorii Strashko 	 * 3) reset RX FDQ in a special way
1341d7024191SGrygorii Strashko 	 */
1342d7024191SGrygorii Strashko 	occ_rx = k3_ringacc_ring_get_occ(flow->ringrxfdq);
1343d7024191SGrygorii Strashko 	dev_dbg(dev, "RX reset flow %u occ_rx_fdq %u\n", flow_num, occ_rx);
1344d7024191SGrygorii Strashko 
1345d7024191SGrygorii Strashko 	for (i = 0; i < occ_rx; i++) {
1346d7024191SGrygorii Strashko 		ret = k3_ringacc_ring_pop(flow->ringrxfdq, &desc_dma);
1347d7024191SGrygorii Strashko 		if (ret) {
1348*5b65781dSVignesh Raghavendra 			if (ret != -ENODATA)
1349d7024191SGrygorii Strashko 				dev_err(dev, "RX reset pop %d\n", ret);
1350d7024191SGrygorii Strashko 			break;
1351d7024191SGrygorii Strashko 		}
1352d7024191SGrygorii Strashko 		cleanup(data, desc_dma);
1353d7024191SGrygorii Strashko 	}
1354d7024191SGrygorii Strashko 
1355d7024191SGrygorii Strashko 	k3_ringacc_ring_reset_dma(flow->ringrxfdq, occ_rx);
1356*5b65781dSVignesh Raghavendra 
1357*5b65781dSVignesh Raghavendra do_reset:
1358*5b65781dSVignesh Raghavendra 	k3_ringacc_ring_reset(flow->ringrx);
1359d7024191SGrygorii Strashko }
1360d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_reset_rx_chn);
1361d7024191SGrygorii Strashko 
1362d7024191SGrygorii Strashko int k3_udma_glue_push_rx_chn(struct k3_udma_glue_rx_channel *rx_chn,
1363d7024191SGrygorii Strashko 			     u32 flow_num, struct cppi5_host_desc_t *desc_rx,
1364d7024191SGrygorii Strashko 			     dma_addr_t desc_dma)
1365d7024191SGrygorii Strashko {
1366d7024191SGrygorii Strashko 	struct k3_udma_glue_rx_flow *flow = &rx_chn->flows[flow_num];
1367d7024191SGrygorii Strashko 
1368d7024191SGrygorii Strashko 	return k3_ringacc_ring_push(flow->ringrxfdq, &desc_dma);
1369d7024191SGrygorii Strashko }
1370d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_push_rx_chn);
1371d7024191SGrygorii Strashko 
1372d7024191SGrygorii Strashko int k3_udma_glue_pop_rx_chn(struct k3_udma_glue_rx_channel *rx_chn,
1373d7024191SGrygorii Strashko 			    u32 flow_num, dma_addr_t *desc_dma)
1374d7024191SGrygorii Strashko {
1375d7024191SGrygorii Strashko 	struct k3_udma_glue_rx_flow *flow = &rx_chn->flows[flow_num];
1376d7024191SGrygorii Strashko 
1377d7024191SGrygorii Strashko 	return k3_ringacc_ring_pop(flow->ringrx, desc_dma);
1378d7024191SGrygorii Strashko }
1379d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_pop_rx_chn);
1380d7024191SGrygorii Strashko 
1381d7024191SGrygorii Strashko int k3_udma_glue_rx_get_irq(struct k3_udma_glue_rx_channel *rx_chn,
1382d7024191SGrygorii Strashko 			    u32 flow_num)
1383d7024191SGrygorii Strashko {
1384d7024191SGrygorii Strashko 	struct k3_udma_glue_rx_flow *flow;
1385d7024191SGrygorii Strashko 
1386d7024191SGrygorii Strashko 	flow = &rx_chn->flows[flow_num];
1387d7024191SGrygorii Strashko 
1388*5b65781dSVignesh Raghavendra 	if (xudma_is_pktdma(rx_chn->common.udmax)) {
1389*5b65781dSVignesh Raghavendra 		flow->virq = xudma_pktdma_rflow_get_irq(rx_chn->common.udmax,
1390*5b65781dSVignesh Raghavendra 							flow->udma_rflow_id);
1391*5b65781dSVignesh Raghavendra 	} else {
1392d7024191SGrygorii Strashko 		flow->virq = k3_ringacc_get_ring_irq_num(flow->ringrx);
1393*5b65781dSVignesh Raghavendra 	}
1394d7024191SGrygorii Strashko 
1395d7024191SGrygorii Strashko 	return flow->virq;
1396d7024191SGrygorii Strashko }
1397d7024191SGrygorii Strashko EXPORT_SYMBOL_GPL(k3_udma_glue_rx_get_irq);
1398426506a7SPeter Ujfalusi 
1399426506a7SPeter Ujfalusi struct device *
1400426506a7SPeter Ujfalusi 	k3_udma_glue_rx_get_dma_device(struct k3_udma_glue_rx_channel *rx_chn)
1401426506a7SPeter Ujfalusi {
1402*5b65781dSVignesh Raghavendra 	if (xudma_is_pktdma(rx_chn->common.udmax) &&
1403*5b65781dSVignesh Raghavendra 	    (rx_chn->common.atype_asel == 14 || rx_chn->common.atype_asel == 15))
1404*5b65781dSVignesh Raghavendra 		return &rx_chn->common.chan_dev;
1405*5b65781dSVignesh Raghavendra 
1406426506a7SPeter Ujfalusi 	return xudma_get_device(rx_chn->common.udmax);
1407426506a7SPeter Ujfalusi }
1408426506a7SPeter Ujfalusi EXPORT_SYMBOL_GPL(k3_udma_glue_rx_get_dma_device);
1409*5b65781dSVignesh Raghavendra 
1410*5b65781dSVignesh Raghavendra void k3_udma_glue_rx_dma_to_cppi5_addr(struct k3_udma_glue_rx_channel *rx_chn,
1411*5b65781dSVignesh Raghavendra 				       dma_addr_t *addr)
1412*5b65781dSVignesh Raghavendra {
1413*5b65781dSVignesh Raghavendra 	if (!xudma_is_pktdma(rx_chn->common.udmax) ||
1414*5b65781dSVignesh Raghavendra 	    !rx_chn->common.atype_asel)
1415*5b65781dSVignesh Raghavendra 		return;
1416*5b65781dSVignesh Raghavendra 
1417*5b65781dSVignesh Raghavendra 	*addr |= (u64)rx_chn->common.atype_asel << K3_ADDRESS_ASEL_SHIFT;
1418*5b65781dSVignesh Raghavendra }
1419*5b65781dSVignesh Raghavendra EXPORT_SYMBOL_GPL(k3_udma_glue_rx_dma_to_cppi5_addr);
1420*5b65781dSVignesh Raghavendra 
1421*5b65781dSVignesh Raghavendra void k3_udma_glue_rx_cppi5_to_dma_addr(struct k3_udma_glue_rx_channel *rx_chn,
1422*5b65781dSVignesh Raghavendra 				       dma_addr_t *addr)
1423*5b65781dSVignesh Raghavendra {
1424*5b65781dSVignesh Raghavendra 	if (!xudma_is_pktdma(rx_chn->common.udmax) ||
1425*5b65781dSVignesh Raghavendra 	    !rx_chn->common.atype_asel)
1426*5b65781dSVignesh Raghavendra 		return;
1427*5b65781dSVignesh Raghavendra 
1428*5b65781dSVignesh Raghavendra 	*addr &= (u64)GENMASK(K3_ADDRESS_ASEL_SHIFT - 1, 0);
1429*5b65781dSVignesh Raghavendra }
1430*5b65781dSVignesh Raghavendra EXPORT_SYMBOL_GPL(k3_udma_glue_rx_cppi5_to_dma_addr);
1431*5b65781dSVignesh Raghavendra 
1432*5b65781dSVignesh Raghavendra static int __init k3_udma_glue_class_init(void)
1433*5b65781dSVignesh Raghavendra {
1434*5b65781dSVignesh Raghavendra 	return class_register(&k3_udma_glue_devclass);
1435*5b65781dSVignesh Raghavendra }
1436*5b65781dSVignesh Raghavendra arch_initcall(k3_udma_glue_class_init);
1437