mirror of https://github.com/OpenIPC/firmware.git
1290 lines
36 KiB
Diff
1290 lines
36 KiB
Diff
--- linux-4.9.37/drivers/dma/edmac_goke.c 1970-01-01 03:00:00.000000000 +0300
|
|
+++ linux-4.9.y/drivers/dma/edmac_goke.c 2021-06-07 13:01:33.000000000 +0300
|
|
@@ -0,0 +1,1286 @@
|
|
+/*
|
|
+ * Copyright (c) Hunan Goke,Chengdu Goke,Shandong Goke. 2021. All rights reserved.
|
|
+ */
|
|
+
|
|
+#include <linux/debugfs.h>
|
|
+#include <linux/delay.h>
|
|
+#include <linux/clk.h>
|
|
+#include <linux/reset.h>
|
|
+#include <linux/platform_device.h>
|
|
+#include <linux/device.h>
|
|
+#include <linux/dmaengine.h>
|
|
+#include <linux/dmapool.h>
|
|
+#include <linux/dma-mapping.h>
|
|
+#include <linux/export.h>
|
|
+#include <linux/init.h>
|
|
+#include <linux/interrupt.h>
|
|
+#include <linux/module.h>
|
|
+#include <linux/of.h>
|
|
+#include <linux/of_dma.h>
|
|
+#include <linux/pm_runtime.h>
|
|
+#include <linux/seq_file.h>
|
|
+#include <linux/slab.h>
|
|
+#include <linux/io.h>
|
|
+#include <linux/regmap.h>
|
|
+#include <linux/mfd/syscon.h>
|
|
+
|
|
+#include "edmac_goke.h"
|
|
+#include "dmaengine.h"
|
|
+#include "virt-dma.h"
|
|
+
|
|
+#define DRIVER_NAME "edmac-goke"
|
|
+
|
|
+int edmac_trace_level = EDMAC_TRACE_LEVEL;
|
|
+
|
|
+typedef struct edmac_lli {
|
|
+ u64 next_lli;
|
|
+ u32 reserved[5];
|
|
+ u32 count;
|
|
+ u64 src_addr;
|
|
+ u64 dest_addr;
|
|
+ u32 config;
|
|
+ u32 pad[3];
|
|
+} edmac_lli;
|
|
+
|
|
+struct edmac_sg {
|
|
+ dma_addr_t src_addr;
|
|
+ dma_addr_t dst_addr;
|
|
+ size_t len;
|
|
+ struct list_head node;
|
|
+};
|
|
+
|
|
+struct transfer_desc {
|
|
+ struct virt_dma_desc virt_desc;
|
|
+
|
|
+ dma_addr_t llis_busaddr;
|
|
+ u64 *llis_vaddr;
|
|
+ u32 ccfg;
|
|
+ size_t size;
|
|
+ bool done;
|
|
+ bool cyclic;
|
|
+};
|
|
+
|
|
+enum edmac_dma_chan_state {
|
|
+ EDMAC_CHAN_IDLE,
|
|
+ EDMAC_CHAN_RUNNING,
|
|
+ EDMAC_CHAN_PAUSED,
|
|
+ EDMAC_CHAN_WAITING,
|
|
+};
|
|
+
|
|
+struct edmac_dma_chan {
|
|
+ bool slave;
|
|
+ int signal;
|
|
+ int id;
|
|
+ struct virt_dma_chan virt_chan;
|
|
+ struct edmac_phy_chan *phychan;
|
|
+ struct dma_slave_config cfg;
|
|
+ struct transfer_desc *at;
|
|
+ struct edmac_driver_data *host;
|
|
+ enum edmac_dma_chan_state state;
|
|
+};
|
|
+
|
|
+struct edmac_phy_chan {
|
|
+ unsigned int id;
|
|
+ void __iomem *base;
|
|
+ spinlock_t lock;
|
|
+ struct edmac_dma_chan *serving;
|
|
+};
|
|
+
|
|
+struct edmac_driver_data {
|
|
+ struct platform_device *dev;
|
|
+ struct dma_device slave;
|
|
+ struct dma_device memcpy;
|
|
+ void __iomem *base;
|
|
+ struct regmap *misc_regmap;
|
|
+ void __iomem *crg_ctrl;
|
|
+ struct edmac_phy_chan *phy_chans;
|
|
+ struct dma_pool *pool;
|
|
+ unsigned int misc_ctrl_base;
|
|
+ int irq;
|
|
+ unsigned int id;
|
|
+ struct clk *clk;
|
|
+ struct clk *axi_clk;
|
|
+ struct reset_control *rstc;
|
|
+ unsigned int channels;
|
|
+ unsigned int slave_requests;
|
|
+ unsigned int max_transfer_size;
|
|
+};
|
|
+
|
|
+#ifdef DEBUG_EDMAC
|
|
+void dump_lli(u64 *llis_vaddr, unsigned int num)
|
|
+{
|
|
+
|
|
+ edmac_lli *plli = (edmac_lli *)llis_vaddr;
|
|
+ unsigned int i;
|
|
+
|
|
+ edmac_trace(3, "lli num = 0%d\n", num);
|
|
+ for (i = 0; i < num; i++) {
|
|
+ printk("lli%d:lli_L: 0x%llx\n", i, plli[i].next_lli & 0xffffffff);
|
|
+ printk("lli%d:lli_H: 0x%llx\n", i, plli[i].next_lli >> 32 & 0xffffffff);
|
|
+ printk("lli%d:count: 0x%llx\n", i, plli[i].count);
|
|
+ printk("lli%d:src_addr_L: 0x%llx\n", i, plli[i].src_addr & 0xffffffff);
|
|
+ printk("lli%d:src_addr_H: 0x%llx\n", i, plli[i].src_addr >> 32 & 0xffffffff);
|
|
+ printk("lli%d:dst_addr_L: 0x%llx\n", i, plli[i].dest_addr & 0xffffffff);
|
|
+ printk("lli%d:dst_addr_H: 0x%llx\n", i, plli[i].dest_addr >> 32 & 0xffffffff);
|
|
+ printk("lli%d:CONFIG: 0x%llx\n", i, plli[i].config);
|
|
+ }
|
|
+}
|
|
+
|
|
+#else
|
|
+void dump_lli(u64 *llis_vaddr, unsigned int num)
|
|
+{
|
|
+}
|
|
+#endif
|
|
+
|
|
+static inline struct edmac_dma_chan *to_edamc_chan(struct dma_chan *chan)
|
|
+{
|
|
+ return container_of(chan, struct edmac_dma_chan, virt_chan.chan);
|
|
+}
|
|
+
|
|
+static inline struct transfer_desc *to_edmac_transfer_desc(struct dma_async_tx_descriptor *tx)
|
|
+{
|
|
+ return container_of(tx, struct transfer_desc, virt_desc.tx);
|
|
+}
|
|
+
|
|
+static struct dma_chan *edmac_find_chan_id(struct edmac_driver_data *edmac,
|
|
+ int request_num)
|
|
+{
|
|
+ struct edmac_dma_chan *edmac_dma_chan = NULL;
|
|
+
|
|
+ list_for_each_entry(edmac_dma_chan, &edmac->slave.channels, virt_chan.chan.device_node) {
|
|
+ if (edmac_dma_chan->id == request_num) {
|
|
+ return &edmac_dma_chan->virt_chan.chan;
|
|
+ }
|
|
+ }
|
|
+ return NULL;
|
|
+}
|
|
+
|
|
+static struct dma_chan *edma_of_xlate(struct of_phandle_args *dma_spec,
|
|
+ struct of_dma *ofdma)
|
|
+{
|
|
+ struct edmac_driver_data *edmac = ofdma->of_dma_data;
|
|
+ struct edmac_dma_chan *edmac_dma_chan = NULL;
|
|
+ struct dma_chan *dma_chan = NULL;
|
|
+ struct regmap *misc = NULL;
|
|
+ unsigned int signal = 0, request_num = 0;
|
|
+ unsigned int reg = 0, offset = 0;
|
|
+
|
|
+ if (!edmac) {
|
|
+ return NULL;
|
|
+ }
|
|
+
|
|
+ misc = edmac->misc_regmap;
|
|
+
|
|
+ if (dma_spec->args_count != 2) {
|
|
+ edmac_error("args count not true!\n");
|
|
+ return NULL;
|
|
+ }
|
|
+
|
|
+ request_num = dma_spec->args[0];
|
|
+ signal = dma_spec->args[1];
|
|
+
|
|
+ edmac_trace(3, "host->id = %d,signal = %d, request_num = %d\n", edmac->id, signal, request_num);
|
|
+
|
|
+ if (misc != NULL) {
|
|
+#ifdef CONFIG_ACCESS_M7_DEV
|
|
+ offset = edmac->misc_ctrl_base;
|
|
+ reg = 0xc0;
|
|
+ regmap_write(misc, offset, reg);
|
|
+#else
|
|
+ offset = edmac->misc_ctrl_base + (request_num & (~0x3));
|
|
+ regmap_read(misc, offset, ®);
|
|
+ reg &= ~(0x3f << ((request_num & 0x3) << 3));
|
|
+ reg |= signal << ((request_num & 0x3) << 3);
|
|
+ regmap_write(misc, offset, reg);
|
|
+#endif
|
|
+ }
|
|
+
|
|
+ edmac_trace(3, "offset = 0x%x, reg = 0x%x\n", offset, reg);
|
|
+
|
|
+ dma_chan = edmac_find_chan_id(edmac, request_num);
|
|
+ if (!dma_chan) {
|
|
+ edmac_error("DMA slave channel is not found!\n");
|
|
+ return NULL;
|
|
+ }
|
|
+
|
|
+ edmac_dma_chan = to_edamc_chan(dma_chan);
|
|
+ edmac_dma_chan->signal = request_num;
|
|
+
|
|
+ return dma_get_slave_channel(dma_chan);
|
|
+}
|
|
+
|
|
+
|
|
+static int get_of_probe(struct edmac_driver_data *edmac)
|
|
+{
|
|
+ struct resource *res = NULL;
|
|
+ struct platform_device *platdev = edmac->dev;
|
|
+ struct device_node *np = platdev->dev.of_node;
|
|
+ int ret;
|
|
+
|
|
+ ret = of_property_read_u32((&platdev->dev)->of_node,
|
|
+ "devid", &(edmac->id));
|
|
+ if (ret) {
|
|
+ edmac_error("get edmac id fail\n");
|
|
+ return -ENODEV;
|
|
+ }
|
|
+
|
|
+ edmac->clk = devm_clk_get(&(platdev->dev), "apb_pclk");
|
|
+ if (IS_ERR(edmac->clk)) {
|
|
+ return PTR_ERR(edmac->clk);
|
|
+ }
|
|
+
|
|
+ edmac->axi_clk = devm_clk_get(&(platdev->dev), "axi_aclk");
|
|
+ if (IS_ERR(edmac->axi_clk)) {
|
|
+ return PTR_ERR(edmac->axi_clk);
|
|
+ }
|
|
+
|
|
+ edmac->rstc = devm_reset_control_get(&(platdev->dev), "dma-reset");
|
|
+ if (IS_ERR(edmac->rstc)) {
|
|
+ return PTR_ERR(edmac->rstc);
|
|
+ }
|
|
+
|
|
+ res = platform_get_resource(platdev, IORESOURCE_MEM, 0);
|
|
+ if (!res) {
|
|
+ edmac_error("no reg resource\n");
|
|
+ return -ENODEV;
|
|
+ }
|
|
+
|
|
+ edmac->base = devm_ioremap_resource(&(platdev->dev), res);
|
|
+ if (IS_ERR(edmac->base)) {
|
|
+ return PTR_ERR(edmac->base);
|
|
+ }
|
|
+#if defined(CONFIG_ARCH_GK7205V200) || defined(CONFIG_ARCH_GK7205V300) || \
|
|
+ defined(CONFIG_ARCH_GK7202V300) || defined(CONFIG_ARCH_GK7605V100)
|
|
+ edmac->misc_regmap = 0;
|
|
+ (void)np;
|
|
+#else
|
|
+ edmac->misc_regmap = syscon_regmap_lookup_by_phandle(np, "misc_regmap");
|
|
+ if (IS_ERR(edmac->misc_regmap)) {
|
|
+ return PTR_ERR(edmac->misc_regmap);
|
|
+ }
|
|
+
|
|
+ ret = of_property_read_u32((&platdev->dev)->of_node,
|
|
+ "misc_ctrl_base", &(edmac->misc_ctrl_base));
|
|
+ if (ret) {
|
|
+ edmac_error( "get dma-misc_ctrl_base fail\n");
|
|
+ return -ENODEV;
|
|
+ }
|
|
+#endif
|
|
+ edmac->irq = platform_get_irq(platdev, 0);
|
|
+ if (unlikely(edmac->irq < 0)) {
|
|
+ return -ENODEV;
|
|
+ }
|
|
+
|
|
+ ret = of_property_read_u32((&platdev->dev)->of_node,
|
|
+ "dma-channels", &(edmac->channels));
|
|
+ if (ret) {
|
|
+ edmac_error( "get dma-channels fail\n");
|
|
+ return -ENODEV;
|
|
+ }
|
|
+ ret = of_property_read_u32((&platdev->dev)->of_node,
|
|
+ "dma-requests", &(edmac->slave_requests));
|
|
+ if (ret) {
|
|
+ edmac_error( "get dma-requests fail\n");
|
|
+ return -ENODEV;
|
|
+ }
|
|
+ edmac_trace(2, "dma-channels = %d, dma-requests = %d\n",
|
|
+ edmac->channels, edmac->slave_requests);
|
|
+ return of_dma_controller_register(platdev->dev.of_node, edma_of_xlate, edmac);
|
|
+}
|
|
+
|
|
+static void edmac_free_chan_resources(struct dma_chan *chan)
|
|
+{
|
|
+ vchan_free_chan_resources(to_virt_chan(chan));
|
|
+}
|
|
+
|
|
+static enum dma_status edmac_tx_status(struct dma_chan *chan,
|
|
+ dma_cookie_t cookie, struct dma_tx_state *txstate)
|
|
+{
|
|
+ enum dma_status ret = DMA_COMPLETE;
|
|
+ struct edmac_dma_chan *edmac_dma_chan = to_edamc_chan(chan);
|
|
+ struct edmac_phy_chan *phychan = edmac_dma_chan->phychan;
|
|
+ struct edmac_driver_data *edmac = edmac_dma_chan->host;
|
|
+ struct virt_dma_desc *vd = NULL;
|
|
+ struct transfer_desc *tsf_desc = NULL;
|
|
+ unsigned long flags;
|
|
+ size_t bytes = 0;
|
|
+ u64 curr_lli = 0, curr_residue_bytes = 0, temp = 0;
|
|
+ edmac_lli *plli = NULL;
|
|
+ unsigned int i = 0, index = 0;
|
|
+
|
|
+ ret = dma_cookie_status(chan, cookie, txstate);
|
|
+ if (ret == DMA_COMPLETE) {
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ spin_lock_irqsave(&edmac_dma_chan->virt_chan.lock, flags);
|
|
+ vd = vchan_find_desc(&edmac_dma_chan->virt_chan, cookie);
|
|
+ if (vd) {
|
|
+ /* no been trasfer */
|
|
+ tsf_desc = to_edmac_transfer_desc(&vd->tx);
|
|
+ bytes = tsf_desc->size;
|
|
+ } else {
|
|
+ /* trasfering */
|
|
+ tsf_desc = edmac_dma_chan->at;
|
|
+
|
|
+ if (!phychan || !tsf_desc) {
|
|
+ spin_unlock_irqrestore(&edmac_dma_chan->virt_chan.lock, flags);
|
|
+ goto out;
|
|
+ }
|
|
+ curr_lli = (edmac_readl(edmac->base + EDMAC_Cx_LLI_L(phychan->id)) & (~(EDMAC_LLI_ALIGN - 1)));
|
|
+ curr_lli |= ((u64)(edmac_readl(edmac->base + EDMAC_Cx_LLI_H(phychan->id)) & 0xffffffff) << 32);
|
|
+ curr_residue_bytes = edmac_readl(edmac->base + EDMAC_Cx_CURR_CNT0(phychan->id));
|
|
+ if (curr_lli == 0) {
|
|
+ /* It means non-lli mode */
|
|
+ bytes = curr_residue_bytes;
|
|
+ } else {
|
|
+ /* It means lli mode */
|
|
+ index = (curr_lli - tsf_desc->llis_busaddr) / sizeof(edmac_lli) - 1;
|
|
+ plli = (edmac_lli *)(tsf_desc->llis_vaddr);
|
|
+ for (i = 0; i < index; i++) {
|
|
+ temp += plli[i].count;
|
|
+ }
|
|
+ temp += plli[i].count - curr_residue_bytes;
|
|
+ bytes = tsf_desc->size - temp;
|
|
+ }
|
|
+ }
|
|
+ spin_unlock_irqrestore(&edmac_dma_chan->virt_chan.lock, flags);
|
|
+
|
|
+ dma_set_residue(txstate, bytes);
|
|
+
|
|
+ if (edmac_dma_chan->state == EDMAC_CHAN_PAUSED && ret == DMA_IN_PROGRESS) {
|
|
+ ret = DMA_PAUSED;
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+out:
|
|
+ return ret;
|
|
+}
|
|
+
|
|
+static struct edmac_phy_chan *edmac_get_phy_channel(
|
|
+ struct edmac_driver_data *edmac,
|
|
+ struct edmac_dma_chan *edmac_dma_chan)
|
|
+{
|
|
+ struct edmac_phy_chan *ch = NULL;
|
|
+ unsigned long flags;
|
|
+ int i;
|
|
+
|
|
+ for (i = 0; i < edmac->channels; i++) {
|
|
+ ch = &edmac->phy_chans[i];
|
|
+
|
|
+ spin_lock_irqsave(&ch->lock, flags);
|
|
+
|
|
+ if (!ch->serving) {
|
|
+ ch->serving = edmac_dma_chan;
|
|
+ spin_unlock_irqrestore(&ch->lock, flags);
|
|
+ break;
|
|
+ }
|
|
+ spin_unlock_irqrestore(&ch->lock, flags);
|
|
+ }
|
|
+
|
|
+ if (i == edmac->channels) {
|
|
+ return NULL;
|
|
+ }
|
|
+
|
|
+ return ch;
|
|
+}
|
|
+
|
|
+static void edmac_write_lli(struct edmac_driver_data *edmac,
|
|
+ struct edmac_phy_chan *phychan,
|
|
+ struct transfer_desc *tsf_desc)
|
|
+{
|
|
+
|
|
+ edmac_lli *plli = (edmac_lli *)tsf_desc->llis_vaddr;
|
|
+
|
|
+ if (plli->next_lli != 0x0) {
|
|
+ edmac_writel((plli->next_lli & 0xffffffff) | EDMAC_LLI_ENABLE, edmac->base + EDMAC_Cx_LLI_L(phychan->id));
|
|
+ } else {
|
|
+ edmac_writel((plli->next_lli & 0xffffffff), edmac->base + EDMAC_Cx_LLI_L(phychan->id));
|
|
+ }
|
|
+
|
|
+ edmac_writel(((plli->next_lli >> 32) & 0xffffffff), edmac->base + EDMAC_Cx_LLI_H(phychan->id));
|
|
+ edmac_writel(plli->count, edmac->base + EDMAC_Cx_CNT0(phychan->id));
|
|
+ edmac_writel(plli->src_addr & 0xffffffff, edmac->base + EDMAC_Cx_SRC_ADDR_L(phychan->id));
|
|
+ edmac_writel((plli->src_addr >> 32) & 0xffffffff, edmac->base + EDMAC_Cx_SRC_ADDR_H(phychan->id));
|
|
+ edmac_writel(plli->dest_addr & 0xffffffff, edmac->base + EDMAC_Cx_DEST_ADDR_L(phychan->id));
|
|
+ edmac_writel((plli->dest_addr >> 32) & 0xffffffff, edmac->base + EDMAC_Cx_DEST_ADDR_H(phychan->id));
|
|
+ edmac_writel(plli->config, edmac->base + EDMAC_Cx_CONFIG(phychan->id));
|
|
+}
|
|
+
|
|
+static void edmac_start_next_txd(struct edmac_dma_chan *edmac_dma_chan)
|
|
+{
|
|
+ struct edmac_driver_data *edmac = edmac_dma_chan->host;
|
|
+ struct edmac_phy_chan *phychan = edmac_dma_chan->phychan;
|
|
+ struct virt_dma_desc *vd = vchan_next_desc(&edmac_dma_chan->virt_chan);
|
|
+ struct transfer_desc *tsf_desc = to_edmac_transfer_desc(&vd->tx);
|
|
+ unsigned int val = 0;
|
|
+
|
|
+ list_del(&tsf_desc->virt_desc.node);
|
|
+
|
|
+ edmac_dma_chan->at = tsf_desc;
|
|
+
|
|
+ edmac_write_lli(edmac, phychan, tsf_desc);
|
|
+
|
|
+ val = edmac_readl(edmac->base + EDMAC_Cx_CONFIG(phychan->id));
|
|
+
|
|
+ edmac_trace(2, " EDMAC_Cx_CONFIG = 0x%x\n", val);
|
|
+ edmac_writel(val | EDMAC_CxCONFIG_LLI_START, edmac->base + EDMAC_Cx_CONFIG(phychan->id));
|
|
+}
|
|
+
|
|
+static void edmac_start(struct edmac_dma_chan * edmac_dma_chan)
|
|
+{
|
|
+ struct edmac_driver_data *edmac = edmac_dma_chan->host;
|
|
+ struct edmac_phy_chan *ch;
|
|
+
|
|
+ ch = edmac_get_phy_channel(edmac, edmac_dma_chan);
|
|
+ if (!ch) {
|
|
+ edmac_error("no phy channel available !\n");
|
|
+ edmac_dma_chan->state = EDMAC_CHAN_WAITING;
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ edmac_dma_chan->phychan = ch;
|
|
+ edmac_dma_chan->state = EDMAC_CHAN_RUNNING;
|
|
+
|
|
+ edmac_start_next_txd(edmac_dma_chan);
|
|
+}
|
|
+
|
|
+static void edmac_issue_pending(struct dma_chan *chan)
|
|
+{
|
|
+ struct edmac_dma_chan *edmac_dma_chan = to_edamc_chan(chan);
|
|
+ unsigned long flags;
|
|
+
|
|
+ spin_lock_irqsave(&edmac_dma_chan->virt_chan.lock, flags);
|
|
+ if (vchan_issue_pending(&edmac_dma_chan->virt_chan)) {
|
|
+ if (!edmac_dma_chan->phychan && edmac_dma_chan->state != EDMAC_CHAN_WAITING) {
|
|
+ edmac_start(edmac_dma_chan);
|
|
+ }
|
|
+ }
|
|
+ spin_unlock_irqrestore(&edmac_dma_chan->virt_chan.lock, flags);
|
|
+}
|
|
+
|
|
+static void edmac_free_txd_list(struct edmac_dma_chan *edmac_dma_chan)
|
|
+{
|
|
+ LIST_HEAD(head);
|
|
+
|
|
+ vchan_get_all_descriptors(&edmac_dma_chan->virt_chan, &head);
|
|
+ vchan_dma_desc_free_list(&edmac_dma_chan->virt_chan, &head);
|
|
+}
|
|
+
|
|
+static int edmac_config(struct dma_chan *chan,
|
|
+ struct dma_slave_config *config)
|
|
+{
|
|
+ struct edmac_dma_chan *edmac_dma_chan = to_edamc_chan(chan);
|
|
+
|
|
+ if (!edmac_dma_chan->slave) {
|
|
+ edmac_error("slave is null!");
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ edmac_dma_chan->cfg = *config;
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static void edmac_pause_phy_chan(struct edmac_dma_chan *edmac_dma_chan)
|
|
+{
|
|
+ struct edmac_driver_data *edmac = edmac_dma_chan->host;
|
|
+ struct edmac_phy_chan *phychan = edmac_dma_chan->phychan;
|
|
+ unsigned int val;
|
|
+ int timeout;
|
|
+
|
|
+
|
|
+ val = edmac_readl(edmac->base + EDMAC_Cx_CONFIG(phychan->id));
|
|
+ val &= ~CCFG_EN;
|
|
+ edmac_writel(val, edmac->base + EDMAC_Cx_CONFIG(phychan->id));
|
|
+
|
|
+ /* Wait for channel inactive */
|
|
+ for (timeout = 2000; timeout > 0; timeout--) {
|
|
+ if (!(0x1 << phychan->id & edmac_readl(edmac->base + EDMAC_CH_STAT))) {
|
|
+ break;
|
|
+ }
|
|
+ edmac_writel(val, edmac->base + EDMAC_Cx_CONFIG(phychan->id));
|
|
+ udelay(1);
|
|
+ }
|
|
+
|
|
+ if (timeout == 0) {
|
|
+ edmac_error(":channel%u timeout waiting for pause, timeout:%d\n",
|
|
+ phychan->id, timeout);
|
|
+ }
|
|
+}
|
|
+
|
|
+static int edmac_pause(struct dma_chan *chan)
|
|
+{
|
|
+ struct edmac_dma_chan *edmac_dma_chan = to_edamc_chan(chan);
|
|
+ unsigned long flags;
|
|
+
|
|
+ spin_lock_irqsave(&edmac_dma_chan->virt_chan.lock, flags);
|
|
+
|
|
+ if (!edmac_dma_chan->phychan) {
|
|
+ spin_unlock_irqrestore(&edmac_dma_chan->virt_chan.lock, flags);
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ edmac_pause_phy_chan(edmac_dma_chan);
|
|
+ edmac_dma_chan->state = EDMAC_CHAN_PAUSED;
|
|
+ spin_unlock_irqrestore(&edmac_dma_chan->virt_chan.lock, flags);
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static void edmac_resume_phy_chan(struct edmac_dma_chan *edmac_dma_chan)
|
|
+{
|
|
+ struct edmac_driver_data *edmac = edmac_dma_chan->host;
|
|
+ struct edmac_phy_chan *phychan = edmac_dma_chan->phychan;
|
|
+ unsigned int val;
|
|
+
|
|
+ val = edmac_readl(edmac->base + EDMAC_Cx_CONFIG(phychan->id));
|
|
+ val |= CCFG_EN;
|
|
+ edmac_writel(val, edmac->base + EDMAC_Cx_CONFIG(phychan->id));
|
|
+}
|
|
+
|
|
+static int edmac_resume(struct dma_chan *chan)
|
|
+{
|
|
+ struct edmac_dma_chan *edmac_dma_chan = to_edamc_chan(chan);
|
|
+ unsigned long flags;
|
|
+
|
|
+ spin_lock_irqsave(&edmac_dma_chan->virt_chan.lock, flags);
|
|
+
|
|
+ if (!edmac_dma_chan->phychan) {
|
|
+ spin_unlock_irqrestore(&edmac_dma_chan->virt_chan.lock, flags);
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ edmac_resume_phy_chan(edmac_dma_chan);
|
|
+ edmac_dma_chan->state = EDMAC_CHAN_RUNNING;
|
|
+ spin_unlock_irqrestore(&edmac_dma_chan->virt_chan.lock, flags);
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+void edmac_phy_free(struct edmac_dma_chan *chan);
|
|
+static void edmac_desc_free(struct virt_dma_desc *vd);
|
|
+static int edmac_terminate_all(struct dma_chan *chan)
|
|
+{
|
|
+ struct edmac_dma_chan *edmac_dma_chan = to_edamc_chan(chan);
|
|
+ unsigned long flags;
|
|
+
|
|
+ spin_lock_irqsave(&edmac_dma_chan->virt_chan.lock, flags);
|
|
+ if (!edmac_dma_chan->phychan && !edmac_dma_chan->at) {
|
|
+ spin_unlock_irqrestore(&edmac_dma_chan->virt_chan.lock, flags);
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ edmac_dma_chan->state = EDMAC_CHAN_IDLE;
|
|
+
|
|
+ if (edmac_dma_chan->phychan) {
|
|
+ edmac_phy_free(edmac_dma_chan);
|
|
+ }
|
|
+
|
|
+ if (edmac_dma_chan->at) {
|
|
+ edmac_desc_free(&edmac_dma_chan->at->virt_desc);
|
|
+ edmac_dma_chan->at = NULL;
|
|
+ }
|
|
+ edmac_free_txd_list(edmac_dma_chan);
|
|
+
|
|
+ spin_unlock_irqrestore(&edmac_dma_chan->virt_chan.lock, flags);
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static struct transfer_desc *edmac_get_tsf_desc(struct edmac_driver_data *plchan)
|
|
+{
|
|
+ struct transfer_desc *tsf_desc = kzalloc(sizeof(struct transfer_desc), GFP_NOWAIT);
|
|
+
|
|
+ if (tsf_desc) {
|
|
+ tsf_desc->ccfg = 0;
|
|
+ }
|
|
+
|
|
+ return tsf_desc;
|
|
+}
|
|
+
|
|
+static void edmac_free_tsf_desc(struct edmac_driver_data *edmac,
|
|
+ struct transfer_desc *tsf_desc)
|
|
+{
|
|
+ if (tsf_desc->llis_vaddr) {
|
|
+ dma_pool_free(edmac->pool, tsf_desc->llis_vaddr, tsf_desc->llis_busaddr);
|
|
+ }
|
|
+
|
|
+ kfree(tsf_desc);
|
|
+}
|
|
+
|
|
+static u32 get_width(enum dma_slave_buswidth width)
|
|
+{
|
|
+ switch (width) {
|
|
+ case DMA_SLAVE_BUSWIDTH_1_BYTE:
|
|
+ return EDMAC_WIDTH_8BIT;
|
|
+ case DMA_SLAVE_BUSWIDTH_2_BYTES:
|
|
+ return EDMAC_WIDTH_16BIT;
|
|
+ case DMA_SLAVE_BUSWIDTH_4_BYTES:
|
|
+ return EDMAC_WIDTH_32BIT;
|
|
+ case DMA_SLAVE_BUSWIDTH_8_BYTES:
|
|
+ return EDMAC_WIDTH_64BIT;
|
|
+ default:
|
|
+ edmac_error("check here, width warning!\n");
|
|
+ return ~0;
|
|
+ }
|
|
+}
|
|
+
|
|
+struct transfer_desc *edmac_init_tsf_desc (
|
|
+ struct dma_chan *chan,
|
|
+ enum dma_transfer_direction direction,
|
|
+ dma_addr_t *slave_addr)
|
|
+{
|
|
+ struct edmac_dma_chan *edmac_dma_chan = to_edamc_chan(chan);
|
|
+ struct edmac_driver_data *edmac = edmac_dma_chan->host;
|
|
+ struct transfer_desc *tsf_desc;
|
|
+ unsigned int config = 0, burst = 0;
|
|
+ unsigned int addr_width = 0, maxburst = 0;
|
|
+ unsigned int width = 0;
|
|
+
|
|
+ tsf_desc = edmac_get_tsf_desc(edmac);
|
|
+ if (!tsf_desc) {
|
|
+ edmac_error("get tsf desc fail!\n");
|
|
+ return NULL;
|
|
+ }
|
|
+
|
|
+ if (direction == DMA_MEM_TO_DEV) {
|
|
+ config = EDMAC_CONFIG_SRC_INC;
|
|
+ *slave_addr = edmac_dma_chan->cfg.dst_addr;
|
|
+ addr_width = edmac_dma_chan->cfg.dst_addr_width;
|
|
+ maxburst = edmac_dma_chan->cfg.dst_maxburst;
|
|
+ } else if (direction == DMA_DEV_TO_MEM) {
|
|
+ config = EDMAC_CONFIG_DST_INC;
|
|
+ *slave_addr = edmac_dma_chan->cfg.src_addr;
|
|
+ addr_width = edmac_dma_chan->cfg.src_addr_width;
|
|
+ maxburst = edmac_dma_chan->cfg.src_maxburst;
|
|
+ } else {
|
|
+ edmac_free_tsf_desc(edmac, tsf_desc);
|
|
+ edmac_error("direction unsupported!\n");
|
|
+ return NULL;
|
|
+ }
|
|
+
|
|
+ edmac_trace(3, "addr_width = 0x%x\n", addr_width);
|
|
+ width = get_width(addr_width);
|
|
+ edmac_trace(3, "width = 0x%x\n", width);
|
|
+ config |= width << EDMAC_CONFIG_SRC_WIDTH_SHIFT;
|
|
+ config |= width << EDMAC_CONFIG_DST_WIDTH_SHIFT;
|
|
+ edmac_trace(2, "tsf_desc->ccfg = 0x%x\n", config);
|
|
+
|
|
+ edmac_trace(3, "maxburst = 0x%x\n", maxburst);
|
|
+ if (maxburst > (EDMAC_MAX_BURST_WIDTH)) {
|
|
+ burst |= (EDMAC_MAX_BURST_WIDTH - 1);
|
|
+ } else if (maxburst == 0) {
|
|
+ burst |= EDMAC_MIN_BURST_WIDTH;
|
|
+ } else {
|
|
+ burst |= (maxburst - 1);
|
|
+ }
|
|
+ edmac_trace(3, "burst = 0x%x\n", burst);
|
|
+ config |= burst << EDMAC_CONFIG_SRC_BURST_SHIFT;
|
|
+ config |= burst << EDMAC_CONFIG_DST_BURST_SHIFT;
|
|
+
|
|
+ if (edmac_dma_chan->signal >= 0) {
|
|
+ edmac_trace(2, "edmac_dma_chan->signal = %d\n", edmac_dma_chan->signal);
|
|
+ config |= (unsigned int)edmac_dma_chan->signal << EDMAC_CXCONFIG_SIGNAL_SHIFT;
|
|
+ }
|
|
+
|
|
+ config |= EDMAC_CXCONFIG_DEV_MEM_TYPE << EDMAC_CXCONFIG_TSF_TYPE_SHIFT;
|
|
+ tsf_desc->ccfg = config;
|
|
+ edmac_trace(2, "tsf_desc->ccfg = 0x%x\n", tsf_desc->ccfg);
|
|
+
|
|
+ return tsf_desc;
|
|
+}
|
|
+
|
|
+static void edmac_fill_desc(struct transfer_desc *tsf_desc, dma_addr_t src,
|
|
+ dma_addr_t dst, unsigned int length, unsigned int num)
|
|
+{
|
|
+ edmac_lli *plli;
|
|
+
|
|
+ plli = (edmac_lli *)(tsf_desc->llis_vaddr);
|
|
+ memset(&plli[num], 0x0, sizeof(edmac_lli));
|
|
+
|
|
+ plli[num].src_addr = src;
|
|
+ plli[num].dest_addr = dst;
|
|
+ plli[num].config = tsf_desc->ccfg;
|
|
+ plli[num].count = length;
|
|
+ tsf_desc->size += length;
|
|
+
|
|
+ if (num > 0) {
|
|
+ plli[num - 1].next_lli = (tsf_desc->llis_busaddr + (num) * sizeof(edmac_lli)) & (~(EDMAC_LLI_ALIGN - 1));
|
|
+ plli[num - 1].next_lli |= EDMAC_LLI_ENABLE;
|
|
+ }
|
|
+}
|
|
+
|
|
+static struct dma_async_tx_descriptor *edmac_perp_slave_sg(
|
|
+ struct dma_chan *chan, struct scatterlist *sgl,
|
|
+ unsigned int sg_len, enum dma_transfer_direction direction,
|
|
+ unsigned long flags, void *context)
|
|
+{
|
|
+ struct edmac_dma_chan *edmac_dma_chan = to_edamc_chan(chan);
|
|
+ struct edmac_driver_data *edmac = edmac_dma_chan->host;
|
|
+ struct transfer_desc *tsf_desc = NULL;
|
|
+ struct scatterlist *sg = NULL;
|
|
+ int tmp = 0;
|
|
+ dma_addr_t src = 0, dst = 0, addr = 0, slave_addr = 0;
|
|
+ unsigned int length = 0, num = 0;
|
|
+
|
|
+ edmac_lli *last_plli = NULL;
|
|
+
|
|
+ if (sgl == NULL) {
|
|
+ edmac_error("sgl is null!\n");
|
|
+ return NULL;
|
|
+ }
|
|
+
|
|
+ tsf_desc = edmac_init_tsf_desc(chan, direction, &slave_addr);
|
|
+ if (!tsf_desc) {
|
|
+ edmac_error("desc init fail\n");
|
|
+ return NULL;
|
|
+ }
|
|
+
|
|
+ tsf_desc->llis_vaddr = dma_pool_alloc(edmac->pool, GFP_NOWAIT, &tsf_desc->llis_busaddr);
|
|
+ if (!tsf_desc->llis_vaddr) {
|
|
+ edmac_free_tsf_desc(edmac, tsf_desc);
|
|
+ edmac_error("malloc memory from pool fail !\n");
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ for_each_sg(sgl, sg, sg_len, tmp) {
|
|
+ addr = sg_dma_address(sg);
|
|
+ length = sg_dma_len(sg);
|
|
+ if (direction == DMA_MEM_TO_DEV) {
|
|
+ src = addr;
|
|
+ dst = slave_addr;
|
|
+ } else if (direction == DMA_DEV_TO_MEM) {
|
|
+ src = slave_addr;
|
|
+ dst = addr;
|
|
+ }
|
|
+ edmac_fill_desc(tsf_desc, src, dst, length, num);
|
|
+ num++;
|
|
+ }
|
|
+
|
|
+ last_plli = (edmac_lli *)((unsigned long)tsf_desc->llis_vaddr + (num - 1) * sizeof(edmac_lli));
|
|
+ last_plli->next_lli |= EDMAC_LLI_DISABLE;
|
|
+ dump_lli(tsf_desc->llis_vaddr, num);
|
|
+
|
|
+ return vchan_tx_prep(&edmac_dma_chan->virt_chan, &tsf_desc->virt_desc, flags);
|
|
+}
|
|
+
|
|
+static struct dma_async_tx_descriptor *edmac_prep_dma_memcpy(
|
|
+ struct dma_chan *chan, dma_addr_t dest, dma_addr_t src,
|
|
+ size_t len, unsigned long flags)
|
|
+{
|
|
+ struct edmac_dma_chan *edmac_dma_chan = to_edamc_chan(chan);
|
|
+ struct edmac_driver_data *edmac = edmac_dma_chan->host;
|
|
+ struct transfer_desc *tsf_desc = NULL;
|
|
+ u32 config = 0;
|
|
+ size_t num = 0;
|
|
+ size_t length = 0;
|
|
+ edmac_lli *last_plli = NULL;
|
|
+
|
|
+ if (!len) {
|
|
+ return NULL;
|
|
+ }
|
|
+
|
|
+ tsf_desc = edmac_get_tsf_desc(edmac);
|
|
+ if (!tsf_desc) {
|
|
+ edmac_error("get tsf desc fail!\n");
|
|
+ return NULL;
|
|
+ }
|
|
+
|
|
+ tsf_desc->llis_vaddr = dma_pool_alloc(edmac->pool, GFP_NOWAIT, &tsf_desc->llis_busaddr);
|
|
+ if (!tsf_desc->llis_vaddr) {
|
|
+ edmac_free_tsf_desc(edmac, tsf_desc);
|
|
+ edmac_error("malloc memory from pool fail !\n");
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ config |= EDMAC_CONFIG_SRC_INC | EDMAC_CONFIG_DST_INC;
|
|
+ config |= EDMAC_CXCONFIG_MEM_TYPE << EDMAC_CXCONFIG_TSF_TYPE_SHIFT;
|
|
+
|
|
+ /* max burst width is 16 ,but reg value set 0xf */
|
|
+ config |= (EDMAC_MAX_BURST_WIDTH - 1) << EDMAC_CONFIG_SRC_BURST_SHIFT;
|
|
+ config |= (EDMAC_MAX_BURST_WIDTH - 1) << EDMAC_CONFIG_DST_BURST_SHIFT;
|
|
+
|
|
+ tsf_desc->ccfg = config;
|
|
+
|
|
+ do {
|
|
+ length = min_t(size_t, len, MAX_TRANSFER_BYTES);
|
|
+ edmac_fill_desc(tsf_desc, src, dest, length, num);
|
|
+
|
|
+ src += length;
|
|
+ dest += length;
|
|
+ len -= length;
|
|
+ num++;
|
|
+ } while(len);
|
|
+
|
|
+ last_plli = (edmac_lli *)((unsigned long)tsf_desc->llis_vaddr + (num - 1) * sizeof(edmac_lli));
|
|
+ last_plli->next_lli |= EDMAC_LLI_DISABLE;
|
|
+ dump_lli(tsf_desc->llis_vaddr, num);
|
|
+
|
|
+ return vchan_tx_prep(&edmac_dma_chan->virt_chan, &tsf_desc->virt_desc, flags);
|
|
+}
|
|
+
|
|
+
|
|
+
|
|
+static struct dma_async_tx_descriptor *edma_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 edmac_dma_chan *edmac_dma_chan = to_edamc_chan(chan);
|
|
+ struct edmac_driver_data *edmac = edmac_dma_chan->host;
|
|
+ struct transfer_desc *tsf_desc;
|
|
+ dma_addr_t src = 0, dst = 0, addr = 0, slave_addr = 0;
|
|
+ size_t length = 0, since = 0, total = 0, num = 0, len = 0;
|
|
+ edmac_lli *last_plli = NULL;
|
|
+ edmac_lli *plli = NULL;
|
|
+
|
|
+ tsf_desc = edmac_init_tsf_desc(chan, direction, &slave_addr);
|
|
+ if (!tsf_desc) {
|
|
+ edmac_error("desc init fail\n");
|
|
+ return NULL;
|
|
+ }
|
|
+
|
|
+ tsf_desc->llis_vaddr = dma_pool_alloc(edmac->pool, GFP_NOWAIT, &tsf_desc->llis_busaddr);
|
|
+ if (!tsf_desc->llis_vaddr) {
|
|
+ edmac_free_tsf_desc(edmac, tsf_desc);
|
|
+ edmac_error("malloc memory from pool fail !\n");
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
+ tsf_desc->cyclic = true;
|
|
+ addr = buf_addr;
|
|
+ total = buf_len;
|
|
+
|
|
+ if (period_len < MAX_TRANSFER_BYTES) {
|
|
+ len = period_len;
|
|
+ }
|
|
+ do {
|
|
+ length = min_t(size_t, total, len);
|
|
+
|
|
+ if (direction == DMA_MEM_TO_DEV) {
|
|
+ src = addr;
|
|
+ dst = slave_addr;
|
|
+ } else if (direction == DMA_DEV_TO_MEM) {
|
|
+ src = slave_addr;
|
|
+ dst = addr;
|
|
+ }
|
|
+
|
|
+ edmac_fill_desc(tsf_desc, src, dst, length, num);
|
|
+
|
|
+ since += length;
|
|
+ if (since >= period_len) {
|
|
+ plli = (edmac_lli *)((unsigned long)tsf_desc->llis_vaddr + (num) * sizeof(edmac_lli));
|
|
+ plli->config |= EDMAC_CXCONFIG_ITC_EN << EDMAC_CXCONFIG_ITC_EN_SHIFT;
|
|
+ since -= period_len;
|
|
+ }
|
|
+ addr += length;
|
|
+ total -= length;
|
|
+ num++;
|
|
+ } while(total);
|
|
+
|
|
+ last_plli = (edmac_lli *)((unsigned long)tsf_desc->llis_vaddr + (num - 1) * sizeof(edmac_lli));
|
|
+
|
|
+ last_plli->next_lli = (unsigned long)(tsf_desc->llis_vaddr);
|
|
+
|
|
+ dump_lli(tsf_desc->llis_vaddr, num);
|
|
+
|
|
+ return vchan_tx_prep(&edmac_dma_chan->virt_chan, &tsf_desc->virt_desc, flags);
|
|
+}
|
|
+
|
|
+
|
|
+static void edmac_phy_reassign(struct edmac_phy_chan *phy_chan,
|
|
+ struct edmac_dma_chan *chan)
|
|
+{
|
|
+ phy_chan->serving = chan;
|
|
+ chan->phychan = phy_chan;
|
|
+ chan->state = EDMAC_CHAN_RUNNING;
|
|
+
|
|
+ edmac_start_next_txd(chan);
|
|
+}
|
|
+
|
|
+static void edmac_terminate_phy_chan(struct edmac_driver_data *edmac,
|
|
+ struct edmac_dma_chan *edmac_dma_chan)
|
|
+{
|
|
+ unsigned int val;
|
|
+ struct edmac_phy_chan *phychan = edmac_dma_chan->phychan;
|
|
+
|
|
+ edmac_pause_phy_chan(edmac_dma_chan);
|
|
+
|
|
+ val = 0x1 << phychan->id;
|
|
+
|
|
+ edmac_writel(val, edmac->base + EDMAC_INT_TC1_RAW);
|
|
+ edmac_writel(val, edmac->base + EDMAC_INT_ERR1_RAW);
|
|
+ edmac_writel(val, edmac->base + EDMAC_INT_ERR2_RAW);
|
|
+}
|
|
+
|
|
+void edmac_phy_free(struct edmac_dma_chan *chan)
|
|
+{
|
|
+ struct edmac_driver_data *edmac = chan->host;
|
|
+ struct edmac_dma_chan *p = NULL;
|
|
+ struct edmac_dma_chan *next = NULL;
|
|
+
|
|
+ list_for_each_entry(p, &edmac->memcpy.channels, virt_chan.chan.device_node) {
|
|
+ if (p->state == EDMAC_CHAN_WAITING) {
|
|
+ next = p;
|
|
+ break;
|
|
+ }
|
|
+ }
|
|
+
|
|
+ if (!next) {
|
|
+ list_for_each_entry(p, &edmac->slave.channels, virt_chan.chan.device_node) {
|
|
+ if (p->state == EDMAC_CHAN_WAITING) {
|
|
+ next = p;
|
|
+ break;
|
|
+ }
|
|
+ }
|
|
+ }
|
|
+ edmac_terminate_phy_chan(edmac, chan);
|
|
+
|
|
+ if (next) {
|
|
+ spin_lock(&next->virt_chan.lock);
|
|
+ edmac_phy_reassign(chan->phychan, next);
|
|
+ spin_unlock(&next->virt_chan.lock);
|
|
+ } else {
|
|
+ chan->phychan->serving = NULL;
|
|
+ }
|
|
+
|
|
+ chan->phychan = NULL;
|
|
+ chan->state = EDMAC_CHAN_IDLE;
|
|
+}
|
|
+
|
|
+static irqreturn_t edmac_irq(int irq, void *dev)
|
|
+{
|
|
+ struct edmac_driver_data *edmac = (struct edmac_driver_data *)dev;
|
|
+ struct edmac_dma_chan *chan = NULL;
|
|
+ struct edmac_phy_chan *phy_chan = NULL;
|
|
+ struct transfer_desc * tsf_desc = NULL;
|
|
+
|
|
+ u32 mask = 0;
|
|
+ unsigned int channel_err_status[3];
|
|
+ unsigned int channel_status = 0;
|
|
+ unsigned int temp = 0;
|
|
+ unsigned int channel_tc_status = -1;
|
|
+ unsigned int i = 0;
|
|
+
|
|
+ channel_status = edmac_readl(edmac->base + EDMAC_INT_STAT);
|
|
+
|
|
+ if (!channel_status) {
|
|
+ edmac_error("channel_status = 0x%x\n", channel_status);
|
|
+ return IRQ_NONE;
|
|
+ }
|
|
+
|
|
+ for (i = 0; i < edmac->channels; i++) {
|
|
+ temp = (channel_status >> i) & 0x1;
|
|
+ if (temp) {
|
|
+ phy_chan = &edmac->phy_chans[i];
|
|
+ chan = phy_chan->serving;
|
|
+ if (!chan) {
|
|
+ edmac_error("error interrupt on chan: %d!\n", i);
|
|
+ continue;
|
|
+ }
|
|
+ tsf_desc = chan->at;
|
|
+
|
|
+ channel_tc_status = edmac_readl(edmac->base + EDMAC_INT_TC1_RAW);
|
|
+ channel_tc_status = (channel_tc_status >> i) & 0x01;
|
|
+ if (channel_tc_status) {
|
|
+ edmac_writel(channel_tc_status << i, edmac->base + EDMAC_INT_TC1_RAW);
|
|
+ }
|
|
+
|
|
+ channel_tc_status = edmac_readl(edmac->base + EDMAC_INT_TC2);
|
|
+ channel_tc_status = (channel_tc_status >> i) & 0x01;
|
|
+ if (channel_tc_status) {
|
|
+ edmac_writel(channel_tc_status << i, edmac->base + EDMAC_INT_TC2_RAW);
|
|
+ }
|
|
+
|
|
+ channel_err_status[0] = edmac_readl(edmac->base + EDMAC_INT_ERR1);
|
|
+ channel_err_status[0] = (channel_err_status[0] >> i) & 0x01;
|
|
+ channel_err_status[1] = edmac_readl(edmac->base + EDMAC_INT_ERR2);
|
|
+ channel_err_status[1] = (channel_err_status[1] >> i) & 0x01;
|
|
+ channel_err_status[2] = edmac_readl(edmac->base + EDMAC_INT_ERR3);
|
|
+ channel_err_status[2] = (channel_err_status[2] >> i) & 0x01;
|
|
+ if (channel_err_status[0] | channel_err_status[1] | channel_err_status[2]) {
|
|
+ edmac_error("Error in edmac %d finish!,ERR1 = 0x%x,ERR2 = 0x%x,ERR3 = 0x%x\n",
|
|
+ i, channel_err_status[0], channel_err_status[1], channel_err_status[2]);
|
|
+ edmac_writel(1 << i, edmac->base + EDMAC_INT_ERR1_RAW);
|
|
+ edmac_writel(1 << i, edmac->base + EDMAC_INT_ERR2_RAW);
|
|
+ edmac_writel(1 << i, edmac->base + EDMAC_INT_ERR3_RAW);
|
|
+ }
|
|
+
|
|
+ spin_lock(&chan->virt_chan.lock);
|
|
+
|
|
+ if (tsf_desc->cyclic) {
|
|
+ vchan_cyclic_callback(&tsf_desc->virt_desc);
|
|
+ spin_unlock(&chan->virt_chan.lock);
|
|
+ continue;
|
|
+ }
|
|
+ chan->at = NULL;
|
|
+ tsf_desc->done = true;
|
|
+ vchan_cookie_complete(&tsf_desc->virt_desc);
|
|
+
|
|
+ if (vchan_next_desc(&chan->virt_chan)) {
|
|
+ edmac_start_next_txd(chan);
|
|
+ } else {
|
|
+ edmac_phy_free(chan);
|
|
+ }
|
|
+
|
|
+ spin_unlock(&chan->virt_chan.lock);
|
|
+ mask |= (1 << i);
|
|
+ }
|
|
+ }
|
|
+
|
|
+ return mask ? IRQ_HANDLED : IRQ_NONE;
|
|
+}
|
|
+
|
|
+static void edmac_dma_slave_init(struct edmac_dma_chan *chan)
|
|
+{
|
|
+ chan->slave = true;
|
|
+}
|
|
+
|
|
+static void edmac_desc_free(struct virt_dma_desc *vd)
|
|
+{
|
|
+ struct transfer_desc *tsf_desc = to_edmac_transfer_desc(&vd->tx);
|
|
+ struct edmac_dma_chan * edmac_dma_chan = to_edamc_chan(vd->tx.chan);
|
|
+
|
|
+ dma_descriptor_unmap(&vd->tx);
|
|
+ edmac_free_tsf_desc(edmac_dma_chan->host, tsf_desc);
|
|
+}
|
|
+
|
|
+static int edmac_init_virt_channels(struct edmac_driver_data *edmac,
|
|
+ struct dma_device *dmadev, unsigned int channels, bool slave)
|
|
+{
|
|
+ struct edmac_dma_chan *chan = NULL;
|
|
+ int i;
|
|
+ INIT_LIST_HEAD(&dmadev->channels);
|
|
+
|
|
+ for (i = 0; i < channels; i++) {
|
|
+ chan = kzalloc(sizeof(struct edmac_dma_chan), GFP_KERNEL);
|
|
+ if (!chan) {
|
|
+ edmac_error("fail to allocate memory for virt channels!");
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ chan->host = edmac;
|
|
+ chan->state = EDMAC_CHAN_IDLE;
|
|
+ chan->signal = -1;
|
|
+
|
|
+ if (slave) {
|
|
+ chan->id = i;
|
|
+ edmac_dma_slave_init(chan);
|
|
+ }
|
|
+ chan->virt_chan.desc_free = edmac_desc_free;
|
|
+ vchan_init(&chan->virt_chan, dmadev);
|
|
+ }
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+void edmac_free_virt_channels(struct dma_device *dmadev)
|
|
+{
|
|
+ struct edmac_dma_chan *chan = NULL;
|
|
+ struct edmac_dma_chan *next = NULL;
|
|
+
|
|
+ list_for_each_entry_safe(chan,
|
|
+ next, &dmadev->channels, virt_chan.chan.device_node) {
|
|
+ list_del(&chan->virt_chan.chan.device_node);
|
|
+ kfree(chan);
|
|
+ }
|
|
+}
|
|
+
|
|
+
|
|
+#define MAX_TSFR_LLIS 32
|
|
+#define EDMACV300_LLI_WORDS 16
|
|
+#define EDMACV300_POOL_ALIGN 16
|
|
+
|
|
+static int __init edmac_probe(struct platform_device *pdev)
|
|
+{
|
|
+
|
|
+ int ret = 0, i = 0;
|
|
+ struct edmac_driver_data *edmac = NULL;
|
|
+ size_t trasfer_size = 0;
|
|
+
|
|
+ ret = dma_set_mask_and_coherent(&(pdev->dev), DMA_BIT_MASK(64));
|
|
+ if (ret) {
|
|
+ return ret;
|
|
+ }
|
|
+
|
|
+ edmac = kzalloc(sizeof(*edmac), GFP_KERNEL);
|
|
+ if (!edmac) {
|
|
+ edmac_error("malloc for edmac fail!");
|
|
+ ret = -ENOMEM;
|
|
+ return ret;
|
|
+ }
|
|
+ edmac->dev = pdev;
|
|
+
|
|
+ ret = get_of_probe(edmac);
|
|
+ if (ret) {
|
|
+ edmac_error("get dts info fail!");
|
|
+ goto free_edmac;
|
|
+ }
|
|
+
|
|
+
|
|
+ clk_prepare_enable(edmac->clk);
|
|
+ clk_prepare_enable(edmac->axi_clk);
|
|
+
|
|
+ reset_control_deassert(edmac->rstc);
|
|
+
|
|
+ dma_cap_set(DMA_MEMCPY, edmac->memcpy.cap_mask);
|
|
+ edmac->memcpy.dev = &pdev->dev;
|
|
+ edmac->memcpy.device_free_chan_resources = edmac_free_chan_resources;
|
|
+ edmac->memcpy.device_prep_dma_memcpy = edmac_prep_dma_memcpy;
|
|
+ edmac->memcpy.device_tx_status = edmac_tx_status;
|
|
+ edmac->memcpy.device_issue_pending = edmac_issue_pending;
|
|
+ edmac->memcpy.device_config = edmac_config;
|
|
+ edmac->memcpy.device_pause = edmac_pause;
|
|
+ edmac->memcpy.device_resume = edmac_resume;
|
|
+ edmac->memcpy.device_terminate_all = edmac_terminate_all;
|
|
+ edmac->memcpy.directions = BIT(DMA_MEM_TO_MEM);
|
|
+ edmac->memcpy.residue_granularity = DMA_RESIDUE_GRANULARITY_SEGMENT;
|
|
+
|
|
+ dma_cap_set(DMA_SLAVE, edmac->slave.cap_mask);
|
|
+ dma_cap_set(DMA_CYCLIC, edmac->slave.cap_mask);
|
|
+ edmac->slave.dev = &pdev->dev;
|
|
+ edmac->slave.device_free_chan_resources = edmac_free_chan_resources;
|
|
+ edmac->slave.device_tx_status = edmac_tx_status;
|
|
+ edmac->slave.device_issue_pending = edmac_issue_pending;
|
|
+ edmac->slave.device_prep_slave_sg = edmac_perp_slave_sg;
|
|
+ edmac->slave.device_prep_dma_cyclic = edma_prep_dma_cyclic;
|
|
+ edmac->slave.device_config = edmac_config;
|
|
+ edmac->slave.device_resume = edmac_resume;
|
|
+ edmac->slave.device_pause = edmac_pause;
|
|
+ edmac->slave.device_terminate_all = edmac_terminate_all;
|
|
+ edmac->slave.directions = BIT(DMA_DEV_TO_MEM) | BIT(DMA_MEM_TO_DEV);
|
|
+ edmac->slave.residue_granularity = DMA_RESIDUE_GRANULARITY_SEGMENT;
|
|
+
|
|
+ edmac->max_transfer_size = MAX_TRANSFER_BYTES;
|
|
+ trasfer_size = MAX_TSFR_LLIS * EDMACV300_LLI_WORDS * sizeof(u32);
|
|
+
|
|
+ edmac->pool = dma_pool_create(DRIVER_NAME, &(pdev->dev),
|
|
+ trasfer_size, EDMACV300_POOL_ALIGN, 0);
|
|
+ if (!edmac->pool) {
|
|
+ edmac_error("create pool fail!");
|
|
+ ret = -ENOMEM;
|
|
+ goto free_edmac;
|
|
+ }
|
|
+
|
|
+ edmac_writel(EDMAC_ALL_CHAN_CLR, edmac->base + EDMAC_INT_TC1_RAW);
|
|
+ edmac_writel(EDMAC_ALL_CHAN_CLR, edmac->base + EDMAC_INT_TC2_RAW);
|
|
+ edmac_writel(EDMAC_ALL_CHAN_CLR, edmac->base + EDMAC_INT_ERR1_RAW);
|
|
+ edmac_writel(EDMAC_ALL_CHAN_CLR, edmac->base + EDMAC_INT_ERR2_RAW);
|
|
+ edmac_writel(EDMAC_ALL_CHAN_CLR, edmac->base + EDMAC_INT_ERR3_RAW);
|
|
+
|
|
+ edmac_writel(EDMAC_INT_ENABLE_ALL_CHAN, edmac->base + EDMAC_INT_TC1_MASK);
|
|
+ edmac_writel(EDMAC_INT_ENABLE_ALL_CHAN, edmac->base + EDMAC_INT_TC2_MASK);
|
|
+ edmac_writel(EDMAC_INT_ENABLE_ALL_CHAN, edmac->base + EDMAC_INT_ERR1_MASK);
|
|
+ edmac_writel(EDMAC_INT_ENABLE_ALL_CHAN, edmac->base + EDMAC_INT_ERR2_MASK);
|
|
+ edmac_writel(EDMAC_INT_ENABLE_ALL_CHAN, edmac->base + EDMAC_INT_ERR3_MASK);
|
|
+
|
|
+ ret = request_irq(edmac->irq, edmac_irq, 0, DRIVER_NAME, edmac);
|
|
+ if (ret) {
|
|
+ edmac_error("fail to request irq");
|
|
+ goto free_pool;
|
|
+ }
|
|
+
|
|
+ edmac->phy_chans = kzalloc((edmac->channels * sizeof(struct edmac_phy_chan)),
|
|
+ GFP_KERNEL);
|
|
+ if (!edmac->phy_chans) {
|
|
+ edmac_error("malloc for phy chans fail!");
|
|
+ ret = -ENOMEM;
|
|
+ goto free_irq_res;
|
|
+ }
|
|
+
|
|
+ /* initialize the phy chan */
|
|
+ for (i = 0; i < edmac->channels; i++) {
|
|
+ struct edmac_phy_chan* phy_ch = &edmac->phy_chans[i];
|
|
+ phy_ch->id = i;
|
|
+ phy_ch->base = edmac->base + EDMAC_Cx_BASE(i);
|
|
+ spin_lock_init(&phy_ch->lock);
|
|
+ phy_ch->serving = NULL;
|
|
+ }
|
|
+
|
|
+ /* initialize the memory virt chan */
|
|
+ ret = edmac_init_virt_channels(edmac, &edmac->memcpy, edmac->channels, false);
|
|
+ if (ret) {
|
|
+ edmac_error("fail to init memory virt channels!");
|
|
+ goto free_phychans;
|
|
+ }
|
|
+
|
|
+ /* initialize the slave virt chan */
|
|
+ ret = edmac_init_virt_channels(edmac, &edmac->slave, edmac->slave_requests, true);
|
|
+ if (ret) {
|
|
+ edmac_error("fail to init slave virt channels!");
|
|
+ goto free_memory_virt_channels;
|
|
+
|
|
+ }
|
|
+
|
|
+ ret = dma_async_device_register(&edmac->memcpy);
|
|
+ if (ret) {
|
|
+ edmac_error(
|
|
+ "%s failed to register memcpy as an async device - %d\n",
|
|
+ __func__, ret);
|
|
+ goto free_slave_virt_channels;
|
|
+ }
|
|
+
|
|
+ ret = dma_async_device_register(&edmac->slave);
|
|
+ if (ret) {
|
|
+ edmac_error(
|
|
+ "%s failed to register slave as an async device - %d\n",
|
|
+ __func__, ret);
|
|
+ goto free_memcpy_device;
|
|
+ }
|
|
+
|
|
+ return 0;
|
|
+
|
|
+free_memcpy_device:
|
|
+ dma_async_device_unregister(&edmac->memcpy);
|
|
+free_slave_virt_channels:
|
|
+ edmac_free_virt_channels(&edmac->slave);
|
|
+free_memory_virt_channels:
|
|
+ edmac_free_virt_channels(&edmac->memcpy);
|
|
+free_phychans:
|
|
+ kfree(edmac->phy_chans);
|
|
+free_irq_res:
|
|
+ free_irq(edmac->irq, edmac);
|
|
+free_pool:
|
|
+ dma_pool_destroy(edmac->pool);
|
|
+free_edmac:
|
|
+ kfree(edmac);
|
|
+
|
|
+ return ret;
|
|
+}
|
|
+
|
|
+
|
|
+static int edma_remove(struct platform_device *pdev)
|
|
+{
|
|
+ int err = 0;
|
|
+ return err;
|
|
+}
|
|
+
|
|
+
|
|
+static const struct of_device_id edmac_match[] = {
|
|
+ { .compatible = "goke,edmac" },
|
|
+ {},
|
|
+};
|
|
+
|
|
+
|
|
+static struct platform_driver edmac_driver = {
|
|
+ .remove = edma_remove,
|
|
+ .driver = {
|
|
+ .name = "edmac",
|
|
+ .of_match_table = edmac_match,
|
|
+ },
|
|
+};
|
|
+
|
|
+static int __init edmac_init(void)
|
|
+{
|
|
+ return platform_driver_probe(&edmac_driver, edmac_probe);
|
|
+}
|
|
+subsys_initcall(edmac_init);
|
|
+
|
|
+static void __exit edmac_exit(void)
|
|
+{
|
|
+ platform_driver_unregister(&edmac_driver);
|
|
+}
|
|
+module_exit(edmac_exit);
|
|
+
|
|
+MODULE_LICENSE("GPL");
|
|
+MODULE_AUTHOR("Goke");
|