| // SPDX-License-Identifier: GPL-2.0 |
| /* |
| * External DMA controller driver for UniPhier SoCs |
| * Copyright 2019 Socionext Inc. |
| * Author: Kunihiko Hayashi <hayashi.kunihiko@socionext.com> |
| */ |
| |
| #include <linux/bitops.h> |
| #include <linux/bitfield.h> |
| #include <linux/iopoll.h> |
| #include <linux/module.h> |
| #include <linux/of.h> |
| #include <linux/of_dma.h> |
| #include <linux/platform_device.h> |
| #include <linux/slab.h> |
| |
| #include "dmaengine.h" |
| #include "virt-dma.h" |
| |
| #define XDMAC_CH_WIDTH 0x100 |
| |
| #define XDMAC_TFA 0x08 |
| #define XDMAC_TFA_MCNT_MASK GENMASK(23, 16) |
| #define XDMAC_TFA_MASK GENMASK(5, 0) |
| #define XDMAC_SADM 0x10 |
| #define XDMAC_SADM_STW_MASK GENMASK(25, 24) |
| #define XDMAC_SADM_SAM BIT(4) |
| #define XDMAC_SADM_SAM_FIXED XDMAC_SADM_SAM |
| #define XDMAC_SADM_SAM_INC 0 |
| #define XDMAC_DADM 0x14 |
| #define XDMAC_DADM_DTW_MASK XDMAC_SADM_STW_MASK |
| #define XDMAC_DADM_DAM XDMAC_SADM_SAM |
| #define XDMAC_DADM_DAM_FIXED XDMAC_SADM_SAM_FIXED |
| #define XDMAC_DADM_DAM_INC XDMAC_SADM_SAM_INC |
| #define XDMAC_EXSAD 0x18 |
| #define XDMAC_EXDAD 0x1c |
| #define XDMAC_SAD 0x20 |
| #define XDMAC_DAD 0x24 |
| #define XDMAC_ITS 0x28 |
| #define XDMAC_ITS_MASK GENMASK(25, 0) |
| #define XDMAC_TNUM 0x2c |
| #define XDMAC_TNUM_MASK GENMASK(15, 0) |
| #define XDMAC_TSS 0x30 |
| #define XDMAC_TSS_REQ BIT(0) |
| #define XDMAC_IEN 0x34 |
| #define XDMAC_IEN_ERRIEN BIT(1) |
| #define XDMAC_IEN_ENDIEN BIT(0) |
| #define XDMAC_STAT 0x40 |
| #define XDMAC_STAT_TENF BIT(0) |
| #define XDMAC_IR 0x44 |
| #define XDMAC_IR_ERRF BIT(1) |
| #define XDMAC_IR_ENDF BIT(0) |
| #define XDMAC_ID 0x48 |
| #define XDMAC_ID_ERRIDF BIT(1) |
| #define XDMAC_ID_ENDIDF BIT(0) |
| |
| #define XDMAC_MAX_CHANS 16 |
| #define XDMAC_INTERVAL_CLKS 20 |
| #define XDMAC_MAX_WORDS XDMAC_TNUM_MASK |
| |
| /* cut lower bit for maintain alignment of maximum transfer size */ |
| #define XDMAC_MAX_WORD_SIZE (XDMAC_ITS_MASK & ~GENMASK(3, 0)) |
| |
| #define UNIPHIER_XDMAC_BUSWIDTHS \ |
| (BIT(DMA_SLAVE_BUSWIDTH_1_BYTE) | \ |
| BIT(DMA_SLAVE_BUSWIDTH_2_BYTES) | \ |
| BIT(DMA_SLAVE_BUSWIDTH_4_BYTES) | \ |
| BIT(DMA_SLAVE_BUSWIDTH_8_BYTES)) |
| |
| struct uniphier_xdmac_desc_node { |
| dma_addr_t src; |
| dma_addr_t dst; |
| u32 burst_size; |
| u32 nr_burst; |
| }; |
| |
| struct uniphier_xdmac_desc { |
| struct virt_dma_desc vd; |
| |
| unsigned int nr_node; |
| unsigned int cur_node; |
| enum dma_transfer_direction dir; |
| struct uniphier_xdmac_desc_node nodes[] __counted_by(nr_node); |
| }; |
| |
| struct uniphier_xdmac_chan { |
| struct virt_dma_chan vc; |
| struct uniphier_xdmac_device *xdev; |
| struct uniphier_xdmac_desc *xd; |
| void __iomem *reg_ch_base; |
| struct dma_slave_config sconfig; |
| int id; |
| unsigned int req_factor; |
| }; |
| |
| struct uniphier_xdmac_device { |
| struct dma_device ddev; |
| void __iomem *reg_base; |
| int nr_chans; |
| struct uniphier_xdmac_chan channels[] __counted_by(nr_chans); |
| }; |
| |
| static struct uniphier_xdmac_chan * |
| to_uniphier_xdmac_chan(struct virt_dma_chan *vc) |
| { |
| return container_of(vc, struct uniphier_xdmac_chan, vc); |
| } |
| |
| static struct uniphier_xdmac_desc * |
| to_uniphier_xdmac_desc(struct virt_dma_desc *vd) |
| { |
| return container_of(vd, struct uniphier_xdmac_desc, vd); |
| } |
| |
| /* xc->vc.lock must be held by caller */ |
| static struct uniphier_xdmac_desc * |
| uniphier_xdmac_next_desc(struct uniphier_xdmac_chan *xc) |
| { |
| struct virt_dma_desc *vd; |
| |
| vd = vchan_next_desc(&xc->vc); |
| if (!vd) |
| return NULL; |
| |
| list_del(&vd->node); |
| |
| return to_uniphier_xdmac_desc(vd); |
| } |
| |
| /* xc->vc.lock must be held by caller */ |
| static void uniphier_xdmac_chan_start(struct uniphier_xdmac_chan *xc, |
| struct uniphier_xdmac_desc *xd) |
| { |
| u32 src_mode, src_width; |
| u32 dst_mode, dst_width; |
| dma_addr_t src_addr, dst_addr; |
| u32 val, its, tnum; |
| enum dma_slave_buswidth buswidth; |
| |
| src_addr = xd->nodes[xd->cur_node].src; |
| dst_addr = xd->nodes[xd->cur_node].dst; |
| its = xd->nodes[xd->cur_node].burst_size; |
| tnum = xd->nodes[xd->cur_node].nr_burst; |
| |
| /* |
| * The width of MEM side must be 4 or 8 bytes, that does not |
| * affect that of DEV side and transfer size. |
| */ |
| if (xd->dir == DMA_DEV_TO_MEM) { |
| src_mode = XDMAC_SADM_SAM_FIXED; |
| buswidth = xc->sconfig.src_addr_width; |
| } else { |
| src_mode = XDMAC_SADM_SAM_INC; |
| buswidth = DMA_SLAVE_BUSWIDTH_8_BYTES; |
| } |
| src_width = FIELD_PREP(XDMAC_SADM_STW_MASK, __ffs(buswidth)); |
| |
| if (xd->dir == DMA_MEM_TO_DEV) { |
| dst_mode = XDMAC_DADM_DAM_FIXED; |
| buswidth = xc->sconfig.dst_addr_width; |
| } else { |
| dst_mode = XDMAC_DADM_DAM_INC; |
| buswidth = DMA_SLAVE_BUSWIDTH_8_BYTES; |
| } |
| dst_width = FIELD_PREP(XDMAC_DADM_DTW_MASK, __ffs(buswidth)); |
| |
| /* setup transfer factor */ |
| val = FIELD_PREP(XDMAC_TFA_MCNT_MASK, XDMAC_INTERVAL_CLKS); |
| val |= FIELD_PREP(XDMAC_TFA_MASK, xc->req_factor); |
| writel(val, xc->reg_ch_base + XDMAC_TFA); |
| |
| /* setup the channel */ |
| writel(lower_32_bits(src_addr), xc->reg_ch_base + XDMAC_SAD); |
| writel(upper_32_bits(src_addr), xc->reg_ch_base + XDMAC_EXSAD); |
| |
| writel(lower_32_bits(dst_addr), xc->reg_ch_base + XDMAC_DAD); |
| writel(upper_32_bits(dst_addr), xc->reg_ch_base + XDMAC_EXDAD); |
| |
| src_mode |= src_width; |
| dst_mode |= dst_width; |
| writel(src_mode, xc->reg_ch_base + XDMAC_SADM); |
| writel(dst_mode, xc->reg_ch_base + XDMAC_DADM); |
| |
| writel(its, xc->reg_ch_base + XDMAC_ITS); |
| writel(tnum, xc->reg_ch_base + XDMAC_TNUM); |
| |
| /* enable interrupt */ |
| writel(XDMAC_IEN_ENDIEN | XDMAC_IEN_ERRIEN, |
| xc->reg_ch_base + XDMAC_IEN); |
| |
| /* start XDMAC */ |
| val = readl(xc->reg_ch_base + XDMAC_TSS); |
| val |= XDMAC_TSS_REQ; |
| writel(val, xc->reg_ch_base + XDMAC_TSS); |
| } |
| |
| /* xc->vc.lock must be held by caller */ |
| static int uniphier_xdmac_chan_stop(struct uniphier_xdmac_chan *xc) |
| { |
| u32 val; |
| |
| /* disable interrupt */ |
| val = readl(xc->reg_ch_base + XDMAC_IEN); |
| val &= ~(XDMAC_IEN_ENDIEN | XDMAC_IEN_ERRIEN); |
| writel(val, xc->reg_ch_base + XDMAC_IEN); |
| |
| /* stop XDMAC */ |
| val = readl(xc->reg_ch_base + XDMAC_TSS); |
| val &= ~XDMAC_TSS_REQ; |
| writel(0, xc->reg_ch_base + XDMAC_TSS); |
| |
| /* wait until transfer is stopped */ |
| return readl_poll_timeout_atomic(xc->reg_ch_base + XDMAC_STAT, val, |
| !(val & XDMAC_STAT_TENF), 100, 1000); |
| } |
| |
| /* xc->vc.lock must be held by caller */ |
| static void uniphier_xdmac_start(struct uniphier_xdmac_chan *xc) |
| { |
| struct uniphier_xdmac_desc *xd; |
| |
| xd = uniphier_xdmac_next_desc(xc); |
| if (xd) |
| uniphier_xdmac_chan_start(xc, xd); |
| |
| /* set desc to chan regardless of xd is null */ |
| xc->xd = xd; |
| } |
| |
| static void uniphier_xdmac_chan_irq(struct uniphier_xdmac_chan *xc) |
| { |
| u32 stat; |
| int ret; |
| |
| spin_lock(&xc->vc.lock); |
| |
| stat = readl(xc->reg_ch_base + XDMAC_ID); |
| |
| if (stat & XDMAC_ID_ERRIDF) { |
| ret = uniphier_xdmac_chan_stop(xc); |
| if (ret) |
| dev_err(xc->xdev->ddev.dev, |
| "DMA transfer error with aborting issue\n"); |
| else |
| dev_err(xc->xdev->ddev.dev, |
| "DMA transfer error\n"); |
| |
| } else if ((stat & XDMAC_ID_ENDIDF) && xc->xd) { |
| xc->xd->cur_node++; |
| if (xc->xd->cur_node >= xc->xd->nr_node) { |
| vchan_cookie_complete(&xc->xd->vd); |
| uniphier_xdmac_start(xc); |
| } else { |
| uniphier_xdmac_chan_start(xc, xc->xd); |
| } |
| } |
| |
| /* write bits to clear */ |
| writel(stat, xc->reg_ch_base + XDMAC_IR); |
| |
| spin_unlock(&xc->vc.lock); |
| } |
| |
| static irqreturn_t uniphier_xdmac_irq_handler(int irq, void *dev_id) |
| { |
| struct uniphier_xdmac_device *xdev = dev_id; |
| int i; |
| |
| for (i = 0; i < xdev->nr_chans; i++) |
| uniphier_xdmac_chan_irq(&xdev->channels[i]); |
| |
| return IRQ_HANDLED; |
| } |
| |
| static void uniphier_xdmac_free_chan_resources(struct dma_chan *chan) |
| { |
| vchan_free_chan_resources(to_virt_chan(chan)); |
| } |
| |
| static struct dma_async_tx_descriptor * |
| uniphier_xdmac_prep_dma_memcpy(struct dma_chan *chan, dma_addr_t dst, |
| dma_addr_t src, size_t len, unsigned long flags) |
| { |
| struct virt_dma_chan *vc = to_virt_chan(chan); |
| struct uniphier_xdmac_desc *xd; |
| unsigned int nr; |
| size_t burst_size, tlen; |
| int i; |
| |
| if (len > XDMAC_MAX_WORD_SIZE * XDMAC_MAX_WORDS) |
| return NULL; |
| |
| nr = 1 + len / XDMAC_MAX_WORD_SIZE; |
| |
| xd = kzalloc(struct_size(xd, nodes, nr), GFP_NOWAIT); |
| if (!xd) |
| return NULL; |
| xd->nr_node = nr; |
| |
| for (i = 0; i < nr; i++) { |
| burst_size = min_t(size_t, len, XDMAC_MAX_WORD_SIZE); |
| xd->nodes[i].src = src; |
| xd->nodes[i].dst = dst; |
| xd->nodes[i].burst_size = burst_size; |
| xd->nodes[i].nr_burst = len / burst_size; |
| tlen = rounddown(len, burst_size); |
| src += tlen; |
| dst += tlen; |
| len -= tlen; |
| } |
| |
| xd->dir = DMA_MEM_TO_MEM; |
| xd->cur_node = 0; |
| |
| return vchan_tx_prep(vc, &xd->vd, flags); |
| } |
| |
| static struct dma_async_tx_descriptor * |
| uniphier_xdmac_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 virt_dma_chan *vc = to_virt_chan(chan); |
| struct uniphier_xdmac_chan *xc = to_uniphier_xdmac_chan(vc); |
| struct uniphier_xdmac_desc *xd; |
| struct scatterlist *sg; |
| enum dma_slave_buswidth buswidth; |
| u32 maxburst; |
| int i; |
| |
| if (!is_slave_direction(direction)) |
| return NULL; |
| |
| if (direction == DMA_DEV_TO_MEM) { |
| buswidth = xc->sconfig.src_addr_width; |
| maxburst = xc->sconfig.src_maxburst; |
| } else { |
| buswidth = xc->sconfig.dst_addr_width; |
| maxburst = xc->sconfig.dst_maxburst; |
| } |
| |
| if (!maxburst) |
| maxburst = 1; |
| if (maxburst > xc->xdev->ddev.max_burst) { |
| dev_err(xc->xdev->ddev.dev, |
| "Exceed maximum number of burst words\n"); |
| return NULL; |
| } |
| |
| xd = kzalloc(struct_size(xd, nodes, sg_len), GFP_NOWAIT); |
| if (!xd) |
| return NULL; |
| xd->nr_node = sg_len; |
| |
| for_each_sg(sgl, sg, sg_len, i) { |
| xd->nodes[i].src = (direction == DMA_DEV_TO_MEM) |
| ? xc->sconfig.src_addr : sg_dma_address(sg); |
| xd->nodes[i].dst = (direction == DMA_MEM_TO_DEV) |
| ? xc->sconfig.dst_addr : sg_dma_address(sg); |
| xd->nodes[i].burst_size = maxburst * buswidth; |
| xd->nodes[i].nr_burst = |
| sg_dma_len(sg) / xd->nodes[i].burst_size; |
| |
| /* |
| * Currently transfer that size doesn't align the unit size |
| * (the number of burst words * bus-width) is not allowed, |
| * because the driver does not support the way to transfer |
| * residue size. As a matter of fact, in order to transfer |
| * arbitrary size, 'src_maxburst' or 'dst_maxburst' of |
| * dma_slave_config must be 1. |
| */ |
| if (sg_dma_len(sg) % xd->nodes[i].burst_size) { |
| dev_err(xc->xdev->ddev.dev, |
| "Unaligned transfer size: %d", sg_dma_len(sg)); |
| kfree(xd); |
| return NULL; |
| } |
| |
| if (xd->nodes[i].nr_burst > XDMAC_MAX_WORDS) { |
| dev_err(xc->xdev->ddev.dev, |
| "Exceed maximum transfer size"); |
| kfree(xd); |
| return NULL; |
| } |
| } |
| |
| xd->dir = direction; |
| xd->cur_node = 0; |
| |
| return vchan_tx_prep(vc, &xd->vd, flags); |
| } |
| |
| static int uniphier_xdmac_slave_config(struct dma_chan *chan, |
| struct dma_slave_config *config) |
| { |
| struct virt_dma_chan *vc = to_virt_chan(chan); |
| struct uniphier_xdmac_chan *xc = to_uniphier_xdmac_chan(vc); |
| |
| memcpy(&xc->sconfig, config, sizeof(*config)); |
| |
| return 0; |
| } |
| |
| static int uniphier_xdmac_terminate_all(struct dma_chan *chan) |
| { |
| struct virt_dma_chan *vc = to_virt_chan(chan); |
| struct uniphier_xdmac_chan *xc = to_uniphier_xdmac_chan(vc); |
| unsigned long flags; |
| int ret = 0; |
| LIST_HEAD(head); |
| |
| spin_lock_irqsave(&vc->lock, flags); |
| |
| if (xc->xd) { |
| vchan_terminate_vdesc(&xc->xd->vd); |
| xc->xd = NULL; |
| ret = uniphier_xdmac_chan_stop(xc); |
| } |
| |
| vchan_get_all_descriptors(vc, &head); |
| |
| spin_unlock_irqrestore(&vc->lock, flags); |
| |
| vchan_dma_desc_free_list(vc, &head); |
| |
| return ret; |
| } |
| |
| static void uniphier_xdmac_synchronize(struct dma_chan *chan) |
| { |
| vchan_synchronize(to_virt_chan(chan)); |
| } |
| |
| static void uniphier_xdmac_issue_pending(struct dma_chan *chan) |
| { |
| struct virt_dma_chan *vc = to_virt_chan(chan); |
| struct uniphier_xdmac_chan *xc = to_uniphier_xdmac_chan(vc); |
| unsigned long flags; |
| |
| spin_lock_irqsave(&vc->lock, flags); |
| |
| if (vchan_issue_pending(vc) && !xc->xd) |
| uniphier_xdmac_start(xc); |
| |
| spin_unlock_irqrestore(&vc->lock, flags); |
| } |
| |
| static void uniphier_xdmac_desc_free(struct virt_dma_desc *vd) |
| { |
| kfree(to_uniphier_xdmac_desc(vd)); |
| } |
| |
| static void uniphier_xdmac_chan_init(struct uniphier_xdmac_device *xdev, |
| int ch) |
| { |
| struct uniphier_xdmac_chan *xc = &xdev->channels[ch]; |
| |
| xc->xdev = xdev; |
| xc->reg_ch_base = xdev->reg_base + XDMAC_CH_WIDTH * ch; |
| xc->vc.desc_free = uniphier_xdmac_desc_free; |
| |
| vchan_init(&xc->vc, &xdev->ddev); |
| } |
| |
| static struct dma_chan *of_dma_uniphier_xlate(struct of_phandle_args *dma_spec, |
| struct of_dma *ofdma) |
| { |
| struct uniphier_xdmac_device *xdev = ofdma->of_dma_data; |
| int chan_id = dma_spec->args[0]; |
| |
| if (chan_id >= xdev->nr_chans) |
| return NULL; |
| |
| xdev->channels[chan_id].id = chan_id; |
| xdev->channels[chan_id].req_factor = dma_spec->args[1]; |
| |
| return dma_get_slave_channel(&xdev->channels[chan_id].vc.chan); |
| } |
| |
| static int uniphier_xdmac_probe(struct platform_device *pdev) |
| { |
| struct uniphier_xdmac_device *xdev; |
| struct device *dev = &pdev->dev; |
| struct dma_device *ddev; |
| int irq; |
| int nr_chans; |
| int i, ret; |
| |
| if (of_property_read_u32(dev->of_node, "dma-channels", &nr_chans)) |
| return -EINVAL; |
| if (nr_chans > XDMAC_MAX_CHANS) |
| nr_chans = XDMAC_MAX_CHANS; |
| |
| xdev = devm_kzalloc(dev, struct_size(xdev, channels, nr_chans), |
| GFP_KERNEL); |
| if (!xdev) |
| return -ENOMEM; |
| |
| xdev->nr_chans = nr_chans; |
| xdev->reg_base = devm_platform_ioremap_resource(pdev, 0); |
| if (IS_ERR(xdev->reg_base)) |
| return PTR_ERR(xdev->reg_base); |
| |
| ddev = &xdev->ddev; |
| ddev->dev = dev; |
| dma_cap_zero(ddev->cap_mask); |
| dma_cap_set(DMA_MEMCPY, ddev->cap_mask); |
| dma_cap_set(DMA_SLAVE, ddev->cap_mask); |
| ddev->src_addr_widths = UNIPHIER_XDMAC_BUSWIDTHS; |
| ddev->dst_addr_widths = UNIPHIER_XDMAC_BUSWIDTHS; |
| ddev->directions = BIT(DMA_DEV_TO_MEM) | BIT(DMA_MEM_TO_DEV) | |
| BIT(DMA_MEM_TO_MEM); |
| ddev->residue_granularity = DMA_RESIDUE_GRANULARITY_BURST; |
| ddev->max_burst = XDMAC_MAX_WORDS; |
| ddev->device_free_chan_resources = uniphier_xdmac_free_chan_resources; |
| ddev->device_prep_dma_memcpy = uniphier_xdmac_prep_dma_memcpy; |
| ddev->device_prep_slave_sg = uniphier_xdmac_prep_slave_sg; |
| ddev->device_config = uniphier_xdmac_slave_config; |
| ddev->device_terminate_all = uniphier_xdmac_terminate_all; |
| ddev->device_synchronize = uniphier_xdmac_synchronize; |
| ddev->device_tx_status = dma_cookie_status; |
| ddev->device_issue_pending = uniphier_xdmac_issue_pending; |
| INIT_LIST_HEAD(&ddev->channels); |
| |
| for (i = 0; i < nr_chans; i++) |
| uniphier_xdmac_chan_init(xdev, i); |
| |
| irq = platform_get_irq(pdev, 0); |
| if (irq < 0) |
| return irq; |
| |
| ret = devm_request_irq(dev, irq, uniphier_xdmac_irq_handler, |
| IRQF_SHARED, "xdmac", xdev); |
| if (ret) { |
| dev_err(dev, "Failed to request IRQ\n"); |
| return ret; |
| } |
| |
| ret = dma_async_device_register(ddev); |
| if (ret) { |
| dev_err(dev, "Failed to register XDMA device\n"); |
| return ret; |
| } |
| |
| ret = of_dma_controller_register(dev->of_node, |
| of_dma_uniphier_xlate, xdev); |
| if (ret) { |
| dev_err(dev, "Failed to register XDMA controller\n"); |
| goto out_unregister_dmac; |
| } |
| |
| platform_set_drvdata(pdev, xdev); |
| |
| dev_info(&pdev->dev, "UniPhier XDMAC driver (%d channels)\n", |
| nr_chans); |
| |
| return 0; |
| |
| out_unregister_dmac: |
| dma_async_device_unregister(ddev); |
| |
| return ret; |
| } |
| |
| static void uniphier_xdmac_remove(struct platform_device *pdev) |
| { |
| struct uniphier_xdmac_device *xdev = platform_get_drvdata(pdev); |
| struct dma_device *ddev = &xdev->ddev; |
| struct dma_chan *chan; |
| int ret; |
| |
| /* |
| * Before reaching here, almost all descriptors have been freed by the |
| * ->device_free_chan_resources() hook. However, each channel might |
| * be still holding one descriptor that was on-flight at that moment. |
| * Terminate it to make sure this hardware is no longer running. Then, |
| * free the channel resources once again to avoid memory leak. |
| */ |
| list_for_each_entry(chan, &ddev->channels, device_node) { |
| ret = dmaengine_terminate_sync(chan); |
| if (ret) { |
| /* |
| * This results in resource leakage and maybe also |
| * use-after-free errors as e.g. *xdev is kfreed. |
| */ |
| dev_alert(&pdev->dev, "Failed to terminate channel %d (%pe)\n", |
| chan->chan_id, ERR_PTR(ret)); |
| return; |
| } |
| uniphier_xdmac_free_chan_resources(chan); |
| } |
| |
| of_dma_controller_free(pdev->dev.of_node); |
| dma_async_device_unregister(ddev); |
| } |
| |
| static const struct of_device_id uniphier_xdmac_match[] = { |
| { .compatible = "socionext,uniphier-xdmac" }, |
| { /* sentinel */ } |
| }; |
| MODULE_DEVICE_TABLE(of, uniphier_xdmac_match); |
| |
| static struct platform_driver uniphier_xdmac_driver = { |
| .probe = uniphier_xdmac_probe, |
| .remove = uniphier_xdmac_remove, |
| .driver = { |
| .name = "uniphier-xdmac", |
| .of_match_table = uniphier_xdmac_match, |
| }, |
| }; |
| module_platform_driver(uniphier_xdmac_driver); |
| |
| MODULE_AUTHOR("Kunihiko Hayashi <hayashi.kunihiko@socionext.com>"); |
| MODULE_DESCRIPTION("UniPhier external DMA controller driver"); |
| MODULE_LICENSE("GPL v2"); |