xref: /freebsd/sys/dev/xdma/xdma.c (revision e2e050c8ef733138fc6a9e514e4b856fefbc3ff1)
185debf7fSRuslan Bukin /*-
2101869a8SRuslan Bukin  * SPDX-License-Identifier: BSD-2-Clause
3101869a8SRuslan Bukin  *
4101869a8SRuslan Bukin  * Copyright (c) 2016-2019 Ruslan Bukin <br@bsdpad.com>
585debf7fSRuslan Bukin  *
685debf7fSRuslan Bukin  * This software was developed by SRI International and the University of
785debf7fSRuslan Bukin  * Cambridge Computer Laboratory under DARPA/AFRL contract FA8750-10-C-0237
885debf7fSRuslan Bukin  * ("CTSRD"), as part of the DARPA CRASH research programme.
985debf7fSRuslan Bukin  *
1085debf7fSRuslan Bukin  * Redistribution and use in source and binary forms, with or without
1185debf7fSRuslan Bukin  * modification, are permitted provided that the following conditions
1285debf7fSRuslan Bukin  * are met:
1385debf7fSRuslan Bukin  * 1. Redistributions of source code must retain the above copyright
1485debf7fSRuslan Bukin  *    notice, this list of conditions and the following disclaimer.
1585debf7fSRuslan Bukin  * 2. Redistributions in binary form must reproduce the above copyright
1685debf7fSRuslan Bukin  *    notice, this list of conditions and the following disclaimer in the
1785debf7fSRuslan Bukin  *    documentation and/or other materials provided with the distribution.
1885debf7fSRuslan Bukin  *
1985debf7fSRuslan Bukin  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
2085debf7fSRuslan Bukin  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
2185debf7fSRuslan Bukin  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
2285debf7fSRuslan Bukin  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
2385debf7fSRuslan Bukin  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2485debf7fSRuslan Bukin  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2585debf7fSRuslan Bukin  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2685debf7fSRuslan Bukin  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
2785debf7fSRuslan Bukin  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
2885debf7fSRuslan Bukin  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
2985debf7fSRuslan Bukin  * SUCH DAMAGE.
3085debf7fSRuslan Bukin  */
3185debf7fSRuslan Bukin 
3285debf7fSRuslan Bukin #include <sys/cdefs.h>
3385debf7fSRuslan Bukin __FBSDID("$FreeBSD$");
3485debf7fSRuslan Bukin 
3585debf7fSRuslan Bukin #include "opt_platform.h"
3685debf7fSRuslan Bukin #include <sys/param.h>
3785debf7fSRuslan Bukin #include <sys/conf.h>
3885debf7fSRuslan Bukin #include <sys/bus.h>
3985debf7fSRuslan Bukin #include <sys/kernel.h>
4085debf7fSRuslan Bukin #include <sys/queue.h>
4185debf7fSRuslan Bukin #include <sys/kobj.h>
4285debf7fSRuslan Bukin #include <sys/malloc.h>
4385debf7fSRuslan Bukin #include <sys/limits.h>
4485debf7fSRuslan Bukin #include <sys/lock.h>
45*e2e050c8SConrad Meyer #include <sys/mutex.h>
4685debf7fSRuslan Bukin #include <sys/sysctl.h>
4785debf7fSRuslan Bukin #include <sys/systm.h>
4885debf7fSRuslan Bukin 
4985debf7fSRuslan Bukin #include <machine/bus.h>
5085debf7fSRuslan Bukin 
5185debf7fSRuslan Bukin #ifdef FDT
5285debf7fSRuslan Bukin #include <dev/fdt/fdt_common.h>
5385debf7fSRuslan Bukin #include <dev/ofw/ofw_bus.h>
5485debf7fSRuslan Bukin #include <dev/ofw/ofw_bus_subr.h>
5585debf7fSRuslan Bukin #endif
5685debf7fSRuslan Bukin 
5785debf7fSRuslan Bukin #include <dev/xdma/xdma.h>
5885debf7fSRuslan Bukin 
5985debf7fSRuslan Bukin #include <xdma_if.h>
6085debf7fSRuslan Bukin 
6185debf7fSRuslan Bukin /*
6285debf7fSRuslan Bukin  * Multiple xDMA controllers may work with single DMA device,
6385debf7fSRuslan Bukin  * so we have global lock for physical channel management.
6485debf7fSRuslan Bukin  */
65101869a8SRuslan Bukin static struct mtx xdma_mtx;
6685debf7fSRuslan Bukin 
67101869a8SRuslan Bukin #define	XDMA_LOCK()			mtx_lock(&xdma_mtx)
68101869a8SRuslan Bukin #define	XDMA_UNLOCK()			mtx_unlock(&xdma_mtx)
69101869a8SRuslan Bukin #define	XDMA_ASSERT_LOCKED()		mtx_assert(&xdma_mtx, MA_OWNED)
70101869a8SRuslan Bukin 
71101869a8SRuslan Bukin #define	FDT_REG_CELLS	4
7285debf7fSRuslan Bukin 
7385debf7fSRuslan Bukin /*
7485debf7fSRuslan Bukin  * Allocate virtual xDMA channel.
7585debf7fSRuslan Bukin  */
7685debf7fSRuslan Bukin xdma_channel_t *
773d5b3b0aSRuslan Bukin xdma_channel_alloc(xdma_controller_t *xdma, uint32_t caps)
7885debf7fSRuslan Bukin {
7985debf7fSRuslan Bukin 	xdma_channel_t *xchan;
8085debf7fSRuslan Bukin 	int ret;
8185debf7fSRuslan Bukin 
8285debf7fSRuslan Bukin 	xchan = malloc(sizeof(xdma_channel_t), M_XDMA, M_WAITOK | M_ZERO);
8385debf7fSRuslan Bukin 	xchan->xdma = xdma;
843d5b3b0aSRuslan Bukin 	xchan->caps = caps;
8585debf7fSRuslan Bukin 
8685debf7fSRuslan Bukin 	XDMA_LOCK();
8785debf7fSRuslan Bukin 
8885debf7fSRuslan Bukin 	/* Request a real channel from hardware driver. */
8985debf7fSRuslan Bukin 	ret = XDMA_CHANNEL_ALLOC(xdma->dma_dev, xchan);
9085debf7fSRuslan Bukin 	if (ret != 0) {
9185debf7fSRuslan Bukin 		device_printf(xdma->dev,
9285debf7fSRuslan Bukin 		    "%s: Can't request hardware channel.\n", __func__);
9385debf7fSRuslan Bukin 		XDMA_UNLOCK();
9485debf7fSRuslan Bukin 		free(xchan, M_XDMA);
9585debf7fSRuslan Bukin 
9685debf7fSRuslan Bukin 		return (NULL);
9785debf7fSRuslan Bukin 	}
9885debf7fSRuslan Bukin 
9985debf7fSRuslan Bukin 	TAILQ_INIT(&xchan->ie_handlers);
1003d5b3b0aSRuslan Bukin 
101101869a8SRuslan Bukin 	mtx_init(&xchan->mtx_lock, "xDMA chan", NULL, MTX_DEF);
102101869a8SRuslan Bukin 	mtx_init(&xchan->mtx_qin_lock, "xDMA qin", NULL, MTX_DEF);
103101869a8SRuslan Bukin 	mtx_init(&xchan->mtx_qout_lock, "xDMA qout", NULL, MTX_DEF);
104101869a8SRuslan Bukin 	mtx_init(&xchan->mtx_bank_lock, "xDMA bank", NULL, MTX_DEF);
105101869a8SRuslan Bukin 	mtx_init(&xchan->mtx_proc_lock, "xDMA proc", NULL, MTX_DEF);
1063d5b3b0aSRuslan Bukin 
1073d5b3b0aSRuslan Bukin 	TAILQ_INIT(&xchan->bank);
1083d5b3b0aSRuslan Bukin 	TAILQ_INIT(&xchan->queue_in);
1093d5b3b0aSRuslan Bukin 	TAILQ_INIT(&xchan->queue_out);
1103d5b3b0aSRuslan Bukin 	TAILQ_INIT(&xchan->processing);
11185debf7fSRuslan Bukin 
11285debf7fSRuslan Bukin 	TAILQ_INSERT_TAIL(&xdma->channels, xchan, xchan_next);
11385debf7fSRuslan Bukin 
11485debf7fSRuslan Bukin 	XDMA_UNLOCK();
11585debf7fSRuslan Bukin 
11685debf7fSRuslan Bukin 	return (xchan);
11785debf7fSRuslan Bukin }
11885debf7fSRuslan Bukin 
11985debf7fSRuslan Bukin int
12085debf7fSRuslan Bukin xdma_channel_free(xdma_channel_t *xchan)
12185debf7fSRuslan Bukin {
12285debf7fSRuslan Bukin 	xdma_controller_t *xdma;
12385debf7fSRuslan Bukin 	int err;
12485debf7fSRuslan Bukin 
12585debf7fSRuslan Bukin 	xdma = xchan->xdma;
1263d5b3b0aSRuslan Bukin 	KASSERT(xdma != NULL, ("xdma is NULL"));
12785debf7fSRuslan Bukin 
12885debf7fSRuslan Bukin 	XDMA_LOCK();
12985debf7fSRuslan Bukin 
13085debf7fSRuslan Bukin 	/* Free the real DMA channel. */
13185debf7fSRuslan Bukin 	err = XDMA_CHANNEL_FREE(xdma->dma_dev, xchan);
13285debf7fSRuslan Bukin 	if (err != 0) {
13385debf7fSRuslan Bukin 		device_printf(xdma->dev,
13485debf7fSRuslan Bukin 		    "%s: Can't free real hw channel.\n", __func__);
13585debf7fSRuslan Bukin 		XDMA_UNLOCK();
13685debf7fSRuslan Bukin 		return (-1);
13785debf7fSRuslan Bukin 	}
13885debf7fSRuslan Bukin 
1393d5b3b0aSRuslan Bukin 	if (xchan->flags & XCHAN_TYPE_SG)
1403d5b3b0aSRuslan Bukin 		xdma_channel_free_sg(xchan);
1413d5b3b0aSRuslan Bukin 
14285debf7fSRuslan Bukin 	xdma_teardown_all_intr(xchan);
14385debf7fSRuslan Bukin 
144101869a8SRuslan Bukin 	mtx_destroy(&xchan->mtx_lock);
145101869a8SRuslan Bukin 	mtx_destroy(&xchan->mtx_qin_lock);
146101869a8SRuslan Bukin 	mtx_destroy(&xchan->mtx_qout_lock);
147101869a8SRuslan Bukin 	mtx_destroy(&xchan->mtx_bank_lock);
148101869a8SRuslan Bukin 	mtx_destroy(&xchan->mtx_proc_lock);
14985debf7fSRuslan Bukin 
15085debf7fSRuslan Bukin 	TAILQ_REMOVE(&xdma->channels, xchan, xchan_next);
15185debf7fSRuslan Bukin 
15285debf7fSRuslan Bukin 	free(xchan, M_XDMA);
15385debf7fSRuslan Bukin 
15485debf7fSRuslan Bukin 	XDMA_UNLOCK();
15585debf7fSRuslan Bukin 
15685debf7fSRuslan Bukin 	return (0);
15785debf7fSRuslan Bukin }
15885debf7fSRuslan Bukin 
15985debf7fSRuslan Bukin int
1603d5b3b0aSRuslan Bukin xdma_setup_intr(xdma_channel_t *xchan,
1613d5b3b0aSRuslan Bukin     int (*cb)(void *, xdma_transfer_status_t *),
1623d5b3b0aSRuslan Bukin     void *arg, void **ihandler)
16385debf7fSRuslan Bukin {
16485debf7fSRuslan Bukin 	struct xdma_intr_handler *ih;
16585debf7fSRuslan Bukin 	xdma_controller_t *xdma;
16685debf7fSRuslan Bukin 
16785debf7fSRuslan Bukin 	xdma = xchan->xdma;
16885debf7fSRuslan Bukin 	KASSERT(xdma != NULL, ("xdma is NULL"));
16985debf7fSRuslan Bukin 
17085debf7fSRuslan Bukin 	/* Sanity check. */
17185debf7fSRuslan Bukin 	if (cb == NULL) {
17285debf7fSRuslan Bukin 		device_printf(xdma->dev,
17385debf7fSRuslan Bukin 		    "%s: Can't setup interrupt handler.\n",
17485debf7fSRuslan Bukin 		    __func__);
17585debf7fSRuslan Bukin 
17685debf7fSRuslan Bukin 		return (-1);
17785debf7fSRuslan Bukin 	}
17885debf7fSRuslan Bukin 
17985debf7fSRuslan Bukin 	ih = malloc(sizeof(struct xdma_intr_handler),
18085debf7fSRuslan Bukin 	    M_XDMA, M_WAITOK | M_ZERO);
18185debf7fSRuslan Bukin 	ih->cb = cb;
18285debf7fSRuslan Bukin 	ih->cb_user = arg;
18385debf7fSRuslan Bukin 
1843d5b3b0aSRuslan Bukin 	XCHAN_LOCK(xchan);
18585debf7fSRuslan Bukin 	TAILQ_INSERT_TAIL(&xchan->ie_handlers, ih, ih_next);
1863d5b3b0aSRuslan Bukin 	XCHAN_UNLOCK(xchan);
18785debf7fSRuslan Bukin 
1883d5b3b0aSRuslan Bukin 	if (ihandler != NULL)
18985debf7fSRuslan Bukin 		*ihandler = ih;
19085debf7fSRuslan Bukin 
19185debf7fSRuslan Bukin 	return (0);
19285debf7fSRuslan Bukin }
19385debf7fSRuslan Bukin 
19485debf7fSRuslan Bukin int
19585debf7fSRuslan Bukin xdma_teardown_intr(xdma_channel_t *xchan, struct xdma_intr_handler *ih)
19685debf7fSRuslan Bukin {
19785debf7fSRuslan Bukin 	xdma_controller_t *xdma;
19885debf7fSRuslan Bukin 
19985debf7fSRuslan Bukin 	xdma = xchan->xdma;
20085debf7fSRuslan Bukin 	KASSERT(xdma != NULL, ("xdma is NULL"));
20185debf7fSRuslan Bukin 
20285debf7fSRuslan Bukin 	/* Sanity check. */
20385debf7fSRuslan Bukin 	if (ih == NULL) {
20485debf7fSRuslan Bukin 		device_printf(xdma->dev,
20585debf7fSRuslan Bukin 		    "%s: Can't teardown interrupt.\n", __func__);
20685debf7fSRuslan Bukin 		return (-1);
20785debf7fSRuslan Bukin 	}
20885debf7fSRuslan Bukin 
20985debf7fSRuslan Bukin 	TAILQ_REMOVE(&xchan->ie_handlers, ih, ih_next);
21085debf7fSRuslan Bukin 	free(ih, M_XDMA);
21185debf7fSRuslan Bukin 
21285debf7fSRuslan Bukin 	return (0);
21385debf7fSRuslan Bukin }
21485debf7fSRuslan Bukin 
21585debf7fSRuslan Bukin int
21685debf7fSRuslan Bukin xdma_teardown_all_intr(xdma_channel_t *xchan)
21785debf7fSRuslan Bukin {
21885debf7fSRuslan Bukin 	struct xdma_intr_handler *ih_tmp;
21985debf7fSRuslan Bukin 	struct xdma_intr_handler *ih;
22085debf7fSRuslan Bukin 	xdma_controller_t *xdma;
22185debf7fSRuslan Bukin 
22285debf7fSRuslan Bukin 	xdma = xchan->xdma;
22385debf7fSRuslan Bukin 	KASSERT(xdma != NULL, ("xdma is NULL"));
22485debf7fSRuslan Bukin 
22585debf7fSRuslan Bukin 	TAILQ_FOREACH_SAFE(ih, &xchan->ie_handlers, ih_next, ih_tmp) {
22685debf7fSRuslan Bukin 		TAILQ_REMOVE(&xchan->ie_handlers, ih, ih_next);
22785debf7fSRuslan Bukin 		free(ih, M_XDMA);
22885debf7fSRuslan Bukin 	}
22985debf7fSRuslan Bukin 
23085debf7fSRuslan Bukin 	return (0);
23185debf7fSRuslan Bukin }
23285debf7fSRuslan Bukin 
23385debf7fSRuslan Bukin int
2343d5b3b0aSRuslan Bukin xdma_request(xdma_channel_t *xchan, struct xdma_request *req)
23585debf7fSRuslan Bukin {
23685debf7fSRuslan Bukin 	xdma_controller_t *xdma;
23785debf7fSRuslan Bukin 	int ret;
23885debf7fSRuslan Bukin 
23985debf7fSRuslan Bukin 	xdma = xchan->xdma;
24085debf7fSRuslan Bukin 
2413d5b3b0aSRuslan Bukin 	KASSERT(xdma != NULL, ("xdma is NULL"));
24285debf7fSRuslan Bukin 
24385debf7fSRuslan Bukin 	XCHAN_LOCK(xchan);
2443d5b3b0aSRuslan Bukin 	ret = XDMA_CHANNEL_REQUEST(xdma->dma_dev, xchan, req);
24585debf7fSRuslan Bukin 	if (ret != 0) {
24685debf7fSRuslan Bukin 		device_printf(xdma->dev,
2473d5b3b0aSRuslan Bukin 		    "%s: Can't request a transfer.\n", __func__);
2483d5b3b0aSRuslan Bukin 		XCHAN_UNLOCK(xchan);
2493d5b3b0aSRuslan Bukin 
25085debf7fSRuslan Bukin 		return (-1);
25185debf7fSRuslan Bukin 	}
2523d5b3b0aSRuslan Bukin 	XCHAN_UNLOCK(xchan);
25385debf7fSRuslan Bukin 
25485debf7fSRuslan Bukin 	return (0);
25585debf7fSRuslan Bukin }
25685debf7fSRuslan Bukin 
25785debf7fSRuslan Bukin int
2583d5b3b0aSRuslan Bukin xdma_control(xdma_channel_t *xchan, enum xdma_command cmd)
25985debf7fSRuslan Bukin {
26085debf7fSRuslan Bukin 	xdma_controller_t *xdma;
26185debf7fSRuslan Bukin 	int ret;
26285debf7fSRuslan Bukin 
26385debf7fSRuslan Bukin 	xdma = xchan->xdma;
26485debf7fSRuslan Bukin 	KASSERT(xdma != NULL, ("xdma is NULL"));
26585debf7fSRuslan Bukin 
2663d5b3b0aSRuslan Bukin 	ret = XDMA_CHANNEL_CONTROL(xdma->dma_dev, xchan, cmd);
26785debf7fSRuslan Bukin 	if (ret != 0) {
26885debf7fSRuslan Bukin 		device_printf(xdma->dev,
2693d5b3b0aSRuslan Bukin 		    "%s: Can't process command.\n", __func__);
27085debf7fSRuslan Bukin 		return (-1);
27185debf7fSRuslan Bukin 	}
27285debf7fSRuslan Bukin 
27385debf7fSRuslan Bukin 	return (0);
27485debf7fSRuslan Bukin }
27585debf7fSRuslan Bukin 
27685debf7fSRuslan Bukin void
2773d5b3b0aSRuslan Bukin xdma_callback(xdma_channel_t *xchan, xdma_transfer_status_t *status)
27885debf7fSRuslan Bukin {
2793d5b3b0aSRuslan Bukin 	struct xdma_intr_handler *ih_tmp;
2803d5b3b0aSRuslan Bukin 	struct xdma_intr_handler *ih;
2813d5b3b0aSRuslan Bukin 	xdma_controller_t *xdma;
28285debf7fSRuslan Bukin 
2833d5b3b0aSRuslan Bukin 	xdma = xchan->xdma;
2843d5b3b0aSRuslan Bukin 	KASSERT(xdma != NULL, ("xdma is NULL"));
2853d5b3b0aSRuslan Bukin 
2863d5b3b0aSRuslan Bukin 	TAILQ_FOREACH_SAFE(ih, &xchan->ie_handlers, ih_next, ih_tmp)
2873d5b3b0aSRuslan Bukin 		if (ih->cb != NULL)
2883d5b3b0aSRuslan Bukin 			ih->cb(ih->cb_user, status);
2893d5b3b0aSRuslan Bukin 
2903d5b3b0aSRuslan Bukin 	if (xchan->flags & XCHAN_TYPE_SG)
2913d5b3b0aSRuslan Bukin 		xdma_queue_submit(xchan);
29285debf7fSRuslan Bukin }
29385debf7fSRuslan Bukin 
29485debf7fSRuslan Bukin #ifdef FDT
29585debf7fSRuslan Bukin /*
29685debf7fSRuslan Bukin  * Notify the DMA driver we have machine-dependent data in FDT.
29785debf7fSRuslan Bukin  */
29885debf7fSRuslan Bukin static int
29985debf7fSRuslan Bukin xdma_ofw_md_data(xdma_controller_t *xdma, pcell_t *cells, int ncells)
30085debf7fSRuslan Bukin {
30185debf7fSRuslan Bukin 	uint32_t ret;
30285debf7fSRuslan Bukin 
3033d5b3b0aSRuslan Bukin 	ret = XDMA_OFW_MD_DATA(xdma->dma_dev,
3043d5b3b0aSRuslan Bukin 	    cells, ncells, (void **)&xdma->data);
30585debf7fSRuslan Bukin 
30685debf7fSRuslan Bukin 	return (ret);
30785debf7fSRuslan Bukin }
30885debf7fSRuslan Bukin 
309101869a8SRuslan Bukin static int
310101869a8SRuslan Bukin xdma_handle_mem_node(vmem_t *vmem, phandle_t memory)
311101869a8SRuslan Bukin {
312101869a8SRuslan Bukin 	pcell_t reg[FDT_REG_CELLS * FDT_MEM_REGIONS];
313101869a8SRuslan Bukin 	pcell_t *regp;
314101869a8SRuslan Bukin 	int addr_cells, size_cells;
315101869a8SRuslan Bukin 	int i, reg_len, ret, tuple_size, tuples;
316f4ab98c5SConrad Meyer 	u_long mem_start, mem_size;
317101869a8SRuslan Bukin 
318101869a8SRuslan Bukin 	if ((ret = fdt_addrsize_cells(OF_parent(memory), &addr_cells,
319101869a8SRuslan Bukin 	    &size_cells)) != 0)
320101869a8SRuslan Bukin 		return (ret);
321101869a8SRuslan Bukin 
322101869a8SRuslan Bukin 	if (addr_cells > 2)
323101869a8SRuslan Bukin 		return (ERANGE);
324101869a8SRuslan Bukin 
325101869a8SRuslan Bukin 	tuple_size = sizeof(pcell_t) * (addr_cells + size_cells);
326101869a8SRuslan Bukin 	reg_len = OF_getproplen(memory, "reg");
327101869a8SRuslan Bukin 	if (reg_len <= 0 || reg_len > sizeof(reg))
328101869a8SRuslan Bukin 		return (ERANGE);
329101869a8SRuslan Bukin 
330101869a8SRuslan Bukin 	if (OF_getprop(memory, "reg", reg, reg_len) <= 0)
331101869a8SRuslan Bukin 		return (ENXIO);
332101869a8SRuslan Bukin 
333101869a8SRuslan Bukin 	tuples = reg_len / tuple_size;
334101869a8SRuslan Bukin 	regp = (pcell_t *)&reg;
335101869a8SRuslan Bukin 	for (i = 0; i < tuples; i++) {
336101869a8SRuslan Bukin 		ret = fdt_data_to_res(regp, addr_cells, size_cells,
337101869a8SRuslan Bukin 		    &mem_start, &mem_size);
338101869a8SRuslan Bukin 		if (ret != 0)
339101869a8SRuslan Bukin 			return (ret);
340101869a8SRuslan Bukin 
341101869a8SRuslan Bukin 		vmem_add(vmem, mem_start, mem_size, 0);
342101869a8SRuslan Bukin 		regp += addr_cells + size_cells;
343101869a8SRuslan Bukin 	}
344101869a8SRuslan Bukin 
345101869a8SRuslan Bukin 	return (0);
346101869a8SRuslan Bukin }
347101869a8SRuslan Bukin 
348101869a8SRuslan Bukin vmem_t *
349101869a8SRuslan Bukin xdma_get_memory(device_t dev)
350101869a8SRuslan Bukin {
351101869a8SRuslan Bukin 	phandle_t mem_node, node;
352101869a8SRuslan Bukin 	pcell_t mem_handle;
353101869a8SRuslan Bukin 	vmem_t *vmem;
354101869a8SRuslan Bukin 
355101869a8SRuslan Bukin 	node = ofw_bus_get_node(dev);
356101869a8SRuslan Bukin 	if (node <= 0) {
357101869a8SRuslan Bukin 		device_printf(dev,
358101869a8SRuslan Bukin 		    "%s called on not ofw based device.\n", __func__);
359101869a8SRuslan Bukin 		return (NULL);
360101869a8SRuslan Bukin 	}
361101869a8SRuslan Bukin 
362101869a8SRuslan Bukin 	if (!OF_hasprop(node, "memory-region"))
363101869a8SRuslan Bukin 		return (NULL);
364101869a8SRuslan Bukin 
365101869a8SRuslan Bukin 	if (OF_getencprop(node, "memory-region", (void *)&mem_handle,
366101869a8SRuslan Bukin 	    sizeof(mem_handle)) <= 0)
367101869a8SRuslan Bukin 		return (NULL);
368101869a8SRuslan Bukin 
369101869a8SRuslan Bukin 	vmem = vmem_create("xDMA vmem", 0, 0, PAGE_SIZE,
370101869a8SRuslan Bukin 	    PAGE_SIZE, M_BESTFIT | M_WAITOK);
371101869a8SRuslan Bukin 	if (vmem == NULL)
372101869a8SRuslan Bukin 		return (NULL);
373101869a8SRuslan Bukin 
374101869a8SRuslan Bukin 	mem_node = OF_node_from_xref(mem_handle);
375101869a8SRuslan Bukin 	if (xdma_handle_mem_node(vmem, mem_node) != 0) {
376101869a8SRuslan Bukin 		vmem_destroy(vmem);
377101869a8SRuslan Bukin 		return (NULL);
378101869a8SRuslan Bukin 	}
379101869a8SRuslan Bukin 
380101869a8SRuslan Bukin 	return (vmem);
381101869a8SRuslan Bukin }
382101869a8SRuslan Bukin 
383101869a8SRuslan Bukin void
384101869a8SRuslan Bukin xdma_put_memory(vmem_t *vmem)
385101869a8SRuslan Bukin {
386101869a8SRuslan Bukin 
387101869a8SRuslan Bukin 	vmem_destroy(vmem);
388101869a8SRuslan Bukin }
389101869a8SRuslan Bukin 
390101869a8SRuslan Bukin void
391101869a8SRuslan Bukin xchan_set_memory(xdma_channel_t *xchan, vmem_t *vmem)
392101869a8SRuslan Bukin {
393101869a8SRuslan Bukin 
394101869a8SRuslan Bukin 	xchan->vmem = vmem;
395101869a8SRuslan Bukin }
396101869a8SRuslan Bukin 
39785debf7fSRuslan Bukin /*
39885debf7fSRuslan Bukin  * Allocate xdma controller.
39985debf7fSRuslan Bukin  */
40085debf7fSRuslan Bukin xdma_controller_t *
40185debf7fSRuslan Bukin xdma_ofw_get(device_t dev, const char *prop)
40285debf7fSRuslan Bukin {
40385debf7fSRuslan Bukin 	phandle_t node, parent;
40485debf7fSRuslan Bukin 	xdma_controller_t *xdma;
40585debf7fSRuslan Bukin 	device_t dma_dev;
40685debf7fSRuslan Bukin 	pcell_t *cells;
40785debf7fSRuslan Bukin 	int ncells;
40885debf7fSRuslan Bukin 	int error;
40985debf7fSRuslan Bukin 	int ndmas;
41085debf7fSRuslan Bukin 	int idx;
41185debf7fSRuslan Bukin 
41285debf7fSRuslan Bukin 	node = ofw_bus_get_node(dev);
4133d5b3b0aSRuslan Bukin 	if (node <= 0)
41485debf7fSRuslan Bukin 		device_printf(dev,
41585debf7fSRuslan Bukin 		    "%s called on not ofw based device.\n", __func__);
41685debf7fSRuslan Bukin 
41785debf7fSRuslan Bukin 	error = ofw_bus_parse_xref_list_get_length(node,
41885debf7fSRuslan Bukin 	    "dmas", "#dma-cells", &ndmas);
41985debf7fSRuslan Bukin 	if (error) {
42085debf7fSRuslan Bukin 		device_printf(dev,
42185debf7fSRuslan Bukin 		    "%s can't get dmas list.\n", __func__);
42285debf7fSRuslan Bukin 		return (NULL);
42385debf7fSRuslan Bukin 	}
42485debf7fSRuslan Bukin 
42585debf7fSRuslan Bukin 	if (ndmas == 0) {
42685debf7fSRuslan Bukin 		device_printf(dev,
42785debf7fSRuslan Bukin 		    "%s dmas list is empty.\n", __func__);
42885debf7fSRuslan Bukin 		return (NULL);
42985debf7fSRuslan Bukin 	}
43085debf7fSRuslan Bukin 
43185debf7fSRuslan Bukin 	error = ofw_bus_find_string_index(node, "dma-names", prop, &idx);
43285debf7fSRuslan Bukin 	if (error != 0) {
43385debf7fSRuslan Bukin 		device_printf(dev,
43485debf7fSRuslan Bukin 		    "%s can't find string index.\n", __func__);
43585debf7fSRuslan Bukin 		return (NULL);
43685debf7fSRuslan Bukin 	}
43785debf7fSRuslan Bukin 
43885debf7fSRuslan Bukin 	error = ofw_bus_parse_xref_list_alloc(node, "dmas", "#dma-cells",
43985debf7fSRuslan Bukin 	    idx, &parent, &ncells, &cells);
44085debf7fSRuslan Bukin 	if (error != 0) {
44185debf7fSRuslan Bukin 		device_printf(dev,
44285debf7fSRuslan Bukin 		    "%s can't get dma device xref.\n", __func__);
44385debf7fSRuslan Bukin 		return (NULL);
44485debf7fSRuslan Bukin 	}
44585debf7fSRuslan Bukin 
44685debf7fSRuslan Bukin 	dma_dev = OF_device_from_xref(parent);
44785debf7fSRuslan Bukin 	if (dma_dev == NULL) {
44885debf7fSRuslan Bukin 		device_printf(dev,
44985debf7fSRuslan Bukin 		    "%s can't get dma device.\n", __func__);
45085debf7fSRuslan Bukin 		return (NULL);
45185debf7fSRuslan Bukin 	}
45285debf7fSRuslan Bukin 
4533d5b3b0aSRuslan Bukin 	xdma = malloc(sizeof(struct xdma_controller),
4543d5b3b0aSRuslan Bukin 	    M_XDMA, M_WAITOK | M_ZERO);
45585debf7fSRuslan Bukin 	xdma->dev = dev;
45685debf7fSRuslan Bukin 	xdma->dma_dev = dma_dev;
45785debf7fSRuslan Bukin 
45885debf7fSRuslan Bukin 	TAILQ_INIT(&xdma->channels);
45985debf7fSRuslan Bukin 
46085debf7fSRuslan Bukin 	xdma_ofw_md_data(xdma, cells, ncells);
46185debf7fSRuslan Bukin 	free(cells, M_OFWPROP);
46285debf7fSRuslan Bukin 
46385debf7fSRuslan Bukin 	return (xdma);
46485debf7fSRuslan Bukin }
46585debf7fSRuslan Bukin #endif
46685debf7fSRuslan Bukin 
46785debf7fSRuslan Bukin /*
46885debf7fSRuslan Bukin  * Free xDMA controller object.
46985debf7fSRuslan Bukin  */
47085debf7fSRuslan Bukin int
47185debf7fSRuslan Bukin xdma_put(xdma_controller_t *xdma)
47285debf7fSRuslan Bukin {
47385debf7fSRuslan Bukin 
47485debf7fSRuslan Bukin 	XDMA_LOCK();
47585debf7fSRuslan Bukin 
47685debf7fSRuslan Bukin 	/* Ensure no channels allocated. */
47785debf7fSRuslan Bukin 	if (!TAILQ_EMPTY(&xdma->channels)) {
47885debf7fSRuslan Bukin 		device_printf(xdma->dev, "%s: Can't free xDMA\n", __func__);
47985debf7fSRuslan Bukin 		return (-1);
48085debf7fSRuslan Bukin 	}
48185debf7fSRuslan Bukin 
48285debf7fSRuslan Bukin 	free(xdma->data, M_DEVBUF);
48385debf7fSRuslan Bukin 	free(xdma, M_XDMA);
48485debf7fSRuslan Bukin 
48585debf7fSRuslan Bukin 	XDMA_UNLOCK();
48685debf7fSRuslan Bukin 
48785debf7fSRuslan Bukin 	return (0);
48885debf7fSRuslan Bukin }
48985debf7fSRuslan Bukin 
49085debf7fSRuslan Bukin static void
49185debf7fSRuslan Bukin xdma_init(void)
49285debf7fSRuslan Bukin {
49385debf7fSRuslan Bukin 
494101869a8SRuslan Bukin 	mtx_init(&xdma_mtx, "xDMA", NULL, MTX_DEF);
49585debf7fSRuslan Bukin }
49685debf7fSRuslan Bukin 
49785debf7fSRuslan Bukin SYSINIT(xdma, SI_SUB_DRIVERS, SI_ORDER_FIRST, xdma_init, NULL);
498