xref: /dpdk/lib/eventdev/rte_event_dma_adapter.c (revision cfa81500ac39f473e4026d51e989dfc356b61856)
1346eb2fdSAmit Prakash Shukla /* SPDX-License-Identifier: BSD-3-Clause
2346eb2fdSAmit Prakash Shukla  * Copyright (c) 2023 Marvell.
3346eb2fdSAmit Prakash Shukla  */
4346eb2fdSAmit Prakash Shukla 
5346eb2fdSAmit Prakash Shukla #include <eventdev_pmd.h>
63c89e8c4SAmit Prakash Shukla #include <rte_service_component.h>
7346eb2fdSAmit Prakash Shukla 
8346eb2fdSAmit Prakash Shukla #include "rte_event_dma_adapter.h"
9346eb2fdSAmit Prakash Shukla 
10346eb2fdSAmit Prakash Shukla #define DMA_BATCH_SIZE 32
11346eb2fdSAmit Prakash Shukla #define DMA_DEFAULT_MAX_NB 128
12346eb2fdSAmit Prakash Shukla #define DMA_ADAPTER_NAME_LEN 32
13346eb2fdSAmit Prakash Shukla #define DMA_ADAPTER_BUFFER_SIZE 1024
14346eb2fdSAmit Prakash Shukla 
15346eb2fdSAmit Prakash Shukla #define DMA_ADAPTER_OPS_BUFFER_SIZE (DMA_BATCH_SIZE + DMA_BATCH_SIZE)
16346eb2fdSAmit Prakash Shukla 
17346eb2fdSAmit Prakash Shukla #define DMA_ADAPTER_ARRAY "event_dma_adapter_array"
18346eb2fdSAmit Prakash Shukla 
19346eb2fdSAmit Prakash Shukla /* Macros to check for valid adapter */
20346eb2fdSAmit Prakash Shukla #define EVENT_DMA_ADAPTER_ID_VALID_OR_ERR_RET(id, retval) \
21346eb2fdSAmit Prakash Shukla 	do { \
22346eb2fdSAmit Prakash Shukla 		if (!edma_adapter_valid_id(id)) { \
23ae282b06SDavid Marchand 			RTE_EDEV_LOG_ERR("Invalid DMA adapter id = %d", id); \
24346eb2fdSAmit Prakash Shukla 			return retval; \
25346eb2fdSAmit Prakash Shukla 		} \
26346eb2fdSAmit Prakash Shukla 	} while (0)
27346eb2fdSAmit Prakash Shukla 
28346eb2fdSAmit Prakash Shukla /* DMA ops circular buffer */
29c6552d9aSTyler Retzlaff struct __rte_cache_aligned dma_ops_circular_buffer {
30346eb2fdSAmit Prakash Shukla 	/* Index of head element */
31346eb2fdSAmit Prakash Shukla 	uint16_t head;
32346eb2fdSAmit Prakash Shukla 
33346eb2fdSAmit Prakash Shukla 	/* Index of tail element */
34346eb2fdSAmit Prakash Shukla 	uint16_t tail;
35346eb2fdSAmit Prakash Shukla 
36346eb2fdSAmit Prakash Shukla 	/* Number of elements in buffer */
37346eb2fdSAmit Prakash Shukla 	uint16_t count;
38346eb2fdSAmit Prakash Shukla 
39346eb2fdSAmit Prakash Shukla 	/* Size of circular buffer */
40346eb2fdSAmit Prakash Shukla 	uint16_t size;
41346eb2fdSAmit Prakash Shukla 
42346eb2fdSAmit Prakash Shukla 	/* Pointer to hold rte_event_dma_adapter_op for processing */
43346eb2fdSAmit Prakash Shukla 	struct rte_event_dma_adapter_op **op_buffer;
44c6552d9aSTyler Retzlaff };
45346eb2fdSAmit Prakash Shukla 
46112bf805SAmit Prakash Shukla /* Vchan information */
47c6552d9aSTyler Retzlaff struct __rte_cache_aligned dma_vchan_info {
48112bf805SAmit Prakash Shukla 	/* Set to indicate vchan queue is enabled */
49112bf805SAmit Prakash Shukla 	bool vq_enabled;
50112bf805SAmit Prakash Shukla 
51112bf805SAmit Prakash Shukla 	/* Circular buffer for batching DMA ops to dma_dev */
52112bf805SAmit Prakash Shukla 	struct dma_ops_circular_buffer dma_buf;
53c6552d9aSTyler Retzlaff };
54112bf805SAmit Prakash Shukla 
55346eb2fdSAmit Prakash Shukla /* DMA device information */
56c6552d9aSTyler Retzlaff struct __rte_cache_aligned dma_device_info {
57112bf805SAmit Prakash Shukla 	/* Pointer to vchan queue info */
58112bf805SAmit Prakash Shukla 	struct dma_vchan_info *vchanq;
59112bf805SAmit Prakash Shukla 
60112bf805SAmit Prakash Shukla 	/* Pointer to vchan queue info.
61112bf805SAmit Prakash Shukla 	 * This holds ops passed by application till the
62112bf805SAmit Prakash Shukla 	 * dma completion is done.
63112bf805SAmit Prakash Shukla 	 */
64112bf805SAmit Prakash Shukla 	struct dma_vchan_info *tqmap;
65112bf805SAmit Prakash Shukla 
66112bf805SAmit Prakash Shukla 	/* If num_vchanq > 0, the start callback will
67112bf805SAmit Prakash Shukla 	 * be invoked if not already invoked
68112bf805SAmit Prakash Shukla 	 */
69112bf805SAmit Prakash Shukla 	uint16_t num_vchanq;
70112bf805SAmit Prakash Shukla 
71346eb2fdSAmit Prakash Shukla 	/* Number of vchans configured for a DMA device. */
72346eb2fdSAmit Prakash Shukla 	uint16_t num_dma_dev_vchan;
733c89e8c4SAmit Prakash Shukla 
743c89e8c4SAmit Prakash Shukla 	/* Next queue pair to be processed */
753c89e8c4SAmit Prakash Shukla 	uint16_t next_vchan_id;
763c89e8c4SAmit Prakash Shukla 
77abbe0ddaSAmit Prakash Shukla 	/* Set to indicate processing has been started */
78abbe0ddaSAmit Prakash Shukla 	uint8_t dev_started;
79abbe0ddaSAmit Prakash Shukla 
80abbe0ddaSAmit Prakash Shukla 	/* Set to indicate dmadev->eventdev packet
81abbe0ddaSAmit Prakash Shukla 	 * transfer uses a hardware mechanism
82abbe0ddaSAmit Prakash Shukla 	 */
83abbe0ddaSAmit Prakash Shukla 	uint8_t internal_event_port;
84c6552d9aSTyler Retzlaff };
85346eb2fdSAmit Prakash Shukla 
86c6552d9aSTyler Retzlaff struct __rte_cache_aligned event_dma_adapter {
87346eb2fdSAmit Prakash Shukla 	/* Event device identifier */
88346eb2fdSAmit Prakash Shukla 	uint8_t eventdev_id;
89346eb2fdSAmit Prakash Shukla 
90346eb2fdSAmit Prakash Shukla 	/* Event port identifier */
91346eb2fdSAmit Prakash Shukla 	uint8_t event_port_id;
92346eb2fdSAmit Prakash Shukla 
93346eb2fdSAmit Prakash Shukla 	/* Adapter mode */
94346eb2fdSAmit Prakash Shukla 	enum rte_event_dma_adapter_mode mode;
95346eb2fdSAmit Prakash Shukla 
96346eb2fdSAmit Prakash Shukla 	/* Memory allocation name */
97346eb2fdSAmit Prakash Shukla 	char mem_name[DMA_ADAPTER_NAME_LEN];
98346eb2fdSAmit Prakash Shukla 
99346eb2fdSAmit Prakash Shukla 	/* Socket identifier cached from eventdev */
100346eb2fdSAmit Prakash Shukla 	int socket_id;
101346eb2fdSAmit Prakash Shukla 
102346eb2fdSAmit Prakash Shukla 	/* Lock to serialize config updates with service function */
103346eb2fdSAmit Prakash Shukla 	rte_spinlock_t lock;
104346eb2fdSAmit Prakash Shukla 
1053c89e8c4SAmit Prakash Shukla 	/* Next dma device to be processed */
1063c89e8c4SAmit Prakash Shukla 	uint16_t next_dmadev_id;
1073c89e8c4SAmit Prakash Shukla 
108346eb2fdSAmit Prakash Shukla 	/* DMA device structure array */
109346eb2fdSAmit Prakash Shukla 	struct dma_device_info *dma_devs;
110346eb2fdSAmit Prakash Shukla 
111346eb2fdSAmit Prakash Shukla 	/* Circular buffer for processing DMA ops to eventdev */
112346eb2fdSAmit Prakash Shukla 	struct dma_ops_circular_buffer ebuf;
113346eb2fdSAmit Prakash Shukla 
114346eb2fdSAmit Prakash Shukla 	/* Configuration callback for rte_service configuration */
115346eb2fdSAmit Prakash Shukla 	rte_event_dma_adapter_conf_cb conf_cb;
116346eb2fdSAmit Prakash Shukla 
117346eb2fdSAmit Prakash Shukla 	/* Configuration callback argument */
118346eb2fdSAmit Prakash Shukla 	void *conf_arg;
119346eb2fdSAmit Prakash Shukla 
120346eb2fdSAmit Prakash Shukla 	/* Set if  default_cb is being used */
121346eb2fdSAmit Prakash Shukla 	int default_cb_arg;
122112bf805SAmit Prakash Shukla 
123112bf805SAmit Prakash Shukla 	/* No. of vchan queue configured */
124112bf805SAmit Prakash Shukla 	uint16_t nb_vchanq;
1253c89e8c4SAmit Prakash Shukla 
1263c89e8c4SAmit Prakash Shukla 	/* Per adapter EAL service ID */
1273c89e8c4SAmit Prakash Shukla 	uint32_t service_id;
1283c89e8c4SAmit Prakash Shukla 
1293c89e8c4SAmit Prakash Shukla 	/* Service initialization state */
1303c89e8c4SAmit Prakash Shukla 	uint8_t service_initialized;
1313c89e8c4SAmit Prakash Shukla 
1323c89e8c4SAmit Prakash Shukla 	/* Max DMA ops processed in any service function invocation */
1333c89e8c4SAmit Prakash Shukla 	uint32_t max_nb;
1343c89e8c4SAmit Prakash Shukla 
1353c89e8c4SAmit Prakash Shukla 	/* Store event port's implicit release capability */
1363c89e8c4SAmit Prakash Shukla 	uint8_t implicit_release_disabled;
1373c89e8c4SAmit Prakash Shukla 
1383c89e8c4SAmit Prakash Shukla 	/* Flag to indicate backpressure at dma_dev
1393c89e8c4SAmit Prakash Shukla 	 * Stop further dequeuing events from eventdev
1403c89e8c4SAmit Prakash Shukla 	 */
1413c89e8c4SAmit Prakash Shukla 	bool stop_enq_to_dma_dev;
1423c89e8c4SAmit Prakash Shukla 
1433c89e8c4SAmit Prakash Shukla 	/* Loop counter to flush dma ops */
1443c89e8c4SAmit Prakash Shukla 	uint16_t transmit_loop_count;
1453c37e8c6SAmit Prakash Shukla 
1463c37e8c6SAmit Prakash Shukla 	/* Per instance stats structure */
1473c37e8c6SAmit Prakash Shukla 	struct rte_event_dma_adapter_stats dma_stats;
148c6552d9aSTyler Retzlaff };
149346eb2fdSAmit Prakash Shukla 
150346eb2fdSAmit Prakash Shukla static struct event_dma_adapter **event_dma_adapter;
151346eb2fdSAmit Prakash Shukla 
152346eb2fdSAmit Prakash Shukla static inline int
edma_adapter_valid_id(uint8_t id)153346eb2fdSAmit Prakash Shukla edma_adapter_valid_id(uint8_t id)
154346eb2fdSAmit Prakash Shukla {
155346eb2fdSAmit Prakash Shukla 	return id < RTE_EVENT_DMA_ADAPTER_MAX_INSTANCE;
156346eb2fdSAmit Prakash Shukla }
157346eb2fdSAmit Prakash Shukla 
158346eb2fdSAmit Prakash Shukla static inline struct event_dma_adapter *
edma_id_to_adapter(uint8_t id)159346eb2fdSAmit Prakash Shukla edma_id_to_adapter(uint8_t id)
160346eb2fdSAmit Prakash Shukla {
161346eb2fdSAmit Prakash Shukla 	return event_dma_adapter ? event_dma_adapter[id] : NULL;
162346eb2fdSAmit Prakash Shukla }
163346eb2fdSAmit Prakash Shukla 
164346eb2fdSAmit Prakash Shukla static int
edma_array_init(void)165346eb2fdSAmit Prakash Shukla edma_array_init(void)
166346eb2fdSAmit Prakash Shukla {
167346eb2fdSAmit Prakash Shukla 	const struct rte_memzone *mz;
168346eb2fdSAmit Prakash Shukla 	uint32_t sz;
169346eb2fdSAmit Prakash Shukla 
170346eb2fdSAmit Prakash Shukla 	mz = rte_memzone_lookup(DMA_ADAPTER_ARRAY);
171346eb2fdSAmit Prakash Shukla 	if (mz == NULL) {
172346eb2fdSAmit Prakash Shukla 		sz = sizeof(struct event_dma_adapter *) * RTE_EVENT_DMA_ADAPTER_MAX_INSTANCE;
173346eb2fdSAmit Prakash Shukla 		sz = RTE_ALIGN(sz, RTE_CACHE_LINE_SIZE);
174346eb2fdSAmit Prakash Shukla 
175346eb2fdSAmit Prakash Shukla 		mz = rte_memzone_reserve_aligned(DMA_ADAPTER_ARRAY, sz, rte_socket_id(), 0,
176346eb2fdSAmit Prakash Shukla 						 RTE_CACHE_LINE_SIZE);
177346eb2fdSAmit Prakash Shukla 		if (mz == NULL) {
178346eb2fdSAmit Prakash Shukla 			RTE_EDEV_LOG_ERR("Failed to reserve memzone : %s, err = %d",
179346eb2fdSAmit Prakash Shukla 					 DMA_ADAPTER_ARRAY, rte_errno);
180346eb2fdSAmit Prakash Shukla 			return -rte_errno;
181346eb2fdSAmit Prakash Shukla 		}
182346eb2fdSAmit Prakash Shukla 	}
183346eb2fdSAmit Prakash Shukla 
184346eb2fdSAmit Prakash Shukla 	event_dma_adapter = mz->addr;
185346eb2fdSAmit Prakash Shukla 
186346eb2fdSAmit Prakash Shukla 	return 0;
187346eb2fdSAmit Prakash Shukla }
188346eb2fdSAmit Prakash Shukla 
1893c89e8c4SAmit Prakash Shukla static inline bool
edma_circular_buffer_batch_ready(struct dma_ops_circular_buffer * bufp)1903c89e8c4SAmit Prakash Shukla edma_circular_buffer_batch_ready(struct dma_ops_circular_buffer *bufp)
1913c89e8c4SAmit Prakash Shukla {
1923c89e8c4SAmit Prakash Shukla 	return bufp->count >= DMA_BATCH_SIZE;
1933c89e8c4SAmit Prakash Shukla }
1943c89e8c4SAmit Prakash Shukla 
1953c89e8c4SAmit Prakash Shukla static inline bool
edma_circular_buffer_space_for_batch(struct dma_ops_circular_buffer * bufp)1963c89e8c4SAmit Prakash Shukla edma_circular_buffer_space_for_batch(struct dma_ops_circular_buffer *bufp)
1973c89e8c4SAmit Prakash Shukla {
1983c89e8c4SAmit Prakash Shukla 	return (bufp->size - bufp->count) >= DMA_BATCH_SIZE;
1993c89e8c4SAmit Prakash Shukla }
2003c89e8c4SAmit Prakash Shukla 
201346eb2fdSAmit Prakash Shukla static inline int
edma_circular_buffer_init(const char * name,struct dma_ops_circular_buffer * buf,uint16_t sz)202346eb2fdSAmit Prakash Shukla edma_circular_buffer_init(const char *name, struct dma_ops_circular_buffer *buf, uint16_t sz)
203346eb2fdSAmit Prakash Shukla {
204346eb2fdSAmit Prakash Shukla 	buf->op_buffer = rte_zmalloc(name, sizeof(struct rte_event_dma_adapter_op *) * sz, 0);
205346eb2fdSAmit Prakash Shukla 	if (buf->op_buffer == NULL)
206346eb2fdSAmit Prakash Shukla 		return -ENOMEM;
207346eb2fdSAmit Prakash Shukla 
208346eb2fdSAmit Prakash Shukla 	buf->size = sz;
209346eb2fdSAmit Prakash Shukla 
210346eb2fdSAmit Prakash Shukla 	return 0;
211346eb2fdSAmit Prakash Shukla }
212346eb2fdSAmit Prakash Shukla 
213346eb2fdSAmit Prakash Shukla static inline void
edma_circular_buffer_free(struct dma_ops_circular_buffer * buf)214346eb2fdSAmit Prakash Shukla edma_circular_buffer_free(struct dma_ops_circular_buffer *buf)
215346eb2fdSAmit Prakash Shukla {
216346eb2fdSAmit Prakash Shukla 	rte_free(buf->op_buffer);
217346eb2fdSAmit Prakash Shukla }
218346eb2fdSAmit Prakash Shukla 
2193c89e8c4SAmit Prakash Shukla static inline int
edma_circular_buffer_add(struct dma_ops_circular_buffer * bufp,struct rte_event_dma_adapter_op * op)2203c89e8c4SAmit Prakash Shukla edma_circular_buffer_add(struct dma_ops_circular_buffer *bufp, struct rte_event_dma_adapter_op *op)
2213c89e8c4SAmit Prakash Shukla {
2223c89e8c4SAmit Prakash Shukla 	uint16_t *tail = &bufp->tail;
2233c89e8c4SAmit Prakash Shukla 
2243c89e8c4SAmit Prakash Shukla 	bufp->op_buffer[*tail] = op;
2253c89e8c4SAmit Prakash Shukla 
2263c89e8c4SAmit Prakash Shukla 	/* circular buffer, go round */
2273c89e8c4SAmit Prakash Shukla 	*tail = (*tail + 1) % bufp->size;
2283c89e8c4SAmit Prakash Shukla 	bufp->count++;
2293c89e8c4SAmit Prakash Shukla 
2303c89e8c4SAmit Prakash Shukla 	return 0;
2313c89e8c4SAmit Prakash Shukla }
2323c89e8c4SAmit Prakash Shukla 
2333c89e8c4SAmit Prakash Shukla static inline int
edma_circular_buffer_flush_to_dma_dev(struct event_dma_adapter * adapter,struct dma_ops_circular_buffer * bufp,uint8_t dma_dev_id,uint16_t vchan,uint16_t * nb_ops_flushed)2343c89e8c4SAmit Prakash Shukla edma_circular_buffer_flush_to_dma_dev(struct event_dma_adapter *adapter,
2353c89e8c4SAmit Prakash Shukla 				      struct dma_ops_circular_buffer *bufp, uint8_t dma_dev_id,
2363c89e8c4SAmit Prakash Shukla 				      uint16_t vchan, uint16_t *nb_ops_flushed)
2373c89e8c4SAmit Prakash Shukla {
2383c89e8c4SAmit Prakash Shukla 	struct rte_event_dma_adapter_op *op;
2393c89e8c4SAmit Prakash Shukla 	uint16_t *head = &bufp->head;
2403c89e8c4SAmit Prakash Shukla 	uint16_t *tail = &bufp->tail;
241588dcac2SPavan Nikhilesh 	struct dma_vchan_info *tq;
2423c89e8c4SAmit Prakash Shukla 	uint16_t n;
2433c89e8c4SAmit Prakash Shukla 	uint16_t i;
2443c89e8c4SAmit Prakash Shukla 	int ret;
2453c89e8c4SAmit Prakash Shukla 
2463c89e8c4SAmit Prakash Shukla 	if (*tail > *head)
2473c89e8c4SAmit Prakash Shukla 		n = *tail - *head;
2483c89e8c4SAmit Prakash Shukla 	else if (*tail < *head)
2493c89e8c4SAmit Prakash Shukla 		n = bufp->size - *head;
2503c89e8c4SAmit Prakash Shukla 	else {
2513c89e8c4SAmit Prakash Shukla 		*nb_ops_flushed = 0;
2523c89e8c4SAmit Prakash Shukla 		return 0; /* buffer empty */
2533c89e8c4SAmit Prakash Shukla 	}
2543c89e8c4SAmit Prakash Shukla 
2553c89e8c4SAmit Prakash Shukla 	tq = &adapter->dma_devs[dma_dev_id].tqmap[vchan];
2563c89e8c4SAmit Prakash Shukla 
2573c89e8c4SAmit Prakash Shukla 	for (i = 0; i < n; i++)	{
2583c89e8c4SAmit Prakash Shukla 		op = bufp->op_buffer[*head];
2593c89e8c4SAmit Prakash Shukla 		if (op->nb_src == 1 && op->nb_dst == 1)
260588dcac2SPavan Nikhilesh 			ret = rte_dma_copy(dma_dev_id, vchan, op->src_dst_seg[0].addr,
261588dcac2SPavan Nikhilesh 					   op->src_dst_seg[1].addr, op->src_dst_seg[0].length,
262588dcac2SPavan Nikhilesh 					   op->flags);
2633c89e8c4SAmit Prakash Shukla 		else
264588dcac2SPavan Nikhilesh 			ret = rte_dma_copy_sg(dma_dev_id, vchan, &op->src_dst_seg[0],
265588dcac2SPavan Nikhilesh 					      &op->src_dst_seg[op->nb_src], op->nb_src, op->nb_dst,
266588dcac2SPavan Nikhilesh 					      op->flags);
2673c89e8c4SAmit Prakash Shukla 		if (ret < 0)
2683c89e8c4SAmit Prakash Shukla 			break;
2693c89e8c4SAmit Prakash Shukla 
2703c89e8c4SAmit Prakash Shukla 		/* Enqueue in transaction queue. */
2713c89e8c4SAmit Prakash Shukla 		edma_circular_buffer_add(&tq->dma_buf, op);
2723c89e8c4SAmit Prakash Shukla 
2733c89e8c4SAmit Prakash Shukla 		*head = (*head + 1) % bufp->size;
2743c89e8c4SAmit Prakash Shukla 	}
2753c89e8c4SAmit Prakash Shukla 
2763c89e8c4SAmit Prakash Shukla 	*nb_ops_flushed = i;
2773c89e8c4SAmit Prakash Shukla 	bufp->count -= *nb_ops_flushed;
2783c89e8c4SAmit Prakash Shukla 	if (!bufp->count) {
2793c89e8c4SAmit Prakash Shukla 		*head = 0;
2803c89e8c4SAmit Prakash Shukla 		*tail = 0;
2813c89e8c4SAmit Prakash Shukla 	}
2823c89e8c4SAmit Prakash Shukla 
2833c89e8c4SAmit Prakash Shukla 	return *nb_ops_flushed == n ? 0 : -1;
2843c89e8c4SAmit Prakash Shukla }
2853c89e8c4SAmit Prakash Shukla 
286346eb2fdSAmit Prakash Shukla static int
edma_default_config_cb(uint8_t id,uint8_t evdev_id,struct rte_event_dma_adapter_conf * conf,void * arg)287346eb2fdSAmit Prakash Shukla edma_default_config_cb(uint8_t id, uint8_t evdev_id, struct rte_event_dma_adapter_conf *conf,
288346eb2fdSAmit Prakash Shukla 		       void *arg)
289346eb2fdSAmit Prakash Shukla {
290346eb2fdSAmit Prakash Shukla 	struct rte_event_port_conf *port_conf;
291346eb2fdSAmit Prakash Shukla 	struct rte_event_dev_config dev_conf;
292346eb2fdSAmit Prakash Shukla 	struct event_dma_adapter *adapter;
293346eb2fdSAmit Prakash Shukla 	struct rte_eventdev *dev;
294346eb2fdSAmit Prakash Shukla 	uint8_t port_id;
295346eb2fdSAmit Prakash Shukla 	int started;
296346eb2fdSAmit Prakash Shukla 	int ret;
297346eb2fdSAmit Prakash Shukla 
298346eb2fdSAmit Prakash Shukla 	adapter = edma_id_to_adapter(id);
299346eb2fdSAmit Prakash Shukla 	if (adapter == NULL)
300346eb2fdSAmit Prakash Shukla 		return -EINVAL;
301346eb2fdSAmit Prakash Shukla 
302346eb2fdSAmit Prakash Shukla 	dev = &rte_eventdevs[adapter->eventdev_id];
303346eb2fdSAmit Prakash Shukla 	dev_conf = dev->data->dev_conf;
304346eb2fdSAmit Prakash Shukla 
305346eb2fdSAmit Prakash Shukla 	started = dev->data->dev_started;
306346eb2fdSAmit Prakash Shukla 	if (started)
307346eb2fdSAmit Prakash Shukla 		rte_event_dev_stop(evdev_id);
308346eb2fdSAmit Prakash Shukla 
309346eb2fdSAmit Prakash Shukla 	port_id = dev_conf.nb_event_ports;
310346eb2fdSAmit Prakash Shukla 	dev_conf.nb_event_ports += 1;
311346eb2fdSAmit Prakash Shukla 
312346eb2fdSAmit Prakash Shukla 	port_conf = arg;
313346eb2fdSAmit Prakash Shukla 	if (port_conf->event_port_cfg & RTE_EVENT_PORT_CFG_SINGLE_LINK)
314346eb2fdSAmit Prakash Shukla 		dev_conf.nb_single_link_event_port_queues += 1;
315346eb2fdSAmit Prakash Shukla 
316346eb2fdSAmit Prakash Shukla 	ret = rte_event_dev_configure(evdev_id, &dev_conf);
317346eb2fdSAmit Prakash Shukla 	if (ret) {
318ae282b06SDavid Marchand 		RTE_EDEV_LOG_ERR("Failed to configure event dev %u", evdev_id);
319346eb2fdSAmit Prakash Shukla 		if (started) {
320346eb2fdSAmit Prakash Shukla 			if (rte_event_dev_start(evdev_id))
321346eb2fdSAmit Prakash Shukla 				return -EIO;
322346eb2fdSAmit Prakash Shukla 		}
323346eb2fdSAmit Prakash Shukla 		return ret;
324346eb2fdSAmit Prakash Shukla 	}
325346eb2fdSAmit Prakash Shukla 
326346eb2fdSAmit Prakash Shukla 	ret = rte_event_port_setup(evdev_id, port_id, port_conf);
327346eb2fdSAmit Prakash Shukla 	if (ret) {
328ae282b06SDavid Marchand 		RTE_EDEV_LOG_ERR("Failed to setup event port %u", port_id);
329346eb2fdSAmit Prakash Shukla 		return ret;
330346eb2fdSAmit Prakash Shukla 	}
331346eb2fdSAmit Prakash Shukla 
332346eb2fdSAmit Prakash Shukla 	conf->event_port_id = port_id;
333346eb2fdSAmit Prakash Shukla 	conf->max_nb = DMA_DEFAULT_MAX_NB;
334346eb2fdSAmit Prakash Shukla 	if (started)
335346eb2fdSAmit Prakash Shukla 		ret = rte_event_dev_start(evdev_id);
336346eb2fdSAmit Prakash Shukla 
337346eb2fdSAmit Prakash Shukla 	adapter->default_cb_arg = 1;
338346eb2fdSAmit Prakash Shukla 	adapter->event_port_id = conf->event_port_id;
339346eb2fdSAmit Prakash Shukla 
340346eb2fdSAmit Prakash Shukla 	return ret;
341346eb2fdSAmit Prakash Shukla }
342346eb2fdSAmit Prakash Shukla 
343346eb2fdSAmit Prakash Shukla int
rte_event_dma_adapter_create_ext(uint8_t id,uint8_t evdev_id,rte_event_dma_adapter_conf_cb conf_cb,enum rte_event_dma_adapter_mode mode,void * conf_arg)344346eb2fdSAmit Prakash Shukla rte_event_dma_adapter_create_ext(uint8_t id, uint8_t evdev_id,
345346eb2fdSAmit Prakash Shukla 				 rte_event_dma_adapter_conf_cb conf_cb,
346346eb2fdSAmit Prakash Shukla 				 enum rte_event_dma_adapter_mode mode, void *conf_arg)
347346eb2fdSAmit Prakash Shukla {
348346eb2fdSAmit Prakash Shukla 	struct rte_event_dev_info dev_info;
349346eb2fdSAmit Prakash Shukla 	struct event_dma_adapter *adapter;
350346eb2fdSAmit Prakash Shukla 	char name[DMA_ADAPTER_NAME_LEN];
351346eb2fdSAmit Prakash Shukla 	struct rte_dma_info info;
352346eb2fdSAmit Prakash Shukla 	uint16_t num_dma_dev;
353346eb2fdSAmit Prakash Shukla 	int socket_id;
354346eb2fdSAmit Prakash Shukla 	uint8_t i;
355346eb2fdSAmit Prakash Shukla 	int ret;
356346eb2fdSAmit Prakash Shukla 
357346eb2fdSAmit Prakash Shukla 	EVENT_DMA_ADAPTER_ID_VALID_OR_ERR_RET(id, -EINVAL);
358346eb2fdSAmit Prakash Shukla 	RTE_EVENTDEV_VALID_DEVID_OR_ERR_RET(evdev_id, -EINVAL);
359346eb2fdSAmit Prakash Shukla 
360346eb2fdSAmit Prakash Shukla 	if (conf_cb == NULL)
361346eb2fdSAmit Prakash Shukla 		return -EINVAL;
362346eb2fdSAmit Prakash Shukla 
363346eb2fdSAmit Prakash Shukla 	if (event_dma_adapter == NULL) {
364346eb2fdSAmit Prakash Shukla 		ret = edma_array_init();
365346eb2fdSAmit Prakash Shukla 		if (ret)
366346eb2fdSAmit Prakash Shukla 			return ret;
367346eb2fdSAmit Prakash Shukla 	}
368346eb2fdSAmit Prakash Shukla 
369346eb2fdSAmit Prakash Shukla 	adapter = edma_id_to_adapter(id);
370346eb2fdSAmit Prakash Shukla 	if (adapter != NULL) {
371346eb2fdSAmit Prakash Shukla 		RTE_EDEV_LOG_ERR("ML adapter ID %d already exists!", id);
372346eb2fdSAmit Prakash Shukla 		return -EEXIST;
373346eb2fdSAmit Prakash Shukla 	}
374346eb2fdSAmit Prakash Shukla 
375346eb2fdSAmit Prakash Shukla 	socket_id = rte_event_dev_socket_id(evdev_id);
376346eb2fdSAmit Prakash Shukla 	snprintf(name, DMA_ADAPTER_NAME_LEN, "rte_event_dma_adapter_%d", id);
377346eb2fdSAmit Prakash Shukla 	adapter = rte_zmalloc_socket(name, sizeof(struct event_dma_adapter), RTE_CACHE_LINE_SIZE,
378346eb2fdSAmit Prakash Shukla 				     socket_id);
379346eb2fdSAmit Prakash Shukla 	if (adapter == NULL) {
380346eb2fdSAmit Prakash Shukla 		RTE_EDEV_LOG_ERR("Failed to get mem for event ML adapter!");
381346eb2fdSAmit Prakash Shukla 		return -ENOMEM;
382346eb2fdSAmit Prakash Shukla 	}
383346eb2fdSAmit Prakash Shukla 
384346eb2fdSAmit Prakash Shukla 	if (edma_circular_buffer_init("edma_circular_buffer", &adapter->ebuf,
385346eb2fdSAmit Prakash Shukla 				      DMA_ADAPTER_BUFFER_SIZE)) {
386346eb2fdSAmit Prakash Shukla 		RTE_EDEV_LOG_ERR("Failed to get memory for event adapter circular buffer");
387346eb2fdSAmit Prakash Shukla 		rte_free(adapter);
388346eb2fdSAmit Prakash Shukla 		return -ENOMEM;
389346eb2fdSAmit Prakash Shukla 	}
390346eb2fdSAmit Prakash Shukla 
391346eb2fdSAmit Prakash Shukla 	ret = rte_event_dev_info_get(evdev_id, &dev_info);
392346eb2fdSAmit Prakash Shukla 	if (ret < 0) {
393346eb2fdSAmit Prakash Shukla 		RTE_EDEV_LOG_ERR("Failed to get info for eventdev %d: %s", evdev_id,
394346eb2fdSAmit Prakash Shukla 				 dev_info.driver_name);
395346eb2fdSAmit Prakash Shukla 		edma_circular_buffer_free(&adapter->ebuf);
396346eb2fdSAmit Prakash Shukla 		rte_free(adapter);
397346eb2fdSAmit Prakash Shukla 		return ret;
398346eb2fdSAmit Prakash Shukla 	}
399346eb2fdSAmit Prakash Shukla 
400346eb2fdSAmit Prakash Shukla 	num_dma_dev = rte_dma_count_avail();
401346eb2fdSAmit Prakash Shukla 
402346eb2fdSAmit Prakash Shukla 	adapter->eventdev_id = evdev_id;
403346eb2fdSAmit Prakash Shukla 	adapter->mode = mode;
404346eb2fdSAmit Prakash Shukla 	rte_strscpy(adapter->mem_name, name, DMA_ADAPTER_NAME_LEN);
405346eb2fdSAmit Prakash Shukla 	adapter->socket_id = socket_id;
406346eb2fdSAmit Prakash Shukla 	adapter->conf_cb = conf_cb;
407346eb2fdSAmit Prakash Shukla 	adapter->conf_arg = conf_arg;
408346eb2fdSAmit Prakash Shukla 	adapter->dma_devs = rte_zmalloc_socket(adapter->mem_name,
409346eb2fdSAmit Prakash Shukla 					       num_dma_dev * sizeof(struct dma_device_info), 0,
410346eb2fdSAmit Prakash Shukla 					       socket_id);
411346eb2fdSAmit Prakash Shukla 	if (adapter->dma_devs == NULL) {
412ae282b06SDavid Marchand 		RTE_EDEV_LOG_ERR("Failed to get memory for DMA devices");
413346eb2fdSAmit Prakash Shukla 		edma_circular_buffer_free(&adapter->ebuf);
414346eb2fdSAmit Prakash Shukla 		rte_free(adapter);
415346eb2fdSAmit Prakash Shukla 		return -ENOMEM;
416346eb2fdSAmit Prakash Shukla 	}
417346eb2fdSAmit Prakash Shukla 
418346eb2fdSAmit Prakash Shukla 	rte_spinlock_init(&adapter->lock);
419346eb2fdSAmit Prakash Shukla 	for (i = 0; i < num_dma_dev; i++) {
420346eb2fdSAmit Prakash Shukla 		ret = rte_dma_info_get(i, &info);
421346eb2fdSAmit Prakash Shukla 		if (ret) {
422ae282b06SDavid Marchand 			RTE_EDEV_LOG_ERR("Failed to get dma device info");
423346eb2fdSAmit Prakash Shukla 			edma_circular_buffer_free(&adapter->ebuf);
424346eb2fdSAmit Prakash Shukla 			rte_free(adapter);
425346eb2fdSAmit Prakash Shukla 			return ret;
426346eb2fdSAmit Prakash Shukla 		}
427346eb2fdSAmit Prakash Shukla 
428346eb2fdSAmit Prakash Shukla 		adapter->dma_devs[i].num_dma_dev_vchan = info.nb_vchans;
429346eb2fdSAmit Prakash Shukla 	}
430346eb2fdSAmit Prakash Shukla 
431346eb2fdSAmit Prakash Shukla 	event_dma_adapter[id] = adapter;
432346eb2fdSAmit Prakash Shukla 
433346eb2fdSAmit Prakash Shukla 	return 0;
434346eb2fdSAmit Prakash Shukla }
435346eb2fdSAmit Prakash Shukla 
436346eb2fdSAmit Prakash Shukla int
rte_event_dma_adapter_create(uint8_t id,uint8_t evdev_id,struct rte_event_port_conf * port_config,enum rte_event_dma_adapter_mode mode)437346eb2fdSAmit Prakash Shukla rte_event_dma_adapter_create(uint8_t id, uint8_t evdev_id, struct rte_event_port_conf *port_config,
438346eb2fdSAmit Prakash Shukla 			    enum rte_event_dma_adapter_mode mode)
439346eb2fdSAmit Prakash Shukla {
440346eb2fdSAmit Prakash Shukla 	struct rte_event_port_conf *pc;
441346eb2fdSAmit Prakash Shukla 	int ret;
442346eb2fdSAmit Prakash Shukla 
443346eb2fdSAmit Prakash Shukla 	if (port_config == NULL)
444346eb2fdSAmit Prakash Shukla 		return -EINVAL;
445346eb2fdSAmit Prakash Shukla 
446346eb2fdSAmit Prakash Shukla 	EVENT_DMA_ADAPTER_ID_VALID_OR_ERR_RET(id, -EINVAL);
447346eb2fdSAmit Prakash Shukla 
448346eb2fdSAmit Prakash Shukla 	pc = rte_malloc(NULL, sizeof(struct rte_event_port_conf), 0);
449346eb2fdSAmit Prakash Shukla 	if (pc == NULL)
450346eb2fdSAmit Prakash Shukla 		return -ENOMEM;
451346eb2fdSAmit Prakash Shukla 
452346eb2fdSAmit Prakash Shukla 	rte_memcpy(pc, port_config, sizeof(struct rte_event_port_conf));
453346eb2fdSAmit Prakash Shukla 	ret = rte_event_dma_adapter_create_ext(id, evdev_id, edma_default_config_cb, mode, pc);
454346eb2fdSAmit Prakash Shukla 	if (ret != 0)
455346eb2fdSAmit Prakash Shukla 		rte_free(pc);
456346eb2fdSAmit Prakash Shukla 
457346eb2fdSAmit Prakash Shukla 	return ret;
458346eb2fdSAmit Prakash Shukla }
459346eb2fdSAmit Prakash Shukla 
460346eb2fdSAmit Prakash Shukla int
rte_event_dma_adapter_free(uint8_t id)461346eb2fdSAmit Prakash Shukla rte_event_dma_adapter_free(uint8_t id)
462346eb2fdSAmit Prakash Shukla {
463346eb2fdSAmit Prakash Shukla 	struct event_dma_adapter *adapter;
464346eb2fdSAmit Prakash Shukla 
465346eb2fdSAmit Prakash Shukla 	EVENT_DMA_ADAPTER_ID_VALID_OR_ERR_RET(id, -EINVAL);
466346eb2fdSAmit Prakash Shukla 
467346eb2fdSAmit Prakash Shukla 	adapter = edma_id_to_adapter(id);
468346eb2fdSAmit Prakash Shukla 	if (adapter == NULL)
469346eb2fdSAmit Prakash Shukla 		return -EINVAL;
470346eb2fdSAmit Prakash Shukla 
471346eb2fdSAmit Prakash Shukla 	rte_free(adapter->conf_arg);
472346eb2fdSAmit Prakash Shukla 	rte_free(adapter->dma_devs);
473346eb2fdSAmit Prakash Shukla 	edma_circular_buffer_free(&adapter->ebuf);
474346eb2fdSAmit Prakash Shukla 	rte_free(adapter);
475346eb2fdSAmit Prakash Shukla 	event_dma_adapter[id] = NULL;
476346eb2fdSAmit Prakash Shukla 
477346eb2fdSAmit Prakash Shukla 	return 0;
478346eb2fdSAmit Prakash Shukla }
479112bf805SAmit Prakash Shukla 
480e6ced132SAmit Prakash Shukla int
rte_event_dma_adapter_event_port_get(uint8_t id,uint8_t * event_port_id)481e6ced132SAmit Prakash Shukla rte_event_dma_adapter_event_port_get(uint8_t id, uint8_t *event_port_id)
482e6ced132SAmit Prakash Shukla {
483e6ced132SAmit Prakash Shukla 	struct event_dma_adapter *adapter;
484e6ced132SAmit Prakash Shukla 
485e6ced132SAmit Prakash Shukla 	EVENT_DMA_ADAPTER_ID_VALID_OR_ERR_RET(id, -EINVAL);
486e6ced132SAmit Prakash Shukla 
487e6ced132SAmit Prakash Shukla 	adapter = edma_id_to_adapter(id);
488e6ced132SAmit Prakash Shukla 	if (adapter == NULL || event_port_id == NULL)
489e6ced132SAmit Prakash Shukla 		return -EINVAL;
490e6ced132SAmit Prakash Shukla 
491e6ced132SAmit Prakash Shukla 	*event_port_id = adapter->event_port_id;
492e6ced132SAmit Prakash Shukla 
493e6ced132SAmit Prakash Shukla 	return 0;
494e6ced132SAmit Prakash Shukla }
495e6ced132SAmit Prakash Shukla 
4963c89e8c4SAmit Prakash Shukla static inline unsigned int
edma_enq_to_dma_dev(struct event_dma_adapter * adapter,struct rte_event * ev,unsigned int cnt)4973c89e8c4SAmit Prakash Shukla edma_enq_to_dma_dev(struct event_dma_adapter *adapter, struct rte_event *ev, unsigned int cnt)
4983c89e8c4SAmit Prakash Shukla {
4993c37e8c6SAmit Prakash Shukla 	struct rte_event_dma_adapter_stats *stats = &adapter->dma_stats;
5003c89e8c4SAmit Prakash Shukla 	struct dma_vchan_info *vchan_qinfo = NULL;
5013c89e8c4SAmit Prakash Shukla 	struct rte_event_dma_adapter_op *dma_op;
5023c89e8c4SAmit Prakash Shukla 	uint16_t vchan, nb_enqueued = 0;
5033c89e8c4SAmit Prakash Shukla 	int16_t dma_dev_id;
5043c89e8c4SAmit Prakash Shukla 	unsigned int i, n;
5053c89e8c4SAmit Prakash Shukla 	int ret;
5063c89e8c4SAmit Prakash Shukla 
5073c89e8c4SAmit Prakash Shukla 	ret = 0;
5083c89e8c4SAmit Prakash Shukla 	n = 0;
5093c37e8c6SAmit Prakash Shukla 	stats->event_deq_count += cnt;
5103c89e8c4SAmit Prakash Shukla 
5113c89e8c4SAmit Prakash Shukla 	for (i = 0; i < cnt; i++) {
5123c89e8c4SAmit Prakash Shukla 		dma_op = ev[i].event_ptr;
5133c89e8c4SAmit Prakash Shukla 		if (dma_op == NULL)
5143c89e8c4SAmit Prakash Shukla 			continue;
5153c89e8c4SAmit Prakash Shukla 
516588dcac2SPavan Nikhilesh 		dma_op->impl_opaque[0] = ev[i].event;
5173c89e8c4SAmit Prakash Shukla 		dma_dev_id = dma_op->dma_dev_id;
5183c89e8c4SAmit Prakash Shukla 		vchan = dma_op->vchan;
5193c89e8c4SAmit Prakash Shukla 		vchan_qinfo = &adapter->dma_devs[dma_dev_id].vchanq[vchan];
5203c89e8c4SAmit Prakash Shukla 		if (!vchan_qinfo->vq_enabled) {
5213c89e8c4SAmit Prakash Shukla 			if (dma_op != NULL && dma_op->op_mp != NULL)
5223c89e8c4SAmit Prakash Shukla 				rte_mempool_put(dma_op->op_mp, dma_op);
5233c89e8c4SAmit Prakash Shukla 			continue;
5243c89e8c4SAmit Prakash Shukla 		}
5253c89e8c4SAmit Prakash Shukla 		edma_circular_buffer_add(&vchan_qinfo->dma_buf, dma_op);
5263c89e8c4SAmit Prakash Shukla 
5273c89e8c4SAmit Prakash Shukla 		if (edma_circular_buffer_batch_ready(&vchan_qinfo->dma_buf)) {
5283c89e8c4SAmit Prakash Shukla 			ret = edma_circular_buffer_flush_to_dma_dev(adapter, &vchan_qinfo->dma_buf,
5293c89e8c4SAmit Prakash Shukla 								    dma_dev_id, vchan,
5303c89e8c4SAmit Prakash Shukla 								    &nb_enqueued);
5313c37e8c6SAmit Prakash Shukla 			stats->dma_enq_count += nb_enqueued;
5323c89e8c4SAmit Prakash Shukla 			n += nb_enqueued;
5333c89e8c4SAmit Prakash Shukla 
5343c89e8c4SAmit Prakash Shukla 			/**
5353c89e8c4SAmit Prakash Shukla 			 * If some dma ops failed to flush to dma_dev and
5363c89e8c4SAmit Prakash Shukla 			 * space for another batch is not available, stop
5373c89e8c4SAmit Prakash Shukla 			 * dequeue from eventdev momentarily
5383c89e8c4SAmit Prakash Shukla 			 */
5393c89e8c4SAmit Prakash Shukla 			if (unlikely(ret < 0 &&
5403c89e8c4SAmit Prakash Shukla 				     !edma_circular_buffer_space_for_batch(&vchan_qinfo->dma_buf)))
5413c89e8c4SAmit Prakash Shukla 				adapter->stop_enq_to_dma_dev = true;
5423c89e8c4SAmit Prakash Shukla 		}
5433c89e8c4SAmit Prakash Shukla 	}
5443c89e8c4SAmit Prakash Shukla 
5453c89e8c4SAmit Prakash Shukla 	return n;
5463c89e8c4SAmit Prakash Shukla }
5473c89e8c4SAmit Prakash Shukla 
5483c89e8c4SAmit Prakash Shukla static unsigned int
edma_adapter_dev_flush(struct event_dma_adapter * adapter,int16_t dma_dev_id,uint16_t * nb_ops_flushed)5493c89e8c4SAmit Prakash Shukla edma_adapter_dev_flush(struct event_dma_adapter *adapter, int16_t dma_dev_id,
5503c89e8c4SAmit Prakash Shukla 		       uint16_t *nb_ops_flushed)
5513c89e8c4SAmit Prakash Shukla {
5523c89e8c4SAmit Prakash Shukla 	struct dma_vchan_info *vchan_info;
5533c89e8c4SAmit Prakash Shukla 	struct dma_device_info *dev_info;
5543c89e8c4SAmit Prakash Shukla 	uint16_t nb = 0, nb_enqueued = 0;
5553c89e8c4SAmit Prakash Shukla 	uint16_t vchan, nb_vchans;
5563c89e8c4SAmit Prakash Shukla 
5573c89e8c4SAmit Prakash Shukla 	dev_info = &adapter->dma_devs[dma_dev_id];
5583c89e8c4SAmit Prakash Shukla 	nb_vchans = dev_info->num_vchanq;
5593c89e8c4SAmit Prakash Shukla 
5603c89e8c4SAmit Prakash Shukla 	for (vchan = 0; vchan < nb_vchans; vchan++) {
5613c89e8c4SAmit Prakash Shukla 
5623c89e8c4SAmit Prakash Shukla 		vchan_info = &dev_info->vchanq[vchan];
5633c89e8c4SAmit Prakash Shukla 		if (unlikely(vchan_info == NULL || !vchan_info->vq_enabled))
5643c89e8c4SAmit Prakash Shukla 			continue;
5653c89e8c4SAmit Prakash Shukla 
5663c89e8c4SAmit Prakash Shukla 		edma_circular_buffer_flush_to_dma_dev(adapter, &vchan_info->dma_buf, dma_dev_id,
5673c89e8c4SAmit Prakash Shukla 						      vchan, &nb_enqueued);
5683c89e8c4SAmit Prakash Shukla 		*nb_ops_flushed += vchan_info->dma_buf.count;
5693c89e8c4SAmit Prakash Shukla 		nb += nb_enqueued;
5703c89e8c4SAmit Prakash Shukla 	}
5713c89e8c4SAmit Prakash Shukla 
5723c89e8c4SAmit Prakash Shukla 	return nb;
5733c89e8c4SAmit Prakash Shukla }
5743c89e8c4SAmit Prakash Shukla 
5753c89e8c4SAmit Prakash Shukla static unsigned int
edma_adapter_enq_flush(struct event_dma_adapter * adapter)5763c89e8c4SAmit Prakash Shukla edma_adapter_enq_flush(struct event_dma_adapter *adapter)
5773c89e8c4SAmit Prakash Shukla {
5783c37e8c6SAmit Prakash Shukla 	struct rte_event_dma_adapter_stats *stats = &adapter->dma_stats;
5793c89e8c4SAmit Prakash Shukla 	int16_t dma_dev_id;
5803c89e8c4SAmit Prakash Shukla 	uint16_t nb_enqueued = 0;
5813c89e8c4SAmit Prakash Shukla 	uint16_t nb_ops_flushed = 0;
5823c89e8c4SAmit Prakash Shukla 	uint16_t num_dma_dev = rte_dma_count_avail();
5833c89e8c4SAmit Prakash Shukla 
5843c89e8c4SAmit Prakash Shukla 	for (dma_dev_id = 0; dma_dev_id < num_dma_dev; dma_dev_id++)
5853c89e8c4SAmit Prakash Shukla 		nb_enqueued += edma_adapter_dev_flush(adapter, dma_dev_id, &nb_ops_flushed);
5863c89e8c4SAmit Prakash Shukla 	/**
5873c89e8c4SAmit Prakash Shukla 	 * Enable dequeue from eventdev if all ops from circular
5883c89e8c4SAmit Prakash Shukla 	 * buffer flushed to dma_dev
5893c89e8c4SAmit Prakash Shukla 	 */
5903c89e8c4SAmit Prakash Shukla 	if (!nb_ops_flushed)
5913c89e8c4SAmit Prakash Shukla 		adapter->stop_enq_to_dma_dev = false;
5923c89e8c4SAmit Prakash Shukla 
5933c37e8c6SAmit Prakash Shukla 	stats->dma_enq_count += nb_enqueued;
5943c37e8c6SAmit Prakash Shukla 
5953c89e8c4SAmit Prakash Shukla 	return nb_enqueued;
5963c89e8c4SAmit Prakash Shukla }
5973c89e8c4SAmit Prakash Shukla 
5983c89e8c4SAmit Prakash Shukla /* Flush an instance's enqueue buffers every DMA_ENQ_FLUSH_THRESHOLD
5993c89e8c4SAmit Prakash Shukla  * iterations of edma_adapter_enq_run()
6003c89e8c4SAmit Prakash Shukla  */
6013c89e8c4SAmit Prakash Shukla #define DMA_ENQ_FLUSH_THRESHOLD 1024
6023c89e8c4SAmit Prakash Shukla 
6033c89e8c4SAmit Prakash Shukla static int
edma_adapter_enq_run(struct event_dma_adapter * adapter,unsigned int max_enq)6043c89e8c4SAmit Prakash Shukla edma_adapter_enq_run(struct event_dma_adapter *adapter, unsigned int max_enq)
6053c89e8c4SAmit Prakash Shukla {
6063c37e8c6SAmit Prakash Shukla 	struct rte_event_dma_adapter_stats *stats = &adapter->dma_stats;
6073c89e8c4SAmit Prakash Shukla 	uint8_t event_port_id = adapter->event_port_id;
6083c89e8c4SAmit Prakash Shukla 	uint8_t event_dev_id = adapter->eventdev_id;
6093c89e8c4SAmit Prakash Shukla 	struct rte_event ev[DMA_BATCH_SIZE];
6103c89e8c4SAmit Prakash Shukla 	unsigned int nb_enq, nb_enqueued;
6113c89e8c4SAmit Prakash Shukla 	uint16_t n;
6123c89e8c4SAmit Prakash Shukla 
6133c89e8c4SAmit Prakash Shukla 	if (adapter->mode == RTE_EVENT_DMA_ADAPTER_OP_NEW)
6143c89e8c4SAmit Prakash Shukla 		return 0;
6153c89e8c4SAmit Prakash Shukla 
6163c89e8c4SAmit Prakash Shukla 	nb_enqueued = 0;
6173c89e8c4SAmit Prakash Shukla 	for (nb_enq = 0; nb_enq < max_enq; nb_enq += n) {
6183c89e8c4SAmit Prakash Shukla 
6193c89e8c4SAmit Prakash Shukla 		if (unlikely(adapter->stop_enq_to_dma_dev)) {
6203c89e8c4SAmit Prakash Shukla 			nb_enqueued += edma_adapter_enq_flush(adapter);
6213c89e8c4SAmit Prakash Shukla 
6223c89e8c4SAmit Prakash Shukla 			if (unlikely(adapter->stop_enq_to_dma_dev))
6233c89e8c4SAmit Prakash Shukla 				break;
6243c89e8c4SAmit Prakash Shukla 		}
6253c89e8c4SAmit Prakash Shukla 
6263c37e8c6SAmit Prakash Shukla 		stats->event_poll_count++;
6273c89e8c4SAmit Prakash Shukla 		n = rte_event_dequeue_burst(event_dev_id, event_port_id, ev, DMA_BATCH_SIZE, 0);
6283c89e8c4SAmit Prakash Shukla 
6293c89e8c4SAmit Prakash Shukla 		if (!n)
6303c89e8c4SAmit Prakash Shukla 			break;
6313c89e8c4SAmit Prakash Shukla 
6323c89e8c4SAmit Prakash Shukla 		nb_enqueued += edma_enq_to_dma_dev(adapter, ev, n);
6333c89e8c4SAmit Prakash Shukla 	}
6343c89e8c4SAmit Prakash Shukla 
6353c89e8c4SAmit Prakash Shukla 	if ((++adapter->transmit_loop_count & (DMA_ENQ_FLUSH_THRESHOLD - 1)) == 0)
6363c89e8c4SAmit Prakash Shukla 		nb_enqueued += edma_adapter_enq_flush(adapter);
6373c89e8c4SAmit Prakash Shukla 
6383c89e8c4SAmit Prakash Shukla 	return nb_enqueued;
6393c89e8c4SAmit Prakash Shukla }
6403c89e8c4SAmit Prakash Shukla 
6413c89e8c4SAmit Prakash Shukla #define DMA_ADAPTER_MAX_EV_ENQ_RETRIES 100
6423c89e8c4SAmit Prakash Shukla 
6433c89e8c4SAmit Prakash Shukla static inline uint16_t
edma_ops_enqueue_burst(struct event_dma_adapter * adapter,struct rte_event_dma_adapter_op ** ops,uint16_t num)6443c89e8c4SAmit Prakash Shukla edma_ops_enqueue_burst(struct event_dma_adapter *adapter, struct rte_event_dma_adapter_op **ops,
6453c89e8c4SAmit Prakash Shukla 		       uint16_t num)
6463c89e8c4SAmit Prakash Shukla {
6473c37e8c6SAmit Prakash Shukla 	struct rte_event_dma_adapter_stats *stats = &adapter->dma_stats;
6483c89e8c4SAmit Prakash Shukla 	uint8_t event_port_id = adapter->event_port_id;
6493c89e8c4SAmit Prakash Shukla 	uint8_t event_dev_id = adapter->eventdev_id;
6503c89e8c4SAmit Prakash Shukla 	struct rte_event events[DMA_BATCH_SIZE];
6513c89e8c4SAmit Prakash Shukla 	uint16_t nb_enqueued, nb_ev;
6523c89e8c4SAmit Prakash Shukla 	uint8_t retry;
6533c89e8c4SAmit Prakash Shukla 	uint8_t i;
6543c89e8c4SAmit Prakash Shukla 
6553c89e8c4SAmit Prakash Shukla 	nb_ev = 0;
6563c89e8c4SAmit Prakash Shukla 	retry = 0;
6573c89e8c4SAmit Prakash Shukla 	nb_enqueued = 0;
6583c89e8c4SAmit Prakash Shukla 	num = RTE_MIN(num, DMA_BATCH_SIZE);
6593c89e8c4SAmit Prakash Shukla 	for (i = 0; i < num; i++) {
6603c89e8c4SAmit Prakash Shukla 		struct rte_event *ev = &events[nb_ev++];
6613c89e8c4SAmit Prakash Shukla 
662588dcac2SPavan Nikhilesh 		ev->event = ops[i]->impl_opaque[0];
6633c89e8c4SAmit Prakash Shukla 		ev->event_ptr = ops[i];
6643c89e8c4SAmit Prakash Shukla 		ev->event_type = RTE_EVENT_TYPE_DMADEV;
6653c89e8c4SAmit Prakash Shukla 		if (adapter->implicit_release_disabled)
6663c89e8c4SAmit Prakash Shukla 			ev->op = RTE_EVENT_OP_FORWARD;
6673c89e8c4SAmit Prakash Shukla 		else
6683c89e8c4SAmit Prakash Shukla 			ev->op = RTE_EVENT_OP_NEW;
669*cfa81500SPavan Nikhilesh 		ev->event = ops[i]->event_meta;
6703c89e8c4SAmit Prakash Shukla 	}
6713c89e8c4SAmit Prakash Shukla 
6723c89e8c4SAmit Prakash Shukla 	do {
6733c89e8c4SAmit Prakash Shukla 		nb_enqueued += rte_event_enqueue_burst(event_dev_id, event_port_id,
6743c89e8c4SAmit Prakash Shukla 						       &events[nb_enqueued], nb_ev - nb_enqueued);
6753c89e8c4SAmit Prakash Shukla 
6763c89e8c4SAmit Prakash Shukla 	} while (retry++ < DMA_ADAPTER_MAX_EV_ENQ_RETRIES && nb_enqueued < nb_ev);
6773c89e8c4SAmit Prakash Shukla 
6783c37e8c6SAmit Prakash Shukla 	stats->event_enq_fail_count += nb_ev - nb_enqueued;
6793c37e8c6SAmit Prakash Shukla 	stats->event_enq_count += nb_enqueued;
6803c37e8c6SAmit Prakash Shukla 	stats->event_enq_retry_count += retry - 1;
6813c37e8c6SAmit Prakash Shukla 
6823c89e8c4SAmit Prakash Shukla 	return nb_enqueued;
6833c89e8c4SAmit Prakash Shukla }
6843c89e8c4SAmit Prakash Shukla 
6853c89e8c4SAmit Prakash Shukla static int
edma_circular_buffer_flush_to_evdev(struct event_dma_adapter * adapter,struct dma_ops_circular_buffer * bufp,uint16_t * enqueue_count)6863c89e8c4SAmit Prakash Shukla edma_circular_buffer_flush_to_evdev(struct event_dma_adapter *adapter,
6873c89e8c4SAmit Prakash Shukla 				    struct dma_ops_circular_buffer *bufp,
6883c89e8c4SAmit Prakash Shukla 				    uint16_t *enqueue_count)
6893c89e8c4SAmit Prakash Shukla {
6903c89e8c4SAmit Prakash Shukla 	struct rte_event_dma_adapter_op **ops = bufp->op_buffer;
6913c89e8c4SAmit Prakash Shukla 	uint16_t n = 0, nb_ops_flushed;
6923c89e8c4SAmit Prakash Shukla 	uint16_t *head = &bufp->head;
6933c89e8c4SAmit Prakash Shukla 	uint16_t *tail = &bufp->tail;
6943c89e8c4SAmit Prakash Shukla 
6953c89e8c4SAmit Prakash Shukla 	if (*tail > *head)
6963c89e8c4SAmit Prakash Shukla 		n = *tail - *head;
6973c89e8c4SAmit Prakash Shukla 	else if (*tail < *head)
6983c89e8c4SAmit Prakash Shukla 		n = bufp->size - *head;
6993c89e8c4SAmit Prakash Shukla 	else {
7003c89e8c4SAmit Prakash Shukla 		if (enqueue_count)
7013c89e8c4SAmit Prakash Shukla 			*enqueue_count = 0;
7023c89e8c4SAmit Prakash Shukla 		return 0; /* buffer empty */
7033c89e8c4SAmit Prakash Shukla 	}
7043c89e8c4SAmit Prakash Shukla 
7053c89e8c4SAmit Prakash Shukla 	if (enqueue_count && n > *enqueue_count)
7063c89e8c4SAmit Prakash Shukla 		n = *enqueue_count;
7073c89e8c4SAmit Prakash Shukla 
7083c89e8c4SAmit Prakash Shukla 	nb_ops_flushed = edma_ops_enqueue_burst(adapter, &ops[*head], n);
7093c89e8c4SAmit Prakash Shukla 	if (enqueue_count)
7103c89e8c4SAmit Prakash Shukla 		*enqueue_count = nb_ops_flushed;
7113c89e8c4SAmit Prakash Shukla 
7123c89e8c4SAmit Prakash Shukla 	bufp->count -= nb_ops_flushed;
7133c89e8c4SAmit Prakash Shukla 	if (!bufp->count) {
7143c89e8c4SAmit Prakash Shukla 		*head = 0;
7153c89e8c4SAmit Prakash Shukla 		*tail = 0;
7163c89e8c4SAmit Prakash Shukla 		return 0; /* buffer empty */
7173c89e8c4SAmit Prakash Shukla 	}
7183c89e8c4SAmit Prakash Shukla 
7193c89e8c4SAmit Prakash Shukla 	*head = (*head + nb_ops_flushed) % bufp->size;
7203c89e8c4SAmit Prakash Shukla 	return 1;
7213c89e8c4SAmit Prakash Shukla }
7223c89e8c4SAmit Prakash Shukla 
7233c89e8c4SAmit Prakash Shukla static void
edma_ops_buffer_flush(struct event_dma_adapter * adapter)7243c89e8c4SAmit Prakash Shukla edma_ops_buffer_flush(struct event_dma_adapter *adapter)
7253c89e8c4SAmit Prakash Shukla {
7263c89e8c4SAmit Prakash Shukla 	if (likely(adapter->ebuf.count == 0))
7273c89e8c4SAmit Prakash Shukla 		return;
7283c89e8c4SAmit Prakash Shukla 
7293c89e8c4SAmit Prakash Shukla 	while (edma_circular_buffer_flush_to_evdev(adapter, &adapter->ebuf, NULL))
7303c89e8c4SAmit Prakash Shukla 		;
7313c89e8c4SAmit Prakash Shukla }
7323c89e8c4SAmit Prakash Shukla 
7333c89e8c4SAmit Prakash Shukla static inline unsigned int
edma_adapter_deq_run(struct event_dma_adapter * adapter,unsigned int max_deq)7343c89e8c4SAmit Prakash Shukla edma_adapter_deq_run(struct event_dma_adapter *adapter, unsigned int max_deq)
7353c89e8c4SAmit Prakash Shukla {
7363c37e8c6SAmit Prakash Shukla 	struct rte_event_dma_adapter_stats *stats = &adapter->dma_stats;
7373c89e8c4SAmit Prakash Shukla 	struct dma_vchan_info *vchan_info;
7383c89e8c4SAmit Prakash Shukla 	struct dma_ops_circular_buffer *tq_buf;
7393c89e8c4SAmit Prakash Shukla 	struct rte_event_dma_adapter_op *ops;
7403c89e8c4SAmit Prakash Shukla 	uint16_t n, nb_deq, nb_enqueued, i;
7413c89e8c4SAmit Prakash Shukla 	struct dma_device_info *dev_info;
7423c89e8c4SAmit Prakash Shukla 	uint16_t vchan, num_vchan;
7433c89e8c4SAmit Prakash Shukla 	uint16_t num_dma_dev;
7443c89e8c4SAmit Prakash Shukla 	int16_t dma_dev_id;
7453c89e8c4SAmit Prakash Shukla 	uint16_t index;
7463c89e8c4SAmit Prakash Shukla 	bool done;
7473c89e8c4SAmit Prakash Shukla 	bool err;
7483c89e8c4SAmit Prakash Shukla 
7493c89e8c4SAmit Prakash Shukla 	nb_deq = 0;
7503c89e8c4SAmit Prakash Shukla 	edma_ops_buffer_flush(adapter);
7513c89e8c4SAmit Prakash Shukla 
7523c89e8c4SAmit Prakash Shukla 	num_dma_dev = rte_dma_count_avail();
7533c89e8c4SAmit Prakash Shukla 	do {
7543c89e8c4SAmit Prakash Shukla 		done = true;
7553c89e8c4SAmit Prakash Shukla 
7563c89e8c4SAmit Prakash Shukla 		for (dma_dev_id = adapter->next_dmadev_id; dma_dev_id < num_dma_dev; dma_dev_id++) {
7573c89e8c4SAmit Prakash Shukla 			uint16_t queues = 0;
7583c89e8c4SAmit Prakash Shukla 			dev_info = &adapter->dma_devs[dma_dev_id];
7593c89e8c4SAmit Prakash Shukla 			num_vchan = dev_info->num_vchanq;
7603c89e8c4SAmit Prakash Shukla 
7613c89e8c4SAmit Prakash Shukla 			for (vchan = dev_info->next_vchan_id; queues < num_vchan;
7623c89e8c4SAmit Prakash Shukla 			     vchan = (vchan + 1) % num_vchan, queues++) {
7633c89e8c4SAmit Prakash Shukla 
7643c89e8c4SAmit Prakash Shukla 				vchan_info = &dev_info->vchanq[vchan];
7653c89e8c4SAmit Prakash Shukla 				if (unlikely(vchan_info == NULL || !vchan_info->vq_enabled))
7663c89e8c4SAmit Prakash Shukla 					continue;
7673c89e8c4SAmit Prakash Shukla 
7683c89e8c4SAmit Prakash Shukla 				n = rte_dma_completed(dma_dev_id, vchan, DMA_BATCH_SIZE,
7693c89e8c4SAmit Prakash Shukla 						&index, &err);
7703c89e8c4SAmit Prakash Shukla 				if (!n)
7713c89e8c4SAmit Prakash Shukla 					continue;
7723c89e8c4SAmit Prakash Shukla 
7733c89e8c4SAmit Prakash Shukla 				done = false;
7743c37e8c6SAmit Prakash Shukla 				stats->dma_deq_count += n;
7753c89e8c4SAmit Prakash Shukla 
7763c89e8c4SAmit Prakash Shukla 				tq_buf = &dev_info->tqmap[vchan].dma_buf;
7773c89e8c4SAmit Prakash Shukla 
7783c89e8c4SAmit Prakash Shukla 				nb_enqueued = n;
7793c89e8c4SAmit Prakash Shukla 				if (unlikely(!adapter->ebuf.count))
7803c89e8c4SAmit Prakash Shukla 					edma_circular_buffer_flush_to_evdev(adapter, tq_buf,
7813c89e8c4SAmit Prakash Shukla 									    &nb_enqueued);
7823c89e8c4SAmit Prakash Shukla 
7833c89e8c4SAmit Prakash Shukla 				if (likely(nb_enqueued == n))
7843c89e8c4SAmit Prakash Shukla 					goto check;
7853c89e8c4SAmit Prakash Shukla 
7863c89e8c4SAmit Prakash Shukla 				/* Failed to enqueue events case */
7873c89e8c4SAmit Prakash Shukla 				for (i = nb_enqueued; i < n; i++) {
7883c89e8c4SAmit Prakash Shukla 					ops = tq_buf->op_buffer[tq_buf->head];
7893c89e8c4SAmit Prakash Shukla 					edma_circular_buffer_add(&adapter->ebuf, ops);
7903c89e8c4SAmit Prakash Shukla 					tq_buf->head = (tq_buf->head + 1) % tq_buf->size;
7913c89e8c4SAmit Prakash Shukla 				}
7923c89e8c4SAmit Prakash Shukla 
7933c89e8c4SAmit Prakash Shukla check:
7943c89e8c4SAmit Prakash Shukla 				nb_deq += n;
7953c89e8c4SAmit Prakash Shukla 				if (nb_deq >= max_deq) {
7963c89e8c4SAmit Prakash Shukla 					if ((vchan + 1) == num_vchan)
7973c89e8c4SAmit Prakash Shukla 						adapter->next_dmadev_id =
7983c89e8c4SAmit Prakash Shukla 								(dma_dev_id + 1) % num_dma_dev;
7993c89e8c4SAmit Prakash Shukla 
8003c89e8c4SAmit Prakash Shukla 					dev_info->next_vchan_id = (vchan + 1) % num_vchan;
8013c89e8c4SAmit Prakash Shukla 
8023c89e8c4SAmit Prakash Shukla 					return nb_deq;
8033c89e8c4SAmit Prakash Shukla 				}
8043c89e8c4SAmit Prakash Shukla 			}
8053c89e8c4SAmit Prakash Shukla 		}
8063c89e8c4SAmit Prakash Shukla 		adapter->next_dmadev_id = 0;
8073c89e8c4SAmit Prakash Shukla 
8083c89e8c4SAmit Prakash Shukla 	} while (done == false);
8093c89e8c4SAmit Prakash Shukla 
8103c89e8c4SAmit Prakash Shukla 	return nb_deq;
8113c89e8c4SAmit Prakash Shukla }
8123c89e8c4SAmit Prakash Shukla 
8133c89e8c4SAmit Prakash Shukla static int
edma_adapter_run(struct event_dma_adapter * adapter,unsigned int max_ops)8143c89e8c4SAmit Prakash Shukla edma_adapter_run(struct event_dma_adapter *adapter, unsigned int max_ops)
8153c89e8c4SAmit Prakash Shukla {
8163c89e8c4SAmit Prakash Shukla 	unsigned int ops_left = max_ops;
8173c89e8c4SAmit Prakash Shukla 
8183c89e8c4SAmit Prakash Shukla 	while (ops_left > 0) {
8193c89e8c4SAmit Prakash Shukla 		unsigned int e_cnt, d_cnt;
8203c89e8c4SAmit Prakash Shukla 
8213c89e8c4SAmit Prakash Shukla 		e_cnt = edma_adapter_deq_run(adapter, ops_left);
8223c89e8c4SAmit Prakash Shukla 		ops_left -= RTE_MIN(ops_left, e_cnt);
8233c89e8c4SAmit Prakash Shukla 
8243c89e8c4SAmit Prakash Shukla 		d_cnt = edma_adapter_enq_run(adapter, ops_left);
8253c89e8c4SAmit Prakash Shukla 		ops_left -= RTE_MIN(ops_left, d_cnt);
8263c89e8c4SAmit Prakash Shukla 
8273c89e8c4SAmit Prakash Shukla 		if (e_cnt == 0 && d_cnt == 0)
8283c89e8c4SAmit Prakash Shukla 			break;
8293c89e8c4SAmit Prakash Shukla 	}
8303c89e8c4SAmit Prakash Shukla 
8313c89e8c4SAmit Prakash Shukla 	if (ops_left == max_ops) {
8323c89e8c4SAmit Prakash Shukla 		rte_event_maintain(adapter->eventdev_id, adapter->event_port_id, 0);
8333c89e8c4SAmit Prakash Shukla 		return -EAGAIN;
8343c89e8c4SAmit Prakash Shukla 	} else
8353c89e8c4SAmit Prakash Shukla 		return 0;
8363c89e8c4SAmit Prakash Shukla }
8373c89e8c4SAmit Prakash Shukla 
8383c89e8c4SAmit Prakash Shukla static int
edma_service_func(void * args)8393c89e8c4SAmit Prakash Shukla edma_service_func(void *args)
8403c89e8c4SAmit Prakash Shukla {
8413c89e8c4SAmit Prakash Shukla 	struct event_dma_adapter *adapter = args;
8423c89e8c4SAmit Prakash Shukla 	int ret;
8433c89e8c4SAmit Prakash Shukla 
8443c89e8c4SAmit Prakash Shukla 	if (rte_spinlock_trylock(&adapter->lock) == 0)
8453c89e8c4SAmit Prakash Shukla 		return 0;
8463c89e8c4SAmit Prakash Shukla 	ret = edma_adapter_run(adapter, adapter->max_nb);
8473c89e8c4SAmit Prakash Shukla 	rte_spinlock_unlock(&adapter->lock);
8483c89e8c4SAmit Prakash Shukla 
8493c89e8c4SAmit Prakash Shukla 	return ret;
8503c89e8c4SAmit Prakash Shukla }
8513c89e8c4SAmit Prakash Shukla 
8523c89e8c4SAmit Prakash Shukla static int
edma_init_service(struct event_dma_adapter * adapter,uint8_t id)8533c89e8c4SAmit Prakash Shukla edma_init_service(struct event_dma_adapter *adapter, uint8_t id)
8543c89e8c4SAmit Prakash Shukla {
8553c89e8c4SAmit Prakash Shukla 	struct rte_event_dma_adapter_conf adapter_conf;
8563c89e8c4SAmit Prakash Shukla 	struct rte_service_spec service;
8573c89e8c4SAmit Prakash Shukla 	uint32_t impl_rel;
8583c89e8c4SAmit Prakash Shukla 	int ret;
8593c89e8c4SAmit Prakash Shukla 
8603c89e8c4SAmit Prakash Shukla 	if (adapter->service_initialized)
8613c89e8c4SAmit Prakash Shukla 		return 0;
8623c89e8c4SAmit Prakash Shukla 
8633c89e8c4SAmit Prakash Shukla 	memset(&service, 0, sizeof(service));
8643c89e8c4SAmit Prakash Shukla 	snprintf(service.name, DMA_ADAPTER_NAME_LEN, "rte_event_dma_adapter_%d", id);
8653c89e8c4SAmit Prakash Shukla 	service.socket_id = adapter->socket_id;
8663c89e8c4SAmit Prakash Shukla 	service.callback = edma_service_func;
8673c89e8c4SAmit Prakash Shukla 	service.callback_userdata = adapter;
8683c89e8c4SAmit Prakash Shukla 
8693c89e8c4SAmit Prakash Shukla 	/* Service function handles locking for queue add/del updates */
8703c89e8c4SAmit Prakash Shukla 	service.capabilities = RTE_SERVICE_CAP_MT_SAFE;
8713c89e8c4SAmit Prakash Shukla 	ret = rte_service_component_register(&service, &adapter->service_id);
8723c89e8c4SAmit Prakash Shukla 	if (ret) {
8733c89e8c4SAmit Prakash Shukla 		RTE_EDEV_LOG_ERR("failed to register service %s err = %" PRId32, service.name, ret);
8743c89e8c4SAmit Prakash Shukla 		return ret;
8753c89e8c4SAmit Prakash Shukla 	}
8763c89e8c4SAmit Prakash Shukla 
8773c89e8c4SAmit Prakash Shukla 	ret = adapter->conf_cb(id, adapter->eventdev_id, &adapter_conf, adapter->conf_arg);
8783c89e8c4SAmit Prakash Shukla 	if (ret) {
8793c89e8c4SAmit Prakash Shukla 		RTE_EDEV_LOG_ERR("configuration callback failed err = %" PRId32, ret);
8803c89e8c4SAmit Prakash Shukla 		return ret;
8813c89e8c4SAmit Prakash Shukla 	}
8823c89e8c4SAmit Prakash Shukla 
8833c89e8c4SAmit Prakash Shukla 	adapter->max_nb = adapter_conf.max_nb;
8843c89e8c4SAmit Prakash Shukla 	adapter->event_port_id = adapter_conf.event_port_id;
8853c89e8c4SAmit Prakash Shukla 
8863c89e8c4SAmit Prakash Shukla 	if (rte_event_port_attr_get(adapter->eventdev_id, adapter->event_port_id,
8873c89e8c4SAmit Prakash Shukla 				    RTE_EVENT_PORT_ATTR_IMPLICIT_RELEASE_DISABLE, &impl_rel)) {
8883c89e8c4SAmit Prakash Shukla 		RTE_EDEV_LOG_ERR("Failed to get port info for eventdev %" PRId32,
8893c89e8c4SAmit Prakash Shukla 				 adapter->eventdev_id);
8903c89e8c4SAmit Prakash Shukla 		edma_circular_buffer_free(&adapter->ebuf);
8913c89e8c4SAmit Prakash Shukla 		rte_free(adapter);
8923c89e8c4SAmit Prakash Shukla 		return -EINVAL;
8933c89e8c4SAmit Prakash Shukla 	}
8943c89e8c4SAmit Prakash Shukla 
8953c89e8c4SAmit Prakash Shukla 	adapter->implicit_release_disabled = (uint8_t)impl_rel;
8963c89e8c4SAmit Prakash Shukla 	adapter->service_initialized = 1;
8973c89e8c4SAmit Prakash Shukla 
8983c89e8c4SAmit Prakash Shukla 	return ret;
8993c89e8c4SAmit Prakash Shukla }
9003c89e8c4SAmit Prakash Shukla 
901112bf805SAmit Prakash Shukla static void
edma_update_vchanq_info(struct event_dma_adapter * adapter,struct dma_device_info * dev_info,uint16_t vchan,uint8_t add)902112bf805SAmit Prakash Shukla edma_update_vchanq_info(struct event_dma_adapter *adapter, struct dma_device_info *dev_info,
903112bf805SAmit Prakash Shukla 			uint16_t vchan, uint8_t add)
904112bf805SAmit Prakash Shukla {
905112bf805SAmit Prakash Shukla 	struct dma_vchan_info *vchan_info;
906112bf805SAmit Prakash Shukla 	struct dma_vchan_info *tqmap_info;
907112bf805SAmit Prakash Shukla 	int enabled;
908112bf805SAmit Prakash Shukla 	uint16_t i;
909112bf805SAmit Prakash Shukla 
910112bf805SAmit Prakash Shukla 	if (dev_info->vchanq == NULL)
911112bf805SAmit Prakash Shukla 		return;
912112bf805SAmit Prakash Shukla 
913112bf805SAmit Prakash Shukla 	if (vchan == RTE_DMA_ALL_VCHAN) {
914112bf805SAmit Prakash Shukla 		for (i = 0; i < dev_info->num_dma_dev_vchan; i++)
915112bf805SAmit Prakash Shukla 			edma_update_vchanq_info(adapter, dev_info, i, add);
916112bf805SAmit Prakash Shukla 	} else {
917112bf805SAmit Prakash Shukla 		tqmap_info = &dev_info->tqmap[vchan];
918112bf805SAmit Prakash Shukla 		vchan_info = &dev_info->vchanq[vchan];
919112bf805SAmit Prakash Shukla 		enabled = vchan_info->vq_enabled;
920112bf805SAmit Prakash Shukla 		if (add) {
921112bf805SAmit Prakash Shukla 			adapter->nb_vchanq += !enabled;
922112bf805SAmit Prakash Shukla 			dev_info->num_vchanq += !enabled;
923112bf805SAmit Prakash Shukla 		} else {
924112bf805SAmit Prakash Shukla 			adapter->nb_vchanq -= enabled;
925112bf805SAmit Prakash Shukla 			dev_info->num_vchanq -= enabled;
926112bf805SAmit Prakash Shukla 		}
927112bf805SAmit Prakash Shukla 		vchan_info->vq_enabled = !!add;
928112bf805SAmit Prakash Shukla 		tqmap_info->vq_enabled = !!add;
929112bf805SAmit Prakash Shukla 	}
930112bf805SAmit Prakash Shukla }
931112bf805SAmit Prakash Shukla 
9323c89e8c4SAmit Prakash Shukla static int
edma_add_vchan(struct event_dma_adapter * adapter,int16_t dma_dev_id,uint16_t vchan)9333c89e8c4SAmit Prakash Shukla edma_add_vchan(struct event_dma_adapter *adapter, int16_t dma_dev_id, uint16_t vchan)
9343c89e8c4SAmit Prakash Shukla {
9353c89e8c4SAmit Prakash Shukla 	struct dma_device_info *dev_info = &adapter->dma_devs[dma_dev_id];
9363c89e8c4SAmit Prakash Shukla 	struct dma_vchan_info *vchanq;
9373c89e8c4SAmit Prakash Shukla 	struct dma_vchan_info *tqmap;
9383c89e8c4SAmit Prakash Shukla 	uint16_t nb_vchans;
9393c89e8c4SAmit Prakash Shukla 	uint32_t i;
9403c89e8c4SAmit Prakash Shukla 
9413c89e8c4SAmit Prakash Shukla 	if (dev_info->vchanq == NULL) {
9423c89e8c4SAmit Prakash Shukla 		nb_vchans = dev_info->num_dma_dev_vchan;
9433c89e8c4SAmit Prakash Shukla 
9443c89e8c4SAmit Prakash Shukla 		dev_info->vchanq = rte_zmalloc_socket(adapter->mem_name,
9453c89e8c4SAmit Prakash Shukla 				nb_vchans * sizeof(struct dma_vchan_info),
9463c89e8c4SAmit Prakash Shukla 				0, adapter->socket_id);
9473c89e8c4SAmit Prakash Shukla 		if (dev_info->vchanq == NULL)
9483c89e8c4SAmit Prakash Shukla 			return -ENOMEM;
9493c89e8c4SAmit Prakash Shukla 
9503c89e8c4SAmit Prakash Shukla 		dev_info->tqmap = rte_zmalloc_socket(adapter->mem_name,
9513c89e8c4SAmit Prakash Shukla 				nb_vchans * sizeof(struct dma_vchan_info),
9523c89e8c4SAmit Prakash Shukla 				0, adapter->socket_id);
9533c89e8c4SAmit Prakash Shukla 		if (dev_info->tqmap == NULL)
9543c89e8c4SAmit Prakash Shukla 			return -ENOMEM;
9553c89e8c4SAmit Prakash Shukla 
9563c89e8c4SAmit Prakash Shukla 		for (i = 0; i < nb_vchans; i++) {
9573c89e8c4SAmit Prakash Shukla 			vchanq = &dev_info->vchanq[i];
9583c89e8c4SAmit Prakash Shukla 
9593c89e8c4SAmit Prakash Shukla 			if (edma_circular_buffer_init("dma_dev_circular_buffer", &vchanq->dma_buf,
9603c89e8c4SAmit Prakash Shukla 						DMA_ADAPTER_OPS_BUFFER_SIZE)) {
9613c89e8c4SAmit Prakash Shukla 				RTE_EDEV_LOG_ERR("Failed to get memory for dma_dev buffer");
9623c89e8c4SAmit Prakash Shukla 				rte_free(vchanq);
9633c89e8c4SAmit Prakash Shukla 				return -ENOMEM;
9643c89e8c4SAmit Prakash Shukla 			}
9653c89e8c4SAmit Prakash Shukla 
9663c89e8c4SAmit Prakash Shukla 			tqmap = &dev_info->tqmap[i];
9673c89e8c4SAmit Prakash Shukla 			if (edma_circular_buffer_init("dma_dev_circular_trans_buf", &tqmap->dma_buf,
9683c89e8c4SAmit Prakash Shukla 						DMA_ADAPTER_OPS_BUFFER_SIZE)) {
9693c89e8c4SAmit Prakash Shukla 				RTE_EDEV_LOG_ERR(
9703c89e8c4SAmit Prakash Shukla 					"Failed to get memory for dma_dev transaction buffer");
9713c89e8c4SAmit Prakash Shukla 				rte_free(tqmap);
9723c89e8c4SAmit Prakash Shukla 				return -ENOMEM;
9733c89e8c4SAmit Prakash Shukla 			}
9743c89e8c4SAmit Prakash Shukla 		}
9753c89e8c4SAmit Prakash Shukla 	}
9763c89e8c4SAmit Prakash Shukla 
9773c89e8c4SAmit Prakash Shukla 	if (vchan == RTE_DMA_ALL_VCHAN) {
9783c89e8c4SAmit Prakash Shukla 		for (i = 0; i < dev_info->num_dma_dev_vchan; i++)
9793c89e8c4SAmit Prakash Shukla 			edma_update_vchanq_info(adapter, dev_info, i, 1);
9803c89e8c4SAmit Prakash Shukla 	} else
9813c89e8c4SAmit Prakash Shukla 		edma_update_vchanq_info(adapter, dev_info, vchan, 1);
9823c89e8c4SAmit Prakash Shukla 
9833c89e8c4SAmit Prakash Shukla 	return 0;
9843c89e8c4SAmit Prakash Shukla }
9853c89e8c4SAmit Prakash Shukla 
986112bf805SAmit Prakash Shukla int
rte_event_dma_adapter_vchan_add(uint8_t id,int16_t dma_dev_id,uint16_t vchan,const struct rte_event * event)987112bf805SAmit Prakash Shukla rte_event_dma_adapter_vchan_add(uint8_t id, int16_t dma_dev_id, uint16_t vchan,
988112bf805SAmit Prakash Shukla 				const struct rte_event *event)
989112bf805SAmit Prakash Shukla {
990112bf805SAmit Prakash Shukla 	struct event_dma_adapter *adapter;
991112bf805SAmit Prakash Shukla 	struct dma_device_info *dev_info;
992112bf805SAmit Prakash Shukla 	struct rte_eventdev *dev;
993112bf805SAmit Prakash Shukla 	uint32_t cap;
994112bf805SAmit Prakash Shukla 	int ret;
995112bf805SAmit Prakash Shukla 
996112bf805SAmit Prakash Shukla 	EVENT_DMA_ADAPTER_ID_VALID_OR_ERR_RET(id, -EINVAL);
997112bf805SAmit Prakash Shukla 
998112bf805SAmit Prakash Shukla 	if (!rte_dma_is_valid(dma_dev_id)) {
999112bf805SAmit Prakash Shukla 		RTE_EDEV_LOG_ERR("Invalid dma_dev_id = %" PRIu8, dma_dev_id);
1000112bf805SAmit Prakash Shukla 		return -EINVAL;
1001112bf805SAmit Prakash Shukla 	}
1002112bf805SAmit Prakash Shukla 
1003112bf805SAmit Prakash Shukla 	adapter = edma_id_to_adapter(id);
1004112bf805SAmit Prakash Shukla 	if (adapter == NULL)
1005112bf805SAmit Prakash Shukla 		return -EINVAL;
1006112bf805SAmit Prakash Shukla 
1007112bf805SAmit Prakash Shukla 	dev = &rte_eventdevs[adapter->eventdev_id];
1008112bf805SAmit Prakash Shukla 	ret = rte_event_dma_adapter_caps_get(adapter->eventdev_id, dma_dev_id, &cap);
1009112bf805SAmit Prakash Shukla 	if (ret) {
1010112bf805SAmit Prakash Shukla 		RTE_EDEV_LOG_ERR("Failed to get adapter caps dev %u dma_dev %u", id, dma_dev_id);
1011112bf805SAmit Prakash Shukla 		return ret;
1012112bf805SAmit Prakash Shukla 	}
1013112bf805SAmit Prakash Shukla 
1014112bf805SAmit Prakash Shukla 	if ((cap & RTE_EVENT_DMA_ADAPTER_CAP_INTERNAL_PORT_VCHAN_EV_BIND) && (event == NULL)) {
1015112bf805SAmit Prakash Shukla 		RTE_EDEV_LOG_ERR("Event can not be NULL for dma_dev_id = %u", dma_dev_id);
1016112bf805SAmit Prakash Shukla 		return -EINVAL;
1017112bf805SAmit Prakash Shukla 	}
1018112bf805SAmit Prakash Shukla 
1019112bf805SAmit Prakash Shukla 	dev_info = &adapter->dma_devs[dma_dev_id];
1020112bf805SAmit Prakash Shukla 	if (vchan != RTE_DMA_ALL_VCHAN && vchan >= dev_info->num_dma_dev_vchan) {
1021112bf805SAmit Prakash Shukla 		RTE_EDEV_LOG_ERR("Invalid vhcan %u", vchan);
1022112bf805SAmit Prakash Shukla 		return -EINVAL;
1023112bf805SAmit Prakash Shukla 	}
1024112bf805SAmit Prakash Shukla 
1025112bf805SAmit Prakash Shukla 	/* In case HW cap is RTE_EVENT_DMA_ADAPTER_CAP_INTERNAL_PORT_OP_FWD, no
1026112bf805SAmit Prakash Shukla 	 * need of service core as HW supports event forward capability.
1027112bf805SAmit Prakash Shukla 	 */
1028112bf805SAmit Prakash Shukla 	if ((cap & RTE_EVENT_DMA_ADAPTER_CAP_INTERNAL_PORT_OP_FWD) ||
1029112bf805SAmit Prakash Shukla 	    (cap & RTE_EVENT_DMA_ADAPTER_CAP_INTERNAL_PORT_VCHAN_EV_BIND &&
1030112bf805SAmit Prakash Shukla 	     adapter->mode == RTE_EVENT_DMA_ADAPTER_OP_NEW) ||
1031112bf805SAmit Prakash Shukla 	    (cap & RTE_EVENT_DMA_ADAPTER_CAP_INTERNAL_PORT_OP_NEW &&
1032112bf805SAmit Prakash Shukla 	     adapter->mode == RTE_EVENT_DMA_ADAPTER_OP_NEW)) {
1033112bf805SAmit Prakash Shukla 		if (*dev->dev_ops->dma_adapter_vchan_add == NULL)
1034112bf805SAmit Prakash Shukla 			return -ENOTSUP;
1035112bf805SAmit Prakash Shukla 		if (dev_info->vchanq == NULL) {
1036112bf805SAmit Prakash Shukla 			dev_info->vchanq = rte_zmalloc_socket(adapter->mem_name,
1037112bf805SAmit Prakash Shukla 							dev_info->num_dma_dev_vchan *
1038112bf805SAmit Prakash Shukla 							sizeof(struct dma_vchan_info),
1039112bf805SAmit Prakash Shukla 							0, adapter->socket_id);
1040112bf805SAmit Prakash Shukla 			if (dev_info->vchanq == NULL) {
1041c07da8e3SDavid Marchand 				RTE_EDEV_LOG_ERR("Queue pair add not supported");
1042112bf805SAmit Prakash Shukla 				return -ENOMEM;
1043112bf805SAmit Prakash Shukla 			}
1044112bf805SAmit Prakash Shukla 		}
1045112bf805SAmit Prakash Shukla 
1046112bf805SAmit Prakash Shukla 		if (dev_info->tqmap == NULL) {
1047112bf805SAmit Prakash Shukla 			dev_info->tqmap = rte_zmalloc_socket(adapter->mem_name,
1048112bf805SAmit Prakash Shukla 						dev_info->num_dma_dev_vchan *
1049112bf805SAmit Prakash Shukla 						sizeof(struct dma_vchan_info),
1050112bf805SAmit Prakash Shukla 						0, adapter->socket_id);
1051112bf805SAmit Prakash Shukla 			if (dev_info->tqmap == NULL) {
1052c07da8e3SDavid Marchand 				RTE_EDEV_LOG_ERR("tq pair add not supported");
1053112bf805SAmit Prakash Shukla 				return -ENOMEM;
1054112bf805SAmit Prakash Shukla 			}
1055112bf805SAmit Prakash Shukla 		}
1056112bf805SAmit Prakash Shukla 
1057112bf805SAmit Prakash Shukla 		ret = (*dev->dev_ops->dma_adapter_vchan_add)(dev, dma_dev_id, vchan, event);
1058112bf805SAmit Prakash Shukla 		if (ret)
1059112bf805SAmit Prakash Shukla 			return ret;
1060112bf805SAmit Prakash Shukla 
1061112bf805SAmit Prakash Shukla 		else
1062112bf805SAmit Prakash Shukla 			edma_update_vchanq_info(adapter, &adapter->dma_devs[dma_dev_id], vchan, 1);
1063112bf805SAmit Prakash Shukla 	}
1064112bf805SAmit Prakash Shukla 
10653c89e8c4SAmit Prakash Shukla 	/* In case HW cap is RTE_EVENT_DMA_ADAPTER_CAP_INTERNAL_PORT_OP_NEW, or SW adapter, initiate
10663c89e8c4SAmit Prakash Shukla 	 * services so the application can choose which ever way it wants to use the adapter.
10673c89e8c4SAmit Prakash Shukla 	 *
10683c89e8c4SAmit Prakash Shukla 	 * Case 1: RTE_EVENT_DMA_ADAPTER_CAP_INTERNAL_PORT_OP_NEW. Application may wants to use one
10693c89e8c4SAmit Prakash Shukla 	 * of below two modes
10703c89e8c4SAmit Prakash Shukla 	 *
10713c89e8c4SAmit Prakash Shukla 	 * a. OP_FORWARD mode -> HW Dequeue + SW enqueue
10723c89e8c4SAmit Prakash Shukla 	 * b. OP_NEW mode -> HW Dequeue
10733c89e8c4SAmit Prakash Shukla 	 *
10743c89e8c4SAmit Prakash Shukla 	 * Case 2: No HW caps, use SW adapter
10753c89e8c4SAmit Prakash Shukla 	 *
10763c89e8c4SAmit Prakash Shukla 	 * a. OP_FORWARD mode -> SW enqueue & dequeue
10773c89e8c4SAmit Prakash Shukla 	 * b. OP_NEW mode -> SW Dequeue
10783c89e8c4SAmit Prakash Shukla 	 */
10793c89e8c4SAmit Prakash Shukla 	if ((cap & RTE_EVENT_DMA_ADAPTER_CAP_INTERNAL_PORT_OP_NEW &&
10803c89e8c4SAmit Prakash Shukla 	     !(cap & RTE_EVENT_DMA_ADAPTER_CAP_INTERNAL_PORT_OP_FWD) &&
10813c89e8c4SAmit Prakash Shukla 	     adapter->mode == RTE_EVENT_DMA_ADAPTER_OP_FORWARD) ||
10823c89e8c4SAmit Prakash Shukla 	    (!(cap & RTE_EVENT_DMA_ADAPTER_CAP_INTERNAL_PORT_OP_NEW) &&
10833c89e8c4SAmit Prakash Shukla 	     !(cap & RTE_EVENT_DMA_ADAPTER_CAP_INTERNAL_PORT_OP_FWD) &&
10843c89e8c4SAmit Prakash Shukla 	     !(cap & RTE_EVENT_DMA_ADAPTER_CAP_INTERNAL_PORT_VCHAN_EV_BIND))) {
10853c89e8c4SAmit Prakash Shukla 		rte_spinlock_lock(&adapter->lock);
10863c89e8c4SAmit Prakash Shukla 		ret = edma_init_service(adapter, id);
10873c89e8c4SAmit Prakash Shukla 		if (ret == 0)
10883c89e8c4SAmit Prakash Shukla 			ret = edma_add_vchan(adapter, dma_dev_id, vchan);
10893c89e8c4SAmit Prakash Shukla 		rte_spinlock_unlock(&adapter->lock);
10903c89e8c4SAmit Prakash Shukla 
10913c89e8c4SAmit Prakash Shukla 		if (ret)
10923c89e8c4SAmit Prakash Shukla 			return ret;
10933c89e8c4SAmit Prakash Shukla 
10943c89e8c4SAmit Prakash Shukla 		rte_service_component_runstate_set(adapter->service_id, 1);
10953c89e8c4SAmit Prakash Shukla 	}
10963c89e8c4SAmit Prakash Shukla 
1097112bf805SAmit Prakash Shukla 	return 0;
1098112bf805SAmit Prakash Shukla }
1099112bf805SAmit Prakash Shukla 
1100112bf805SAmit Prakash Shukla int
rte_event_dma_adapter_vchan_del(uint8_t id,int16_t dma_dev_id,uint16_t vchan)1101112bf805SAmit Prakash Shukla rte_event_dma_adapter_vchan_del(uint8_t id, int16_t dma_dev_id, uint16_t vchan)
1102112bf805SAmit Prakash Shukla {
1103112bf805SAmit Prakash Shukla 	struct event_dma_adapter *adapter;
1104112bf805SAmit Prakash Shukla 	struct dma_device_info *dev_info;
1105112bf805SAmit Prakash Shukla 	struct rte_eventdev *dev;
1106112bf805SAmit Prakash Shukla 	uint32_t cap;
1107112bf805SAmit Prakash Shukla 	int ret;
1108112bf805SAmit Prakash Shukla 
1109112bf805SAmit Prakash Shukla 	EVENT_DMA_ADAPTER_ID_VALID_OR_ERR_RET(id, -EINVAL);
1110112bf805SAmit Prakash Shukla 
1111112bf805SAmit Prakash Shukla 	if (!rte_dma_is_valid(dma_dev_id)) {
1112112bf805SAmit Prakash Shukla 		RTE_EDEV_LOG_ERR("Invalid dma_dev_id = %" PRIu8, dma_dev_id);
1113112bf805SAmit Prakash Shukla 		return -EINVAL;
1114112bf805SAmit Prakash Shukla 	}
1115112bf805SAmit Prakash Shukla 
1116112bf805SAmit Prakash Shukla 	adapter = edma_id_to_adapter(id);
1117112bf805SAmit Prakash Shukla 	if (adapter == NULL)
1118112bf805SAmit Prakash Shukla 		return -EINVAL;
1119112bf805SAmit Prakash Shukla 
1120112bf805SAmit Prakash Shukla 	dev = &rte_eventdevs[adapter->eventdev_id];
1121112bf805SAmit Prakash Shukla 	ret = rte_event_dma_adapter_caps_get(adapter->eventdev_id, dma_dev_id, &cap);
1122112bf805SAmit Prakash Shukla 	if (ret)
1123112bf805SAmit Prakash Shukla 		return ret;
1124112bf805SAmit Prakash Shukla 
1125112bf805SAmit Prakash Shukla 	dev_info = &adapter->dma_devs[dma_dev_id];
1126112bf805SAmit Prakash Shukla 
1127112bf805SAmit Prakash Shukla 	if (vchan != RTE_DMA_ALL_VCHAN && vchan >= dev_info->num_dma_dev_vchan) {
1128112bf805SAmit Prakash Shukla 		RTE_EDEV_LOG_ERR("Invalid vhcan %" PRIu16, vchan);
1129112bf805SAmit Prakash Shukla 		return -EINVAL;
1130112bf805SAmit Prakash Shukla 	}
1131112bf805SAmit Prakash Shukla 
1132112bf805SAmit Prakash Shukla 	if ((cap & RTE_EVENT_DMA_ADAPTER_CAP_INTERNAL_PORT_OP_FWD) ||
1133112bf805SAmit Prakash Shukla 	    (cap & RTE_EVENT_DMA_ADAPTER_CAP_INTERNAL_PORT_OP_NEW &&
1134112bf805SAmit Prakash Shukla 	     adapter->mode == RTE_EVENT_DMA_ADAPTER_OP_NEW)) {
1135112bf805SAmit Prakash Shukla 		if (*dev->dev_ops->dma_adapter_vchan_del == NULL)
1136112bf805SAmit Prakash Shukla 			return -ENOTSUP;
1137112bf805SAmit Prakash Shukla 		ret = (*dev->dev_ops->dma_adapter_vchan_del)(dev, dma_dev_id, vchan);
1138112bf805SAmit Prakash Shukla 		if (ret == 0) {
1139112bf805SAmit Prakash Shukla 			edma_update_vchanq_info(adapter, dev_info, vchan, 0);
1140112bf805SAmit Prakash Shukla 			if (dev_info->num_vchanq == 0) {
1141112bf805SAmit Prakash Shukla 				rte_free(dev_info->vchanq);
1142112bf805SAmit Prakash Shukla 				dev_info->vchanq = NULL;
1143112bf805SAmit Prakash Shukla 			}
1144112bf805SAmit Prakash Shukla 		}
1145112bf805SAmit Prakash Shukla 	} else {
1146112bf805SAmit Prakash Shukla 		if (adapter->nb_vchanq == 0)
1147112bf805SAmit Prakash Shukla 			return 0;
1148112bf805SAmit Prakash Shukla 
1149112bf805SAmit Prakash Shukla 		rte_spinlock_lock(&adapter->lock);
1150112bf805SAmit Prakash Shukla 		edma_update_vchanq_info(adapter, dev_info, vchan, 0);
1151112bf805SAmit Prakash Shukla 
1152112bf805SAmit Prakash Shukla 		if (dev_info->num_vchanq == 0) {
1153112bf805SAmit Prakash Shukla 			rte_free(dev_info->vchanq);
1154112bf805SAmit Prakash Shukla 			rte_free(dev_info->tqmap);
1155112bf805SAmit Prakash Shukla 			dev_info->vchanq = NULL;
1156112bf805SAmit Prakash Shukla 			dev_info->tqmap = NULL;
1157112bf805SAmit Prakash Shukla 		}
1158112bf805SAmit Prakash Shukla 
1159112bf805SAmit Prakash Shukla 		rte_spinlock_unlock(&adapter->lock);
11603c89e8c4SAmit Prakash Shukla 		rte_service_component_runstate_set(adapter->service_id, adapter->nb_vchanq);
1161112bf805SAmit Prakash Shukla 	}
1162112bf805SAmit Prakash Shukla 
1163112bf805SAmit Prakash Shukla 	return ret;
1164112bf805SAmit Prakash Shukla }
1165abbe0ddaSAmit Prakash Shukla 
116605881462SAmit Prakash Shukla int
rte_event_dma_adapter_service_id_get(uint8_t id,uint32_t * service_id)116705881462SAmit Prakash Shukla rte_event_dma_adapter_service_id_get(uint8_t id, uint32_t *service_id)
116805881462SAmit Prakash Shukla {
116905881462SAmit Prakash Shukla 	struct event_dma_adapter *adapter;
117005881462SAmit Prakash Shukla 
117105881462SAmit Prakash Shukla 	EVENT_DMA_ADAPTER_ID_VALID_OR_ERR_RET(id, -EINVAL);
117205881462SAmit Prakash Shukla 
117305881462SAmit Prakash Shukla 	adapter = edma_id_to_adapter(id);
117405881462SAmit Prakash Shukla 	if (adapter == NULL || service_id == NULL)
117505881462SAmit Prakash Shukla 		return -EINVAL;
117605881462SAmit Prakash Shukla 
117705881462SAmit Prakash Shukla 	if (adapter->service_initialized)
117805881462SAmit Prakash Shukla 		*service_id = adapter->service_id;
117905881462SAmit Prakash Shukla 
118005881462SAmit Prakash Shukla 	return adapter->service_initialized ? 0 : -ESRCH;
118105881462SAmit Prakash Shukla }
118205881462SAmit Prakash Shukla 
1183abbe0ddaSAmit Prakash Shukla static int
edma_adapter_ctrl(uint8_t id,int start)1184abbe0ddaSAmit Prakash Shukla edma_adapter_ctrl(uint8_t id, int start)
1185abbe0ddaSAmit Prakash Shukla {
1186abbe0ddaSAmit Prakash Shukla 	struct event_dma_adapter *adapter;
1187abbe0ddaSAmit Prakash Shukla 	struct dma_device_info *dev_info;
1188abbe0ddaSAmit Prakash Shukla 	struct rte_eventdev *dev;
1189abbe0ddaSAmit Prakash Shukla 	uint16_t num_dma_dev;
1190abbe0ddaSAmit Prakash Shukla 	int stop = !start;
1191abbe0ddaSAmit Prakash Shukla 	int use_service;
1192abbe0ddaSAmit Prakash Shukla 	uint32_t i;
1193abbe0ddaSAmit Prakash Shukla 
1194abbe0ddaSAmit Prakash Shukla 	use_service = 0;
1195abbe0ddaSAmit Prakash Shukla 	EVENT_DMA_ADAPTER_ID_VALID_OR_ERR_RET(id, -EINVAL);
1196abbe0ddaSAmit Prakash Shukla 	adapter = edma_id_to_adapter(id);
1197abbe0ddaSAmit Prakash Shukla 	if (adapter == NULL)
1198abbe0ddaSAmit Prakash Shukla 		return -EINVAL;
1199abbe0ddaSAmit Prakash Shukla 
1200abbe0ddaSAmit Prakash Shukla 	num_dma_dev = rte_dma_count_avail();
1201abbe0ddaSAmit Prakash Shukla 	dev = &rte_eventdevs[adapter->eventdev_id];
1202abbe0ddaSAmit Prakash Shukla 
1203abbe0ddaSAmit Prakash Shukla 	for (i = 0; i < num_dma_dev; i++) {
1204abbe0ddaSAmit Prakash Shukla 		dev_info = &adapter->dma_devs[i];
1205abbe0ddaSAmit Prakash Shukla 		/* start check for num queue pairs */
1206abbe0ddaSAmit Prakash Shukla 		if (start && !dev_info->num_vchanq)
1207abbe0ddaSAmit Prakash Shukla 			continue;
1208abbe0ddaSAmit Prakash Shukla 		/* stop check if dev has been started */
1209abbe0ddaSAmit Prakash Shukla 		if (stop && !dev_info->dev_started)
1210abbe0ddaSAmit Prakash Shukla 			continue;
1211abbe0ddaSAmit Prakash Shukla 		use_service |= !dev_info->internal_event_port;
1212abbe0ddaSAmit Prakash Shukla 		dev_info->dev_started = start;
1213abbe0ddaSAmit Prakash Shukla 		if (dev_info->internal_event_port == 0)
1214abbe0ddaSAmit Prakash Shukla 			continue;
1215abbe0ddaSAmit Prakash Shukla 		start ? (*dev->dev_ops->dma_adapter_start)(dev, i) :
1216abbe0ddaSAmit Prakash Shukla 			(*dev->dev_ops->dma_adapter_stop)(dev, i);
1217abbe0ddaSAmit Prakash Shukla 	}
1218abbe0ddaSAmit Prakash Shukla 
1219abbe0ddaSAmit Prakash Shukla 	if (use_service)
1220abbe0ddaSAmit Prakash Shukla 		rte_service_runstate_set(adapter->service_id, start);
1221abbe0ddaSAmit Prakash Shukla 
1222abbe0ddaSAmit Prakash Shukla 	return 0;
1223abbe0ddaSAmit Prakash Shukla }
1224abbe0ddaSAmit Prakash Shukla 
1225abbe0ddaSAmit Prakash Shukla int
rte_event_dma_adapter_start(uint8_t id)1226abbe0ddaSAmit Prakash Shukla rte_event_dma_adapter_start(uint8_t id)
1227abbe0ddaSAmit Prakash Shukla {
1228abbe0ddaSAmit Prakash Shukla 	struct event_dma_adapter *adapter;
1229abbe0ddaSAmit Prakash Shukla 
1230abbe0ddaSAmit Prakash Shukla 	EVENT_DMA_ADAPTER_ID_VALID_OR_ERR_RET(id, -EINVAL);
1231abbe0ddaSAmit Prakash Shukla 
1232abbe0ddaSAmit Prakash Shukla 	adapter = edma_id_to_adapter(id);
1233abbe0ddaSAmit Prakash Shukla 	if (adapter == NULL)
1234abbe0ddaSAmit Prakash Shukla 		return -EINVAL;
1235abbe0ddaSAmit Prakash Shukla 
1236abbe0ddaSAmit Prakash Shukla 	return edma_adapter_ctrl(id, 1);
1237abbe0ddaSAmit Prakash Shukla }
1238abbe0ddaSAmit Prakash Shukla 
1239abbe0ddaSAmit Prakash Shukla int
rte_event_dma_adapter_stop(uint8_t id)1240abbe0ddaSAmit Prakash Shukla rte_event_dma_adapter_stop(uint8_t id)
1241abbe0ddaSAmit Prakash Shukla {
1242abbe0ddaSAmit Prakash Shukla 	return edma_adapter_ctrl(id, 0);
1243abbe0ddaSAmit Prakash Shukla }
12442c6e23cdSAmit Prakash Shukla 
12452c6e23cdSAmit Prakash Shukla #define DEFAULT_MAX_NB 128
12462c6e23cdSAmit Prakash Shukla 
12472c6e23cdSAmit Prakash Shukla int
rte_event_dma_adapter_runtime_params_init(struct rte_event_dma_adapter_runtime_params * params)12482c6e23cdSAmit Prakash Shukla rte_event_dma_adapter_runtime_params_init(struct rte_event_dma_adapter_runtime_params *params)
12492c6e23cdSAmit Prakash Shukla {
12502c6e23cdSAmit Prakash Shukla 	if (params == NULL)
12512c6e23cdSAmit Prakash Shukla 		return -EINVAL;
12522c6e23cdSAmit Prakash Shukla 
12532c6e23cdSAmit Prakash Shukla 	memset(params, 0, sizeof(*params));
12542c6e23cdSAmit Prakash Shukla 	params->max_nb = DEFAULT_MAX_NB;
12552c6e23cdSAmit Prakash Shukla 
12562c6e23cdSAmit Prakash Shukla 	return 0;
12572c6e23cdSAmit Prakash Shukla }
12582c6e23cdSAmit Prakash Shukla 
12592c6e23cdSAmit Prakash Shukla static int
dma_adapter_cap_check(struct event_dma_adapter * adapter)12602c6e23cdSAmit Prakash Shukla dma_adapter_cap_check(struct event_dma_adapter *adapter)
12612c6e23cdSAmit Prakash Shukla {
12622c6e23cdSAmit Prakash Shukla 	uint32_t caps;
12632c6e23cdSAmit Prakash Shukla 	int ret;
12642c6e23cdSAmit Prakash Shukla 
12652c6e23cdSAmit Prakash Shukla 	if (!adapter->nb_vchanq)
12662c6e23cdSAmit Prakash Shukla 		return -EINVAL;
12672c6e23cdSAmit Prakash Shukla 
12682c6e23cdSAmit Prakash Shukla 	ret = rte_event_dma_adapter_caps_get(adapter->eventdev_id, adapter->next_dmadev_id, &caps);
12692c6e23cdSAmit Prakash Shukla 	if (ret) {
12702c6e23cdSAmit Prakash Shukla 		RTE_EDEV_LOG_ERR("Failed to get adapter caps dev %" PRIu8 " cdev %" PRIu8,
12712c6e23cdSAmit Prakash Shukla 				 adapter->eventdev_id, adapter->next_dmadev_id);
12722c6e23cdSAmit Prakash Shukla 		return ret;
12732c6e23cdSAmit Prakash Shukla 	}
12742c6e23cdSAmit Prakash Shukla 
12752c6e23cdSAmit Prakash Shukla 	if ((caps & RTE_EVENT_DMA_ADAPTER_CAP_INTERNAL_PORT_OP_FWD) ||
12762c6e23cdSAmit Prakash Shukla 	    (caps & RTE_EVENT_DMA_ADAPTER_CAP_INTERNAL_PORT_OP_NEW))
12772c6e23cdSAmit Prakash Shukla 		return -ENOTSUP;
12782c6e23cdSAmit Prakash Shukla 
12792c6e23cdSAmit Prakash Shukla 	return 0;
12802c6e23cdSAmit Prakash Shukla }
12812c6e23cdSAmit Prakash Shukla 
12822c6e23cdSAmit Prakash Shukla int
rte_event_dma_adapter_runtime_params_set(uint8_t id,struct rte_event_dma_adapter_runtime_params * params)12832c6e23cdSAmit Prakash Shukla rte_event_dma_adapter_runtime_params_set(uint8_t id,
12842c6e23cdSAmit Prakash Shukla 					 struct rte_event_dma_adapter_runtime_params *params)
12852c6e23cdSAmit Prakash Shukla {
12862c6e23cdSAmit Prakash Shukla 	struct event_dma_adapter *adapter;
12872c6e23cdSAmit Prakash Shukla 	int ret;
12882c6e23cdSAmit Prakash Shukla 
12892c6e23cdSAmit Prakash Shukla 	EVENT_DMA_ADAPTER_ID_VALID_OR_ERR_RET(id, -EINVAL);
12902c6e23cdSAmit Prakash Shukla 
12912c6e23cdSAmit Prakash Shukla 	if (params == NULL) {
1292ae282b06SDavid Marchand 		RTE_EDEV_LOG_ERR("params pointer is NULL");
12932c6e23cdSAmit Prakash Shukla 		return -EINVAL;
12942c6e23cdSAmit Prakash Shukla 	}
12952c6e23cdSAmit Prakash Shukla 
12962c6e23cdSAmit Prakash Shukla 	adapter = edma_id_to_adapter(id);
12972c6e23cdSAmit Prakash Shukla 	if (adapter == NULL)
12982c6e23cdSAmit Prakash Shukla 		return -EINVAL;
12992c6e23cdSAmit Prakash Shukla 
13002c6e23cdSAmit Prakash Shukla 	ret = dma_adapter_cap_check(adapter);
13012c6e23cdSAmit Prakash Shukla 	if (ret)
13022c6e23cdSAmit Prakash Shukla 		return ret;
13032c6e23cdSAmit Prakash Shukla 
13042c6e23cdSAmit Prakash Shukla 	rte_spinlock_lock(&adapter->lock);
13052c6e23cdSAmit Prakash Shukla 	adapter->max_nb = params->max_nb;
13062c6e23cdSAmit Prakash Shukla 	rte_spinlock_unlock(&adapter->lock);
13072c6e23cdSAmit Prakash Shukla 
13082c6e23cdSAmit Prakash Shukla 	return 0;
13092c6e23cdSAmit Prakash Shukla }
13102c6e23cdSAmit Prakash Shukla 
13112c6e23cdSAmit Prakash Shukla int
rte_event_dma_adapter_runtime_params_get(uint8_t id,struct rte_event_dma_adapter_runtime_params * params)13122c6e23cdSAmit Prakash Shukla rte_event_dma_adapter_runtime_params_get(uint8_t id,
13132c6e23cdSAmit Prakash Shukla 					 struct rte_event_dma_adapter_runtime_params *params)
13142c6e23cdSAmit Prakash Shukla {
13152c6e23cdSAmit Prakash Shukla 	struct event_dma_adapter *adapter;
13162c6e23cdSAmit Prakash Shukla 	int ret;
13172c6e23cdSAmit Prakash Shukla 
13182c6e23cdSAmit Prakash Shukla 	EVENT_DMA_ADAPTER_ID_VALID_OR_ERR_RET(id, -EINVAL);
13192c6e23cdSAmit Prakash Shukla 
13202c6e23cdSAmit Prakash Shukla 	if (params == NULL) {
1321ae282b06SDavid Marchand 		RTE_EDEV_LOG_ERR("params pointer is NULL");
13222c6e23cdSAmit Prakash Shukla 		return -EINVAL;
13232c6e23cdSAmit Prakash Shukla 	}
13242c6e23cdSAmit Prakash Shukla 
13252c6e23cdSAmit Prakash Shukla 	adapter = edma_id_to_adapter(id);
13262c6e23cdSAmit Prakash Shukla 	if (adapter == NULL)
13272c6e23cdSAmit Prakash Shukla 		return -EINVAL;
13282c6e23cdSAmit Prakash Shukla 
13292c6e23cdSAmit Prakash Shukla 	ret = dma_adapter_cap_check(adapter);
13302c6e23cdSAmit Prakash Shukla 	if (ret)
13312c6e23cdSAmit Prakash Shukla 		return ret;
13322c6e23cdSAmit Prakash Shukla 
13332c6e23cdSAmit Prakash Shukla 	params->max_nb = adapter->max_nb;
13342c6e23cdSAmit Prakash Shukla 
13352c6e23cdSAmit Prakash Shukla 	return 0;
13362c6e23cdSAmit Prakash Shukla }
13373c37e8c6SAmit Prakash Shukla 
13383c37e8c6SAmit Prakash Shukla int
rte_event_dma_adapter_stats_get(uint8_t id,struct rte_event_dma_adapter_stats * stats)13393c37e8c6SAmit Prakash Shukla rte_event_dma_adapter_stats_get(uint8_t id, struct rte_event_dma_adapter_stats *stats)
13403c37e8c6SAmit Prakash Shukla {
13413c37e8c6SAmit Prakash Shukla 	struct rte_event_dma_adapter_stats dev_stats_sum = {0};
13423c37e8c6SAmit Prakash Shukla 	struct rte_event_dma_adapter_stats dev_stats;
13433c37e8c6SAmit Prakash Shukla 	struct event_dma_adapter *adapter;
13443c37e8c6SAmit Prakash Shukla 	struct dma_device_info *dev_info;
13453c37e8c6SAmit Prakash Shukla 	struct rte_eventdev *dev;
13463c37e8c6SAmit Prakash Shukla 	uint16_t num_dma_dev;
13473c37e8c6SAmit Prakash Shukla 	uint32_t i;
13483c37e8c6SAmit Prakash Shukla 	int ret;
13493c37e8c6SAmit Prakash Shukla 
13503c37e8c6SAmit Prakash Shukla 	EVENT_DMA_ADAPTER_ID_VALID_OR_ERR_RET(id, -EINVAL);
13513c37e8c6SAmit Prakash Shukla 
13523c37e8c6SAmit Prakash Shukla 	adapter = edma_id_to_adapter(id);
13533c37e8c6SAmit Prakash Shukla 	if (adapter == NULL || stats == NULL)
13543c37e8c6SAmit Prakash Shukla 		return -EINVAL;
13553c37e8c6SAmit Prakash Shukla 
13563c37e8c6SAmit Prakash Shukla 	num_dma_dev = rte_dma_count_avail();
13573c37e8c6SAmit Prakash Shukla 	dev = &rte_eventdevs[adapter->eventdev_id];
13583c37e8c6SAmit Prakash Shukla 	memset(stats, 0, sizeof(*stats));
13593c37e8c6SAmit Prakash Shukla 	for (i = 0; i < num_dma_dev; i++) {
13603c37e8c6SAmit Prakash Shukla 		dev_info = &adapter->dma_devs[i];
13613c37e8c6SAmit Prakash Shukla 
13623c37e8c6SAmit Prakash Shukla 		if (dev_info->internal_event_port == 0 ||
13633c37e8c6SAmit Prakash Shukla 		    dev->dev_ops->dma_adapter_stats_get == NULL)
13643c37e8c6SAmit Prakash Shukla 			continue;
13653c37e8c6SAmit Prakash Shukla 
13663c37e8c6SAmit Prakash Shukla 		ret = (*dev->dev_ops->dma_adapter_stats_get)(dev, i, &dev_stats);
13673c37e8c6SAmit Prakash Shukla 		if (ret)
13683c37e8c6SAmit Prakash Shukla 			continue;
13693c37e8c6SAmit Prakash Shukla 
13703c37e8c6SAmit Prakash Shukla 		dev_stats_sum.dma_deq_count += dev_stats.dma_deq_count;
13713c37e8c6SAmit Prakash Shukla 		dev_stats_sum.event_enq_count += dev_stats.event_enq_count;
13723c37e8c6SAmit Prakash Shukla 	}
13733c37e8c6SAmit Prakash Shukla 
13743c37e8c6SAmit Prakash Shukla 	if (adapter->service_initialized)
13753c37e8c6SAmit Prakash Shukla 		*stats = adapter->dma_stats;
13763c37e8c6SAmit Prakash Shukla 
13773c37e8c6SAmit Prakash Shukla 	stats->dma_deq_count += dev_stats_sum.dma_deq_count;
13783c37e8c6SAmit Prakash Shukla 	stats->event_enq_count += dev_stats_sum.event_enq_count;
13793c37e8c6SAmit Prakash Shukla 
13803c37e8c6SAmit Prakash Shukla 	return 0;
13813c37e8c6SAmit Prakash Shukla }
13823c37e8c6SAmit Prakash Shukla 
13833c37e8c6SAmit Prakash Shukla int
rte_event_dma_adapter_stats_reset(uint8_t id)13843c37e8c6SAmit Prakash Shukla rte_event_dma_adapter_stats_reset(uint8_t id)
13853c37e8c6SAmit Prakash Shukla {
13863c37e8c6SAmit Prakash Shukla 	struct event_dma_adapter *adapter;
13873c37e8c6SAmit Prakash Shukla 	struct dma_device_info *dev_info;
13883c37e8c6SAmit Prakash Shukla 	struct rte_eventdev *dev;
13893c37e8c6SAmit Prakash Shukla 	uint16_t num_dma_dev;
13903c37e8c6SAmit Prakash Shukla 	uint32_t i;
13913c37e8c6SAmit Prakash Shukla 
13923c37e8c6SAmit Prakash Shukla 	EVENT_DMA_ADAPTER_ID_VALID_OR_ERR_RET(id, -EINVAL);
13933c37e8c6SAmit Prakash Shukla 
13943c37e8c6SAmit Prakash Shukla 	adapter = edma_id_to_adapter(id);
13953c37e8c6SAmit Prakash Shukla 	if (adapter == NULL)
13963c37e8c6SAmit Prakash Shukla 		return -EINVAL;
13973c37e8c6SAmit Prakash Shukla 
13983c37e8c6SAmit Prakash Shukla 	num_dma_dev = rte_dma_count_avail();
13993c37e8c6SAmit Prakash Shukla 	dev = &rte_eventdevs[adapter->eventdev_id];
14003c37e8c6SAmit Prakash Shukla 	for (i = 0; i < num_dma_dev; i++) {
14013c37e8c6SAmit Prakash Shukla 		dev_info = &adapter->dma_devs[i];
14023c37e8c6SAmit Prakash Shukla 
14033c37e8c6SAmit Prakash Shukla 		if (dev_info->internal_event_port == 0 ||
14043c37e8c6SAmit Prakash Shukla 		    dev->dev_ops->dma_adapter_stats_reset == NULL)
14053c37e8c6SAmit Prakash Shukla 			continue;
14063c37e8c6SAmit Prakash Shukla 
14073c37e8c6SAmit Prakash Shukla 		(*dev->dev_ops->dma_adapter_stats_reset)(dev, i);
14083c37e8c6SAmit Prakash Shukla 	}
14093c37e8c6SAmit Prakash Shukla 
14103c37e8c6SAmit Prakash Shukla 	memset(&adapter->dma_stats, 0, sizeof(adapter->dma_stats));
14113c37e8c6SAmit Prakash Shukla 
14123c37e8c6SAmit Prakash Shukla 	return 0;
14133c37e8c6SAmit Prakash Shukla }
1414a7884827SAmit Prakash Shukla 
1415a7884827SAmit Prakash Shukla uint16_t
rte_event_dma_adapter_enqueue(uint8_t dev_id,uint8_t port_id,struct rte_event ev[],uint16_t nb_events)1416a7884827SAmit Prakash Shukla rte_event_dma_adapter_enqueue(uint8_t dev_id, uint8_t port_id, struct rte_event ev[],
1417a7884827SAmit Prakash Shukla 			      uint16_t nb_events)
1418a7884827SAmit Prakash Shukla {
1419a7884827SAmit Prakash Shukla 	const struct rte_event_fp_ops *fp_ops;
1420a7884827SAmit Prakash Shukla 	void *port;
1421a7884827SAmit Prakash Shukla 
1422a7884827SAmit Prakash Shukla 	fp_ops = &rte_event_fp_ops[dev_id];
1423a7884827SAmit Prakash Shukla 	port = fp_ops->data[port_id];
1424a7884827SAmit Prakash Shukla 
1425a7884827SAmit Prakash Shukla 	return fp_ops->dma_enqueue(port, ev, nb_events);
1426a7884827SAmit Prakash Shukla }
1427