[linux-yocto] [PATCH 02/65] dma: add owl dma driver
Jiang Lu
lu.jiang at windriver.com
Wed Dec 21 01:16:03 PST 2016
From: wurui <wurui at actions-semi.com>
commit b0695faf218afafb617add43741fe39d79f2de98 from
https://github.com/xapp-le/kernel.git
Change-Id: Ib9b94dfc49c300866fe7e17c3e824dac2419d4a3
---
drivers/dma/Kconfig | 7 +
drivers/dma/Makefile | 2 +
drivers/dma/owl_hdmac.c | 1967 ++++++++++++++++++++++++++++++++++++++++++
drivers/dma/owl_hdmac_regs.h | 433 ++++++++++
4 files changed, 2409 insertions(+)
mode change 100644 => 100755 drivers/dma/Kconfig
mode change 100644 => 100755 drivers/dma/Makefile
create mode 100755 drivers/dma/owl_hdmac.c
create mode 100755 drivers/dma/owl_hdmac_regs.h
diff --git a/drivers/dma/Kconfig b/drivers/dma/Kconfig
old mode 100644
new mode 100755
index 703df70..66177d9
--- a/drivers/dma/Kconfig
+++ b/drivers/dma/Kconfig
@@ -63,6 +63,13 @@ config AMBA_PL08X
Platform has a PL08x DMAC device
which can provide DMA engine support
+config OWL_DMAC
+ tristate "OWL AHB DMA support"
+ depends on ARCH_OWL
+ select DMA_ENGINE
+ help
+ Support the actions AHB DMA controller.
+
config INTEL_IOATDMA
tristate "Intel I/OAT DMA support"
depends on PCI && X86
diff --git a/drivers/dma/Makefile b/drivers/dma/Makefile
old mode 100644
new mode 100755
index 9078963..d52e86b
--- a/drivers/dma/Makefile
+++ b/drivers/dma/Makefile
@@ -55,3 +55,5 @@ obj-$(CONFIG_NBPFAXI_DMA) += nbpfaxi.o
obj-$(CONFIG_DMA_SUN6I) += sun6i-dma.o
obj-$(CONFIG_IMG_MDC_DMA) += img-mdc-dma.o
obj-$(CONFIG_XGENE_DMA) += xgene-dma.o
+
+obj-$(CONFIG_OWL_DMAC) += owl_hdmac.o
diff --git a/drivers/dma/owl_hdmac.c b/drivers/dma/owl_hdmac.c
new file mode 100755
index 0000000..3a9df67
--- /dev/null
+++ b/drivers/dma/owl_hdmac.c
@@ -0,0 +1,1967 @@
+/*
+ * Driver for the Atmel AHB DMA Controller (aka HDMA or DMAC on AT91 systems)
+ *
+ * Copyright (C) 2008 Atmel Corporation
+ *
+ * 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 supports the Atmel AHB DMA Controller,
+ *
+ * The driver has currently been tested with the Atmel AT91SAM9RL
+ * and AT91SAM9G45 series.
+ */
+
+
+#include <linux/clk.h>
+#include "dmaengine.h"
+#include <linux/dma-mapping.h>
+#include <linux/dmapool.h>
+#include <linux/interrupt.h>
+#include <linux/module.h>
+#include <linux/platform_device.h>
+#include <linux/slab.h>
+#include <linux/delay.h>
+#include <linux/cpu.h>
+#include <linux/of_device.h>
+#include <mach/powergate.h>
+#include <mach/clkname.h>
+#include <mach/module-owl.h>
+
+#if 1
+#define DMA_CACHE_ADDR
+#endif
+
+#include "owl_hdmac_regs.h"
+
+#define STOP_TIMEOUT 200000
+static unsigned long max_timeout;
+
+#define debug_nand_timeout
+#ifdef debug_nand_timeout
+static unsigned int nand_desc;
+static unsigned long time_dostart;
+static unsigned long time_interrupt;
+static unsigned long time_callback;
+static unsigned long time_submit;
+#endif
+
+//#define enable_one_normal
+#ifdef enable_one_normal
+static unsigned char *dma_normal_buf;
+static dma_addr_t dma_normal_phy;
+#endif
+
+/*
+ * Initial number of descriptors to allocate for each channel. This could
+ * be increased during dma usage.
+ */
+static unsigned int init_nr_desc_per_channel = 64;
+module_param(init_nr_desc_per_channel, uint, 0644);
+MODULE_PARM_DESC(init_nr_desc_per_channel,
+ "initial descriptors per channel (default: 64)");
+
+/* prototypes */
+static dma_cookie_t acts_tx_submit(struct dma_async_tx_descriptor *tx);
+
+static struct acts_desc *acts_first_active(struct owl_dma_chan *atchan)
+{
+ return list_first_entry(&atchan->active_list,
+ struct acts_desc, desc_node);
+}
+
+static struct acts_desc *acts_first_queued(struct owl_dma_chan *atchan)
+{
+ return list_first_entry(&atchan->queue,
+ struct acts_desc, desc_node);
+}
+
+#ifdef DMA_CACHE_ADDR
+static void acts_sync_desc_for_cpu(
+ struct owl_dma_chan *atchan,
+ struct acts_desc *desc)
+{
+ struct acts_desc *child;
+
+ list_for_each_entry(child, &desc->tx_list, desc_node)
+ dma_sync_single_for_cpu(chan2parent(&atchan->chan_common),
+ child->txd.phys, sizeof(child->lli),
+ DMA_TO_DEVICE);
+ dma_sync_single_for_cpu(chan2parent(&atchan->chan_common),
+ desc->txd.phys, sizeof(desc->lli),
+ DMA_TO_DEVICE);
+}
+#endif
+
+/**
+ * acts_alloc_descriptor - allocate and return an initilized descriptor
+ * @chan: the channel to allocate descriptors for
+ * @gfp_flags: GFP allocation flags
+ *
+ * Note: The ack-bit is positioned in the descriptor flag at creation time
+ * to make initial allocation more convenient. This bit will be cleared
+ * and control will be given to client at usage time (during
+ * preparation functions).
+ */
+static struct acts_desc *acts_alloc_descriptor(struct dma_chan *chan,
+ gfp_t gfp_flags)
+{
+ struct acts_desc *desc = NULL;
+#ifndef DMA_CACHE_ADDR
+ struct owl_dma *owl_dma = to_owl_dma(chan->device);
+ dma_addr_t phys;
+#endif
+
+#ifdef DMA_CACHE_ADDR
+
+ desc = kzalloc(sizeof(struct acts_desc), GFP_KERNEL);
+ if (desc) {
+ INIT_LIST_HEAD(&desc->tx_list);
+ dma_async_tx_descriptor_init(&desc->txd, chan);
+ desc->txd.tx_submit = acts_tx_submit;
+ desc->txd.flags = DMA_CTRL_ACK;
+ desc->txd.phys = dma_map_single(chan2parent(chan),
+ &desc->lli, sizeof(desc->lli), DMA_TO_DEVICE);
+ /*atchan_desc_put(atchan, desc);*/
+ }
+#else
+ desc = dma_pool_alloc(owl_dma->dma_desc_pool, gfp_flags, &phys);
+ if (desc) {
+ memset(desc, 0, sizeof(struct acts_desc));
+ INIT_LIST_HEAD(&desc->tx_list);
+ dma_async_tx_descriptor_init(&desc->txd, chan);
+ /* txd.flags will be overwritten in prep functions */
+ desc->txd.flags = DMA_CTRL_ACK;
+ desc->txd.tx_submit = acts_tx_submit;
+ desc->txd.phys = phys;
+ }
+#endif
+
+ return desc;
+}
+
+/**
+ * acts_desc_get - get an unused descriptor from free_list
+ * @atchan: channel we want a new descriptor for
+ */
+static struct acts_desc *acts_desc_get(struct owl_dma_chan *atchan)
+{
+ struct acts_desc *desc, *_desc;
+ struct acts_desc *ret = NULL;
+ unsigned int i = 0;
+ LIST_HEAD(tmp_list);
+ unsigned long flags;
+
+ spin_lock_irqsave(&atchan->lock, flags);
+ list_for_each_entry_safe(desc, _desc, &atchan->free_list, desc_node) {
+ i++;
+ if (async_tx_test_ack(&desc->txd)) {
+ list_del(&desc->desc_node);
+ ret = desc;
+ break;
+ }
+ dev_dbg(chan2dev(&atchan->chan_common),
+ "desc %p not ACKed\n", desc);
+ }
+ spin_unlock_irqrestore(&atchan->lock, flags);
+ dev_vdbg(chan2dev(&atchan->chan_common),
+ "scanned %u descriptors on freelist\n", i);
+
+ /* no more descriptor available in initial pool: create one more */
+ if (!ret) {
+ ret = acts_alloc_descriptor(&atchan->chan_common, GFP_ATOMIC);
+ if (ret) {
+ spin_lock_irqsave(&atchan->lock, flags);
+ atchan->descs_allocated++;
+ spin_unlock_irqrestore(&atchan->lock, flags);
+ } else {
+ dev_err(chan2dev(&atchan->chan_common),
+ "not enough descriptors available\n");
+ }
+ }
+
+ return ret;
+}
+
+/**
+ * acts_desc_put - move a descriptor, including any children, to the free list
+ * @atchan: channel we work on
+ * @desc: descriptor, at the head of a chain, to move to free list
+ */
+static void acts_desc_put(struct owl_dma_chan *atchan, struct acts_desc *desc)
+{
+
+ if (desc) {
+ unsigned long flags;
+
+#ifdef DMA_CACHE_ADDR
+ acts_sync_desc_for_cpu(atchan, desc);
+#endif
+
+ spin_lock_irqsave(&atchan->lock, flags);
+ list_splice_init(&desc->tx_list, &atchan->free_list);
+ list_add(&desc->desc_node, &atchan->free_list);
+ spin_unlock_irqrestore(&atchan->lock, flags);
+ }
+}
+
+/**
+ * atc_desc_chain - build chain adding a descripor
+ * @first: address of first descripor of the chain
+ * @prev: address of previous descripor of the chain
+ * @desc: descriptor to queue
+ *
+ * Called from prep_* functions
+ */
+
+static void acts_desc_chain(struct dma_chan *chan, struct acts_desc **first,
+ struct acts_desc **prev, struct acts_desc *desc)
+{
+ if (!(*first)) {
+ *first = desc;
+ } else {
+ /* inform the HW lli about chaining */
+ (*prev)->lli.dscr = desc->txd.phys;
+#ifdef DMA_CACHE_ADDR
+ dma_sync_single_for_device(chan2parent(chan),
+ (*prev)->txd.phys, sizeof((*prev)->lli),
+ DMA_TO_DEVICE);
+#endif
+ /* insert the link descriptor to the LD ring */
+ list_add_tail(&desc->desc_node,
+ &(*first)->tx_list);
+ }
+ *prev = desc;
+}
+
+/**
+ * acts_dostart - starts the DMA engine for real
+ * @atchan: the channel we want to start
+ * @first: first descriptor in the list we want to begin with
+ *
+ * Called with atchan->lock held and bh disabled
+ */
+static void acts_dostart(struct owl_dma_chan *atchan, struct acts_desc *first)
+{
+ /*struct owl_dma *owl_dma =
+ to_owl_dma(atchan->chan_common.device);*/
+ u32 int_ctl;
+
+ if (acts_chan_is_enabled_dump(atchan)) {
+ dev_err(chan2dev(&atchan->chan_common),
+ "BUG: Attempted to start non-idle channel!\n");
+ return;
+ }
+
+#if defined(VERBOSE_DEBUG)
+ vdbg_dump_regs(atchan);
+#endif
+
+#ifdef DMA_CACHE_ADDR
+ acts_sync_desc_for_cpu(atchan, first);
+#endif
+
+ int_ctl = ACTS_INT_CTL_SUPERBLOCK_INT
+ | ACTS_INT_STATUS_SECURE_ERROR | ACTS_INT_STATUS_ALAINED_ERROR;
+ if (acts_chan_is_cyclic(atchan))
+ int_ctl |= ACTS_INT_STATUS_END_BLOCK_INT;
+
+ channel_writel(atchan, MODE, first->mode);
+ channel_writel(atchan, LINKLIST, ACTS_LINKLIST_SRC_DST_VLD);
+ channel_writel(atchan, NEXT_DESC, first->txd.phys);
+
+ channel_writel(atchan, INT_CTL, int_ctl);
+ channel_writel(atchan, INT_STAT, 0x7f);
+
+ channel_writel(atchan, START, 0x1);
+
+#ifdef debug_nand_timeout
+ if (atchan->descs_allocated == nand_desc)
+ time_dostart = jiffies;
+#endif
+
+// vdbg_dump_regs(atchan);
+}
+
+/**
+ * acts_chain_complete - finish work for one transaction chain
+ * @atchan: channel we work on
+ * @desc: descriptor at the head of the chain we want do complete
+ *
+ * Called with atchan->lock held and bh disabled */
+static void
+acts_chain_complete(struct owl_dma_chan *atchan, struct acts_desc *desc)
+{
+ dma_async_tx_callback callback;
+ void *param;
+ struct dma_async_tx_descriptor *txd = &desc->txd;
+
+ dev_vdbg(chan2dev(&atchan->chan_common),
+ "descriptor %u complete\n", txd->cookie);
+
+ if (!acts_chan_is_cyclic(atchan))
+ dma_cookie_complete(txd);
+
+
+ callback = txd->callback;
+ param = txd->callback_param;
+
+#ifdef DMA_CACHE_ADDR
+ acts_sync_desc_for_cpu(atchan, desc);
+#endif
+
+ /* move children to free_list */
+ list_splice_init(&desc->tx_list, &atchan->free_list);
+ /* move myself to free_list */
+ list_move(&desc->desc_node, &atchan->free_list);
+
+ /*
+ * The API requires that no submissions are done from a
+ * callback, so we don't need to drop the lock here
+ */
+ if (!acts_chan_is_cyclic(atchan)) {
+ dma_async_tx_callback callback = txd->callback;
+ void *param = txd->callback_param;
+
+ /*
+ * The API requires that no submissions are done from a
+ * callback, so we don't need to drop the lock here
+ */
+
+#ifdef debug_nand_timeout
+ if (atchan->descs_allocated == nand_desc)
+ time_callback = jiffies;
+
+ if ((atchan->descs_allocated == nand_desc) &&
+ (jiffies_to_msecs(time_callback - time_submit)
+ > 1000)) {
+ pr_alert("\n\nIF you see this, report it to CaiYu(SH),please!!!\n\n");
+ pr_alert("[NAND-DMA]time elapsed from submit to dostart: %d ms\n",
+ jiffies_to_msecs(time_dostart -
+ time_submit));
+ pr_alert("[NAND-DMA]time elapsed from dostart to interrupt: %d ms\n",
+ jiffies_to_msecs(time_interrupt -
+ time_dostart));
+ pr_alert("[NAND-DMA]time elapsed from interrupt to callback: %d ms\n\n\n",
+ jiffies_to_msecs(time_callback -
+ time_interrupt));
+ }
+#endif
+
+ if (callback)
+ callback(param);
+ }
+
+ dma_run_dependencies(txd);
+}
+
+/**
+ * acts_complete_all - finish work for all transactions
+ * @atchan: channel to complete transactions for
+ *
+ * Eventually submit queued descriptors if any
+ *
+ * Assume channel is idle while calling this function
+ * Called with atchan->lock held and bh disabled
+ */
+static void acts_complete_all(struct owl_dma_chan *atchan)
+{
+ struct acts_desc *desc, *_desc;
+ LIST_HEAD(list);
+
+ dev_vdbg(chan2dev(&atchan->chan_common), "complete all\n");
+
+ BUG_ON(acts_chan_is_enabled_dump(atchan));
+
+ /*
+ * Submit queued descriptors ASAP, i.e. before we go through
+ * the completed ones.
+ */
+ if (!list_empty(&atchan->queue))
+ acts_dostart(atchan, acts_first_queued(atchan));
+ /* empty active_list now it is completed */
+ list_splice_init(&atchan->active_list, &list);
+ /* empty queue list by moving descriptors (if any) to active_list */
+ list_splice_init(&atchan->queue, &atchan->active_list);
+
+ list_for_each_entry_safe(desc, _desc, &list, desc_node)
+ acts_chain_complete(atchan, desc);
+}
+
+/**
+ * acts_cleanup_descriptors - cleanup up finished descriptors in active_list
+ * @atchan: channel to be cleaned up
+ *
+ * Called with atchan->lock held and bh disabled
+ */
+static void acts_cleanup_descriptors(struct owl_dma_chan *atchan)
+{
+ /*struct acts_desc *desc, *_desc;*/
+ /*struct acts_desc *child;*/
+
+ dev_err(chan2dev(&atchan->chan_common), "cleanup descriptors\n");
+ return;
+}
+
+/**
+ * acts_advance_work - at the end of a transaction, move forward
+ * @atchan: channel where the transaction ended
+ *
+ * Called with atchan->lock held and bh disabled
+ */
+static void acts_advance_work(struct owl_dma_chan *atchan)
+{
+ dev_vdbg(chan2dev(&atchan->chan_common), "advance_work\n");
+
+ if (list_empty(&atchan->active_list) ||
+ list_is_singular(&atchan->active_list)) {
+ acts_complete_all(atchan);
+ } else {
+ acts_chain_complete(atchan, acts_first_active(atchan));
+ /* advance work */
+ acts_dostart(atchan, acts_first_active(atchan));
+ }
+}
+
+/**
+ * acts_handle_error - handle errors reported by DMA controller
+ * @atchan: channel where error occurs
+ *
+ * Called with atchan->lock held and bh disabled
+ */
+static void acts_handle_error(struct owl_dma_chan *atchan)
+{
+ struct acts_desc *bad_desc;
+ struct acts_desc *child;
+
+ /*
+ * The descriptor currently at the head of the active list is
+ * broked. Since we don't have any way to report errors, we'll
+ * just have to scream loudly and try to carry on.
+ */
+ bad_desc = acts_first_active(atchan);
+ list_del_init(&bad_desc->desc_node);
+
+ /* As we are stopped, take advantage to push queued descriptors
+ * in active_list */
+ list_splice_init(&atchan->queue, atchan->active_list.prev);
+
+ /* Try to restart the controller */
+ if (!list_empty(&atchan->active_list))
+ acts_dostart(atchan, acts_first_active(atchan));
+
+ /*
+ * KERN_CRITICAL may seem harsh, but since this only happens
+ * when someone submits a bad physical address in a
+ * descriptor, we should consider ourselves lucky that the
+ * controller flagged an error instead of scribbling over
+ * random memory locations.
+ */
+ dev_crit(chan2dev(&atchan->chan_common),
+ "Bad descriptor submitted for DMA!\n");
+ dev_crit(chan2dev(&atchan->chan_common),
+ " cookie: %d\n", bad_desc->txd.cookie);
+ acts_dump_lli(atchan, &bad_desc->lli);
+ list_for_each_entry(child, &bad_desc->tx_list, desc_node)
+ acts_dump_lli(atchan, &child->lli);
+
+ /* Pretend the descriptor completed successfully */
+ acts_chain_complete(atchan, bad_desc);
+}
+
+static void acts_handle_cyclic(struct owl_dma_chan *atchan)
+{
+ struct acts_desc *first = acts_first_active(atchan);
+ struct dma_async_tx_descriptor *txd = &first->txd;
+ dma_async_tx_callback callback = txd->callback;
+ void *param = txd->callback_param;
+
+ if (test_and_clear_bit(ACTS_IS_INTERRUPT, &atchan->status)) {
+ dev_vdbg(chan2dev(&atchan->chan_common),
+ "new cyclic period llp 0x%08x\n",
+ channel_readl(atchan, NEXT_DESC));
+
+ if (NULL != callback)
+ callback(param);
+ }
+}
+
+/*-- IRQ & Tasklet ---------------------------------------------------*/
+
+static void acts_tasklet(unsigned long data)
+{
+ struct owl_dma *owl_dma = (struct owl_dma *)data;
+ struct owl_dma_chan *atchan;
+ int i;
+
+ for (i = 0; i < owl_dma->dma_common.chancnt; i++) {
+ atchan = &owl_dma->chan[i];
+ if (test_bit(ACTS_IS_ERROR, &atchan->status))
+ acts_handle_error(atchan);
+ else if (test_bit(ACTS_IS_CYCLIC, &atchan->status)) {
+ if (list_empty(&atchan->active_list)) {
+ pr_warning("WARNING: %s, active_list is empty!\n",
+ __func__);
+ return;
+ }
+ acts_handle_cyclic(atchan);
+ } else if (test_and_clear_bit(ACTS_IS_INTERRUPT,
+ &atchan->status)) {
+ if (acts_chan_is_enabled_dump(atchan)) {
+ dev_err(chan2dev(&atchan->chan_common),
+ "BUG: channel enabled in tasklet\n");
+ return;
+ }
+ acts_advance_work(atchan);
+ }
+ }
+}
+
+static irqreturn_t owl_dma_interrupt(int irq, void *dev_id)
+{
+ struct owl_dma *owl_dma = (struct owl_dma *)dev_id;
+ struct owl_dma_chan *atchan;
+ int i;
+ u32 status, imr, pending, p0, p1, p2, p3;
+ u32 int_ctl, int_status, channel_pending;
+ int ret = IRQ_NONE;
+
+ imr = dma_readl(owl_dma, IRQEN_0);
+ status = dma_readl(owl_dma, IRQPD_0);
+ p0 = status & imr;
+ imr = dma_readl(owl_dma, IRQEN_1);
+ status = dma_readl(owl_dma, IRQPD_1);
+ p1 = status & imr;
+ imr = dma_readl(owl_dma, IRQEN_2);
+ status = dma_readl(owl_dma, IRQPD_2);
+ p2 = status & imr;
+ imr = dma_readl(owl_dma, IRQEN_3);
+ status = dma_readl(owl_dma, IRQPD_3);
+ p3 = status & imr;
+
+ pending = p0 | p1 | p2 | p3;
+
+ dev_vdbg(owl_dma->dma_common.dev,
+ "interrupt: pending = 0x%08x\n",
+ pending);
+
+ pending &= 0x0fff;
+ if (pending) {
+ for (i = 0; i < owl_dma->dma_common.chancnt; i++) {
+ atchan = &owl_dma->chan[i];
+#ifdef debug_nand_timeout
+ if (atchan->descs_allocated == nand_desc)
+ time_interrupt = jiffies;
+#endif
+ if (pending & ACTS_DMA_PENDING_MASK(i)) {
+ spin_lock(&atchan->lock);
+ int_ctl = channel_readl(atchan, INT_CTL);
+ int_status = channel_readl(atchan, INT_STAT);
+
+ /*clear pending bits ASAP*/
+ channel_writel(atchan, INT_STAT, 0x7f);
+ dma_writel(owl_dma, IRQPD_0, (1 << i));
+ dma_writel(owl_dma, IRQPD_1, (1 << i));
+ dma_writel(owl_dma, IRQPD_2, (1 << i));
+ dma_writel(owl_dma, IRQPD_3, (1 << i));
+ spin_unlock(&atchan->lock);
+
+ channel_pending = int_ctl & int_status;
+ if (channel_pending & 0x60) {
+ set_bit(ACTS_IS_ERROR, &atchan->status);
+ pr_err("dma%d transfer error",
+ atchan->chan_common.chan_id);
+ owl_dma_dump_all(&atchan->chan_common);
+ } else if (channel_pending & 0x1f) {
+ set_bit(ACTS_IS_INTERRUPT,
+ &atchan->status);
+ }
+ }
+ }
+
+ /*owl_dma->pending = pending;*/
+ tasklet_schedule(&owl_dma->tasklet);
+ ret = IRQ_HANDLED;
+ }
+ return ret;
+}
+
+/*-- DMA Engine API --------------------------------------------------*/
+
+/**
+ * acts_tx_submit - set the prepared descriptor(s) to be executed by the engine
+ * @desc: descriptor at the head of the transaction chain
+ *
+ * Queue chain if DMA engine is working already
+ *
+ * Cookie increment and adding to active_list or queue must be atomic
+ */
+static dma_cookie_t acts_tx_submit(struct dma_async_tx_descriptor *tx)
+{
+ struct acts_desc *desc = txd_to_acts_desc(tx);
+ struct owl_dma_chan *atchan = to_owl_dma_chan(tx->chan);
+ dma_cookie_t cookie;
+ unsigned long flags;
+
+#ifdef debug_nand_timeout
+ if (atchan->descs_allocated == nand_desc)
+ time_submit = jiffies;
+#endif
+
+ spin_lock_irqsave(&atchan->lock, flags);
+ cookie = dma_cookie_assign(tx);
+
+ if (list_empty(&atchan->active_list)) {
+ dev_vdbg(chan2dev(tx->chan), "tx_submit: started %u\n",
+ desc->txd.cookie);
+ acts_dostart(atchan, desc);
+ list_add_tail(&desc->desc_node, &atchan->active_list);
+ } else {
+ dev_vdbg(chan2dev(tx->chan), "tx_submit: queued %u\n",
+ desc->txd.cookie);
+ list_add_tail(&desc->desc_node, &atchan->queue);
+ }
+
+ spin_unlock_irqrestore(&atchan->lock, flags);
+
+ return cookie;
+}
+
+static void acts_set_ctrl(struct dma_chan *chan, struct acts_desc *desc,
+ struct owl_dma_slave *atslave, u32 mode, u32 ctrlb)
+{
+ unsigned long acp;
+ u32 int_ctl;
+ struct owl_dma_chan *atchan = to_owl_dma_chan(chan);
+
+ acp = atslave->trans_type >> 2;
+ if ((mode & ACTS_SRC_DST_STRIDE) == 0) { /*no stride mode*/
+ /* frame cnt */
+ desc->lli.ctrla |= ACTS_FRAMECNT(0x1);
+ desc->lli.src_stride = 0;
+ desc->lli.dst_stride = 0;
+ } else if ((mode & ACTS_SRC_DST_STRIDE) == ACTS_DST_ADDR_MODE_STRIDE) {
+ /* frame cnt */
+ desc->lli.ctrla |= ACTS_FRAMECNT(atslave->frame_cnt);
+ desc->lli.src_stride = 0;
+ desc->lli.dst_stride = atslave->dst_stride;
+ } else if ((mode & ACTS_SRC_DST_STRIDE) == ACTS_SRC_ADDR_MODE_STRIDE) {
+ /* frame cnt */
+ desc->lli.ctrla |= ACTS_FRAMECNT(atslave->frame_cnt);
+ desc->lli.src_stride = atslave->src_stride;
+ desc->lli.dst_stride = 0;
+ }
+
+ desc->lli.ctrlb = 0;
+ desc->lli.ctrlb |= ctrlb;
+ desc->lli.ctrlb |= (ACTS_LINKLIST_SRC_VLD
+ |ACTS_LINKLIST_DST_VLD);
+
+ int_ctl = ACTS_INT_CTL_SECURE_INT
+ | ACTS_INT_CTL_SUPERBLOCK_INT | ACTS_INT_CTL_ALAINED_INT;
+ if (acts_chan_is_cyclic(atchan))
+ int_ctl |= ACTS_INT_STATUS_END_BLOCK_INT;
+
+ desc->lli.ctrlc = ACTS_DMAINTCTL(int_ctl);
+ if (acp) {
+ desc->lli.ctrlc |= ACTS_DMAINTCTL(atslave->intctl)
+ |ACTS_ACPCTL_1(atslave->acp_attr)
+ |ACTS_ACPCTL_2(atslave->acp_attr)
+ |ACTS_ACPCTL_3(atslave->acp_attr)
+ |ACTS_ACPCTL_4(atslave->acp_attr);
+ } else
+ desc->lli.ctrlb |= ACTS_LINKLIST_DIS_TYPE1;
+
+ dev_vdbg(chan2dev(chan),
+ "desrricptor: d0x%x s0x%x ctrla0x%x ctrlb0x%x ctrlc0x%x\n",
+ desc->lli.daddr, desc->lli.saddr, desc->lli.ctrla,
+ desc->lli.ctrlb, desc->lli.ctrlc);
+
+ dev_vdbg(chan2dev(chan),
+ "desrricptor: dtr0x%x str0x%x num0x%x next0x%x\n",
+ desc->lli.dst_stride, desc->lli.src_stride, desc->lli.const_num,
+ desc->lli.dscr);
+}
+
+/**
+ * acts_prep_dma_memcpy - prepare a memcpy operation
+ * @chan: the channel to prepare operation on
+ * @dest: operation virtual destination address
+ * @src: operation virtual source address
+ * @len: operation length
+ * @flags: tx descriptor status flags
+ */
+static struct dma_async_tx_descriptor *
+acts_prep_dma_memcpy(struct dma_chan *chan, dma_addr_t dest, dma_addr_t src,
+ size_t len, unsigned long flags)
+{
+ struct owl_dma_chan *atchan = to_owl_dma_chan(chan);
+ struct owl_dma_slave *atslave = chan->private;
+ struct acts_desc *desc = NULL;
+ struct acts_desc *first = NULL;
+ struct acts_desc *prev = NULL;
+ size_t xfer_count;
+ size_t offset;
+ unsigned int src_width;
+ unsigned int dst_width;
+ u32 ctrlb;
+ u32 mode;
+
+ dev_vdbg(chan2dev(chan), "prep_dma_memcpy: d0x%x s0x%x l0x%zx f0x%lx\n",
+ dest, src, len, flags);
+
+ if (unlikely(!len)) {
+ dev_dbg(chan2dev(chan), "prep_dma_memcpy: length is zero!\n");
+ return NULL;
+ }
+
+ if (ACTS_SRAM_ADDR(dest) || ACTS_SRAM_ADDR(src)) {
+ if (len > ACTS_SRAM_SIZE) {
+ dev_err(chan2dev(chan),
+ "txfer len exceed the share ram 0x%x\n", len);
+ return NULL;
+ }
+ }
+
+ flags |= DMA_CTRL_ACK;
+ mode = atslave->mode;
+
+
+ if (!((src | dest | len) & 3))
+ src_width = dst_width = 2;
+ else if (!((src | dest | len) & 1))
+ src_width = dst_width = 1;
+ else
+ src_width = dst_width = 0;
+
+
+ ctrlb = ACTS_LINKLIST_CTRLB
+ | ACTS_SRC_ADDR_MODE_INCR
+ | ACTS_DST_ADDR_MODE_INCR
+ | ACTS_DMAMODE_1(mode)
+ | ACTS_DMAMODE_2(mode)
+ | ACTS_DMAMODE_4(mode)
+ | ACTS_DMAMODE_5(mode);
+
+ /*
+ * We can be a lot more clever here, but this should take care
+ * of the most common optimization.
+ */
+
+ /* set the dma mode and src */
+ for (offset = 0; offset < len; offset += xfer_count << src_width) {
+ xfer_count = min_t(size_t, (len - offset) >> src_width,
+ ACTS_BTSIZE_MAX);
+
+ desc = acts_desc_get(atchan);
+ if (!desc)
+ goto err_desc_get;
+
+ desc->lli.saddr = src + offset;
+ desc->lli.daddr = dest + offset;
+ desc->lli.ctrla = 0;
+ desc->lli.ctrla |= xfer_count << src_width; /*frame_len */
+
+ acts_set_ctrl(chan, desc, atslave, mode, ctrlb);
+
+ desc->txd.cookie = 0;
+ /* async_tx_ack(&desc->txd);*/
+
+ acts_desc_chain(chan, &first, &prev, desc);
+
+ }
+
+ first->mode = mode
+ |ACTS_LINKLIST_CTRLB
+ | SRC_INCR
+ | DST_INCR;
+
+ /* First descriptor of the chain embedds additional information */
+ first->txd.cookie = -EBUSY;
+ first->len = len;
+
+ /* set end-of-link to the last link descriptor of list*/
+ set_desc_eol(desc);
+
+#ifdef DMA_CACHE_ADDR
+ dma_sync_single_for_device(chan2parent(chan),
+ prev->txd.phys, sizeof(prev->lli),
+ DMA_TO_DEVICE);
+#endif
+
+ /*desc->txd.flags = flags; *//* client is in control of this ack */
+ first->txd.flags = flags;
+
+ return &first->txd;
+
+err_desc_get:
+ acts_desc_put(atchan, first);
+ return NULL;
+}
+
+/**
+ * acts_prep_slave_sg - prepare descriptors for a DMA_SLAVE transaction
+ * @chan: DMA channel
+ * @sgl: scatterlist to transfer to/from
+ * @sg_len: number of entries in @scatterlist
+ * @direction: DMA direction
+ * @flags: tx descriptor status flags
+ */
+static struct dma_async_tx_descriptor *
+acts_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 owl_dma_chan *atchan = to_owl_dma_chan(chan);
+ struct owl_dma_slave *atslave = chan->private;
+ struct dma_slave_config *sconfig = &atchan->dma_sconfig;
+
+ struct acts_desc *first = NULL;
+ struct acts_desc *prev = NULL;
+ u32 ctrlb;
+ dma_addr_t reg;
+ /*unsigned int reg_width;*/
+ unsigned int mem_width;
+ unsigned int i;
+ struct scatterlist *sg;
+ u32 mode;
+ size_t total_len = 0;
+
+ dev_vdbg(chan2dev(chan), "prep_slave_sg: %s f0x%lx\n",
+ direction == DMA_MEM_TO_DEV ?
+ "TO DEVICE" : "FROM DEVICE",
+ flags);
+
+ if (unlikely(!atslave || !sg_len)) {
+ dev_dbg(chan2dev(chan), "prep_dma_slave_sg: length is zero!\n");
+ return NULL;
+ }
+
+ flags |= DMA_CTRL_ACK;
+ mode = atslave->mode;
+ /*reg_width = atslave->reg_width;*/
+
+ ctrlb = ACTS_LINKLIST_CTRLB
+ | ACTS_DMAMODE_1(mode)
+ | ACTS_DMAMODE_2(mode)
+ | ACTS_DMAMODE_3(mode)
+ | ACTS_DMAMODE_4(mode)
+ | ACTS_DMAMODE_5(mode);
+
+
+ switch (direction) {
+ case DMA_MEM_TO_DEV:
+
+ reg = sconfig->dst_addr;
+ for_each_sg(sgl, sg, sg_len, i) {
+ struct acts_desc *desc;
+ u32 len;
+ u32 mem;
+
+ desc = acts_desc_get(atchan);
+ if (!desc)
+ goto err_desc_get;
+
+ mem = sg_dma_address(sg);
+ len = sg_dma_len(sg);
+ mem_width = 2;
+ if (unlikely(mem & 3 || len & 3))
+ mem_width = 0;
+
+ desc->lli.saddr = mem;
+ desc->lli.daddr = reg;
+
+#ifdef debug_nand_timeout
+ if (reg > 0xb0210000 && reg < 0xb0220000)
+ nand_desc = atchan->descs_allocated;
+#endif
+
+ desc->lli.ctrla = 0;
+ desc->lli.ctrla |= len;
+
+ acts_set_ctrl(chan, desc, atslave, mode, ctrlb);
+ acts_desc_chain(chan, &first, &prev, desc);
+ total_len += len;
+ }
+ break;
+ case DMA_DEV_TO_MEM:
+
+ reg = sconfig->src_addr;
+ for_each_sg(sgl, sg, sg_len, i) {
+ struct acts_desc *desc;
+ u32 len;
+ u32 mem;
+
+ desc = acts_desc_get(atchan);
+ if (!desc)
+ goto err_desc_get;
+
+ mem = sg_dma_address(sg);
+ len = sg_dma_len(sg);
+ mem_width = 2;
+ if (unlikely(mem & 3 || len & 3))
+ mem_width = 0;
+
+ desc->lli.saddr = reg;
+ desc->lli.daddr = mem;
+
+#ifdef debug_nand_timeout
+ if (reg > 0xb0210000 && reg < 0xb0220000)
+ nand_desc = atchan->descs_allocated;
+#endif
+
+ desc->lli.ctrla = 0;
+ desc->lli.ctrla |= len;
+ acts_set_ctrl(chan, desc, atslave, mode, ctrlb);
+ acts_desc_chain(chan, &first, &prev, desc);
+ total_len += len;
+ }
+ break;
+ default:
+ return NULL;
+ }
+
+ /* set end-of-link to the last link descriptor of list*/
+ set_desc_eol(prev);
+
+#ifdef DMA_CACHE_ADDR
+ dma_sync_single_for_device(chan2parent(chan),
+ prev->txd.phys, sizeof(prev->lli),
+ DMA_TO_DEVICE);
+#endif
+
+ /* First descriptor of the chain embedds additional information */
+ first->txd.cookie = -EBUSY;
+ first->len = total_len;
+ first->mode = mode | ACTS_LINKLIST_CTRLB;
+
+ /* last link descriptor of list is responsible of flags */
+ /*prev->txd.flags = flags;*/ /* client is in control of this ack */
+
+ return &first->txd;
+
+err_desc_get:
+ dev_err(chan2dev(chan), "not enough descriptors available\n");
+ acts_desc_put(atchan, first);
+ return NULL;
+}
+
+/**
+ * owl_dma_cyclic_check_values
+ * Check for too big/unaligned periods and unaligned DMA buffer
+ */
+static int
+owl_dma_cyclic_check_values(unsigned int reg_width, dma_addr_t buf_addr,
+ size_t period_len, enum dma_transfer_direction direction)
+{
+ if (period_len > (ACTS_BTSIZE_MAX << reg_width))
+ goto err_out;
+ if (unlikely(period_len & ((1 << reg_width) - 1)))
+ goto err_out;
+ if (unlikely(buf_addr & ((1 << reg_width) - 1)))
+ goto err_out;
+ if (unlikely(!(direction & (DMA_DEV_TO_MEM | DMA_MEM_TO_DEV))))
+ goto err_out;
+
+ return 0;
+
+err_out:
+ return -EINVAL;
+}
+
+/**
+ * acts_prep_dma_cyclic - prepare the cyclic DMA transfer
+ * @chan: the DMA channel to prepare
+ * @buf_addr: physical DMA address where the buffer starts
+ * @buf_len: total number of bytes for the entire buffer
+ * @period_len: number of bytes for each period
+ * @direction: transfer direction, to or from device
+ * @context: transfer context (ignored)
+ */
+static struct dma_async_tx_descriptor *
+acts_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 owl_dma_chan *atchan = to_owl_dma_chan(chan);
+ struct owl_dma_slave *atslave = chan->private;
+ struct dma_slave_config *sconfig = &atchan->dma_sconfig;
+ struct acts_desc *first = NULL;
+ struct acts_desc *prev = NULL;
+ unsigned long was_cyclic;
+ unsigned int reg_width;
+ unsigned int periods = buf_len / period_len;
+ unsigned int i;
+ u32 mode, ctrlb;
+
+ dev_vdbg(chan2dev(chan), "prep_dma_cyclic: %s buf at 0x%08x - %d (%d/%d)\n",
+ direction == DMA_MEM_TO_DEV ? "TO DEVICE" : "FROM DEVICE",
+ buf_addr,
+ periods, buf_len, period_len);
+
+ if (unlikely(!atslave || !buf_len || !period_len)) {
+ dev_dbg(chan2dev(chan), "prep_dma_cyclic: length is zero!\n");
+ return NULL;
+ }
+
+ was_cyclic = test_and_set_bit(ACTS_IS_CYCLIC, &atchan->status);
+ if (was_cyclic) {
+ dev_dbg(chan2dev(chan), "prep_dma_cyclic: channel in use!\n");
+ return NULL;
+ }
+
+ mode = atslave->mode;
+ ctrlb = ACTS_LINKLIST_CTRLB
+ | ACTS_DMAMODE_1(mode)
+ | ACTS_DMAMODE_2(mode)
+ | ACTS_DMAMODE_3(mode)
+ | ACTS_DMAMODE_4(mode)
+ | ACTS_DMAMODE_5(mode);
+
+ reg_width = (mode & ACTS_MODE_BUS_WIDTH) ? 0 : 2;
+
+ /* Check for too big/unaligned periods and unaligned DMA buffer */
+ if (owl_dma_cyclic_check_values(reg_width, buf_addr,
+ period_len, direction))
+ goto err_out;
+
+ /* build cyclic linked list */
+ for (i = 0; i < periods; i++) {
+ struct acts_desc *desc;
+
+ desc = acts_desc_get(atchan);
+ if (!desc)
+ goto err_desc_get;
+
+ switch (direction) {
+ case DMA_MEM_TO_DEV:
+ desc->lli.saddr = buf_addr + (period_len * i);
+ desc->lli.daddr = sconfig->dst_addr;
+ break;
+
+ case DMA_DEV_TO_MEM:
+ desc->lli.saddr = sconfig->src_addr;
+ desc->lli.daddr = buf_addr + (period_len * i);
+
+ break;
+
+ default:
+ goto err_trans_type;
+ }
+ desc->lli.ctrla = 0;
+ /*desc->lli.ctrla |= period_len >> reg_width; */
+ desc->lli.ctrla |= period_len;
+ acts_set_ctrl(chan, desc, atslave, mode, ctrlb);
+ acts_desc_chain(chan, &first, &prev, desc);
+ }
+
+ /* lets make a cyclic list */
+ prev->lli.dscr = first->txd.phys;
+
+#ifdef DMA_CACHE_ADDR
+ dma_sync_single_for_device(chan2parent(chan),
+ prev->txd.phys, sizeof(prev->lli),
+ DMA_TO_DEVICE);
+#endif
+
+ /* First descriptor of the chain embedds additional information */
+ first->txd.cookie = -EBUSY;
+ first->len = buf_len;
+ first->mode = mode | ACTS_LINKLIST_CTRLB;
+
+ return &first->txd;
+
+err_trans_type:
+ dev_err(chan2dev(chan), "wrong transfer directions\n");
+err_desc_get:
+ dev_err(chan2dev(chan), "not enough descriptors available\n");
+ acts_desc_put(atchan, first);
+err_out:
+ clear_bit(ACTS_IS_CYCLIC, &atchan->status);
+ return NULL;
+}
+
+static int set_runtime_config(struct dma_chan *chan,
+ struct dma_slave_config *sconfig)
+{
+ struct owl_dma_chan *atchan = to_owl_dma_chan(chan);
+
+ /* Check if it is chan is configured for slave transfers */
+ if (!chan->private)
+ return -EINVAL;
+
+ memcpy(&atchan->dma_sconfig, sconfig, sizeof(*sconfig));
+
+ return 0;
+}
+
+static int acts_pause(struct dma_chan *chan)
+{
+ struct owl_dma_chan *atchan = to_owl_dma_chan(chan);
+ unsigned long flags, tmp;
+ unsigned long id = chan->chan_id;
+ struct owl_dma *owl_dma = to_owl_dma(atchan->chan_common.device);
+
+ if (acts_chan_is_paused(atchan))
+ return 0;
+
+ spin_lock_irqsave(&atchan->lock, flags);
+ tmp = dma_readl(owl_dma, DBG_SEL);
+ tmp = tmp | (0x1 << (id + 16));
+ dma_writel(owl_dma, DBG_SEL, tmp);
+ set_bit(ACTS_IS_PAUSED, &atchan->status);
+ spin_unlock_irqrestore(&atchan->lock, flags);
+
+ return 0;
+}
+
+static int acts_resume(struct dma_chan *chan)
+{
+ struct owl_dma_chan *atchan = to_owl_dma_chan(chan);
+ unsigned long flags, tmp;
+ unsigned long id = chan->chan_id;
+ struct owl_dma *owl_dma = to_owl_dma(atchan->chan_common.device);
+
+ if (!acts_chan_is_paused(atchan))
+ return 0;
+
+ spin_lock_irqsave(&atchan->lock, flags);
+ tmp = dma_readl(owl_dma, DBG_SEL);
+ tmp = tmp & ~(0x1 << (id + 16));
+ dma_writel(owl_dma, DBG_SEL, tmp);
+ clear_bit(ACTS_IS_PAUSED, &atchan->status);
+ spin_unlock_irqrestore(&atchan->lock, flags);
+
+ return 0;
+}
+
+static int acts_stop(struct dma_chan *chan)
+{
+ struct owl_dma_chan *atchan = to_owl_dma_chan(chan);
+ struct acts_desc *desc, *_desc;
+ unsigned long flags, tmp, timeout = STOP_TIMEOUT;
+ unsigned long id = chan->chan_id;
+ struct owl_dma *owl_dma = to_owl_dma(atchan->chan_common.device);
+
+ LIST_HEAD(list);
+
+ /*
+ * This is only called when something went wrong elsewhere, so
+ * we don't really care about the data. Just disable the
+ * channel. We still have to poll the channel enable bit due
+ * to AHB/HSB limitations.
+ */
+ spin_lock_irqsave(&atchan->lock, flags);
+ channel_writel(atchan, START, 0);
+ spin_unlock_irqrestore(&atchan->lock, flags);
+ while (!(dma_readl(owl_dma, IDLE_STAT) & (0x1 << id))) {
+ udelay(5);
+ timeout--;
+ if (timeout == 0)
+ goto err;
+ }
+ if (STOP_TIMEOUT - timeout > max_timeout) {
+ max_timeout = STOP_TIMEOUT - timeout;
+ pr_err("dma%d stop wating time is %ld, max timeout %ld\n",
+ chan->chan_id, STOP_TIMEOUT - timeout, max_timeout);
+ }
+
+ dma_readl(owl_dma, IDLE_STAT);
+ dma_readl(owl_dma, IDLE_STAT);
+ dma_readl(owl_dma, IDLE_STAT);
+
+ spin_lock_irqsave(&atchan->lock, flags);
+ channel_writel(atchan, INT_STAT, 0x7f);
+ dma_writel(owl_dma, IRQPD_0, (1 << id));
+ dma_writel(owl_dma, IRQPD_1, (1 << id));
+ dma_writel(owl_dma, IRQPD_2, (1 << id));
+ dma_writel(owl_dma, IRQPD_3, (1 << id));
+
+ /* stop the transfer */
+ /* active_list entries will :end up before queued entries */
+ list_splice_init(&atchan->queue, &list);
+ list_splice_init(&atchan->active_list, &list);
+
+ /* Flush all pending and queued descriptors */
+ list_for_each_entry_safe(desc, _desc, &list, desc_node)
+ acts_chain_complete(atchan, desc);
+
+ tmp = dma_readl(owl_dma, DBG_SEL);
+ tmp = tmp & ~(0x1 << (id + 16));
+ dma_writel(owl_dma, DBG_SEL, tmp);
+ clear_bit(ACTS_IS_PAUSED, &atchan->status);
+ clear_bit(ACTS_IS_CYCLIC, &atchan->status);
+ clear_bit(ACTS_IS_INTERRUPT, &atchan->status);
+ spin_unlock_irqrestore(&atchan->lock, flags);
+
+ return 0;
+
+err:
+ pr_err("dma id %d: channel not stop\n", chan->chan_id);
+ pr_err("%s, calling stack is:\n", __func__);
+ pr_err("\t%pF\n", __builtin_return_address(0));
+
+ owl_dma_dump_all(chan);
+ return -EINVAL;
+}
+
+/*This function should be rewirte when using chained mode to implement cyclic*/
+static int acts_read_remain_frame_cnt(struct owl_dma_chan *atchan)
+{
+ struct acts_desc *first;
+ dma_addr_t cur_saddr, saddr;
+ size_t period_len;
+ int periods;
+ int remain;
+ struct acts_desc *tmp;
+
+ if(list_empty(&atchan->active_list))
+ return 0;
+
+ first = acts_first_active(atchan);
+ saddr = first->lli.saddr;
+
+ periods = 1;/*the only desc in active list also counts*/
+ list_for_each_entry(tmp, &first->tx_list, desc_node)
+ periods++;
+
+ period_len = first->lli.ctrla & 0xfffff;
+
+ /*wurui: avoid atm7059a dma cur_src_ptr fast away from src problem*/
+ while(channel_readl(atchan, CUR_SRC_PTR) - channel_readl(atchan, SRC) > period_len
+ || channel_readl(atchan, CUR_SRC_PTR) - channel_readl(atchan, SRC) < 0);
+
+ cur_saddr = channel_readl(atchan, CUR_SRC_PTR);
+
+ remain = periods - (cur_saddr - saddr) / period_len;
+
+ return remain;
+}
+
+static int acts_config(struct dma_chan *chan,
+ struct dma_slave_config *dmaengine_cfg)
+{
+ return set_runtime_config(chan, dmaengine_cfg);
+}
+
+static int acts_fsldma_external_start0(struct dma_chan *chan)
+{
+ struct owl_dma_chan *atchan = to_owl_dma_chan(chan);
+
+ return vdbg_dump_regs(atchan);
+}
+
+static int acts_fsldma_external_start1(struct dma_chan *chan)
+{
+ struct owl_dma_chan *atchan = to_owl_dma_chan(chan);
+
+ return acts_read_remain_cnt(atchan);
+}
+
+static int acts_fsldma_external_start2(struct dma_chan *chan)
+{
+ struct owl_dma_chan *atchan = to_owl_dma_chan(chan);
+
+ return acts_read_remain_frame_cnt(atchan);
+}
+
+/**
+ * acts_tx_status - poll for transaction completion
+ * @chan: DMA channel
+ * @cookie: transaction identifier to check status of
+ * @txstate: if not %NULL updated with transaction state
+ *
+ * If @txstate is passed in, upon return it reflect the driver
+ * internal state and can be used with dma_async_is_complete() to check
+ * the status of multiple cookies without re-checking hardware state.
+ */
+static enum dma_status
+acts_tx_status(struct dma_chan *chan,
+ dma_cookie_t cookie,
+ struct dma_tx_state *txstate)
+{
+ struct owl_dma_chan *atchan = to_owl_dma_chan(chan);
+ dma_cookie_t last_used;
+ dma_cookie_t last_complete;
+ unsigned long flags;
+ enum dma_status ret;
+
+ spin_lock_irqsave(&atchan->lock, flags);
+
+ ret = dma_cookie_status(chan, cookie, txstate);
+ if (ret != DMA_COMPLETE) {
+ acts_cleanup_descriptors(atchan);
+
+ ret = dma_cookie_status(chan, cookie, txstate);
+ }
+
+ last_complete = chan->completed_cookie;
+ last_used = chan->cookie;
+
+ spin_unlock_irqrestore(&atchan->lock, flags);
+
+ if (ret != DMA_COMPLETE)
+ dma_set_residue(txstate, acts_first_active(atchan)->len);
+
+ if (acts_chan_is_paused(atchan))
+ ret = DMA_PAUSED;
+
+ dev_vdbg(chan2dev(chan), "tx_status: %d (d%d, u%d)\n",
+ cookie, last_complete ? last_complete : 0,
+ last_used ? last_used : 0);
+
+ return ret;
+}
+
+/**
+ * acts_issue_pending - try to finish work
+ * @chan: target DMA channel
+ */
+static void acts_issue_pending(struct dma_chan *chan)
+{
+ struct owl_dma_chan *atchan = to_owl_dma_chan(chan);
+ unsigned long flags;
+
+ dev_vdbg(chan2dev(chan), "issue_pending\n");
+
+ if (acts_chan_is_cyclic(atchan))
+ return;
+
+ spin_lock_irqsave(&atchan->lock, flags);
+ if (!acts_chan_is_enabled(atchan))
+ acts_advance_work(atchan);
+ spin_unlock_irqrestore(&atchan->lock, flags);
+}
+
+/**
+ * acts_alloc_chan_resources - allocate resources for DMA channel
+ * @chan: allocate descriptor resources for this channel
+ * @client: current client requesting the channel be ready for requests
+ *
+ * return - the number of allocated descriptors
+ */
+static int acts_alloc_chan_resources(struct dma_chan *chan)
+{
+ struct owl_dma_chan *atchan = to_owl_dma_chan(chan);
+ struct owl_dma *owl_dma = to_owl_dma(chan->device);
+ struct acts_desc *desc;
+ struct owl_dma_slave *atslave;
+ int i;
+ unsigned long flags;
+ u32 cfg;
+ LIST_HEAD(tmp_list);
+
+ dev_vdbg(chan2dev(chan), "alloc_chan_resources\n");
+
+ /* ASSERT: channel is idle */
+ if (acts_chan_is_enabled(atchan)) {
+ dev_dbg(chan2dev(chan), "DMA channel not idle ?\n");
+ return -EIO;
+ }
+
+ atslave = chan->private;
+ if (atslave) {
+ /*
+ * We need controller-specific data to set up slave
+ * transfers.
+ */
+ BUG_ON(!atslave->dma_dev
+ || atslave->dma_dev != owl_dma->dma_common.dev);
+
+ /* if cfg configuration specified take it instad of default */
+ if (atslave->mode)
+ cfg = atslave->mode;
+ }
+
+ /* have we already been set up?
+ * reconfigure channel but no need to reallocate descriptors */
+ if (!list_empty(&atchan->free_list))
+ return atchan->descs_allocated;
+
+ /* Allocate initial pool of descriptors */
+ for (i = 0; i < init_nr_desc_per_channel; i++) {
+ desc = acts_alloc_descriptor(chan, GFP_KERNEL);
+ if (!desc) {
+ dev_err(owl_dma->dma_common.dev,
+ "Only %d initial descriptors\n", i);
+ break;
+ }
+#ifdef DMA_CACHE_ADDR
+ acts_desc_put(atchan, desc);
+#else
+ list_add_tail(&desc->desc_node, &tmp_list);
+#endif
+ }
+
+ spin_lock_irqsave(&atchan->lock, flags);
+ atchan->descs_allocated = i;
+#ifndef DMA_CACHE_ADDR
+ list_splice(&tmp_list, &atchan->free_list);
+#endif
+
+ dma_cookie_init(chan);
+ channel_writel(atchan, MODE, 0x50a00); /*avoid dev to dev */
+ spin_unlock_irqrestore(&atchan->lock, flags);
+ /* channel parameters */
+
+ dev_dbg(chan2dev(chan),
+ "alloc_chan_resources: allocated %d descriptors\n",
+ atchan->descs_allocated);
+
+ return atchan->descs_allocated;
+}
+
+/**
+ * acts_free_chan_resources - free all channel resources
+ * @chan: DMA channel
+ */
+static void acts_free_chan_resources(struct dma_chan *chan)
+{
+ struct owl_dma_chan *atchan = to_owl_dma_chan(chan);
+#ifndef DMA_CACHE_ADDR
+ struct owl_dma *owl_dma = to_owl_dma(chan->device);
+#endif
+ struct acts_desc *desc, *_desc;
+
+ LIST_HEAD(list);
+
+ dev_dbg(chan2dev(chan), "free_chan_resources: (descs allocated=%u)\n",
+ atchan->descs_allocated);
+
+ if (!list_empty(&atchan->active_list))
+ pr_err("dma:%d free the dirty channel\n", chan->chan_id);
+ /* ASSERT: channel is idle */
+ BUG_ON(acts_chan_is_enabled_dump(atchan));
+ BUG_ON(!list_empty(&atchan->active_list));
+ BUG_ON(!list_empty(&atchan->queue));
+
+ list_for_each_entry_safe(desc, _desc, &atchan->free_list, desc_node) {
+ dev_vdbg(chan2dev(chan), " freeing descriptor %p\n", desc);
+#ifdef DMA_CACHE_ADDR
+ dma_unmap_single(chan2parent(chan), desc->txd.phys,
+ sizeof(desc->lli), DMA_TO_DEVICE);
+ list_del(&desc->desc_node);
+ kfree(desc);
+#else
+ list_del(&desc->desc_node);
+ /* free link descriptor */
+ dma_pool_free(owl_dma->dma_desc_pool, desc, desc->txd.phys);
+#endif
+ }
+ chan->private = NULL;
+
+ list_splice_init(&atchan->free_list, &list);
+ atchan->descs_allocated = 0;
+
+ dev_vdbg(chan2dev(chan), "free_chan_resources: done\n");
+}
+
+/*-- Module Management -----------------------------------------------*/
+
+/**
+ * owl_dma_on - /enable/disable DMA controller
+ * @owl_dma: the Acts HDAMC device
+ */
+static void owl_dma_on(struct owl_dma *owl_dma, bool on)
+{
+
+ if (on) {
+ dma_writel(owl_dma, IRQEN_0, 0xfff);
+ dma_writel(owl_dma, IRQEN_1, 0xfff);
+ dma_writel(owl_dma, IRQEN_2, 0xfff);
+ dma_writel(owl_dma, IRQEN_3, 0xfff);
+ dma_writel(owl_dma, NIC_QOS, 0xf0);
+ } else {
+ dma_writel(owl_dma, IRQEN_0, 0);
+ dma_writel(owl_dma, IRQEN_1, 0);
+ dma_writel(owl_dma, IRQEN_2, 0);
+ dma_writel(owl_dma, IRQEN_3, 0);
+ dma_writel(owl_dma, NIC_QOS, 0);
+ }
+ dma_writel(owl_dma, IRQPD_0, 0xfff);
+ dma_writel(owl_dma, IRQPD_1, 0xfff);
+ dma_writel(owl_dma, IRQPD_2, 0xfff);
+ dma_writel(owl_dma, IRQPD_3, 0xfff);
+}
+
+#ifdef enable_one_normal
+static void enable_one_normal_channel(struct owl_dma_chan *atchan)
+{
+ /*
+ *1. fake device -> DCU
+ *2. DRQ trig source set as an undefined value
+ *3. use DMA channel 11
+ */
+ unsigned int mode = 0x1 << 18 | 0x0 << 16 |
+ 0x2 << 10 | 0x0 << 8 | 0x3f << 0;
+ dma_normal_buf = kzalloc(4, GFP_KERNEL);
+ dma_normal_phy = dma_map_single(NULL, dma_normal_buf,
+ 4, DMA_FROM_DEVICE);
+
+ channel_writel(atchan, MODE, mode);
+ channel_writel(atchan, SRC, 0x0);
+ channel_writel(atchan, DST, dma_normal_phy);
+ channel_writel(atchan, FRAMELEN, 0x4);
+ channel_writel(atchan, FRAMECNT, 0x1);
+ channel_writel(atchan, INT_CTL, 0x1);
+ channel_writel(atchan, START, 0x1);
+}
+#endif
+
+/*
+static void dma_set_affinity(int cpu, int irq)
+{
+ struct cpumask cpumask;
+
+ cpumask_clear(&cpumask);
+ cpumask_set_cpu(cpu, &cpumask);
+
+ irq_set_affinity(irq, &cpumask);
+ irq_set_affinity(irq + 1, &cpumask);
+ irq_set_affinity(irq + 2, &cpumask);
+ irq_set_affinity(irq + 3, &cpumask);
+}
+
+static int dma_irq_temp;
+static int __cpuinit setdmairqaffinity_cpu_callback(struct notifier_block *nfb,
+ unsigned long action, void *hcpu)
+{
+ unsigned int cpu = (unsigned long) hcpu;
+
+ switch (action) {
+ case CPU_ONLINE:
+ if (1 == cpu)
+ dma_set_affinity(1, dma_irq_temp);
+ default:
+ break;
+ }
+ return NOTIFY_OK;
+}
+
+static struct notifier_block setdmairqaffinity_cpu_notifier = {
+ .notifier_call = setdmairqaffinity_cpu_callback,
+};
+*/
+
+static struct owl_dma_platform_data owl_dma_platform_data[] = {
+ [0] = {
+ .nr_channels = 12,
+ .cap_mask = {
+ .bits = { 0x301}
+ },
+ },
+};
+
+static const struct of_device_id owl_dma_of_match[] = {
+ {.compatible = "actions,owl-dma", .data = &owl_dma_platform_data[0]},
+ {}
+};
+MODULE_DEVICE_TABLE(of, owl_dma_of_match);
+
+
+static int __init owl_dma_probe(struct platform_device *pdev)
+{
+ const struct owl_dma_platform_data *pdata;
+ const struct of_device_id *id;
+ struct resource *iores;
+ struct owl_dma *owl_dma;
+ size_t size;
+ int irq;
+ int err;
+ int i;
+
+ printk("owl_dma_probe\n");
+ max_timeout = 1;
+
+ /* get DMA Controller parameters from platform */
+ id = of_match_device(owl_dma_of_match, &pdev->dev);
+ if(id == NULL)
+ {
+ printk("owl dma id is null\n");
+ return -EINVAL;
+ }
+ pdata = id->data;
+ if (!pdata || pdata->nr_channels > ACTS_DMA_MAX_NR_CHANNELS)
+ {
+ printk("owl dma pdata error\n");
+ return -EINVAL;
+ }
+
+ iores = platform_get_resource(pdev, IORESOURCE_MEM, 0);
+ if(iores == NULL)
+ {
+ printk("owl dma iores is null\n");
+ return -EINVAL;
+ }
+
+ irq = platform_get_irq(pdev, 0);
+ if (irq < 0)
+ {
+ printk("owl dma irq get failed\n");
+ return irq;
+ }
+
+ size = sizeof(struct owl_dma);
+#ifdef enable_one_normal
+ size += (pdata->nr_channels - 1) * sizeof(struct owl_dma_chan);
+#else
+ size += pdata->nr_channels * sizeof(struct owl_dma_chan);
+#endif
+ owl_dma = devm_kzalloc(&pdev->dev, size, GFP_KERNEL);
+ if (!owl_dma)
+ return -ENOMEM;
+
+ /* discover transaction capabilites from the platform data */
+ owl_dma->dma_common.cap_mask = pdata->cap_mask;
+#ifdef enable_one_normal
+ owl_dma->all_chan_mask = (1 << (pdata->nr_channels - 1)) - 1;
+#else
+ owl_dma->all_chan_mask = (1 << pdata->nr_channels) - 1;
+#endif
+ owl_dma->id = pdev->id;
+
+ iores = platform_get_resource(pdev, IORESOURCE_MEM, 0);
+ if(iores == NULL)
+ {
+ printk("owl dma iores is null\n");
+ return -EINVAL;
+ }
+
+ owl_dma->regs = devm_ioremap_resource(&pdev->dev, iores);
+ if (IS_ERR(owl_dma->regs))
+ {
+ printk("owl dma iobase remap failed\n");
+ return PTR_ERR(owl_dma->regs);
+ }
+
+ owl_powergate_power_on(OWL_POWERGATE_DMA);
+
+ owl_dma->clk = clk_get(&pdev->dev, "CMUMOD_DMAC");
+ if (IS_ERR(owl_dma->clk)) {
+ err = PTR_ERR(owl_dma->clk);
+ goto err_clk;
+ }
+ clk_enable(owl_dma->clk);
+
+ /* force dma off, just in case */
+ owl_dma_on(owl_dma, 0);
+
+
+ err = request_irq(irq, owl_dma_interrupt, 0, "owl_dma0", owl_dma);
+ if (err)
+ goto err_irq;
+ err = request_irq(irq + 1, owl_dma_interrupt, 0, "owl_dma1", owl_dma);
+ if (err)
+ goto err_irq;
+ err = request_irq(irq + 2, owl_dma_interrupt, 0, "owl_dma2", owl_dma);
+ if (err)
+ goto err_irq;
+ err = request_irq(irq + 3, owl_dma_interrupt, 0, "owl_dma3", owl_dma);
+ if (err)
+ goto err_irq;
+
+ /* dma_set_affinity(1, irq); */
+ owl_dma->irq = irq;
+
+ platform_set_drvdata(pdev, owl_dma);
+
+#ifndef DMA_CACHE_ADDR
+ /* create a pool of consistent memory blocks for hardware descriptors */
+ owl_dma->dma_desc_pool = dma_pool_create("acts_hdmac_desc_pool",
+ &pdev->dev, sizeof(struct acts_desc),
+ 4 /* word alignment */, 0);
+ if (!owl_dma->dma_desc_pool) {
+ dev_err(&pdev->dev, "No memory for descriptors dma pool\n");
+ err = -ENOMEM;
+ goto err_pool_create;
+ }
+#endif
+ /* clear any pending interrupt */
+
+ /* initialize channels related values */
+ INIT_LIST_HEAD(&owl_dma->dma_common.channels);
+#ifdef enable_one_normal
+ for (i = 0; i < (pdata->nr_channels - 1);
+ i++, owl_dma->dma_common.chancnt++)
+#else
+ for (i = 0; i < pdata->nr_channels;
+ i++, owl_dma->dma_common.chancnt++)
+#endif
+ {
+ struct owl_dma_chan *atchan = &owl_dma->chan[i];
+
+ atchan->chan_common.device = &owl_dma->dma_common;
+ dma_cookie_init(&atchan->chan_common);
+ atchan->chan_common.chan_id = i;
+ list_add_tail(&atchan->chan_common.device_node,
+ &owl_dma->dma_common.channels);
+
+ atchan->ch_regs = owl_dma->regs + ch_regs(i);
+ spin_lock_init(&atchan->lock);
+ atchan->mask = 1 << i;
+
+ INIT_LIST_HEAD(&atchan->active_list);
+ INIT_LIST_HEAD(&atchan->queue);
+ INIT_LIST_HEAD(&atchan->free_list);
+
+ acts_enable_irq(atchan);
+ }
+
+ tasklet_init(&owl_dma->tasklet, acts_tasklet,
+ (unsigned long)owl_dma);
+
+ /* set base routines */
+ owl_dma->dma_common.device_alloc_chan_resources =
+ acts_alloc_chan_resources;
+ owl_dma->dma_common.device_free_chan_resources =
+ acts_free_chan_resources;
+ owl_dma->dma_common.device_tx_status = acts_tx_status;
+ owl_dma->dma_common.device_issue_pending = acts_issue_pending;
+ owl_dma->dma_common.dev = &pdev->dev;
+
+ /* set prep routines based on capability */
+ if (dma_has_cap(DMA_MEMCPY, owl_dma->dma_common.cap_mask))
+ owl_dma->dma_common.device_prep_dma_memcpy =
+ acts_prep_dma_memcpy;
+
+ if (dma_has_cap(DMA_SLAVE, owl_dma->dma_common.cap_mask)) {
+ owl_dma->dma_common.device_prep_slave_sg =
+ acts_prep_slave_sg;
+ dma_cap_set(DMA_CYCLIC, owl_dma->dma_common.cap_mask);
+ owl_dma->dma_common.device_prep_dma_cyclic =
+ acts_prep_dma_cyclic;
+ owl_dma->dma_common.device_config = acts_config;
+ owl_dma->dma_common.device_terminate_all = acts_stop;
+ owl_dma->dma_common.device_pause = acts_pause;
+ owl_dma->dma_common.device_resume = acts_resume;
+ owl_dma->dma_common.fsldma_external_start0 = acts_fsldma_external_start0;
+ owl_dma->dma_common.fsldma_external_start1 = acts_fsldma_external_start1;
+ owl_dma->dma_common.fsldma_external_start2 = acts_fsldma_external_start2;
+ }
+
+ owl_dma_on(owl_dma, 1); /* enable dma*/
+
+ dev_info(&pdev->dev,
+ "actions AHB DMA Controller ( %s%s%s), %d channels\n",
+ dma_has_cap(DMA_MEMCPY,
+ owl_dma->dma_common.cap_mask) ? "cpy " : "",
+ dma_has_cap(DMA_SLAVE,
+ owl_dma->dma_common.cap_mask) ? "slave " : "",
+ 0 ? "memset " : "",
+ owl_dma->dma_common.chancnt);
+
+ dma_async_device_register(&owl_dma->dma_common);
+
+#ifdef enable_one_normal
+ enable_one_normal_channel(&owl_dma->chan[owl_dma->dma_common.chancnt-1]);
+#endif
+
+ return 0;
+
+#ifndef DMA_CACHE_ADDR
+err_pool_create:
+#endif
+ platform_set_drvdata(pdev, NULL);
+ free_irq(platform_get_irq(pdev, 0), owl_dma);
+
+err_irq:
+ clk_disable(owl_dma->clk);
+ clk_put(owl_dma->clk);
+err_clk:
+ return err;
+}
+
+static int __exit owl_dma_remove(struct platform_device *pdev)
+{
+ struct owl_dma *owl_dma = platform_get_drvdata(pdev);
+ struct dma_chan *chan, *_chan;
+ struct owl_dma_chan *atchan;
+
+ owl_dma_on(owl_dma, 0);
+ dma_async_device_unregister(&owl_dma->dma_common);
+
+#ifndef DMA_CACHE_ADDR
+ dma_pool_destroy(owl_dma->dma_desc_pool);
+#endif
+ platform_set_drvdata(pdev, NULL);
+ free_irq(platform_get_irq(pdev, 0), owl_dma);
+
+ list_for_each_entry_safe(chan, _chan, &owl_dma->dma_common.channels,
+ device_node) {
+ atchan = to_owl_dma_chan(chan);
+
+ /* Disable interrupts */
+ acts_disable_irq(atchan);
+ list_del(&chan->device_node);
+ }
+
+ tasklet_disable(&owl_dma->tasklet);
+ tasklet_kill(&owl_dma->tasklet);
+
+ clk_disable(owl_dma->clk);
+ clk_put(owl_dma->clk);
+
+
+#ifdef enable_one_normal
+ atchan = &owl_dma->chan[owl_dma->dma_common.chancnt-1];
+ channel_writel(atchan, START, 0x0);
+ dma_unmap_single(NULL, dma_normal_phy, 4, DMA_FROM_DEVICE);
+ kfree(dma_normal_buf);
+#endif
+
+ kfree(owl_dma);
+ return 0;
+}
+
+//static void owl_dma_shutdown(struct platform_device *pdev)
+//{
+// struct owl_dma *owl_dma = platform_get_drvdata(pdev);
+//
+// owl_dma_on(platform_get_drvdata(pdev), 0);
+// clk_disable(owl_dma->clk);
+//}
+
+static void acts_suspend_cyclic(struct owl_dma_chan *atchan)
+{
+ struct dma_chan *chan = &atchan->chan_common;
+
+ /* Channel should be paused by user
+ * do it anyway even if it is not done already */
+ if (!acts_chan_is_paused(atchan)) {
+ dev_warn(chan2dev(chan),
+ "cyclic channel not paused, should be done by channel user\n");
+ acts_pause(chan);
+ }
+
+ /* now preserve additional data for cyclic operations */
+ /* next descriptor address in the cyclic list */
+ atchan->save_dscr = channel_readl(atchan, NEXT_DESC);
+
+ /* vdbg_dump_regs(atchan); */
+}
+
+static void acts_resume_cyclic(struct owl_dma_chan *atchan)
+{
+ int int_ctl;
+
+ /* restore channel status for cyclic descriptors list:
+ * next descriptor in the cyclic list at the time of suspend */
+ channel_writel(atchan, SRC, 0);
+ channel_writel(atchan, DST, 0);
+ channel_writel(atchan, NEXT_DESC, atchan->save_dscr);
+ int_ctl = channel_readl(atchan, INT_CTL);
+ channel_writel(atchan, INT_CTL, int_ctl | ACTS_INT_CTL_END_BLOCK_INT);
+
+ /* channel pause status should be removed by channel user
+ * We cannot take the initiative to do it here */
+
+ /* vdbg_dump_regs(atchan); */
+}
+
+static int owl_dma_suspend_noirq(struct device *dev)
+{
+ struct platform_device *pdev = to_platform_device(dev);
+ struct owl_dma *owl_dma = platform_get_drvdata(pdev);
+ struct dma_chan *chan, *_chan;
+
+ /* preserve data */
+ list_for_each_entry_safe(chan, _chan, &owl_dma->dma_common.channels,
+ device_node) {
+ struct owl_dma_chan *atchan = to_owl_dma_chan(chan);
+
+ /* here, all channels should be idled already
+ * (cyclic channel is paused)
+ */
+ if (acts_chan_is_enabled(atchan)
+ && !acts_chan_is_cyclic(atchan)) {
+ vdbg_dump_regs(atchan);
+ pr_alert("%s:DMA channel%d is not idled yet!!!\n",
+ __func__, atchan->chan_common.chan_id);
+ }
+
+ if (acts_chan_is_cyclic(atchan))
+ acts_suspend_cyclic(atchan);
+ atchan->save_mode = channel_readl(atchan, MODE);
+ atchan->save_ll = channel_readl(atchan, LINKLIST);
+ acts_disable_irq(atchan);
+ }
+ owl_dma->save_nicqos = dma_readl(owl_dma, NIC_QOS);
+
+ /* disable DMA controller */
+ owl_dma_on(owl_dma, 0);
+ clk_disable(owl_dma->clk);
+ return 0;
+}
+
+static int owl_dma_resume_noirq(struct device *dev)
+{
+ struct platform_device *pdev = to_platform_device(dev);
+ struct owl_dma *owl_dma = platform_get_drvdata(pdev);
+ struct dma_chan *chan, *_chan;
+
+ module_reset(MODULE_RST_DMAC);
+
+ /* bring back DMA controller */
+ clk_enable(owl_dma->clk);
+ owl_dma_on(owl_dma, 1);
+
+ /* restore saved data */
+ dma_writel(owl_dma, NIC_QOS, owl_dma->save_nicqos);
+ list_for_each_entry_safe(chan, _chan, &owl_dma->dma_common.channels,
+ device_node) {
+ struct owl_dma_chan *atchan = to_owl_dma_chan(chan);
+
+ acts_enable_irq(atchan);
+ channel_writel(atchan, MODE, atchan->save_mode);
+ channel_writel(atchan, LINKLIST, atchan->save_ll);
+ if (acts_chan_is_cyclic(atchan))
+ acts_resume_cyclic(atchan);
+ }
+ return 0;
+}
+
+static const struct dev_pm_ops owl_dma_dev_pm_ops = {
+ .suspend_noirq = owl_dma_suspend_noirq,
+ .resume_noirq = owl_dma_resume_noirq,
+};
+
+static struct platform_driver __refdata owl_dma_driver = {
+ .driver = {
+ .name = "owl_dma",
+ .pm = &owl_dma_dev_pm_ops,
+ .owner = THIS_MODULE,
+ .of_match_table = owl_dma_of_match,
+ },
+ .probe = owl_dma_probe,
+ .remove = __exit_p(owl_dma_remove),
+// .shutdown = owl_dma_shutdown,
+};
+
+static int __init owl_dma_init(void)
+{
+ int err;
+
+ printk("owl_dma_init\n");
+ err = platform_driver_register(&owl_dma_driver);
+ if (err != 0) {
+ printk("register owl dma platform driver error!\n");
+ return err;
+ }
+
+ return err;
+}
+subsys_initcall(owl_dma_init);
+
+static void __exit owl_dma_exit(void)
+{
+ platform_driver_unregister(&owl_dma_driver);
+}
+module_exit(owl_dma_exit);
+
+MODULE_DESCRIPTION("actions AHB DMA Controller driver");
+MODULE_AUTHOR("mdchen");
+MODULE_LICENSE("GPL");
+MODULE_ALIAS("platform:acts_hdmac");
diff --git a/drivers/dma/owl_hdmac_regs.h b/drivers/dma/owl_hdmac_regs.h
new file mode 100755
index 0000000..1911904
--- /dev/null
+++ b/drivers/dma/owl_hdmac_regs.h
@@ -0,0 +1,433 @@
+/*
+ * Header file for the Atmel AHB DMA Controller driver
+ *
+ * Copyright (C) 2008 Atmel Corporation
+ *
+ * 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.
+ */
+#ifndef ACTS_HDMAC_REGS_H
+#define ACTS_HDMAC_REGS_H
+
+#include <mach/hardware.h>
+#include <mach/hdmac-owl.h>
+
+#define ACTS_DMA_MAX_NR_CHANNELS 12
+
+
+#define ACTS_DMA_IRQPD_0 0x00
+#define ACTS_DMA_IRQPD_1 0x04
+#define ACTS_DMA_IRQPD_2 0x08
+#define ACTS_DMA_IRQPD_3 0x0c
+#define ACTS_DMA_IRQEN_0 0x10
+#define ACTS_DMA_IRQEN_1 0x14
+#define ACTS_DMA_IRQEN_2 0x18
+#define ACTS_DMA_IRQEN_3 0x1c
+#define ACTS_DMA_SECURE_ACCESS_CTL 0x20
+#define ACTS_DMA_NIC_QOS 0x24
+#define ACTS_DMA_DBG_SEL 0x28
+#define ACTS_DMA_IDLE_STAT 0x2c
+
+#define ACTS_DMA_CH_REGS_BASE 0x100
+
+#define ACTS_DMA_PENDING_MASK(x) (0x1 << (x))
+
+#define ACTS_BTSIZE_MAX 0x8000
+
+#define ch_regs(x) (ACTS_DMA_CH_REGS_BASE + (x) * 0x100)
+/* Hardware register offset for each channel */
+#define ACTS_MODE_OFFSET 0x00
+#define ACTS_SRC_OFFSET 0x04
+#define ACTS_DST_OFFSET 0x08
+#define ACTS_FRAMELEN_OFFSET 0x0C
+#define ACTS_FRAMECNT_OFFSET 0x10
+#define ACTS_REMAIN_FRAME_OFFSET 0x14
+#define ACTS_REMAIN_CNT_OFFSET 0x18
+#define ACTS_SRC_STRIDE_OFFSET 0x1C
+#define ACTS_DST_STRIDE_OFFSET 0x20
+#define ACTS_START_OFFSET 0x24
+#define ACTS_ACP_ATTR_OFFSET 0x28
+#define ACTS_CHAINED_CTL_OFFSET 0x2c
+#define ACTS_CONSTANT_OFFSET 0x30
+#define ACTS_LINKLIST_OFFSET 0x34
+#define ACTS_NEXT_DESC_OFFSET 0x38
+#define ACTS_CUR_DESC_NUM_OFFSET 0x3C
+#define ACTS_INT_CTL_OFFSET 0x40
+#define ACTS_INT_STAT_OFFSET 0x44
+#define ACTS_CUR_SRC_PTR_OFFSET 0x48
+#define ACTS_CUR_DST_PTR_OFFSET 0x4c
+
+#define ACTS_MODE_BUS_WIDTH 0x10000000
+
+#define ACTS_SRAM_ADDR(x) (((x) & 0xfffe0000) == 0xb4060000)
+#define ACTS_SRAM_SIZE 0x10000
+
+#define DMA_ERROR 0x60
+
+#define ACTS_CHAINED_CTRLB 0x80000000
+#define ACTS_LINKLIST_CTRLB 0x40000000
+#define ACTS_CONSTFILL_CTRLB 0x20000000
+#define ACTS_SRC_ADDR_MODE_INCR 0x00100000
+#define ACTS_SRC_ADDR_MODE_STRIDE 0x00200000
+#define ACTS_DST_ADDR_MODE_INCR 0x00400000
+#define ACTS_DST_ADDR_MODE_STRIDE 0x00800000
+#define ACTS_FC_MEM2MEM 0x00050000
+
+#define ACTS_LINKLIST_SRC_DST_VLD 0x500
+
+#define ACTS_LINKLIST_SRC_CONST 0x00000080
+#define ACTS_LINKLIST_SRC_VLD 0x00000040
+#define ACTS_LINKLIST_DST_VLD 0x00000100
+#define ACTS_LINKLIST_DST_CONST 0x00000200
+#define ACTS_LINKLIST_DIS_TYPE1 0x00000001
+#define ACTS_LINKLIST_DIS_TYPE2 0x00000002
+
+#define ACTS_INT_CTL_SECURE_INT 0x00000040
+#define ACTS_INT_CTL_ALAINED_INT 0x00000020
+#define ACTS_INT_CTL_LAST_FRAME_INT 0x00000010
+#define ACTS_INT_CTL_HALF_FRAME_INT 0x00000008
+#define ACTS_INT_CTL_END_FRAME_INT 0x00000004
+#define ACTS_INT_CTL_SUPERBLOCK_INT 0x00000002
+#define ACTS_INT_CTL_END_BLOCK_INT 0x00000001
+
+
+#define ACTS_INT_STATUS_SECURE_ERROR 0x00000040
+#define ACTS_INT_STATUS_ALAINED_ERROR 0x00000020
+#define ACTS_INT_STATUS_LAST_FRAME 0x00000010
+#define ACTS_INT_STATUS_HALF_FRAME 0x00000008
+#define ACTS_INT_STATUS_END_FRAME 0x00000004
+#define ACTS_INT_STATUS_SUPERBLOCK 0x00000002
+#define ACTS_INT_STATUS_END_BLOCK_INT 0x00000001
+
+#define PAUSE_CHANNEL 0x80000000
+#define ACTS_SRC_DST_STRIDE 0xa0000
+
+/* Bitfield definitions */
+#define ACTS_FRAMECNT(x) (((x) << 20) & 0xfff00000)
+
+/* ctrlb */
+#define ACTS_DMAMODE_1(x) ((x) & 0xF0000000) /*[31:28]*/
+#define ACTS_DMAMODE_2(x) (((x) << 4) & 0xf000000) /*[23:20]*/
+#define ACTS_DMAMODE_3(x) (((x) << 4) & 0x0f00000) /*[20:16]*/
+#define ACTS_DMAMODE_4(x) (((x) << 8) & 0xf0000) /*[11:8]*/
+#define ACTS_DMAMODE_5(x) (((x) << 10) & 0xfc00) /*[5:0]*/
+
+#define ACTS_CHAINEDCTL_2(x) (((x) << 2) & 0x3c)
+
+/*ctrlc*/
+#define ACTS_DMAINTCTL(x) (((x) << 18) & 0x1fc0000)
+#define ACTS_ACPCTL_1(x) (((x) << 13) & 0x3e000) /*acp attr[4:0]*/
+#define ACTS_ACPCTL_2(x) ((x) & 0x1f00) /*acp attr[12:8]*/
+#define ACTS_ACPCTL_3(x) (((x) >> 12) & 0xf0) /*acp attr[19:16]*/
+#define ACTS_ACPCTL_4(x) (((x) >> 24) & 0xf) /*acp attr[19:16]*/
+
+/*-- descriptors -----------------------------------------------------*/
+
+struct acts_lli {
+ dma_addr_t dscr;
+ dma_addr_t saddr;
+ dma_addr_t daddr;
+ u32 ctrla; /*frame len and cnt*/
+ u32 src_stride;
+ u32 dst_stride;
+ u32 ctrlb; /*dma_mode and linklist ctrl */
+ u32 ctrlc; /*acp attribute;*/
+ u32 const_num;
+};
+/**
+ * struct acts_desc - software descriptor
+ * @acts_lli: hardware lli structure
+ * @txd: support for the async_tx api
+ * @desc_node: node on the channed descriptors list
+ * @len: total transaction bytecount
+ */
+
+struct acts_desc {
+ /* FIRST values the hardware uses */
+
+ struct acts_lli lli;
+
+ /* THEN values for driver housekeeping */
+ struct list_head tx_list;
+ struct dma_async_tx_descriptor txd;
+ struct list_head desc_node;
+ size_t len;
+ u32 mode;
+};
+
+static inline struct acts_desc *
+txd_to_acts_desc(struct dma_async_tx_descriptor *txd)
+{
+ return container_of(txd, struct acts_desc, txd);
+}
+
+
+enum acts_status {
+ ACTS_IS_ERROR = 0,
+ ACTS_IS_PAUSED = (1 << 1),
+ ACTS_IS_CYCLIC = (1 << 2),
+ ACTS_IS_INTERRUPT = (1 << 4),
+};
+
+/*-- Channels --------------------------------------------------------*/
+
+/**
+ * struct owl_dma_chan - internal representation of an Atmel HDMAC channel
+ * @chan_common: common dmaengine channel object members
+ * @device: parent device
+ * @ch_regs: memory mapped register base
+ * @mask: channel index in a mask
+ * @error_status: transmit error status information from irq handler
+ * to tasklet (use atomic operations)
+ * @tasklet: bottom half to finish transaction work
+ * @lock: serializes enqueue/dequeue operations to descriptors lists
+ * @completed_cookie: identifier for the most recently completed operation
+ * @active_list: list of descriptors dmaengine is being running on
+ * @queue: list of descriptors ready to be submitted to engine
+ * @free_list: list of descriptors usable by the channel
+ * @descs_allocated: records the actual size of the descriptor pool
+ */
+struct owl_dma_chan {
+ struct dma_chan chan_common;
+ struct owl_dma *device;
+ void __iomem *ch_regs;
+ u32 mask;
+ unsigned long status;
+#ifdef COMPLEX_INT
+ struct tasklet_struct tasklet;
+#endif
+ atomic_t channel_pending;
+ spinlock_t lock;
+ struct dma_slave_config dma_sconfig;
+ u32 save_dscr;
+ u32 save_mode;
+ u32 save_ll;
+
+ /* these other elements are all protected by lock */
+ dma_cookie_t completed_cookie;
+ struct list_head active_list;
+ struct list_head queue;
+ struct list_head free_list;
+ unsigned int descs_allocated;
+};
+
+#define channel_readl(atchan, name) \
+ readl_relaxed((atchan)->ch_regs + ACTS_##name##_OFFSET)
+
+#define channel_writel(atchan, name, val) \
+ writel_relaxed((val), (atchan)->ch_regs + ACTS_##name##_OFFSET)
+
+
+static inline struct owl_dma_chan *to_owl_dma_chan(struct dma_chan *dchan)
+{
+ return container_of(dchan, struct owl_dma_chan, chan_common);
+}
+
+/*-- Controller ------------------------------------------------------*/
+
+/**
+ * struct owl_dma - internal representation of an Atmel HDMA Controller
+ * @chan_common: common dmaengine dma_device object members
+ * @ch_regs: memory mapped register base
+ * @clk: dma controller clock
+ * @all_chan_mask: all channels availlable in a mask
+ * @dma_desc_pool: base of DMA descriptor region (DMA address)
+ * @chan: channels table to store owl_dma_chan structures
+ */
+struct owl_dma {
+ struct dma_device dma_common;
+ void __iomem *regs;
+ struct clk *clk;
+ int irq;
+
+ u32 all_chan_mask;
+ u32 id;
+#ifndef COMPLEX_INT
+ struct tasklet_struct tasklet;
+ atomic_t pending;
+#endif
+ u32 save_nicqos;
+ struct dma_pool *dma_desc_pool;
+ /* AT THE END channels table */
+ struct owl_dma_chan chan[0];
+};
+
+#define dma_readl(atdma, name) \
+ readl_relaxed((atdma)->regs + ACTS_DMA_##name)
+#define dma_writel(atdma, name, val) \
+ writel_relaxed((val), (atdma)->regs + ACTS_DMA_##name)
+
+static inline struct owl_dma *to_owl_dma(struct dma_device *ddev)
+{
+ return container_of(ddev, struct owl_dma, dma_common);
+}
+
+/*-- Helper functions ------------------------------------------------*/
+
+static struct device *chan2dev(struct dma_chan *chan)
+{
+ return &chan->dev->device;
+}
+static struct device *chan2parent(struct dma_chan *chan)
+{
+ return chan->dev->device.parent;
+}
+
+static int vdbg_dump_regs(struct owl_dma_chan *atchan)
+{
+ struct owl_dma *atdma = to_owl_dma(atchan->chan_common.device);
+
+ dev_err(chan2dev(&atchan->chan_common),
+ " channel %d : irqpd = 0x%x, irqen = 0x%x, dbg_sel:0x%x"
+ " idle stat:0x%x\n",
+ atchan->chan_common.chan_id,
+ dma_readl(atdma, IRQPD_0),
+ dma_readl(atdma, IRQEN_0),
+ dma_readl(atdma, DBG_SEL),
+ dma_readl(atdma, IDLE_STAT));
+
+ dev_err(chan2dev(&atchan->chan_common),
+ " channel: s0x%x d0x%x mode0x%x:0x%x next_des0x%x framelen0x%x intctl0x%x intstat0x%x\n",
+ channel_readl(atchan, SRC),
+ channel_readl(atchan, DST),
+ channel_readl(atchan, MODE),
+ channel_readl(atchan, LINKLIST),
+ channel_readl(atchan, NEXT_DESC),
+ channel_readl(atchan, FRAMELEN),
+ channel_readl(atchan, INT_CTL),
+ channel_readl(atchan, INT_STAT));
+ dev_err(chan2dev(&atchan->chan_common),
+ " channel: framecnt0x%x desc num0x%x cur sptr0x%x cur dptr0x%x remaincnt0x%x remainframe0x%x\n"
+ " start:0x%x\n",
+ channel_readl(atchan, FRAMECNT),
+ channel_readl(atchan, CUR_DESC_NUM),
+ channel_readl(atchan, CUR_SRC_PTR),
+ channel_readl(atchan, CUR_DST_PTR),
+ channel_readl(atchan, REMAIN_CNT),
+ channel_readl(atchan, REMAIN_FRAME),
+ channel_readl(atchan, START));
+
+ return 0;
+}
+
+static void acts_dump_lli(struct owl_dma_chan *atchan, struct acts_lli *lli)
+{
+ dev_printk(KERN_CRIT, chan2dev(&atchan->chan_common),
+ " desc: s0x%x d0x%x ctrl0x%x:0x%x ctrlc0x%x l0x%x\n",
+ lli->saddr, lli->daddr,
+ lli->ctrla, lli->ctrlb, lli->ctrlc, lli->dscr);
+}
+
+static void acts_setup_irq(struct owl_dma_chan *atchan, int on)
+{
+ int int_ctl;
+
+ int_ctl = ACTS_INT_CTL_SECURE_INT
+ | ACTS_INT_CTL_ALAINED_INT
+ | ACTS_INT_CTL_SUPERBLOCK_INT;
+
+ if (on)
+ channel_writel(atchan, INT_CTL, int_ctl);
+ else
+ channel_writel(atchan, INT_CTL, 0);
+ channel_writel(atchan, INT_STAT, 0xffff);
+}
+
+static inline void acts_enable_irq(struct owl_dma_chan *atchan)
+{
+ acts_setup_irq(atchan, 1);
+}
+
+static inline void acts_disable_irq(struct owl_dma_chan *atchan)
+{
+ acts_setup_irq(atchan, 0);
+}
+
+/**
+ * atc_chan_is_paused - test channel pause/resume status
+ * @atchan: channel we want to test status
+ */
+static inline int acts_chan_is_paused(struct owl_dma_chan *atchan)
+{
+ return test_bit(ACTS_IS_PAUSED, &atchan->status);
+}
+
+/**
+ * atc_chan_is_cyclic - test if given channel has cyclic property set
+ * @atchan: channel we want to test status
+ */
+static inline int acts_chan_is_cyclic(struct owl_dma_chan *atchan)
+{
+ return test_bit(ACTS_IS_CYCLIC, &atchan->status);
+}
+
+/**
+ * acts_chan_is_enabled - test if given channel is enabled
+ * @atchan: channel we want to test status
+ */
+static inline int acts_chan_is_enabled(struct owl_dma_chan *atchan)
+{
+ struct owl_dma *asoc_dma = to_owl_dma(atchan->chan_common.device);
+ struct dma_chan *chan = &atchan->chan_common;
+ unsigned long id = chan->chan_id;
+ unsigned int idle, pause;
+
+ idle = dma_readl(asoc_dma, IDLE_STAT);
+ idle = idle & (0x1 << id);
+ pause = dma_readl(asoc_dma, DBG_SEL);
+ pause = pause & (0x1 << (id + 16));
+
+ if (idle && !pause)
+ return 0;
+ else
+ return 1;
+}
+
+static inline int acts_chan_is_enabled_dump(struct owl_dma_chan *atchan)
+{
+ struct owl_dma *atdma = to_owl_dma(atchan->chan_common.device);
+
+ if (acts_chan_is_enabled(atchan)) {
+ dev_err(chan2dev(&atchan->chan_common),
+ " channel %d : dbg_sel:0x%x idle stat:0x%x\n",
+ atchan->chan_common.chan_id,
+ dma_readl(atdma, DBG_SEL),
+ dma_readl(atdma, IDLE_STAT));
+
+ dev_err(chan2dev(&atchan->chan_common),
+ " channel: s0x%x d0x%x mode:0x%x frame_len:0x%x frame_cnt0x%x\n",
+ channel_readl(atchan, SRC),
+ channel_readl(atchan, DST),
+ channel_readl(atchan, MODE),
+ channel_readl(atchan, FRAMELEN),
+ channel_readl(atchan, FRAMECNT));
+
+ dev_err(chan2dev(&atchan->chan_common),
+ " channel: acp:0x%x,chain ctl:0x%x,llist:0x%x,int_ctl:0x%x,int_stat:0x%x\n",
+ channel_readl(atchan, ACP_ATTR),
+ channel_readl(atchan, CHAINED_CTL),
+ channel_readl(atchan, LINKLIST),
+ channel_readl(atchan, INT_CTL),
+ channel_readl(atchan, INT_STAT));
+ return 1;
+ } else
+ return 0;
+}
+
+/**
+ * set_desc_eol - set end-of-link to descriptor so it will end transfer
+ * @desc: descriptor, signle or at the end of a chain, to end chain on
+ */
+static void set_desc_eol(struct acts_desc *desc)
+{
+ desc->lli.dscr = 0;
+ desc->lli.ctrlb &= ~ACTS_LINKLIST_CTRLB;
+}
+
+static int acts_read_remain_cnt(struct owl_dma_chan *atchan)
+{
+ return channel_readl(atchan, REMAIN_CNT);
+}
+#endif /* ACTS_HDMAC_REGS_H */
--
2.7.4
More information about the linux-yocto
mailing list