| /* | 
 |  * BCM2835 DMA engine support | 
 |  * | 
 |  * This driver only supports cyclic DMA transfers | 
 |  * as needed for the I2S module. | 
 |  * | 
 |  * Author:      Florian Meier <florian.meier@koalo.de> | 
 |  *              Copyright 2013 | 
 |  * | 
 |  * Based on | 
 |  *	OMAP DMAengine support by Russell King | 
 |  * | 
 |  *	BCM2708 DMA Driver | 
 |  *	Copyright (C) 2010 Broadcom | 
 |  * | 
 |  *	Raspberry Pi PCM I2S ALSA Driver | 
 |  *	Copyright (c) by Phil Poole 2013 | 
 |  * | 
 |  *	MARVELL MMP Peripheral DMA Driver | 
 |  *	Copyright 2012 Marvell International Ltd. | 
 |  * | 
 |  * This program is free software; you can redistribute it and/or modify | 
 |  * it under the terms of the GNU General Public License as published by | 
 |  * the Free Software Foundation; either version 2 of the License, or | 
 |  * (at your option) any later version. | 
 |  * | 
 |  * This program is distributed in the hope that it will be useful, | 
 |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
 |  * GNU General Public License for more details. | 
 |  */ | 
 | #include <linux/dmaengine.h> | 
 | #include <linux/dma-mapping.h> | 
 | #include <linux/dmapool.h> | 
 | #include <linux/err.h> | 
 | #include <linux/init.h> | 
 | #include <linux/interrupt.h> | 
 | #include <linux/list.h> | 
 | #include <linux/module.h> | 
 | #include <linux/platform_device.h> | 
 | #include <linux/slab.h> | 
 | #include <linux/io.h> | 
 | #include <linux/spinlock.h> | 
 | #include <linux/of.h> | 
 | #include <linux/of_dma.h> | 
 |  | 
 | #include "virt-dma.h" | 
 |  | 
 | #define BCM2835_DMA_MAX_DMA_CHAN_SUPPORTED 14 | 
 | #define BCM2835_DMA_CHAN_NAME_SIZE 8 | 
 |  | 
 | struct bcm2835_dmadev { | 
 | 	struct dma_device ddev; | 
 | 	spinlock_t lock; | 
 | 	void __iomem *base; | 
 | 	struct device_dma_parameters dma_parms; | 
 | }; | 
 |  | 
 | struct bcm2835_dma_cb { | 
 | 	uint32_t info; | 
 | 	uint32_t src; | 
 | 	uint32_t dst; | 
 | 	uint32_t length; | 
 | 	uint32_t stride; | 
 | 	uint32_t next; | 
 | 	uint32_t pad[2]; | 
 | }; | 
 |  | 
 | struct bcm2835_cb_entry { | 
 | 	struct bcm2835_dma_cb *cb; | 
 | 	dma_addr_t paddr; | 
 | }; | 
 |  | 
 | struct bcm2835_chan { | 
 | 	struct virt_dma_chan vc; | 
 | 	struct list_head node; | 
 |  | 
 | 	struct dma_slave_config	cfg; | 
 | 	unsigned int dreq; | 
 |  | 
 | 	int ch; | 
 | 	struct bcm2835_desc *desc; | 
 | 	struct dma_pool *cb_pool; | 
 |  | 
 | 	void __iomem *chan_base; | 
 | 	int irq_number; | 
 | 	unsigned int irq_flags; | 
 |  | 
 | 	bool is_lite_channel; | 
 | }; | 
 |  | 
 | struct bcm2835_desc { | 
 | 	struct bcm2835_chan *c; | 
 | 	struct virt_dma_desc vd; | 
 | 	enum dma_transfer_direction dir; | 
 |  | 
 | 	unsigned int frames; | 
 | 	size_t size; | 
 |  | 
 | 	bool cyclic; | 
 |  | 
 | 	struct bcm2835_cb_entry cb_list[]; | 
 | }; | 
 |  | 
 | #define BCM2835_DMA_CS		0x00 | 
 | #define BCM2835_DMA_ADDR	0x04 | 
 | #define BCM2835_DMA_TI		0x08 | 
 | #define BCM2835_DMA_SOURCE_AD	0x0c | 
 | #define BCM2835_DMA_DEST_AD	0x10 | 
 | #define BCM2835_DMA_LEN		0x14 | 
 | #define BCM2835_DMA_STRIDE	0x18 | 
 | #define BCM2835_DMA_NEXTCB	0x1c | 
 | #define BCM2835_DMA_DEBUG	0x20 | 
 |  | 
 | /* DMA CS Control and Status bits */ | 
 | #define BCM2835_DMA_ACTIVE	BIT(0)  /* activate the DMA */ | 
 | #define BCM2835_DMA_END		BIT(1)  /* current CB has ended */ | 
 | #define BCM2835_DMA_INT		BIT(2)  /* interrupt status */ | 
 | #define BCM2835_DMA_DREQ	BIT(3)  /* DREQ state */ | 
 | #define BCM2835_DMA_ISPAUSED	BIT(4)  /* Pause requested or not active */ | 
 | #define BCM2835_DMA_ISHELD	BIT(5)  /* Is held by DREQ flow control */ | 
 | #define BCM2835_DMA_WAITING_FOR_WRITES BIT(6) /* waiting for last | 
 | 					       * AXI-write to ack | 
 | 					       */ | 
 | #define BCM2835_DMA_ERR		BIT(8) | 
 | #define BCM2835_DMA_PRIORITY(x) ((x & 15) << 16) /* AXI priority */ | 
 | #define BCM2835_DMA_PANIC_PRIORITY(x) ((x & 15) << 20) /* panic priority */ | 
 | /* current value of TI.BCM2835_DMA_WAIT_RESP */ | 
 | #define BCM2835_DMA_WAIT_FOR_WRITES BIT(28) | 
 | #define BCM2835_DMA_DIS_DEBUG	BIT(29) /* disable debug pause signal */ | 
 | #define BCM2835_DMA_ABORT	BIT(30) /* Stop current CB, go to next, WO */ | 
 | #define BCM2835_DMA_RESET	BIT(31) /* WO, self clearing */ | 
 |  | 
 | /* Transfer information bits - also bcm2835_cb.info field */ | 
 | #define BCM2835_DMA_INT_EN	BIT(0) | 
 | #define BCM2835_DMA_TDMODE	BIT(1) /* 2D-Mode */ | 
 | #define BCM2835_DMA_WAIT_RESP	BIT(3) /* wait for AXI-write to be acked */ | 
 | #define BCM2835_DMA_D_INC	BIT(4) | 
 | #define BCM2835_DMA_D_WIDTH	BIT(5) /* 128bit writes if set */ | 
 | #define BCM2835_DMA_D_DREQ	BIT(6) /* enable DREQ for destination */ | 
 | #define BCM2835_DMA_D_IGNORE	BIT(7) /* ignore destination writes */ | 
 | #define BCM2835_DMA_S_INC	BIT(8) | 
 | #define BCM2835_DMA_S_WIDTH	BIT(9) /* 128bit writes if set */ | 
 | #define BCM2835_DMA_S_DREQ	BIT(10) /* enable SREQ for source */ | 
 | #define BCM2835_DMA_S_IGNORE	BIT(11) /* ignore source reads - read 0 */ | 
 | #define BCM2835_DMA_BURST_LENGTH(x) ((x & 15) << 12) | 
 | #define BCM2835_DMA_PER_MAP(x)	((x & 31) << 16) /* REQ source */ | 
 | #define BCM2835_DMA_WAIT(x)	((x & 31) << 21) /* add DMA-wait cycles */ | 
 | #define BCM2835_DMA_NO_WIDE_BURSTS BIT(26) /* no 2 beat write bursts */ | 
 |  | 
 | /* debug register bits */ | 
 | #define BCM2835_DMA_DEBUG_LAST_NOT_SET_ERR	BIT(0) | 
 | #define BCM2835_DMA_DEBUG_FIFO_ERR		BIT(1) | 
 | #define BCM2835_DMA_DEBUG_READ_ERR		BIT(2) | 
 | #define BCM2835_DMA_DEBUG_OUTSTANDING_WRITES_SHIFT 4 | 
 | #define BCM2835_DMA_DEBUG_OUTSTANDING_WRITES_BITS 4 | 
 | #define BCM2835_DMA_DEBUG_ID_SHIFT		16 | 
 | #define BCM2835_DMA_DEBUG_ID_BITS		9 | 
 | #define BCM2835_DMA_DEBUG_STATE_SHIFT		16 | 
 | #define BCM2835_DMA_DEBUG_STATE_BITS		9 | 
 | #define BCM2835_DMA_DEBUG_VERSION_SHIFT		25 | 
 | #define BCM2835_DMA_DEBUG_VERSION_BITS		3 | 
 | #define BCM2835_DMA_DEBUG_LITE			BIT(28) | 
 |  | 
 | /* shared registers for all dma channels */ | 
 | #define BCM2835_DMA_INT_STATUS         0xfe0 | 
 | #define BCM2835_DMA_ENABLE             0xff0 | 
 |  | 
 | #define BCM2835_DMA_DATA_TYPE_S8	1 | 
 | #define BCM2835_DMA_DATA_TYPE_S16	2 | 
 | #define BCM2835_DMA_DATA_TYPE_S32	4 | 
 | #define BCM2835_DMA_DATA_TYPE_S128	16 | 
 |  | 
 | /* Valid only for channels 0 - 14, 15 has its own base address */ | 
 | #define BCM2835_DMA_CHAN(n)	((n) << 8) /* Base address */ | 
 | #define BCM2835_DMA_CHANIO(base, n) ((base) + BCM2835_DMA_CHAN(n)) | 
 |  | 
 | /* the max dma length for different channels */ | 
 | #define MAX_DMA_LEN SZ_1G | 
 | #define MAX_LITE_DMA_LEN (SZ_64K - 4) | 
 |  | 
 | static inline size_t bcm2835_dma_max_frame_length(struct bcm2835_chan *c) | 
 | { | 
 | 	/* lite and normal channels have different max frame length */ | 
 | 	return c->is_lite_channel ? MAX_LITE_DMA_LEN : MAX_DMA_LEN; | 
 | } | 
 |  | 
 | /* how many frames of max_len size do we need to transfer len bytes */ | 
 | static inline size_t bcm2835_dma_frames_for_length(size_t len, | 
 | 						   size_t max_len) | 
 | { | 
 | 	return DIV_ROUND_UP(len, max_len); | 
 | } | 
 |  | 
 | static inline struct bcm2835_dmadev *to_bcm2835_dma_dev(struct dma_device *d) | 
 | { | 
 | 	return container_of(d, struct bcm2835_dmadev, ddev); | 
 | } | 
 |  | 
 | static inline struct bcm2835_chan *to_bcm2835_dma_chan(struct dma_chan *c) | 
 | { | 
 | 	return container_of(c, struct bcm2835_chan, vc.chan); | 
 | } | 
 |  | 
 | static inline struct bcm2835_desc *to_bcm2835_dma_desc( | 
 | 		struct dma_async_tx_descriptor *t) | 
 | { | 
 | 	return container_of(t, struct bcm2835_desc, vd.tx); | 
 | } | 
 |  | 
 | static void bcm2835_dma_free_cb_chain(struct bcm2835_desc *desc) | 
 | { | 
 | 	size_t i; | 
 |  | 
 | 	for (i = 0; i < desc->frames; i++) | 
 | 		dma_pool_free(desc->c->cb_pool, desc->cb_list[i].cb, | 
 | 			      desc->cb_list[i].paddr); | 
 |  | 
 | 	kfree(desc); | 
 | } | 
 |  | 
 | static void bcm2835_dma_desc_free(struct virt_dma_desc *vd) | 
 | { | 
 | 	bcm2835_dma_free_cb_chain( | 
 | 		container_of(vd, struct bcm2835_desc, vd)); | 
 | } | 
 |  | 
 | static void bcm2835_dma_create_cb_set_length( | 
 | 	struct bcm2835_chan *chan, | 
 | 	struct bcm2835_dma_cb *control_block, | 
 | 	size_t len, | 
 | 	size_t period_len, | 
 | 	size_t *total_len, | 
 | 	u32 finalextrainfo) | 
 | { | 
 | 	size_t max_len = bcm2835_dma_max_frame_length(chan); | 
 |  | 
 | 	/* set the length taking lite-channel limitations into account */ | 
 | 	control_block->length = min_t(u32, len, max_len); | 
 |  | 
 | 	/* finished if we have no period_length */ | 
 | 	if (!period_len) | 
 | 		return; | 
 |  | 
 | 	/* | 
 | 	 * period_len means: that we need to generate | 
 | 	 * transfers that are terminating at every | 
 | 	 * multiple of period_len - this is typically | 
 | 	 * used to set the interrupt flag in info | 
 | 	 * which is required during cyclic transfers | 
 | 	 */ | 
 |  | 
 | 	/* have we filled in period_length yet? */ | 
 | 	if (*total_len + control_block->length < period_len) { | 
 | 		/* update number of bytes in this period so far */ | 
 | 		*total_len += control_block->length; | 
 | 		return; | 
 | 	} | 
 |  | 
 | 	/* calculate the length that remains to reach period_length */ | 
 | 	control_block->length = period_len - *total_len; | 
 |  | 
 | 	/* reset total_length for next period */ | 
 | 	*total_len = 0; | 
 |  | 
 | 	/* add extrainfo bits in info */ | 
 | 	control_block->info |= finalextrainfo; | 
 | } | 
 |  | 
 | static inline size_t bcm2835_dma_count_frames_for_sg( | 
 | 	struct bcm2835_chan *c, | 
 | 	struct scatterlist *sgl, | 
 | 	unsigned int sg_len) | 
 | { | 
 | 	size_t frames = 0; | 
 | 	struct scatterlist *sgent; | 
 | 	unsigned int i; | 
 | 	size_t plength = bcm2835_dma_max_frame_length(c); | 
 |  | 
 | 	for_each_sg(sgl, sgent, sg_len, i) | 
 | 		frames += bcm2835_dma_frames_for_length( | 
 | 			sg_dma_len(sgent), plength); | 
 |  | 
 | 	return frames; | 
 | } | 
 |  | 
 | /** | 
 |  * bcm2835_dma_create_cb_chain - create a control block and fills data in | 
 |  * | 
 |  * @chan:           the @dma_chan for which we run this | 
 |  * @direction:      the direction in which we transfer | 
 |  * @cyclic:         it is a cyclic transfer | 
 |  * @info:           the default info bits to apply per controlblock | 
 |  * @frames:         number of controlblocks to allocate | 
 |  * @src:            the src address to assign (if the S_INC bit is set | 
 |  *                  in @info, then it gets incremented) | 
 |  * @dst:            the dst address to assign (if the D_INC bit is set | 
 |  *                  in @info, then it gets incremented) | 
 |  * @buf_len:        the full buffer length (may also be 0) | 
 |  * @period_len:     the period length when to apply @finalextrainfo | 
 |  *                  in addition to the last transfer | 
 |  *                  this will also break some control-blocks early | 
 |  * @finalextrainfo: additional bits in last controlblock | 
 |  *                  (or when period_len is reached in case of cyclic) | 
 |  * @gfp:            the GFP flag to use for allocation | 
 |  */ | 
 | static struct bcm2835_desc *bcm2835_dma_create_cb_chain( | 
 | 	struct dma_chan *chan, enum dma_transfer_direction direction, | 
 | 	bool cyclic, u32 info, u32 finalextrainfo, size_t frames, | 
 | 	dma_addr_t src, dma_addr_t dst, size_t buf_len, | 
 | 	size_t period_len, gfp_t gfp) | 
 | { | 
 | 	struct bcm2835_chan *c = to_bcm2835_dma_chan(chan); | 
 | 	size_t len = buf_len, total_len; | 
 | 	size_t frame; | 
 | 	struct bcm2835_desc *d; | 
 | 	struct bcm2835_cb_entry *cb_entry; | 
 | 	struct bcm2835_dma_cb *control_block; | 
 |  | 
 | 	if (!frames) | 
 | 		return NULL; | 
 |  | 
 | 	/* allocate and setup the descriptor. */ | 
 | 	d = kzalloc(sizeof(*d) + frames * sizeof(struct bcm2835_cb_entry), | 
 | 		    gfp); | 
 | 	if (!d) | 
 | 		return NULL; | 
 |  | 
 | 	d->c = c; | 
 | 	d->dir = direction; | 
 | 	d->cyclic = cyclic; | 
 |  | 
 | 	/* | 
 | 	 * Iterate over all frames, create a control block | 
 | 	 * for each frame and link them together. | 
 | 	 */ | 
 | 	for (frame = 0, total_len = 0; frame < frames; d->frames++, frame++) { | 
 | 		cb_entry = &d->cb_list[frame]; | 
 | 		cb_entry->cb = dma_pool_alloc(c->cb_pool, gfp, | 
 | 					      &cb_entry->paddr); | 
 | 		if (!cb_entry->cb) | 
 | 			goto error_cb; | 
 |  | 
 | 		/* fill in the control block */ | 
 | 		control_block = cb_entry->cb; | 
 | 		control_block->info = info; | 
 | 		control_block->src = src; | 
 | 		control_block->dst = dst; | 
 | 		control_block->stride = 0; | 
 | 		control_block->next = 0; | 
 | 		/* set up length in control_block if requested */ | 
 | 		if (buf_len) { | 
 | 			/* calculate length honoring period_length */ | 
 | 			bcm2835_dma_create_cb_set_length( | 
 | 				c, control_block, | 
 | 				len, period_len, &total_len, | 
 | 				cyclic ? finalextrainfo : 0); | 
 |  | 
 | 			/* calculate new remaining length */ | 
 | 			len -= control_block->length; | 
 | 		} | 
 |  | 
 | 		/* link this the last controlblock */ | 
 | 		if (frame) | 
 | 			d->cb_list[frame - 1].cb->next = cb_entry->paddr; | 
 |  | 
 | 		/* update src and dst and length */ | 
 | 		if (src && (info & BCM2835_DMA_S_INC)) | 
 | 			src += control_block->length; | 
 | 		if (dst && (info & BCM2835_DMA_D_INC)) | 
 | 			dst += control_block->length; | 
 |  | 
 | 		/* Length of total transfer */ | 
 | 		d->size += control_block->length; | 
 | 	} | 
 |  | 
 | 	/* the last frame requires extra flags */ | 
 | 	d->cb_list[d->frames - 1].cb->info |= finalextrainfo; | 
 |  | 
 | 	/* detect a size missmatch */ | 
 | 	if (buf_len && (d->size != buf_len)) | 
 | 		goto error_cb; | 
 |  | 
 | 	return d; | 
 | error_cb: | 
 | 	bcm2835_dma_free_cb_chain(d); | 
 |  | 
 | 	return NULL; | 
 | } | 
 |  | 
 | static void bcm2835_dma_fill_cb_chain_with_sg( | 
 | 	struct dma_chan *chan, | 
 | 	enum dma_transfer_direction direction, | 
 | 	struct bcm2835_cb_entry *cb, | 
 | 	struct scatterlist *sgl, | 
 | 	unsigned int sg_len) | 
 | { | 
 | 	struct bcm2835_chan *c = to_bcm2835_dma_chan(chan); | 
 | 	size_t len, max_len; | 
 | 	unsigned int i; | 
 | 	dma_addr_t addr; | 
 | 	struct scatterlist *sgent; | 
 |  | 
 | 	max_len = bcm2835_dma_max_frame_length(c); | 
 | 	for_each_sg(sgl, sgent, sg_len, i) { | 
 | 		for (addr = sg_dma_address(sgent), len = sg_dma_len(sgent); | 
 | 		     len > 0; | 
 | 		     addr += cb->cb->length, len -= cb->cb->length, cb++) { | 
 | 			if (direction == DMA_DEV_TO_MEM) | 
 | 				cb->cb->dst = addr; | 
 | 			else | 
 | 				cb->cb->src = addr; | 
 | 			cb->cb->length = min(len, max_len); | 
 | 		} | 
 | 	} | 
 | } | 
 |  | 
 | static int bcm2835_dma_abort(struct bcm2835_chan *c) | 
 | { | 
 | 	void __iomem *chan_base = c->chan_base; | 
 | 	long int timeout = 10000; | 
 |  | 
 | 	/* | 
 | 	 * A zero control block address means the channel is idle. | 
 | 	 * (The ACTIVE flag in the CS register is not a reliable indicator.) | 
 | 	 */ | 
 | 	if (!readl(chan_base + BCM2835_DMA_ADDR)) | 
 | 		return 0; | 
 |  | 
 | 	/* Write 0 to the active bit - Pause the DMA */ | 
 | 	writel(0, chan_base + BCM2835_DMA_CS); | 
 |  | 
 | 	/* Wait for any current AXI transfer to complete */ | 
 | 	while ((readl(chan_base + BCM2835_DMA_CS) & | 
 | 		BCM2835_DMA_WAITING_FOR_WRITES) && --timeout) | 
 | 		cpu_relax(); | 
 |  | 
 | 	/* Peripheral might be stuck and fail to signal AXI write responses */ | 
 | 	if (!timeout) | 
 | 		dev_err(c->vc.chan.device->dev, | 
 | 			"failed to complete outstanding writes\n"); | 
 |  | 
 | 	writel(BCM2835_DMA_RESET, chan_base + BCM2835_DMA_CS); | 
 | 	return 0; | 
 | } | 
 |  | 
 | static void bcm2835_dma_start_desc(struct bcm2835_chan *c) | 
 | { | 
 | 	struct virt_dma_desc *vd = vchan_next_desc(&c->vc); | 
 | 	struct bcm2835_desc *d; | 
 |  | 
 | 	if (!vd) { | 
 | 		c->desc = NULL; | 
 | 		return; | 
 | 	} | 
 |  | 
 | 	list_del(&vd->node); | 
 |  | 
 | 	c->desc = d = to_bcm2835_dma_desc(&vd->tx); | 
 |  | 
 | 	writel(d->cb_list[0].paddr, c->chan_base + BCM2835_DMA_ADDR); | 
 | 	writel(BCM2835_DMA_ACTIVE, c->chan_base + BCM2835_DMA_CS); | 
 | } | 
 |  | 
 | static irqreturn_t bcm2835_dma_callback(int irq, void *data) | 
 | { | 
 | 	struct bcm2835_chan *c = data; | 
 | 	struct bcm2835_desc *d; | 
 | 	unsigned long flags; | 
 |  | 
 | 	/* check the shared interrupt */ | 
 | 	if (c->irq_flags & IRQF_SHARED) { | 
 | 		/* check if the interrupt is enabled */ | 
 | 		flags = readl(c->chan_base + BCM2835_DMA_CS); | 
 | 		/* if not set then we are not the reason for the irq */ | 
 | 		if (!(flags & BCM2835_DMA_INT)) | 
 | 			return IRQ_NONE; | 
 | 	} | 
 |  | 
 | 	spin_lock_irqsave(&c->vc.lock, flags); | 
 |  | 
 | 	/* | 
 | 	 * Clear the INT flag to receive further interrupts. Keep the channel | 
 | 	 * active in case the descriptor is cyclic or in case the client has | 
 | 	 * already terminated the descriptor and issued a new one. (May happen | 
 | 	 * if this IRQ handler is threaded.) If the channel is finished, it | 
 | 	 * will remain idle despite the ACTIVE flag being set. | 
 | 	 */ | 
 | 	writel(BCM2835_DMA_INT | BCM2835_DMA_ACTIVE, | 
 | 	       c->chan_base + BCM2835_DMA_CS); | 
 |  | 
 | 	d = c->desc; | 
 |  | 
 | 	if (d) { | 
 | 		if (d->cyclic) { | 
 | 			/* call the cyclic callback */ | 
 | 			vchan_cyclic_callback(&d->vd); | 
 | 		} else if (!readl(c->chan_base + BCM2835_DMA_ADDR)) { | 
 | 			vchan_cookie_complete(&c->desc->vd); | 
 | 			bcm2835_dma_start_desc(c); | 
 | 		} | 
 | 	} | 
 |  | 
 | 	spin_unlock_irqrestore(&c->vc.lock, flags); | 
 |  | 
 | 	return IRQ_HANDLED; | 
 | } | 
 |  | 
 | static int bcm2835_dma_alloc_chan_resources(struct dma_chan *chan) | 
 | { | 
 | 	struct bcm2835_chan *c = to_bcm2835_dma_chan(chan); | 
 | 	struct device *dev = c->vc.chan.device->dev; | 
 |  | 
 | 	dev_dbg(dev, "Allocating DMA channel %d\n", c->ch); | 
 |  | 
 | 	c->cb_pool = dma_pool_create(dev_name(dev), dev, | 
 | 				     sizeof(struct bcm2835_dma_cb), 0, 0); | 
 | 	if (!c->cb_pool) { | 
 | 		dev_err(dev, "unable to allocate descriptor pool\n"); | 
 | 		return -ENOMEM; | 
 | 	} | 
 |  | 
 | 	return request_irq(c->irq_number, bcm2835_dma_callback, | 
 | 			   c->irq_flags, "DMA IRQ", c); | 
 | } | 
 |  | 
 | static void bcm2835_dma_free_chan_resources(struct dma_chan *chan) | 
 | { | 
 | 	struct bcm2835_chan *c = to_bcm2835_dma_chan(chan); | 
 |  | 
 | 	vchan_free_chan_resources(&c->vc); | 
 | 	free_irq(c->irq_number, c); | 
 | 	dma_pool_destroy(c->cb_pool); | 
 |  | 
 | 	dev_dbg(c->vc.chan.device->dev, "Freeing DMA channel %u\n", c->ch); | 
 | } | 
 |  | 
 | static size_t bcm2835_dma_desc_size(struct bcm2835_desc *d) | 
 | { | 
 | 	return d->size; | 
 | } | 
 |  | 
 | static size_t bcm2835_dma_desc_size_pos(struct bcm2835_desc *d, dma_addr_t addr) | 
 | { | 
 | 	unsigned int i; | 
 | 	size_t size; | 
 |  | 
 | 	for (size = i = 0; i < d->frames; i++) { | 
 | 		struct bcm2835_dma_cb *control_block = d->cb_list[i].cb; | 
 | 		size_t this_size = control_block->length; | 
 | 		dma_addr_t dma; | 
 |  | 
 | 		if (d->dir == DMA_DEV_TO_MEM) | 
 | 			dma = control_block->dst; | 
 | 		else | 
 | 			dma = control_block->src; | 
 |  | 
 | 		if (size) | 
 | 			size += this_size; | 
 | 		else if (addr >= dma && addr < dma + this_size) | 
 | 			size += dma + this_size - addr; | 
 | 	} | 
 |  | 
 | 	return size; | 
 | } | 
 |  | 
 | static enum dma_status bcm2835_dma_tx_status(struct dma_chan *chan, | 
 | 	dma_cookie_t cookie, struct dma_tx_state *txstate) | 
 | { | 
 | 	struct bcm2835_chan *c = to_bcm2835_dma_chan(chan); | 
 | 	struct virt_dma_desc *vd; | 
 | 	enum dma_status ret; | 
 | 	unsigned long flags; | 
 |  | 
 | 	ret = dma_cookie_status(chan, cookie, txstate); | 
 | 	if (ret == DMA_COMPLETE || !txstate) | 
 | 		return ret; | 
 |  | 
 | 	spin_lock_irqsave(&c->vc.lock, flags); | 
 | 	vd = vchan_find_desc(&c->vc, cookie); | 
 | 	if (vd) { | 
 | 		txstate->residue = | 
 | 			bcm2835_dma_desc_size(to_bcm2835_dma_desc(&vd->tx)); | 
 | 	} else if (c->desc && c->desc->vd.tx.cookie == cookie) { | 
 | 		struct bcm2835_desc *d = c->desc; | 
 | 		dma_addr_t pos; | 
 |  | 
 | 		if (d->dir == DMA_MEM_TO_DEV) | 
 | 			pos = readl(c->chan_base + BCM2835_DMA_SOURCE_AD); | 
 | 		else if (d->dir == DMA_DEV_TO_MEM) | 
 | 			pos = readl(c->chan_base + BCM2835_DMA_DEST_AD); | 
 | 		else | 
 | 			pos = 0; | 
 |  | 
 | 		txstate->residue = bcm2835_dma_desc_size_pos(d, pos); | 
 | 	} else { | 
 | 		txstate->residue = 0; | 
 | 	} | 
 |  | 
 | 	spin_unlock_irqrestore(&c->vc.lock, flags); | 
 |  | 
 | 	return ret; | 
 | } | 
 |  | 
 | static void bcm2835_dma_issue_pending(struct dma_chan *chan) | 
 | { | 
 | 	struct bcm2835_chan *c = to_bcm2835_dma_chan(chan); | 
 | 	unsigned long flags; | 
 |  | 
 | 	spin_lock_irqsave(&c->vc.lock, flags); | 
 | 	if (vchan_issue_pending(&c->vc) && !c->desc) | 
 | 		bcm2835_dma_start_desc(c); | 
 |  | 
 | 	spin_unlock_irqrestore(&c->vc.lock, flags); | 
 | } | 
 |  | 
 | static struct dma_async_tx_descriptor *bcm2835_dma_prep_dma_memcpy( | 
 | 	struct dma_chan *chan, dma_addr_t dst, dma_addr_t src, | 
 | 	size_t len, unsigned long flags) | 
 | { | 
 | 	struct bcm2835_chan *c = to_bcm2835_dma_chan(chan); | 
 | 	struct bcm2835_desc *d; | 
 | 	u32 info = BCM2835_DMA_D_INC | BCM2835_DMA_S_INC; | 
 | 	u32 extra = BCM2835_DMA_INT_EN | BCM2835_DMA_WAIT_RESP; | 
 | 	size_t max_len = bcm2835_dma_max_frame_length(c); | 
 | 	size_t frames; | 
 |  | 
 | 	/* if src, dst or len is not given return with an error */ | 
 | 	if (!src || !dst || !len) | 
 | 		return NULL; | 
 |  | 
 | 	/* calculate number of frames */ | 
 | 	frames = bcm2835_dma_frames_for_length(len, max_len); | 
 |  | 
 | 	/* allocate the CB chain - this also fills in the pointers */ | 
 | 	d = bcm2835_dma_create_cb_chain(chan, DMA_MEM_TO_MEM, false, | 
 | 					info, extra, frames, | 
 | 					src, dst, len, 0, GFP_KERNEL); | 
 | 	if (!d) | 
 | 		return NULL; | 
 |  | 
 | 	return vchan_tx_prep(&c->vc, &d->vd, flags); | 
 | } | 
 |  | 
 | static struct dma_async_tx_descriptor *bcm2835_dma_prep_slave_sg( | 
 | 	struct dma_chan *chan, | 
 | 	struct scatterlist *sgl, unsigned int sg_len, | 
 | 	enum dma_transfer_direction direction, | 
 | 	unsigned long flags, void *context) | 
 | { | 
 | 	struct bcm2835_chan *c = to_bcm2835_dma_chan(chan); | 
 | 	struct bcm2835_desc *d; | 
 | 	dma_addr_t src = 0, dst = 0; | 
 | 	u32 info = BCM2835_DMA_WAIT_RESP; | 
 | 	u32 extra = BCM2835_DMA_INT_EN; | 
 | 	size_t frames; | 
 |  | 
 | 	if (!is_slave_direction(direction)) { | 
 | 		dev_err(chan->device->dev, | 
 | 			"%s: bad direction?\n", __func__); | 
 | 		return NULL; | 
 | 	} | 
 |  | 
 | 	if (c->dreq != 0) | 
 | 		info |= BCM2835_DMA_PER_MAP(c->dreq); | 
 |  | 
 | 	if (direction == DMA_DEV_TO_MEM) { | 
 | 		if (c->cfg.src_addr_width != DMA_SLAVE_BUSWIDTH_4_BYTES) | 
 | 			return NULL; | 
 | 		src = c->cfg.src_addr; | 
 | 		info |= BCM2835_DMA_S_DREQ | BCM2835_DMA_D_INC; | 
 | 	} else { | 
 | 		if (c->cfg.dst_addr_width != DMA_SLAVE_BUSWIDTH_4_BYTES) | 
 | 			return NULL; | 
 | 		dst = c->cfg.dst_addr; | 
 | 		info |= BCM2835_DMA_D_DREQ | BCM2835_DMA_S_INC; | 
 | 	} | 
 |  | 
 | 	/* count frames in sg list */ | 
 | 	frames = bcm2835_dma_count_frames_for_sg(c, sgl, sg_len); | 
 |  | 
 | 	/* allocate the CB chain */ | 
 | 	d = bcm2835_dma_create_cb_chain(chan, direction, false, | 
 | 					info, extra, | 
 | 					frames, src, dst, 0, 0, | 
 | 					GFP_KERNEL); | 
 | 	if (!d) | 
 | 		return NULL; | 
 |  | 
 | 	/* fill in frames with scatterlist pointers */ | 
 | 	bcm2835_dma_fill_cb_chain_with_sg(chan, direction, d->cb_list, | 
 | 					  sgl, sg_len); | 
 |  | 
 | 	return vchan_tx_prep(&c->vc, &d->vd, flags); | 
 | } | 
 |  | 
 | static struct dma_async_tx_descriptor *bcm2835_dma_prep_dma_cyclic( | 
 | 	struct dma_chan *chan, dma_addr_t buf_addr, size_t buf_len, | 
 | 	size_t period_len, enum dma_transfer_direction direction, | 
 | 	unsigned long flags) | 
 | { | 
 | 	struct bcm2835_chan *c = to_bcm2835_dma_chan(chan); | 
 | 	struct bcm2835_desc *d; | 
 | 	dma_addr_t src, dst; | 
 | 	u32 info = BCM2835_DMA_WAIT_RESP; | 
 | 	u32 extra = BCM2835_DMA_INT_EN; | 
 | 	size_t max_len = bcm2835_dma_max_frame_length(c); | 
 | 	size_t frames; | 
 |  | 
 | 	/* Grab configuration */ | 
 | 	if (!is_slave_direction(direction)) { | 
 | 		dev_err(chan->device->dev, "%s: bad direction?\n", __func__); | 
 | 		return NULL; | 
 | 	} | 
 |  | 
 | 	if (!buf_len) { | 
 | 		dev_err(chan->device->dev, | 
 | 			"%s: bad buffer length (= 0)\n", __func__); | 
 | 		return NULL; | 
 | 	} | 
 |  | 
 | 	/* | 
 | 	 * warn if buf_len is not a multiple of period_len - this may leed | 
 | 	 * to unexpected latencies for interrupts and thus audiable clicks | 
 | 	 */ | 
 | 	if (buf_len % period_len) | 
 | 		dev_warn_once(chan->device->dev, | 
 | 			      "%s: buffer_length (%zd) is not a multiple of period_len (%zd)\n", | 
 | 			      __func__, buf_len, period_len); | 
 |  | 
 | 	/* Setup DREQ channel */ | 
 | 	if (c->dreq != 0) | 
 | 		info |= BCM2835_DMA_PER_MAP(c->dreq); | 
 |  | 
 | 	if (direction == DMA_DEV_TO_MEM) { | 
 | 		if (c->cfg.src_addr_width != DMA_SLAVE_BUSWIDTH_4_BYTES) | 
 | 			return NULL; | 
 | 		src = c->cfg.src_addr; | 
 | 		dst = buf_addr; | 
 | 		info |= BCM2835_DMA_S_DREQ | BCM2835_DMA_D_INC; | 
 | 	} else { | 
 | 		if (c->cfg.dst_addr_width != DMA_SLAVE_BUSWIDTH_4_BYTES) | 
 | 			return NULL; | 
 | 		dst = c->cfg.dst_addr; | 
 | 		src = buf_addr; | 
 | 		info |= BCM2835_DMA_D_DREQ | BCM2835_DMA_S_INC; | 
 | 	} | 
 |  | 
 | 	/* calculate number of frames */ | 
 | 	frames = /* number of periods */ | 
 | 		 DIV_ROUND_UP(buf_len, period_len) * | 
 | 		 /* number of frames per period */ | 
 | 		 bcm2835_dma_frames_for_length(period_len, max_len); | 
 |  | 
 | 	/* | 
 | 	 * allocate the CB chain | 
 | 	 * note that we need to use GFP_NOWAIT, as the ALSA i2s dmaengine | 
 | 	 * implementation calls prep_dma_cyclic with interrupts disabled. | 
 | 	 */ | 
 | 	d = bcm2835_dma_create_cb_chain(chan, direction, true, | 
 | 					info, extra, | 
 | 					frames, src, dst, buf_len, | 
 | 					period_len, GFP_NOWAIT); | 
 | 	if (!d) | 
 | 		return NULL; | 
 |  | 
 | 	/* wrap around into a loop */ | 
 | 	d->cb_list[d->frames - 1].cb->next = d->cb_list[0].paddr; | 
 |  | 
 | 	return vchan_tx_prep(&c->vc, &d->vd, flags); | 
 | } | 
 |  | 
 | static int bcm2835_dma_slave_config(struct dma_chan *chan, | 
 | 				    struct dma_slave_config *cfg) | 
 | { | 
 | 	struct bcm2835_chan *c = to_bcm2835_dma_chan(chan); | 
 |  | 
 | 	if ((cfg->direction == DMA_DEV_TO_MEM && | 
 | 	     cfg->src_addr_width != DMA_SLAVE_BUSWIDTH_4_BYTES) || | 
 | 	    (cfg->direction == DMA_MEM_TO_DEV && | 
 | 	     cfg->dst_addr_width != DMA_SLAVE_BUSWIDTH_4_BYTES) || | 
 | 	    !is_slave_direction(cfg->direction)) { | 
 | 		return -EINVAL; | 
 | 	} | 
 |  | 
 | 	c->cfg = *cfg; | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int bcm2835_dma_terminate_all(struct dma_chan *chan) | 
 | { | 
 | 	struct bcm2835_chan *c = to_bcm2835_dma_chan(chan); | 
 | 	struct bcm2835_dmadev *d = to_bcm2835_dma_dev(c->vc.chan.device); | 
 | 	unsigned long flags; | 
 | 	LIST_HEAD(head); | 
 |  | 
 | 	spin_lock_irqsave(&c->vc.lock, flags); | 
 |  | 
 | 	/* Prevent this channel being scheduled */ | 
 | 	spin_lock(&d->lock); | 
 | 	list_del_init(&c->node); | 
 | 	spin_unlock(&d->lock); | 
 |  | 
 | 	/* stop DMA activity */ | 
 | 	if (c->desc) { | 
 | 		vchan_terminate_vdesc(&c->desc->vd); | 
 | 		c->desc = NULL; | 
 | 		bcm2835_dma_abort(c); | 
 | 	} | 
 |  | 
 | 	vchan_get_all_descriptors(&c->vc, &head); | 
 | 	spin_unlock_irqrestore(&c->vc.lock, flags); | 
 | 	vchan_dma_desc_free_list(&c->vc, &head); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static void bcm2835_dma_synchronize(struct dma_chan *chan) | 
 | { | 
 | 	struct bcm2835_chan *c = to_bcm2835_dma_chan(chan); | 
 |  | 
 | 	vchan_synchronize(&c->vc); | 
 | } | 
 |  | 
 | static int bcm2835_dma_chan_init(struct bcm2835_dmadev *d, int chan_id, | 
 | 				 int irq, unsigned int irq_flags) | 
 | { | 
 | 	struct bcm2835_chan *c; | 
 |  | 
 | 	c = devm_kzalloc(d->ddev.dev, sizeof(*c), GFP_KERNEL); | 
 | 	if (!c) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	c->vc.desc_free = bcm2835_dma_desc_free; | 
 | 	vchan_init(&c->vc, &d->ddev); | 
 | 	INIT_LIST_HEAD(&c->node); | 
 |  | 
 | 	c->chan_base = BCM2835_DMA_CHANIO(d->base, chan_id); | 
 | 	c->ch = chan_id; | 
 | 	c->irq_number = irq; | 
 | 	c->irq_flags = irq_flags; | 
 |  | 
 | 	/* check in DEBUG register if this is a LITE channel */ | 
 | 	if (readl(c->chan_base + BCM2835_DMA_DEBUG) & | 
 | 		BCM2835_DMA_DEBUG_LITE) | 
 | 		c->is_lite_channel = true; | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static void bcm2835_dma_free(struct bcm2835_dmadev *od) | 
 | { | 
 | 	struct bcm2835_chan *c, *next; | 
 |  | 
 | 	list_for_each_entry_safe(c, next, &od->ddev.channels, | 
 | 				 vc.chan.device_node) { | 
 | 		list_del(&c->vc.chan.device_node); | 
 | 		tasklet_kill(&c->vc.task); | 
 | 	} | 
 | } | 
 |  | 
 | static const struct of_device_id bcm2835_dma_of_match[] = { | 
 | 	{ .compatible = "brcm,bcm2835-dma", }, | 
 | 	{}, | 
 | }; | 
 | MODULE_DEVICE_TABLE(of, bcm2835_dma_of_match); | 
 |  | 
 | static struct dma_chan *bcm2835_dma_xlate(struct of_phandle_args *spec, | 
 | 					   struct of_dma *ofdma) | 
 | { | 
 | 	struct bcm2835_dmadev *d = ofdma->of_dma_data; | 
 | 	struct dma_chan *chan; | 
 |  | 
 | 	chan = dma_get_any_slave_channel(&d->ddev); | 
 | 	if (!chan) | 
 | 		return NULL; | 
 |  | 
 | 	/* Set DREQ from param */ | 
 | 	to_bcm2835_dma_chan(chan)->dreq = spec->args[0]; | 
 |  | 
 | 	return chan; | 
 | } | 
 |  | 
 | static int bcm2835_dma_probe(struct platform_device *pdev) | 
 | { | 
 | 	struct bcm2835_dmadev *od; | 
 | 	struct resource *res; | 
 | 	void __iomem *base; | 
 | 	int rc; | 
 | 	int i, j; | 
 | 	int irq[BCM2835_DMA_MAX_DMA_CHAN_SUPPORTED + 1]; | 
 | 	int irq_flags; | 
 | 	uint32_t chans_available; | 
 | 	char chan_name[BCM2835_DMA_CHAN_NAME_SIZE]; | 
 |  | 
 | 	if (!pdev->dev.dma_mask) | 
 | 		pdev->dev.dma_mask = &pdev->dev.coherent_dma_mask; | 
 |  | 
 | 	rc = dma_set_mask_and_coherent(&pdev->dev, DMA_BIT_MASK(32)); | 
 | 	if (rc) { | 
 | 		dev_err(&pdev->dev, "Unable to set DMA mask\n"); | 
 | 		return rc; | 
 | 	} | 
 |  | 
 | 	od = devm_kzalloc(&pdev->dev, sizeof(*od), GFP_KERNEL); | 
 | 	if (!od) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	pdev->dev.dma_parms = &od->dma_parms; | 
 | 	dma_set_max_seg_size(&pdev->dev, 0x3FFFFFFF); | 
 |  | 
 | 	res = platform_get_resource(pdev, IORESOURCE_MEM, 0); | 
 | 	base = devm_ioremap_resource(&pdev->dev, res); | 
 | 	if (IS_ERR(base)) | 
 | 		return PTR_ERR(base); | 
 |  | 
 | 	od->base = base; | 
 |  | 
 | 	dma_cap_set(DMA_SLAVE, od->ddev.cap_mask); | 
 | 	dma_cap_set(DMA_PRIVATE, od->ddev.cap_mask); | 
 | 	dma_cap_set(DMA_CYCLIC, od->ddev.cap_mask); | 
 | 	dma_cap_set(DMA_SLAVE, od->ddev.cap_mask); | 
 | 	dma_cap_set(DMA_MEMCPY, od->ddev.cap_mask); | 
 | 	od->ddev.device_alloc_chan_resources = bcm2835_dma_alloc_chan_resources; | 
 | 	od->ddev.device_free_chan_resources = bcm2835_dma_free_chan_resources; | 
 | 	od->ddev.device_tx_status = bcm2835_dma_tx_status; | 
 | 	od->ddev.device_issue_pending = bcm2835_dma_issue_pending; | 
 | 	od->ddev.device_prep_dma_cyclic = bcm2835_dma_prep_dma_cyclic; | 
 | 	od->ddev.device_prep_slave_sg = bcm2835_dma_prep_slave_sg; | 
 | 	od->ddev.device_prep_dma_memcpy = bcm2835_dma_prep_dma_memcpy; | 
 | 	od->ddev.device_config = bcm2835_dma_slave_config; | 
 | 	od->ddev.device_terminate_all = bcm2835_dma_terminate_all; | 
 | 	od->ddev.device_synchronize = bcm2835_dma_synchronize; | 
 | 	od->ddev.src_addr_widths = BIT(DMA_SLAVE_BUSWIDTH_4_BYTES); | 
 | 	od->ddev.dst_addr_widths = BIT(DMA_SLAVE_BUSWIDTH_4_BYTES); | 
 | 	od->ddev.directions = BIT(DMA_DEV_TO_MEM) | BIT(DMA_MEM_TO_DEV) | | 
 | 			      BIT(DMA_MEM_TO_MEM); | 
 | 	od->ddev.residue_granularity = DMA_RESIDUE_GRANULARITY_BURST; | 
 | 	od->ddev.dev = &pdev->dev; | 
 | 	INIT_LIST_HEAD(&od->ddev.channels); | 
 | 	spin_lock_init(&od->lock); | 
 |  | 
 | 	platform_set_drvdata(pdev, od); | 
 |  | 
 | 	/* Request DMA channel mask from device tree */ | 
 | 	if (of_property_read_u32(pdev->dev.of_node, | 
 | 			"brcm,dma-channel-mask", | 
 | 			&chans_available)) { | 
 | 		dev_err(&pdev->dev, "Failed to get channel mask\n"); | 
 | 		rc = -EINVAL; | 
 | 		goto err_no_dma; | 
 | 	} | 
 |  | 
 | 	/* get irqs for each channel that we support */ | 
 | 	for (i = 0; i <= BCM2835_DMA_MAX_DMA_CHAN_SUPPORTED; i++) { | 
 | 		/* skip masked out channels */ | 
 | 		if (!(chans_available & (1 << i))) { | 
 | 			irq[i] = -1; | 
 | 			continue; | 
 | 		} | 
 |  | 
 | 		/* get the named irq */ | 
 | 		snprintf(chan_name, sizeof(chan_name), "dma%i", i); | 
 | 		irq[i] = platform_get_irq_byname(pdev, chan_name); | 
 | 		if (irq[i] >= 0) | 
 | 			continue; | 
 |  | 
 | 		/* legacy device tree case handling */ | 
 | 		dev_warn_once(&pdev->dev, | 
 | 			      "missing interrupt-names property in device tree - legacy interpretation is used\n"); | 
 | 		/* | 
 | 		 * in case of channel >= 11 | 
 | 		 * use the 11th interrupt and that is shared | 
 | 		 */ | 
 | 		irq[i] = platform_get_irq(pdev, i < 11 ? i : 11); | 
 | 	} | 
 |  | 
 | 	/* get irqs for each channel */ | 
 | 	for (i = 0; i <= BCM2835_DMA_MAX_DMA_CHAN_SUPPORTED; i++) { | 
 | 		/* skip channels without irq */ | 
 | 		if (irq[i] < 0) | 
 | 			continue; | 
 |  | 
 | 		/* check if there are other channels that also use this irq */ | 
 | 		irq_flags = 0; | 
 | 		for (j = 0; j <= BCM2835_DMA_MAX_DMA_CHAN_SUPPORTED; j++) | 
 | 			if ((i != j) && (irq[j] == irq[i])) { | 
 | 				irq_flags = IRQF_SHARED; | 
 | 				break; | 
 | 			} | 
 |  | 
 | 		/* initialize the channel */ | 
 | 		rc = bcm2835_dma_chan_init(od, i, irq[i], irq_flags); | 
 | 		if (rc) | 
 | 			goto err_no_dma; | 
 | 	} | 
 |  | 
 | 	dev_dbg(&pdev->dev, "Initialized %i DMA channels\n", i); | 
 |  | 
 | 	/* Device-tree DMA controller registration */ | 
 | 	rc = of_dma_controller_register(pdev->dev.of_node, | 
 | 			bcm2835_dma_xlate, od); | 
 | 	if (rc) { | 
 | 		dev_err(&pdev->dev, "Failed to register DMA controller\n"); | 
 | 		goto err_no_dma; | 
 | 	} | 
 |  | 
 | 	rc = dma_async_device_register(&od->ddev); | 
 | 	if (rc) { | 
 | 		dev_err(&pdev->dev, | 
 | 			"Failed to register slave DMA engine device: %d\n", rc); | 
 | 		goto err_no_dma; | 
 | 	} | 
 |  | 
 | 	dev_dbg(&pdev->dev, "Load BCM2835 DMA engine driver\n"); | 
 |  | 
 | 	return 0; | 
 |  | 
 | err_no_dma: | 
 | 	bcm2835_dma_free(od); | 
 | 	return rc; | 
 | } | 
 |  | 
 | static int bcm2835_dma_remove(struct platform_device *pdev) | 
 | { | 
 | 	struct bcm2835_dmadev *od = platform_get_drvdata(pdev); | 
 |  | 
 | 	dma_async_device_unregister(&od->ddev); | 
 | 	bcm2835_dma_free(od); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static struct platform_driver bcm2835_dma_driver = { | 
 | 	.probe	= bcm2835_dma_probe, | 
 | 	.remove	= bcm2835_dma_remove, | 
 | 	.driver = { | 
 | 		.name = "bcm2835-dma", | 
 | 		.of_match_table = of_match_ptr(bcm2835_dma_of_match), | 
 | 	}, | 
 | }; | 
 |  | 
 | module_platform_driver(bcm2835_dma_driver); | 
 |  | 
 | MODULE_ALIAS("platform:bcm2835-dma"); | 
 | MODULE_DESCRIPTION("BCM2835 DMA engine driver"); | 
 | MODULE_AUTHOR("Florian Meier <florian.meier@koalo.de>"); | 
 | MODULE_LICENSE("GPL v2"); |