Commit ec5b103e authored by Linus Torvalds's avatar Linus Torvalds

Merge branch 'for-linus' of git://git.infradead.org/users/vkoul/slave-dma

Pull slave-dmaengine updates from Vinod Koul:
 "This pull brings:
   - Andy's DW driver updates
   - Guennadi's sh driver updates
   - Pl08x driver fixes from Tomasz & Alban
   - Improvements to mmp_pdma by Daniel
   - TI EDMA fixes by Joel
   - New drivers:
     - Hisilicon k3dma driver
     - Renesas rcar dma driver
  - New API for publishing slave driver capablities
  - Various fixes across the subsystem by Andy, Jingoo, Sachin etc..."

* 'for-linus' of git://git.infradead.org/users/vkoul/slave-dma: (94 commits)
  dma: edma: Remove limits on number of slots
  dma: edma: Leave linked to Null slot instead of DUMMY slot
  dma: edma: Find missed events and issue them
  ARM: edma: Add function to manually trigger an EDMA channel
  dma: edma: Write out and handle MAX_NR_SG at a given time
  dma: edma: Setup parameters to DMA MAX_NR_SG at a time
  dmaengine: pl330: use dma_set_max_seg_size to set the sg limit
  dmaengine: dma_slave_caps: remove sg entries
  dma: replace devm_request_and_ioremap by devm_ioremap_resource
  dma: ste_dma40: Fix potential null pointer dereference
  dma: ste_dma40: Remove duplicate const
  dma: imx-dma: Remove redundant NULL check
  dma: dmagengine: fix function names in comments
  dma: add driver for R-Car HPB-DMAC
  dma: k3dma: use devm_ioremap_resource() instead of devm_request_and_ioremap()
  dma: imx-sdma: Staticize sdma_driver_data structures
  pch_dma: Add MODULE_DEVICE_TABLE
  dmaengine: PL08x: Add cyclic transfer support
  dmaengine: PL08x: Fix reading the byte count in cctl
  dmaengine: PL08x: Add support for different maximum transfer size
  ...
parents d0048f0b 5622ff1a
* Freescale Smart Direct Memory Access (SDMA) Controller for i.MX * Freescale Smart Direct Memory Access (SDMA) Controller for i.MX
Required properties: Required properties:
- compatible : Should be "fsl,<chip>-sdma" - compatible : Should be "fsl,imx31-sdma", "fsl,imx31-to1-sdma",
"fsl,imx31-to2-sdma", "fsl,imx35-sdma", "fsl,imx35-to1-sdma",
"fsl,imx35-to2-sdma", "fsl,imx51-sdma", "fsl,imx53-sdma" or
"fsl,imx6q-sdma". The -to variants should be preferred since they
allow to determnine the correct ROM script addresses needed for
the driver to work without additional firmware.
- reg : Should contain SDMA registers location and length - reg : Should contain SDMA registers location and length
- interrupts : Should contain SDMA interrupt - interrupts : Should contain SDMA interrupt
- #dma-cells : Must be <3>. - #dma-cells : Must be <3>.
......
* Hisilicon K3 DMA controller
See dma.txt first
Required properties:
- compatible: Should be "hisilicon,k3-dma-1.0"
- reg: Should contain DMA registers location and length.
- interrupts: Should contain one interrupt shared by all channel
- #dma-cells: see dma.txt, should be 1, para number
- dma-channels: physical channels supported
- dma-requests: virtual channels supported, each virtual channel
have specific request line
- clocks: clock required
Example:
Controller:
dma0: dma@fcd02000 {
compatible = "hisilicon,k3-dma-1.0";
reg = <0xfcd02000 0x1000>;
#dma-cells = <1>;
dma-channels = <16>;
dma-requests = <27>;
interrupts = <0 12 4>;
clocks = <&pclk>;
status = "disable";
};
Client:
Use specific request line passing from dmax
For example, i2c0 read channel request line is 18, while write channel use 19
i2c0: i2c@fcb08000 {
compatible = "snps,designware-i2c";
dmas = <&dma0 18 /* read channel */
&dma0 19>; /* write channel */
dma-names = "rx", "tx";
};
i2c1: i2c@fcb09000 {
compatible = "snps,designware-i2c";
dmas = <&dma0 20 /* read channel */
&dma0 21>; /* write channel */
dma-names = "rx", "tx";
};
...@@ -22,42 +22,51 @@ Optional properties (currently unused): ...@@ -22,42 +22,51 @@ Optional properties (currently unused):
* DMA controller * DMA controller
Required properties: Required properties:
- compatible: should be "renesas,shdma" - compatible: should be of the form "renesas,shdma-<soc>", where <soc> should
be replaced with the desired SoC model, e.g.
"renesas,shdma-r8a73a4" for the system DMAC on r8a73a4 SoC
Example: Example:
dmac: dma-mux0 { dmac: dma-multiplexer@0 {
compatible = "renesas,shdma-mux"; compatible = "renesas,shdma-mux";
#dma-cells = <1>; #dma-cells = <1>;
dma-channels = <6>; dma-channels = <20>;
dma-requests = <256>; dma-requests = <256>;
reg = <0 0>; /* Needed for AUXDATA */ #address-cells = <2>;
#address-cells = <1>; #size-cells = <2>;
#size-cells = <1>;
ranges; ranges;
dma0: shdma@fe008020 { dma0: dma-controller@e6700020 {
compatible = "renesas,shdma"; compatible = "renesas,shdma-r8a73a4";
reg = <0xfe008020 0x270>, reg = <0 0xe6700020 0 0x89e0>;
<0xfe009000 0xc>;
interrupt-parent = <&gic>; interrupt-parent = <&gic>;
interrupts = <0 34 4 interrupts = <0 220 4
0 28 4 0 200 4
0 29 4 0 201 4
0 30 4 0 202 4
0 31 4 0 203 4
0 32 4 0 204 4
0 33 4>; 0 205 4
0 206 4
0 207 4
0 208 4
0 209 4
0 210 4
0 211 4
0 212 4
0 213 4
0 214 4
0 215 4
0 216 4
0 217 4
0 218 4
0 219 4>;
interrupt-names = "error", interrupt-names = "error",
"ch0", "ch1", "ch2", "ch3", "ch0", "ch1", "ch2", "ch3",
"ch4", "ch5"; "ch4", "ch5", "ch6", "ch7",
}; "ch8", "ch9", "ch10", "ch11",
"ch12", "ch13", "ch14", "ch15",
dma1: shdma@fe018020 { "ch16", "ch17", "ch18", "ch19";
...
};
dma2: shdma@fe028020 {
...
}; };
}; };
......
...@@ -299,3 +299,6 @@ PWM ...@@ -299,3 +299,6 @@ PWM
PHY PHY
devm_usb_get_phy() devm_usb_get_phy()
devm_usb_put_phy() devm_usb_put_phy()
SLAVE DMA ENGINE
devm_acpi_dma_controller_register()
...@@ -7204,6 +7204,7 @@ F: drivers/tty/serial ...@@ -7204,6 +7204,7 @@ F: drivers/tty/serial
SYNOPSYS DESIGNWARE DMAC DRIVER SYNOPSYS DESIGNWARE DMAC DRIVER
M: Viresh Kumar <viresh.linux@gmail.com> M: Viresh Kumar <viresh.linux@gmail.com>
M: Andy Shevchenko <andriy.shevchenko@linux.intel.com>
S: Maintained S: Maintained
F: include/linux/dw_dmac.h F: include/linux/dw_dmac.h
F: drivers/dma/dw/ F: drivers/dma/dw/
......
...@@ -1235,6 +1235,23 @@ void edma_resume(unsigned channel) ...@@ -1235,6 +1235,23 @@ void edma_resume(unsigned channel)
} }
EXPORT_SYMBOL(edma_resume); EXPORT_SYMBOL(edma_resume);
int edma_trigger_channel(unsigned channel)
{
unsigned ctlr;
unsigned int mask;
ctlr = EDMA_CTLR(channel);
channel = EDMA_CHAN_SLOT(channel);
mask = BIT(channel & 0x1f);
edma_shadow0_write_array(ctlr, SH_ESR, (channel >> 5), mask);
pr_debug("EDMA: ESR%d %08x\n", (channel >> 5),
edma_shadow0_read_array(ctlr, SH_ESR, (channel >> 5)));
return 0;
}
EXPORT_SYMBOL(edma_trigger_channel);
/** /**
* edma_start - start dma on a channel * edma_start - start dma on a channel
* @channel: channel being activated * @channel: channel being activated
......
...@@ -61,25 +61,8 @@ void __init mx25_init_irq(void) ...@@ -61,25 +61,8 @@ void __init mx25_init_irq(void)
mxc_init_irq(MX25_IO_ADDRESS(MX25_AVIC_BASE_ADDR)); mxc_init_irq(MX25_IO_ADDRESS(MX25_AVIC_BASE_ADDR));
} }
static struct sdma_script_start_addrs imx25_sdma_script __initdata = {
.ap_2_ap_addr = 729,
.uart_2_mcu_addr = 904,
.per_2_app_addr = 1255,
.mcu_2_app_addr = 834,
.uartsh_2_mcu_addr = 1120,
.per_2_shp_addr = 1329,
.mcu_2_shp_addr = 1048,
.ata_2_mcu_addr = 1560,
.mcu_2_ata_addr = 1479,
.app_2_per_addr = 1189,
.app_2_mcu_addr = 770,
.shp_2_per_addr = 1407,
.shp_2_mcu_addr = 979,
};
static struct sdma_platform_data imx25_sdma_pdata __initdata = { static struct sdma_platform_data imx25_sdma_pdata __initdata = {
.fw_name = "sdma-imx25.bin", .fw_name = "sdma-imx25.bin",
.script_addrs = &imx25_sdma_script,
}; };
static const struct resource imx25_audmux_res[] __initconst = { static const struct resource imx25_audmux_res[] __initconst = {
......
...@@ -103,22 +103,8 @@ void __init mx53_init_irq(void) ...@@ -103,22 +103,8 @@ void __init mx53_init_irq(void)
tzic_init_irq(MX53_IO_ADDRESS(MX53_TZIC_BASE_ADDR)); tzic_init_irq(MX53_IO_ADDRESS(MX53_TZIC_BASE_ADDR));
} }
static struct sdma_script_start_addrs imx51_sdma_script __initdata = {
.ap_2_ap_addr = 642,
.uart_2_mcu_addr = 817,
.mcu_2_app_addr = 747,
.mcu_2_shp_addr = 961,
.ata_2_mcu_addr = 1473,
.mcu_2_ata_addr = 1392,
.app_2_per_addr = 1033,
.app_2_mcu_addr = 683,
.shp_2_per_addr = 1251,
.shp_2_mcu_addr = 892,
};
static struct sdma_platform_data imx51_sdma_pdata __initdata = { static struct sdma_platform_data imx51_sdma_pdata __initdata = {
.fw_name = "sdma-imx51.bin", .fw_name = "sdma-imx51.bin",
.script_addrs = &imx51_sdma_script,
}; };
static const struct resource imx51_audmux_res[] __initconst = { static const struct resource imx51_audmux_res[] __initconst = {
......
...@@ -308,6 +308,15 @@ config DMA_JZ4740 ...@@ -308,6 +308,15 @@ config DMA_JZ4740
select DMA_ENGINE select DMA_ENGINE
select DMA_VIRTUAL_CHANNELS select DMA_VIRTUAL_CHANNELS
config K3_DMA
tristate "Hisilicon K3 DMA support"
depends on ARCH_HI3xxx
select DMA_ENGINE
select DMA_VIRTUAL_CHANNELS
help
Support the DMA engine for Hisilicon K3 platform
devices.
config DMA_ENGINE config DMA_ENGINE
bool bool
......
...@@ -40,3 +40,4 @@ obj-$(CONFIG_DMA_OMAP) += omap-dma.o ...@@ -40,3 +40,4 @@ obj-$(CONFIG_DMA_OMAP) += omap-dma.o
obj-$(CONFIG_MMP_PDMA) += mmp_pdma.o obj-$(CONFIG_MMP_PDMA) += mmp_pdma.o
obj-$(CONFIG_DMA_JZ4740) += dma-jz4740.o obj-$(CONFIG_DMA_JZ4740) += dma-jz4740.o
obj-$(CONFIG_TI_CPPI41) += cppi41.o obj-$(CONFIG_TI_CPPI41) += cppi41.o
obj-$(CONFIG_K3_DMA) += k3dma.o
...@@ -43,7 +43,6 @@ static int acpi_dma_parse_resource_group(const struct acpi_csrt_group *grp, ...@@ -43,7 +43,6 @@ static int acpi_dma_parse_resource_group(const struct acpi_csrt_group *grp,
struct list_head resource_list; struct list_head resource_list;
struct resource_list_entry *rentry; struct resource_list_entry *rentry;
resource_size_t mem = 0, irq = 0; resource_size_t mem = 0, irq = 0;
u32 vendor_id;
int ret; int ret;
if (grp->shared_info_length != sizeof(struct acpi_csrt_shared_info)) if (grp->shared_info_length != sizeof(struct acpi_csrt_shared_info))
...@@ -73,9 +72,8 @@ static int acpi_dma_parse_resource_group(const struct acpi_csrt_group *grp, ...@@ -73,9 +72,8 @@ static int acpi_dma_parse_resource_group(const struct acpi_csrt_group *grp,
if (si->mmio_base_low != mem || si->gsi_interrupt != irq) if (si->mmio_base_low != mem || si->gsi_interrupt != irq)
return 0; return 0;
vendor_id = le32_to_cpu(grp->vendor_id);
dev_dbg(&adev->dev, "matches with %.4s%04X (rev %u)\n", dev_dbg(&adev->dev, "matches with %.4s%04X (rev %u)\n",
(char *)&vendor_id, grp->device_id, grp->revision); (char *)&grp->vendor_id, grp->device_id, grp->revision);
/* Check if the request line range is available */ /* Check if the request line range is available */
if (si->base_request_line == 0 && si->num_handshake_signals == 0) if (si->base_request_line == 0 && si->num_handshake_signals == 0)
......
This diff is collapsed.
...@@ -509,7 +509,33 @@ static struct dma_chan *private_candidate(const dma_cap_mask_t *mask, ...@@ -509,7 +509,33 @@ static struct dma_chan *private_candidate(const dma_cap_mask_t *mask,
} }
/** /**
* dma_request_channel - try to allocate an exclusive channel * dma_request_slave_channel - try to get specific channel exclusively
* @chan: target channel
*/
struct dma_chan *dma_get_slave_channel(struct dma_chan *chan)
{
int err = -EBUSY;
/* lock against __dma_request_channel */
mutex_lock(&dma_list_mutex);
if (chan->client_count == 0) {
err = dma_chan_get(chan);
if (err)
pr_debug("%s: failed to get %s: (%d)\n",
__func__, dma_chan_name(chan), err);
} else
chan = NULL;
mutex_unlock(&dma_list_mutex);
return chan;
}
EXPORT_SYMBOL_GPL(dma_get_slave_channel);
/**
* __dma_request_channel - try to allocate an exclusive channel
* @mask: capabilities that the channel must satisfy * @mask: capabilities that the channel must satisfy
* @fn: optional callback to disposition available channels * @fn: optional callback to disposition available channels
* @fn_param: opaque parameter to pass to dma_filter_fn * @fn_param: opaque parameter to pass to dma_filter_fn
......
...@@ -37,16 +37,22 @@ ...@@ -37,16 +37,22 @@
* which does not support descriptor writeback. * which does not support descriptor writeback.
*/ */
static inline bool is_request_line_unset(struct dw_dma_chan *dwc)
{
return dwc->request_line == (typeof(dwc->request_line))~0;
}
static inline void dwc_set_masters(struct dw_dma_chan *dwc) static inline void dwc_set_masters(struct dw_dma_chan *dwc)
{ {
struct dw_dma *dw = to_dw_dma(dwc->chan.device); struct dw_dma *dw = to_dw_dma(dwc->chan.device);
struct dw_dma_slave *dws = dwc->chan.private; struct dw_dma_slave *dws = dwc->chan.private;
unsigned char mmax = dw->nr_masters - 1; unsigned char mmax = dw->nr_masters - 1;
if (dwc->request_line == ~0) { if (!is_request_line_unset(dwc))
return;
dwc->src_master = min_t(unsigned char, mmax, dwc_get_sms(dws)); dwc->src_master = min_t(unsigned char, mmax, dwc_get_sms(dws));
dwc->dst_master = min_t(unsigned char, mmax, dwc_get_dms(dws)); dwc->dst_master = min_t(unsigned char, mmax, dwc_get_dms(dws));
}
} }
#define DWC_DEFAULT_CTLLO(_chan) ({ \ #define DWC_DEFAULT_CTLLO(_chan) ({ \
...@@ -644,10 +650,13 @@ static void dw_dma_tasklet(unsigned long data) ...@@ -644,10 +650,13 @@ static void dw_dma_tasklet(unsigned long data)
static irqreturn_t dw_dma_interrupt(int irq, void *dev_id) static irqreturn_t dw_dma_interrupt(int irq, void *dev_id)
{ {
struct dw_dma *dw = dev_id; struct dw_dma *dw = dev_id;
u32 status; u32 status = dma_readl(dw, STATUS_INT);
dev_vdbg(dw->dma.dev, "%s: status=0x%x\n", __func__, dev_vdbg(dw->dma.dev, "%s: status=0x%x\n", __func__, status);
dma_readl(dw, STATUS_INT));
/* Check if we have any interrupt from the DMAC */
if (!status)
return IRQ_NONE;
/* /*
* Just disable the interrupts. We'll turn them back on in the * Just disable the interrupts. We'll turn them back on in the
...@@ -984,7 +993,7 @@ set_runtime_config(struct dma_chan *chan, struct dma_slave_config *sconfig) ...@@ -984,7 +993,7 @@ set_runtime_config(struct dma_chan *chan, struct dma_slave_config *sconfig)
dwc->direction = sconfig->direction; dwc->direction = sconfig->direction;
/* Take the request line from slave_id member */ /* Take the request line from slave_id member */
if (dwc->request_line == ~0) if (is_request_line_unset(dwc))
dwc->request_line = sconfig->slave_id; dwc->request_line = sconfig->slave_id;
convert_burst(&dwc->dma_sconfig.src_maxburst); convert_burst(&dwc->dma_sconfig.src_maxburst);
...@@ -1089,16 +1098,16 @@ dwc_tx_status(struct dma_chan *chan, ...@@ -1089,16 +1098,16 @@ dwc_tx_status(struct dma_chan *chan,
enum dma_status ret; enum dma_status ret;
ret = dma_cookie_status(chan, cookie, txstate); ret = dma_cookie_status(chan, cookie, txstate);
if (ret != DMA_SUCCESS) { if (ret == DMA_SUCCESS)
return ret;
dwc_scan_descriptors(to_dw_dma(chan->device), dwc); dwc_scan_descriptors(to_dw_dma(chan->device), dwc);
ret = dma_cookie_status(chan, cookie, txstate); ret = dma_cookie_status(chan, cookie, txstate);
}
if (ret != DMA_SUCCESS) if (ret != DMA_SUCCESS)
dma_set_residue(txstate, dwc_get_residue(dwc)); dma_set_residue(txstate, dwc_get_residue(dwc));
if (dwc->paused) if (dwc->paused && ret == DMA_IN_PROGRESS)
return DMA_PAUSED; return DMA_PAUSED;
return ret; return ret;
...@@ -1560,8 +1569,8 @@ int dw_dma_probe(struct dw_dma_chip *chip, struct dw_dma_platform_data *pdata) ...@@ -1560,8 +1569,8 @@ int dw_dma_probe(struct dw_dma_chip *chip, struct dw_dma_platform_data *pdata)
/* Disable BLOCK interrupts as well */ /* Disable BLOCK interrupts as well */
channel_clear_bit(dw, MASK.BLOCK, dw->all_chan_mask); channel_clear_bit(dw, MASK.BLOCK, dw->all_chan_mask);
err = devm_request_irq(chip->dev, chip->irq, dw_dma_interrupt, 0, err = devm_request_irq(chip->dev, chip->irq, dw_dma_interrupt,
"dw_dmac", dw); IRQF_SHARED, "dw_dmac", dw);
if (err) if (err)
return err; return err;
......
...@@ -253,6 +253,7 @@ static const struct acpi_device_id dw_dma_acpi_id_table[] = { ...@@ -253,6 +253,7 @@ static const struct acpi_device_id dw_dma_acpi_id_table[] = {
{ "INTL9C60", 0 }, { "INTL9C60", 0 },
{ } { }
}; };
MODULE_DEVICE_TABLE(acpi, dw_dma_acpi_id_table);
#endif #endif
#ifdef CONFIG_PM_SLEEP #ifdef CONFIG_PM_SLEEP
......
...@@ -56,6 +56,7 @@ struct edma_desc { ...@@ -56,6 +56,7 @@ struct edma_desc {
struct list_head node; struct list_head node;
int absync; int absync;
int pset_nr; int pset_nr;
int processed;
struct edmacc_param pset[0]; struct edmacc_param pset[0];
}; };
...@@ -69,6 +70,7 @@ struct edma_chan { ...@@ -69,6 +70,7 @@ struct edma_chan {
int ch_num; int ch_num;
bool alloced; bool alloced;
int slot[EDMA_MAX_SLOTS]; int slot[EDMA_MAX_SLOTS];
int missed;
struct dma_slave_config cfg; struct dma_slave_config cfg;
}; };
...@@ -104,22 +106,34 @@ static void edma_desc_free(struct virt_dma_desc *vdesc) ...@@ -104,22 +106,34 @@ static void edma_desc_free(struct virt_dma_desc *vdesc)
/* Dispatch a queued descriptor to the controller (caller holds lock) */ /* Dispatch a queued descriptor to the controller (caller holds lock) */
static void edma_execute(struct edma_chan *echan) static void edma_execute(struct edma_chan *echan)
{ {
struct virt_dma_desc *vdesc = vchan_next_desc(&echan->vchan); struct virt_dma_desc *vdesc;
struct edma_desc *edesc; struct edma_desc *edesc;
int i; struct device *dev = echan->vchan.chan.device->dev;
int i, j, left, nslots;
/* If either we processed all psets or we're still not started */
if (!echan->edesc ||
echan->edesc->pset_nr == echan->edesc->processed) {
/* Get next vdesc */
vdesc = vchan_next_desc(&echan->vchan);
if (!vdesc) { if (!vdesc) {
echan->edesc = NULL; echan->edesc = NULL;
return; return;
} }
list_del(&vdesc->node); list_del(&vdesc->node);
echan->edesc = to_edma_desc(&vdesc->tx);
}
edesc = echan->edesc;
echan->edesc = edesc = to_edma_desc(&vdesc->tx); /* Find out how many left */
left = edesc->pset_nr - edesc->processed;
nslots = min(MAX_NR_SG, left);
/* Write descriptor PaRAM set(s) */ /* Write descriptor PaRAM set(s) */
for (i = 0; i < edesc->pset_nr; i++) { for (i = 0; i < nslots; i++) {
edma_write_slot(echan->slot[i], &edesc->pset[i]); j = i + edesc->processed;
edma_write_slot(echan->slot[i], &edesc->pset[j]);
dev_dbg(echan->vchan.chan.device->dev, dev_dbg(echan->vchan.chan.device->dev,
"\n pset[%d]:\n" "\n pset[%d]:\n"
" chnum\t%d\n" " chnum\t%d\n"
...@@ -132,24 +146,50 @@ static void edma_execute(struct edma_chan *echan) ...@@ -132,24 +146,50 @@ static void edma_execute(struct edma_chan *echan)
" bidx\t%08x\n" " bidx\t%08x\n"
" cidx\t%08x\n" " cidx\t%08x\n"
" lkrld\t%08x\n", " lkrld\t%08x\n",
i, echan->ch_num, echan->slot[i], j, echan->ch_num, echan->slot[i],
edesc->pset[i].opt, edesc->pset[j].opt,
edesc->pset[i].src, edesc->pset[j].src,
edesc->pset[i].dst, edesc->pset[j].dst,
edesc->pset[i].a_b_cnt, edesc->pset[j].a_b_cnt,
edesc->pset[i].ccnt, edesc->pset[j].ccnt,
edesc->pset[i].src_dst_bidx, edesc->pset[j].src_dst_bidx,
edesc->pset[i].src_dst_cidx, edesc->pset[j].src_dst_cidx,
edesc->pset[i].link_bcntrld); edesc->pset[j].link_bcntrld);
/* Link to the previous slot if not the last set */ /* Link to the previous slot if not the last set */
if (i != (edesc->pset_nr - 1)) if (i != (nslots - 1))
edma_link(echan->slot[i], echan->slot[i+1]); edma_link(echan->slot[i], echan->slot[i+1]);
/* Final pset links to the dummy pset */
else
edma_link(echan->slot[i], echan->ecc->dummy_slot);
} }
edesc->processed += nslots;
/*
* If this is either the last set in a set of SG-list transactions
* then setup a link to the dummy slot, this results in all future
* events being absorbed and that's OK because we're done
*/
if (edesc->processed == edesc->pset_nr)
edma_link(echan->slot[nslots-1], echan->ecc->dummy_slot);
edma_resume(echan->ch_num);
if (edesc->processed <= MAX_NR_SG) {
dev_dbg(dev, "first transfer starting %d\n", echan->ch_num);
edma_start(echan->ch_num);
}
/*
* This happens due to setup times between intermediate transfers
* in long SG lists which have to be broken up into transfers of
* MAX_NR_SG
*/
if (echan->missed) {
dev_dbg(dev, "missed event in execute detected\n");
edma_clean_channel(echan->ch_num);
edma_stop(echan->ch_num);
edma_start(echan->ch_num); edma_start(echan->ch_num);
edma_trigger_channel(echan->ch_num);
echan->missed = 0;
}
} }
static int edma_terminate_all(struct edma_chan *echan) static int edma_terminate_all(struct edma_chan *echan)
...@@ -222,9 +262,9 @@ static struct dma_async_tx_descriptor *edma_prep_slave_sg( ...@@ -222,9 +262,9 @@ static struct dma_async_tx_descriptor *edma_prep_slave_sg(
enum dma_slave_buswidth dev_width; enum dma_slave_buswidth dev_width;
u32 burst; u32 burst;
struct scatterlist *sg; struct scatterlist *sg;
int i;
int acnt, bcnt, ccnt, src, dst, cidx; int acnt, bcnt, ccnt, src, dst, cidx;
int src_bidx, dst_bidx, src_cidx, dst_cidx; int src_bidx, dst_bidx, src_cidx, dst_cidx;
int i, nslots;
if (unlikely(!echan || !sgl || !sg_len)) if (unlikely(!echan || !sgl || !sg_len))
return NULL; return NULL;
...@@ -247,12 +287,6 @@ static struct dma_async_tx_descriptor *edma_prep_slave_sg( ...@@ -247,12 +287,6 @@ static struct dma_async_tx_descriptor *edma_prep_slave_sg(
return NULL; return NULL;
} }
if (sg_len > MAX_NR_SG) {
dev_err(dev, "Exceeded max SG segments %d > %d\n",
sg_len, MAX_NR_SG);
return NULL;
}
edesc = kzalloc(sizeof(*edesc) + sg_len * edesc = kzalloc(sizeof(*edesc) + sg_len *
sizeof(edesc->pset[0]), GFP_ATOMIC); sizeof(edesc->pset[0]), GFP_ATOMIC);
if (!edesc) { if (!edesc) {
...@@ -262,8 +296,10 @@ static struct dma_async_tx_descriptor *edma_prep_slave_sg( ...@@ -262,8 +296,10 @@ static struct dma_async_tx_descriptor *edma_prep_slave_sg(
edesc->pset_nr = sg_len; edesc->pset_nr = sg_len;
for_each_sg(sgl, sg, sg_len, i) {
/* Allocate a PaRAM slot, if needed */ /* Allocate a PaRAM slot, if needed */
nslots = min_t(unsigned, MAX_NR_SG, sg_len);
for (i = 0; i < nslots; i++) {
if (echan->slot[i] < 0) { if (echan->slot[i] < 0) {
echan->slot[i] = echan->slot[i] =
edma_alloc_slot(EDMA_CTLR(echan->ch_num), edma_alloc_slot(EDMA_CTLR(echan->ch_num),
...@@ -273,6 +309,10 @@ static struct dma_async_tx_descriptor *edma_prep_slave_sg( ...@@ -273,6 +309,10 @@ static struct dma_async_tx_descriptor *edma_prep_slave_sg(
return NULL; return NULL;
} }
} }
}
/* Configure PaRAM sets for each SG */
for_each_sg(sgl, sg, sg_len, i) {
acnt = dev_width; acnt = dev_width;
...@@ -330,6 +370,12 @@ static struct dma_async_tx_descriptor *edma_prep_slave_sg( ...@@ -330,6 +370,12 @@ static struct dma_async_tx_descriptor *edma_prep_slave_sg(
/* Configure A or AB synchronized transfers */ /* Configure A or AB synchronized transfers */
if (edesc->absync) if (edesc->absync)
edesc->pset[i].opt |= SYNCDIM; edesc->pset[i].opt |= SYNCDIM;
/* If this is the last in a current SG set of transactions,
enable interrupts so that next set is processed */
if (!((i+1) % MAX_NR_SG))
edesc->pset[i].opt |= TCINTEN;
/* If this is the last set, enable completion interrupt flag */ /* If this is the last set, enable completion interrupt flag */
if (i == sg_len - 1) if (i == sg_len - 1)
edesc->pset[i].opt |= TCINTEN; edesc->pset[i].opt |= TCINTEN;
...@@ -355,27 +401,65 @@ static void edma_callback(unsigned ch_num, u16 ch_status, void *data) ...@@ -355,27 +401,65 @@ static void edma_callback(unsigned ch_num, u16 ch_status, void *data)
struct device *dev = echan->vchan.chan.device->dev; struct device *dev = echan->vchan.chan.device->dev;
struct edma_desc *edesc; struct edma_desc *edesc;
unsigned long flags; unsigned long flags;
struct edmacc_param p;
/* Stop the channel */ /* Pause the channel */
edma_stop(echan->ch_num); edma_pause(echan->ch_num);
switch (ch_status) { switch (ch_status) {
case DMA_COMPLETE: case DMA_COMPLETE:
dev_dbg(dev, "transfer complete on channel %d\n", ch_num);
spin_lock_irqsave(&echan->vchan.lock, flags); spin_lock_irqsave(&echan->vchan.lock, flags);
edesc = echan->edesc; edesc = echan->edesc;
if (edesc) { if (edesc) {
edma_execute(echan); if (edesc->processed == edesc->pset_nr) {
dev_dbg(dev, "Transfer complete, stopping channel %d\n", ch_num);
edma_stop(echan->ch_num);
vchan_cookie_complete(&edesc->vdesc); vchan_cookie_complete(&edesc->vdesc);
} else {
dev_dbg(dev, "Intermediate transfer complete on channel %d\n", ch_num);
}
edma_execute(echan);
} }
spin_unlock_irqrestore(&echan->vchan.lock, flags); spin_unlock_irqrestore(&echan->vchan.lock, flags);
break; break;
case DMA_CC_ERROR: case DMA_CC_ERROR:
dev_dbg(dev, "transfer error on channel %d\n", ch_num); spin_lock_irqsave(&echan->vchan.lock, flags);
edma_read_slot(EDMA_CHAN_SLOT(echan->slot[0]), &p);
/*
* Issue later based on missed flag which will be sure
* to happen as:
* (1) we finished transmitting an intermediate slot and
* edma_execute is coming up.
* (2) or we finished current transfer and issue will
* call edma_execute.
*
* Important note: issuing can be dangerous here and
* lead to some nasty recursion when we are in a NULL
* slot. So we avoid doing so and set the missed flag.
*/
if (p.a_b_cnt == 0 && p.ccnt == 0) {
dev_dbg(dev, "Error occurred, looks like slot is null, just setting miss\n");
echan->missed = 1;
} else {
/*
* The slot is already programmed but the event got
* missed, so its safe to issue it here.
*/
dev_dbg(dev, "Error occurred but slot is non-null, TRIGGERING\n");
edma_clean_channel(echan->ch_num);
edma_stop(echan->ch_num);
edma_start(echan->ch_num);
edma_trigger_channel(echan->ch_num);
}
spin_unlock_irqrestore(&echan->vchan.lock, flags);
break; break;
default: default:
break; break;
...@@ -502,8 +586,6 @@ static enum dma_status edma_tx_status(struct dma_chan *chan, ...@@ -502,8 +586,6 @@ static enum dma_status edma_tx_status(struct dma_chan *chan,
} else if (echan->edesc && echan->edesc->vdesc.tx.cookie == cookie) { } else if (echan->edesc && echan->edesc->vdesc.tx.cookie == cookie) {
struct edma_desc *edesc = echan->edesc; struct edma_desc *edesc = echan->edesc;
txstate->residue = edma_desc_size(edesc); txstate->residue = edma_desc_size(edesc);
} else {
txstate->residue = 0;
} }
spin_unlock_irqrestore(&echan->vchan.lock, flags); spin_unlock_irqrestore(&echan->vchan.lock, flags);
......
...@@ -1313,15 +1313,7 @@ static enum dma_status ep93xx_dma_tx_status(struct dma_chan *chan, ...@@ -1313,15 +1313,7 @@ static enum dma_status ep93xx_dma_tx_status(struct dma_chan *chan,
dma_cookie_t cookie, dma_cookie_t cookie,
struct dma_tx_state *state) struct dma_tx_state *state)
{ {
struct ep93xx_dma_chan *edmac = to_ep93xx_dma_chan(chan); return dma_cookie_status(chan, cookie, state);
enum dma_status ret;
unsigned long flags;
spin_lock_irqsave(&edmac->lock, flags);
ret = dma_cookie_status(chan, cookie, state);
spin_unlock_irqrestore(&edmac->lock, flags);
return ret;
} }
/** /**
......
...@@ -979,15 +979,7 @@ static enum dma_status fsl_tx_status(struct dma_chan *dchan, ...@@ -979,15 +979,7 @@ static enum dma_status fsl_tx_status(struct dma_chan *dchan,
dma_cookie_t cookie, dma_cookie_t cookie,
struct dma_tx_state *txstate) struct dma_tx_state *txstate)
{ {
struct fsldma_chan *chan = to_fsl_chan(dchan); return dma_cookie_status(dchan, cookie, txstate);
enum dma_status ret;
unsigned long flags;
spin_lock_irqsave(&chan->desc_lock, flags);
ret = dma_cookie_status(dchan, cookie, txstate);
spin_unlock_irqrestore(&chan->desc_lock, flags);
return ret;
} }
/*----------------------------------------------------------------------------*/ /*----------------------------------------------------------------------------*/
......
...@@ -805,10 +805,8 @@ static void imxdma_free_chan_resources(struct dma_chan *chan) ...@@ -805,10 +805,8 @@ static void imxdma_free_chan_resources(struct dma_chan *chan)
} }
INIT_LIST_HEAD(&imxdmac->ld_free); INIT_LIST_HEAD(&imxdmac->ld_free);
if (imxdmac->sg_list) {
kfree(imxdmac->sg_list); kfree(imxdmac->sg_list);
imxdmac->sg_list = NULL; imxdmac->sg_list = NULL;
}
} }
static struct dma_async_tx_descriptor *imxdma_prep_slave_sg( static struct dma_async_tx_descriptor *imxdma_prep_slave_sg(
......
...@@ -243,7 +243,6 @@ struct sdma_engine; ...@@ -243,7 +243,6 @@ struct sdma_engine;
* @event_id1 for channels that use 2 events * @event_id1 for channels that use 2 events
* @word_size peripheral access size * @word_size peripheral access size
* @buf_tail ID of the buffer that was processed * @buf_tail ID of the buffer that was processed
* @done channel completion
* @num_bd max NUM_BD. number of descriptors currently handling * @num_bd max NUM_BD. number of descriptors currently handling
*/ */
struct sdma_channel { struct sdma_channel {
...@@ -255,7 +254,6 @@ struct sdma_channel { ...@@ -255,7 +254,6 @@ struct sdma_channel {
unsigned int event_id1; unsigned int event_id1;
enum dma_slave_buswidth word_size; enum dma_slave_buswidth word_size;
unsigned int buf_tail; unsigned int buf_tail;
struct completion done;
unsigned int num_bd; unsigned int num_bd;
struct sdma_buffer_descriptor *bd; struct sdma_buffer_descriptor *bd;
dma_addr_t bd_phys; dma_addr_t bd_phys;
...@@ -307,9 +305,10 @@ struct sdma_firmware_header { ...@@ -307,9 +305,10 @@ struct sdma_firmware_header {
u32 ram_code_size; u32 ram_code_size;
}; };
enum sdma_devtype { struct sdma_driver_data {
IMX31_SDMA, /* runs on i.mx31 */ int chnenbl0;
IMX35_SDMA, /* runs on i.mx35 and later */ int num_events;
struct sdma_script_start_addrs *script_addrs;
}; };
struct sdma_engine { struct sdma_engine {
...@@ -318,8 +317,6 @@ struct sdma_engine { ...@@ -318,8 +317,6 @@ struct sdma_engine {
struct sdma_channel channel[MAX_DMA_CHANNELS]; struct sdma_channel channel[MAX_DMA_CHANNELS];
struct sdma_channel_control *channel_control; struct sdma_channel_control *channel_control;
void __iomem *regs; void __iomem *regs;
enum sdma_devtype devtype;
unsigned int num_events;
struct sdma_context_data *context; struct sdma_context_data *context;
dma_addr_t context_phys; dma_addr_t context_phys;
struct dma_device dma_device; struct dma_device dma_device;
...@@ -327,15 +324,118 @@ struct sdma_engine { ...@@ -327,15 +324,118 @@ struct sdma_engine {
struct clk *clk_ahb; struct clk *clk_ahb;
spinlock_t channel_0_lock; spinlock_t channel_0_lock;
struct sdma_script_start_addrs *script_addrs; struct sdma_script_start_addrs *script_addrs;
const struct sdma_driver_data *drvdata;
};
static struct sdma_driver_data sdma_imx31 = {
.chnenbl0 = SDMA_CHNENBL0_IMX31,
.num_events = 32,
};
static struct sdma_script_start_addrs sdma_script_imx25 = {
.ap_2_ap_addr = 729,
.uart_2_mcu_addr = 904,
.per_2_app_addr = 1255,
.mcu_2_app_addr = 834,
.uartsh_2_mcu_addr = 1120,
.per_2_shp_addr = 1329,
.mcu_2_shp_addr = 1048,
.ata_2_mcu_addr = 1560,
.mcu_2_ata_addr = 1479,
.app_2_per_addr = 1189,
.app_2_mcu_addr = 770,
.shp_2_per_addr = 1407,
.shp_2_mcu_addr = 979,
};
static struct sdma_driver_data sdma_imx25 = {
.chnenbl0 = SDMA_CHNENBL0_IMX35,
.num_events = 48,
.script_addrs = &sdma_script_imx25,
};
static struct sdma_driver_data sdma_imx35 = {
.chnenbl0 = SDMA_CHNENBL0_IMX35,
.num_events = 48,
};
static struct sdma_script_start_addrs sdma_script_imx51 = {
.ap_2_ap_addr = 642,
.uart_2_mcu_addr = 817,
.mcu_2_app_addr = 747,
.mcu_2_shp_addr = 961,
.ata_2_mcu_addr = 1473,
.mcu_2_ata_addr = 1392,
.app_2_per_addr = 1033,
.app_2_mcu_addr = 683,
.shp_2_per_addr = 1251,
.shp_2_mcu_addr = 892,
};
static struct sdma_driver_data sdma_imx51 = {
.chnenbl0 = SDMA_CHNENBL0_IMX35,
.num_events = 48,
.script_addrs = &sdma_script_imx51,
};
static struct sdma_script_start_addrs sdma_script_imx53 = {
.ap_2_ap_addr = 642,
.app_2_mcu_addr = 683,
.mcu_2_app_addr = 747,
.uart_2_mcu_addr = 817,
.shp_2_mcu_addr = 891,
.mcu_2_shp_addr = 960,
.uartsh_2_mcu_addr = 1032,
.spdif_2_mcu_addr = 1100,
.mcu_2_spdif_addr = 1134,
.firi_2_mcu_addr = 1193,
.mcu_2_firi_addr = 1290,
};
static struct sdma_driver_data sdma_imx53 = {
.chnenbl0 = SDMA_CHNENBL0_IMX35,
.num_events = 48,
.script_addrs = &sdma_script_imx53,
};
static struct sdma_script_start_addrs sdma_script_imx6q = {
.ap_2_ap_addr = 642,
.uart_2_mcu_addr = 817,
.mcu_2_app_addr = 747,
.per_2_per_addr = 6331,
.uartsh_2_mcu_addr = 1032,
.mcu_2_shp_addr = 960,
.app_2_mcu_addr = 683,
.shp_2_mcu_addr = 891,
.spdif_2_mcu_addr = 1100,
.mcu_2_spdif_addr = 1134,
};
static struct sdma_driver_data sdma_imx6q = {
.chnenbl0 = SDMA_CHNENBL0_IMX35,
.num_events = 48,
.script_addrs = &sdma_script_imx6q,
}; };
static struct platform_device_id sdma_devtypes[] = { static struct platform_device_id sdma_devtypes[] = {
{ {
.name = "imx25-sdma",
.driver_data = (unsigned long)&sdma_imx25,
}, {
.name = "imx31-sdma", .name = "imx31-sdma",
.driver_data = IMX31_SDMA, .driver_data = (unsigned long)&sdma_imx31,
}, { }, {
.name = "imx35-sdma", .name = "imx35-sdma",
.driver_data = IMX35_SDMA, .driver_data = (unsigned long)&sdma_imx35,
}, {
.name = "imx51-sdma",
.driver_data = (unsigned long)&sdma_imx51,
}, {
.name = "imx53-sdma",
.driver_data = (unsigned long)&sdma_imx53,
}, {
.name = "imx6q-sdma",
.driver_data = (unsigned long)&sdma_imx6q,
}, { }, {
/* sentinel */ /* sentinel */
} }
...@@ -343,8 +443,11 @@ static struct platform_device_id sdma_devtypes[] = { ...@@ -343,8 +443,11 @@ static struct platform_device_id sdma_devtypes[] = {
MODULE_DEVICE_TABLE(platform, sdma_devtypes); MODULE_DEVICE_TABLE(platform, sdma_devtypes);
static const struct of_device_id sdma_dt_ids[] = { static const struct of_device_id sdma_dt_ids[] = {
{ .compatible = "fsl,imx31-sdma", .data = &sdma_devtypes[IMX31_SDMA], }, { .compatible = "fsl,imx6q-sdma", .data = &sdma_imx6q, },
{ .compatible = "fsl,imx35-sdma", .data = &sdma_devtypes[IMX35_SDMA], }, { .compatible = "fsl,imx53-sdma", .data = &sdma_imx53, },
{ .compatible = "fsl,imx51-sdma", .data = &sdma_imx51, },
{ .compatible = "fsl,imx35-sdma", .data = &sdma_imx35, },
{ .compatible = "fsl,imx31-sdma", .data = &sdma_imx31, },
{ /* sentinel */ } { /* sentinel */ }
}; };
MODULE_DEVICE_TABLE(of, sdma_dt_ids); MODULE_DEVICE_TABLE(of, sdma_dt_ids);
...@@ -356,8 +459,7 @@ MODULE_DEVICE_TABLE(of, sdma_dt_ids); ...@@ -356,8 +459,7 @@ MODULE_DEVICE_TABLE(of, sdma_dt_ids);
static inline u32 chnenbl_ofs(struct sdma_engine *sdma, unsigned int event) static inline u32 chnenbl_ofs(struct sdma_engine *sdma, unsigned int event)
{ {
u32 chnenbl0 = (sdma->devtype == IMX31_SDMA ? SDMA_CHNENBL0_IMX31 : u32 chnenbl0 = sdma->drvdata->chnenbl0;
SDMA_CHNENBL0_IMX35);
return chnenbl0 + event * 4; return chnenbl0 + event * 4;
} }
...@@ -547,8 +649,6 @@ static void sdma_tasklet(unsigned long data) ...@@ -547,8 +649,6 @@ static void sdma_tasklet(unsigned long data)
{ {
struct sdma_channel *sdmac = (struct sdma_channel *) data; struct sdma_channel *sdmac = (struct sdma_channel *) data;
complete(&sdmac->done);
if (sdmac->flags & IMX_DMA_SG_LOOP) if (sdmac->flags & IMX_DMA_SG_LOOP)
sdma_handle_channel_loop(sdmac); sdma_handle_channel_loop(sdmac);
else else
...@@ -733,7 +833,7 @@ static int sdma_config_channel(struct sdma_channel *sdmac) ...@@ -733,7 +833,7 @@ static int sdma_config_channel(struct sdma_channel *sdmac)
sdmac->per_addr = 0; sdmac->per_addr = 0;
if (sdmac->event_id0) { if (sdmac->event_id0) {
if (sdmac->event_id0 >= sdmac->sdma->num_events) if (sdmac->event_id0 >= sdmac->sdma->drvdata->num_events)
return -EINVAL; return -EINVAL;
sdma_event_enable(sdmac, sdmac->event_id0); sdma_event_enable(sdmac, sdmac->event_id0);
} }
...@@ -812,9 +912,6 @@ static int sdma_request_channel(struct sdma_channel *sdmac) ...@@ -812,9 +912,6 @@ static int sdma_request_channel(struct sdma_channel *sdmac)
sdma->channel_control[channel].current_bd_ptr = sdmac->bd_phys; sdma->channel_control[channel].current_bd_ptr = sdmac->bd_phys;
sdma_set_channel_priority(sdmac, MXC_SDMA_DEFAULT_PRIORITY); sdma_set_channel_priority(sdmac, MXC_SDMA_DEFAULT_PRIORITY);
init_completion(&sdmac->done);
return 0; return 0;
out: out:
...@@ -1124,11 +1221,8 @@ static enum dma_status sdma_tx_status(struct dma_chan *chan, ...@@ -1124,11 +1221,8 @@ static enum dma_status sdma_tx_status(struct dma_chan *chan,
struct dma_tx_state *txstate) struct dma_tx_state *txstate)
{ {
struct sdma_channel *sdmac = to_sdma_chan(chan); struct sdma_channel *sdmac = to_sdma_chan(chan);
dma_cookie_t last_used;
last_used = chan->cookie;
dma_set_tx_state(txstate, chan->completed_cookie, last_used, dma_set_tx_state(txstate, chan->completed_cookie, chan->cookie,
sdmac->chn_count - sdmac->chn_real_count); sdmac->chn_count - sdmac->chn_real_count);
return sdmac->status; return sdmac->status;
...@@ -1218,19 +1312,6 @@ static int __init sdma_init(struct sdma_engine *sdma) ...@@ -1218,19 +1312,6 @@ static int __init sdma_init(struct sdma_engine *sdma)
int i, ret; int i, ret;
dma_addr_t ccb_phys; dma_addr_t ccb_phys;
switch (sdma->devtype) {
case IMX31_SDMA:
sdma->num_events = 32;
break;
case IMX35_SDMA:
sdma->num_events = 48;
break;
default:
dev_err(sdma->dev, "Unknown sdma type %d. aborting\n",
sdma->devtype);
return -ENODEV;
}
clk_enable(sdma->clk_ipg); clk_enable(sdma->clk_ipg);
clk_enable(sdma->clk_ahb); clk_enable(sdma->clk_ahb);
...@@ -1257,7 +1338,7 @@ static int __init sdma_init(struct sdma_engine *sdma) ...@@ -1257,7 +1338,7 @@ static int __init sdma_init(struct sdma_engine *sdma)
MAX_DMA_CHANNELS * sizeof (struct sdma_channel_control)); MAX_DMA_CHANNELS * sizeof (struct sdma_channel_control));
/* disable all channels */ /* disable all channels */
for (i = 0; i < sdma->num_events; i++) for (i = 0; i < sdma->drvdata->num_events; i++)
writel_relaxed(0, sdma->regs + chnenbl_ofs(sdma, i)); writel_relaxed(0, sdma->regs + chnenbl_ofs(sdma, i));
/* All channels have priority 0 */ /* All channels have priority 0 */
...@@ -1335,10 +1416,21 @@ static int __init sdma_probe(struct platform_device *pdev) ...@@ -1335,10 +1416,21 @@ static int __init sdma_probe(struct platform_device *pdev)
int ret; int ret;
int irq; int irq;
struct resource *iores; struct resource *iores;
struct sdma_platform_data *pdata = pdev->dev.platform_data; struct sdma_platform_data *pdata = dev_get_platdata(&pdev->dev);
int i; int i;
struct sdma_engine *sdma; struct sdma_engine *sdma;
s32 *saddr_arr; s32 *saddr_arr;
const struct sdma_driver_data *drvdata = NULL;
if (of_id)
drvdata = of_id->data;
else if (pdev->id_entry)
drvdata = (void *)pdev->id_entry->driver_data;
if (!drvdata) {
dev_err(&pdev->dev, "unable to find driver data\n");
return -EINVAL;
}
sdma = kzalloc(sizeof(*sdma), GFP_KERNEL); sdma = kzalloc(sizeof(*sdma), GFP_KERNEL);
if (!sdma) if (!sdma)
...@@ -1347,6 +1439,7 @@ static int __init sdma_probe(struct platform_device *pdev) ...@@ -1347,6 +1439,7 @@ static int __init sdma_probe(struct platform_device *pdev)
spin_lock_init(&sdma->channel_0_lock); spin_lock_init(&sdma->channel_0_lock);
sdma->dev = &pdev->dev; sdma->dev = &pdev->dev;
sdma->drvdata = drvdata;
iores = platform_get_resource(pdev, IORESOURCE_MEM, 0); iores = platform_get_resource(pdev, IORESOURCE_MEM, 0);
irq = platform_get_irq(pdev, 0); irq = platform_get_irq(pdev, 0);
...@@ -1396,10 +1489,6 @@ static int __init sdma_probe(struct platform_device *pdev) ...@@ -1396,10 +1489,6 @@ static int __init sdma_probe(struct platform_device *pdev)
for (i = 0; i < SDMA_SCRIPT_ADDRS_ARRAY_SIZE_V1; i++) for (i = 0; i < SDMA_SCRIPT_ADDRS_ARRAY_SIZE_V1; i++)
saddr_arr[i] = -EINVAL; saddr_arr[i] = -EINVAL;
if (of_id)
pdev->id_entry = of_id->data;
sdma->devtype = pdev->id_entry->driver_data;
dma_cap_set(DMA_SLAVE, sdma->dma_device.cap_mask); dma_cap_set(DMA_SLAVE, sdma->dma_device.cap_mask);
dma_cap_set(DMA_CYCLIC, sdma->dma_device.cap_mask); dma_cap_set(DMA_CYCLIC, sdma->dma_device.cap_mask);
...@@ -1431,6 +1520,8 @@ static int __init sdma_probe(struct platform_device *pdev) ...@@ -1431,6 +1520,8 @@ static int __init sdma_probe(struct platform_device *pdev)
if (ret) if (ret)
goto err_init; goto err_init;
if (sdma->drvdata->script_addrs)
sdma_add_scripts(sdma, sdma->drvdata->script_addrs);
if (pdata && pdata->script_addrs) if (pdata && pdata->script_addrs)
sdma_add_scripts(sdma, pdata->script_addrs); sdma_add_scripts(sdma, pdata->script_addrs);
......
...@@ -518,7 +518,7 @@ static int iop_adma_alloc_chan_resources(struct dma_chan *chan) ...@@ -518,7 +518,7 @@ static int iop_adma_alloc_chan_resources(struct dma_chan *chan)
struct iop_adma_desc_slot *slot = NULL; struct iop_adma_desc_slot *slot = NULL;
int init = iop_chan->slots_allocated ? 0 : 1; int init = iop_chan->slots_allocated ? 0 : 1;
struct iop_adma_platform_data *plat_data = struct iop_adma_platform_data *plat_data =
iop_chan->device->pdev->dev.platform_data; dev_get_platdata(&iop_chan->device->pdev->dev);
int num_descs_in_pool = plat_data->pool_size/IOP_ADMA_SLOT_SIZE; int num_descs_in_pool = plat_data->pool_size/IOP_ADMA_SLOT_SIZE;
/* Allocate descriptor slots */ /* Allocate descriptor slots */
...@@ -1351,7 +1351,7 @@ static int iop_adma_remove(struct platform_device *dev) ...@@ -1351,7 +1351,7 @@ static int iop_adma_remove(struct platform_device *dev)
struct iop_adma_device *device = platform_get_drvdata(dev); struct iop_adma_device *device = platform_get_drvdata(dev);
struct dma_chan *chan, *_chan; struct dma_chan *chan, *_chan;
struct iop_adma_chan *iop_chan; struct iop_adma_chan *iop_chan;
struct iop_adma_platform_data *plat_data = dev->dev.platform_data; struct iop_adma_platform_data *plat_data = dev_get_platdata(&dev->dev);
dma_async_device_unregister(&device->common); dma_async_device_unregister(&device->common);
...@@ -1376,7 +1376,7 @@ static int iop_adma_probe(struct platform_device *pdev) ...@@ -1376,7 +1376,7 @@ static int iop_adma_probe(struct platform_device *pdev)
struct iop_adma_device *adev; struct iop_adma_device *adev;
struct iop_adma_chan *iop_chan; struct iop_adma_chan *iop_chan;
struct dma_device *dma_dev; struct dma_device *dma_dev;
struct iop_adma_platform_data *plat_data = pdev->dev.platform_data; struct iop_adma_platform_data *plat_data = dev_get_platdata(&pdev->dev);
res = platform_get_resource(pdev, IORESOURCE_MEM, 0); res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
if (!res) if (!res)
......
...@@ -1593,10 +1593,7 @@ static void idmac_free_chan_resources(struct dma_chan *chan) ...@@ -1593,10 +1593,7 @@ static void idmac_free_chan_resources(struct dma_chan *chan)
static enum dma_status idmac_tx_status(struct dma_chan *chan, static enum dma_status idmac_tx_status(struct dma_chan *chan,
dma_cookie_t cookie, struct dma_tx_state *txstate) dma_cookie_t cookie, struct dma_tx_state *txstate)
{ {
dma_set_tx_state(txstate, chan->completed_cookie, chan->cookie, 0); return dma_cookie_status(chan, cookie, txstate);
if (cookie != chan->cookie)
return DMA_ERROR;
return DMA_SUCCESS;
} }
static int __init ipu_idmac_init(struct ipu *ipu) static int __init ipu_idmac_init(struct ipu *ipu)
...@@ -1767,7 +1764,6 @@ static int ipu_remove(struct platform_device *pdev) ...@@ -1767,7 +1764,6 @@ static int ipu_remove(struct platform_device *pdev)
iounmap(ipu->reg_ic); iounmap(ipu->reg_ic);
iounmap(ipu->reg_ipu); iounmap(ipu->reg_ipu);
tasklet_kill(&ipu->tasklet); tasklet_kill(&ipu->tasklet);
platform_set_drvdata(pdev, NULL);
return 0; return 0;
} }
......
This diff is collapsed.
This diff is collapsed.
...@@ -460,7 +460,8 @@ static enum dma_status mmp_tdma_tx_status(struct dma_chan *chan, ...@@ -460,7 +460,8 @@ static enum dma_status mmp_tdma_tx_status(struct dma_chan *chan,
{ {
struct mmp_tdma_chan *tdmac = to_mmp_tdma_chan(chan); struct mmp_tdma_chan *tdmac = to_mmp_tdma_chan(chan);
dma_set_residue(txstate, tdmac->buf_len - tdmac->pos); dma_set_tx_state(txstate, chan->completed_cookie, chan->cookie,
tdmac->buf_len - tdmac->pos);
return tdmac->status; return tdmac->status;
} }
...@@ -549,9 +550,6 @@ static int mmp_tdma_probe(struct platform_device *pdev) ...@@ -549,9 +550,6 @@ static int mmp_tdma_probe(struct platform_device *pdev)
} }
iores = platform_get_resource(pdev, IORESOURCE_MEM, 0); iores = platform_get_resource(pdev, IORESOURCE_MEM, 0);
if (!iores)
return -EINVAL;
tdev->base = devm_ioremap_resource(&pdev->dev, iores); tdev->base = devm_ioremap_resource(&pdev->dev, iores);
if (IS_ERR(tdev->base)) if (IS_ERR(tdev->base))
return PTR_ERR(tdev->base); return PTR_ERR(tdev->base);
......
...@@ -556,15 +556,7 @@ static enum dma_status ...@@ -556,15 +556,7 @@ static enum dma_status
mpc_dma_tx_status(struct dma_chan *chan, dma_cookie_t cookie, mpc_dma_tx_status(struct dma_chan *chan, dma_cookie_t cookie,
struct dma_tx_state *txstate) struct dma_tx_state *txstate)
{ {
struct mpc_dma_chan *mchan = dma_chan_to_mpc_dma_chan(chan); return dma_cookie_status(chan, cookie, txstate);
enum dma_status ret;
unsigned long flags;
spin_lock_irqsave(&mchan->lock, flags);
ret = dma_cookie_status(chan, cookie, txstate);
spin_unlock_irqrestore(&mchan->lock, flags);
return ret;
} }
/* Prepare descriptor for memory to memory copy */ /* Prepare descriptor for memory to memory copy */
......
...@@ -654,7 +654,7 @@ mv_xor_prep_dma_memcpy(struct dma_chan *chan, dma_addr_t dest, dma_addr_t src, ...@@ -654,7 +654,7 @@ mv_xor_prep_dma_memcpy(struct dma_chan *chan, dma_addr_t dest, dma_addr_t src,
dev_dbg(mv_chan_to_devp(mv_chan), dev_dbg(mv_chan_to_devp(mv_chan),
"%s sw_desc %p async_tx %p\n", "%s sw_desc %p async_tx %p\n",
__func__, sw_desc, sw_desc ? &sw_desc->async_tx : 0); __func__, sw_desc, sw_desc ? &sw_desc->async_tx : NULL);
return sw_desc ? &sw_desc->async_tx : NULL; return sw_desc ? &sw_desc->async_tx : NULL;
} }
...@@ -1171,7 +1171,7 @@ static int mv_xor_probe(struct platform_device *pdev) ...@@ -1171,7 +1171,7 @@ static int mv_xor_probe(struct platform_device *pdev)
{ {
const struct mbus_dram_target_info *dram; const struct mbus_dram_target_info *dram;
struct mv_xor_device *xordev; struct mv_xor_device *xordev;
struct mv_xor_platform_data *pdata = pdev->dev.platform_data; struct mv_xor_platform_data *pdata = dev_get_platdata(&pdev->dev);
struct resource *res; struct resource *res;
int i, ret; int i, ret;
......
...@@ -23,7 +23,6 @@ ...@@ -23,7 +23,6 @@
#include <linux/dmaengine.h> #include <linux/dmaengine.h>
#include <linux/delay.h> #include <linux/delay.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/fsl/mxs-dma.h>
#include <linux/stmp_device.h> #include <linux/stmp_device.h>
#include <linux/of.h> #include <linux/of.h>
#include <linux/of_device.h> #include <linux/of_device.h>
...@@ -197,24 +196,6 @@ static struct mxs_dma_chan *to_mxs_dma_chan(struct dma_chan *chan) ...@@ -197,24 +196,6 @@ static struct mxs_dma_chan *to_mxs_dma_chan(struct dma_chan *chan)
return container_of(chan, struct mxs_dma_chan, chan); return container_of(chan, struct mxs_dma_chan, chan);
} }
int mxs_dma_is_apbh(struct dma_chan *chan)
{
struct mxs_dma_chan *mxs_chan = to_mxs_dma_chan(chan);
struct mxs_dma_engine *mxs_dma = mxs_chan->mxs_dma;
return dma_is_apbh(mxs_dma);
}
EXPORT_SYMBOL_GPL(mxs_dma_is_apbh);
int mxs_dma_is_apbx(struct dma_chan *chan)
{
struct mxs_dma_chan *mxs_chan = to_mxs_dma_chan(chan);
struct mxs_dma_engine *mxs_dma = mxs_chan->mxs_dma;
return !dma_is_apbh(mxs_dma);
}
EXPORT_SYMBOL_GPL(mxs_dma_is_apbx);
static void mxs_dma_reset_chan(struct mxs_dma_chan *mxs_chan) static void mxs_dma_reset_chan(struct mxs_dma_chan *mxs_chan)
{ {
struct mxs_dma_engine *mxs_dma = mxs_chan->mxs_dma; struct mxs_dma_engine *mxs_dma = mxs_chan->mxs_dma;
...@@ -349,13 +330,9 @@ static irqreturn_t mxs_dma_int_handler(int irq, void *dev_id) ...@@ -349,13 +330,9 @@ static irqreturn_t mxs_dma_int_handler(int irq, void *dev_id)
static int mxs_dma_alloc_chan_resources(struct dma_chan *chan) static int mxs_dma_alloc_chan_resources(struct dma_chan *chan)
{ {
struct mxs_dma_chan *mxs_chan = to_mxs_dma_chan(chan); struct mxs_dma_chan *mxs_chan = to_mxs_dma_chan(chan);
struct mxs_dma_data *data = chan->private;
struct mxs_dma_engine *mxs_dma = mxs_chan->mxs_dma; struct mxs_dma_engine *mxs_dma = mxs_chan->mxs_dma;
int ret; int ret;
if (data)
mxs_chan->chan_irq = data->chan_irq;
mxs_chan->ccw = dma_alloc_coherent(mxs_dma->dma_device.dev, mxs_chan->ccw = dma_alloc_coherent(mxs_dma->dma_device.dev,
CCW_BLOCK_SIZE, &mxs_chan->ccw_phys, CCW_BLOCK_SIZE, &mxs_chan->ccw_phys,
GFP_KERNEL); GFP_KERNEL);
...@@ -622,10 +599,8 @@ static enum dma_status mxs_dma_tx_status(struct dma_chan *chan, ...@@ -622,10 +599,8 @@ static enum dma_status mxs_dma_tx_status(struct dma_chan *chan,
dma_cookie_t cookie, struct dma_tx_state *txstate) dma_cookie_t cookie, struct dma_tx_state *txstate)
{ {
struct mxs_dma_chan *mxs_chan = to_mxs_dma_chan(chan); struct mxs_dma_chan *mxs_chan = to_mxs_dma_chan(chan);
dma_cookie_t last_used;
last_used = chan->cookie; dma_set_tx_state(txstate, chan->completed_cookie, chan->cookie, 0);
dma_set_tx_state(txstate, chan->completed_cookie, last_used, 0);
return mxs_chan->status; return mxs_chan->status;
} }
......
...@@ -160,7 +160,8 @@ struct dma_chan *of_dma_request_slave_channel(struct device_node *np, ...@@ -160,7 +160,8 @@ struct dma_chan *of_dma_request_slave_channel(struct device_node *np,
count = of_property_count_strings(np, "dma-names"); count = of_property_count_strings(np, "dma-names");
if (count < 0) { if (count < 0) {
pr_err("%s: dma-names property missing or empty\n", __func__); pr_err("%s: dma-names property of node '%s' missing or empty\n",
__func__, np->full_name);
return NULL; return NULL;
} }
......
...@@ -564,14 +564,7 @@ static void pd_free_chan_resources(struct dma_chan *chan) ...@@ -564,14 +564,7 @@ static void pd_free_chan_resources(struct dma_chan *chan)
static enum dma_status pd_tx_status(struct dma_chan *chan, dma_cookie_t cookie, static enum dma_status pd_tx_status(struct dma_chan *chan, dma_cookie_t cookie,
struct dma_tx_state *txstate) struct dma_tx_state *txstate)
{ {
struct pch_dma_chan *pd_chan = to_pd_chan(chan); return dma_cookie_status(chan, cookie, txstate);
enum dma_status ret;
spin_lock_irq(&pd_chan->lock);
ret = dma_cookie_status(chan, cookie, txstate);
spin_unlock_irq(&pd_chan->lock);
return ret;
} }
static void pd_issue_pending(struct dma_chan *chan) static void pd_issue_pending(struct dma_chan *chan)
...@@ -1036,3 +1029,4 @@ MODULE_DESCRIPTION("Intel EG20T PCH / LAPIS Semicon ML7213/ML7223/ML7831 IOH " ...@@ -1036,3 +1029,4 @@ MODULE_DESCRIPTION("Intel EG20T PCH / LAPIS Semicon ML7213/ML7223/ML7831 IOH "
"DMA controller driver"); "DMA controller driver");
MODULE_AUTHOR("Yong Wang <yong.y.wang@intel.com>"); MODULE_AUTHOR("Yong Wang <yong.y.wang@intel.com>");
MODULE_LICENSE("GPL v2"); MODULE_LICENSE("GPL v2");
MODULE_DEVICE_TABLE(pci, pch_dma_id_table);
...@@ -545,6 +545,8 @@ struct dma_pl330_chan { ...@@ -545,6 +545,8 @@ struct dma_pl330_chan {
/* List of to be xfered descriptors */ /* List of to be xfered descriptors */
struct list_head work_list; struct list_head work_list;
/* List of completed descriptors */
struct list_head completed_list;
/* Pointer to the DMAC that manages this channel, /* Pointer to the DMAC that manages this channel,
* NULL if the channel is available to be acquired. * NULL if the channel is available to be acquired.
...@@ -2198,66 +2200,6 @@ to_desc(struct dma_async_tx_descriptor *tx) ...@@ -2198,66 +2200,6 @@ to_desc(struct dma_async_tx_descriptor *tx)
return container_of(tx, struct dma_pl330_desc, txd); return container_of(tx, struct dma_pl330_desc, txd);
} }
static inline void free_desc_list(struct list_head *list)
{
struct dma_pl330_dmac *pdmac;
struct dma_pl330_desc *desc;
struct dma_pl330_chan *pch = NULL;
unsigned long flags;
/* Finish off the work list */
list_for_each_entry(desc, list, node) {
dma_async_tx_callback callback;
void *param;
/* All desc in a list belong to same channel */
pch = desc->pchan;
callback = desc->txd.callback;
param = desc->txd.callback_param;
if (callback)
callback(param);
desc->pchan = NULL;
}
/* pch will be unset if list was empty */
if (!pch)
return;
pdmac = pch->dmac;
spin_lock_irqsave(&pdmac->pool_lock, flags);
list_splice_tail_init(list, &pdmac->desc_pool);
spin_unlock_irqrestore(&pdmac->pool_lock, flags);
}
static inline void handle_cyclic_desc_list(struct list_head *list)
{
struct dma_pl330_desc *desc;
struct dma_pl330_chan *pch = NULL;
unsigned long flags;
list_for_each_entry(desc, list, node) {
dma_async_tx_callback callback;
/* Change status to reload it */
desc->status = PREP;
pch = desc->pchan;
callback = desc->txd.callback;
if (callback)
callback(desc->txd.callback_param);
}
/* pch will be unset if list was empty */
if (!pch)
return;
spin_lock_irqsave(&pch->lock, flags);
list_splice_tail_init(list, &pch->work_list);
spin_unlock_irqrestore(&pch->lock, flags);
}
static inline void fill_queue(struct dma_pl330_chan *pch) static inline void fill_queue(struct dma_pl330_chan *pch)
{ {
struct dma_pl330_desc *desc; struct dma_pl330_desc *desc;
...@@ -2291,7 +2233,6 @@ static void pl330_tasklet(unsigned long data) ...@@ -2291,7 +2233,6 @@ static void pl330_tasklet(unsigned long data)
struct dma_pl330_chan *pch = (struct dma_pl330_chan *)data; struct dma_pl330_chan *pch = (struct dma_pl330_chan *)data;
struct dma_pl330_desc *desc, *_dt; struct dma_pl330_desc *desc, *_dt;
unsigned long flags; unsigned long flags;
LIST_HEAD(list);
spin_lock_irqsave(&pch->lock, flags); spin_lock_irqsave(&pch->lock, flags);
...@@ -2300,7 +2241,7 @@ static void pl330_tasklet(unsigned long data) ...@@ -2300,7 +2241,7 @@ static void pl330_tasklet(unsigned long data)
if (desc->status == DONE) { if (desc->status == DONE) {
if (!pch->cyclic) if (!pch->cyclic)
dma_cookie_complete(&desc->txd); dma_cookie_complete(&desc->txd);
list_move_tail(&desc->node, &list); list_move_tail(&desc->node, &pch->completed_list);
} }
/* Try to submit a req imm. next to the last completed cookie */ /* Try to submit a req imm. next to the last completed cookie */
...@@ -2309,12 +2250,31 @@ static void pl330_tasklet(unsigned long data) ...@@ -2309,12 +2250,31 @@ static void pl330_tasklet(unsigned long data)
/* Make sure the PL330 Channel thread is active */ /* Make sure the PL330 Channel thread is active */
pl330_chan_ctrl(pch->pl330_chid, PL330_OP_START); pl330_chan_ctrl(pch->pl330_chid, PL330_OP_START);
spin_unlock_irqrestore(&pch->lock, flags); while (!list_empty(&pch->completed_list)) {
dma_async_tx_callback callback;
void *callback_param;
if (pch->cyclic) desc = list_first_entry(&pch->completed_list,
handle_cyclic_desc_list(&list); struct dma_pl330_desc, node);
else
free_desc_list(&list); callback = desc->txd.callback;
callback_param = desc->txd.callback_param;
if (pch->cyclic) {
desc->status = PREP;
list_move_tail(&desc->node, &pch->work_list);
} else {
desc->status = FREE;
list_move_tail(&desc->node, &pch->dmac->desc_pool);
}
if (callback) {
spin_unlock_irqrestore(&pch->lock, flags);
callback(callback_param);
spin_lock_irqsave(&pch->lock, flags);
}
}
spin_unlock_irqrestore(&pch->lock, flags);
} }
static void dma_pl330_rqcb(void *token, enum pl330_op_err err) static void dma_pl330_rqcb(void *token, enum pl330_op_err err)
...@@ -2409,7 +2369,7 @@ static int pl330_alloc_chan_resources(struct dma_chan *chan) ...@@ -2409,7 +2369,7 @@ static int pl330_alloc_chan_resources(struct dma_chan *chan)
static int pl330_control(struct dma_chan *chan, enum dma_ctrl_cmd cmd, unsigned long arg) static int pl330_control(struct dma_chan *chan, enum dma_ctrl_cmd cmd, unsigned long arg)
{ {
struct dma_pl330_chan *pch = to_pchan(chan); struct dma_pl330_chan *pch = to_pchan(chan);
struct dma_pl330_desc *desc, *_dt; struct dma_pl330_desc *desc;
unsigned long flags; unsigned long flags;
struct dma_pl330_dmac *pdmac = pch->dmac; struct dma_pl330_dmac *pdmac = pch->dmac;
struct dma_slave_config *slave_config; struct dma_slave_config *slave_config;
...@@ -2423,12 +2383,18 @@ static int pl330_control(struct dma_chan *chan, enum dma_ctrl_cmd cmd, unsigned ...@@ -2423,12 +2383,18 @@ static int pl330_control(struct dma_chan *chan, enum dma_ctrl_cmd cmd, unsigned
pl330_chan_ctrl(pch->pl330_chid, PL330_OP_FLUSH); pl330_chan_ctrl(pch->pl330_chid, PL330_OP_FLUSH);
/* Mark all desc done */ /* Mark all desc done */
list_for_each_entry_safe(desc, _dt, &pch->work_list , node) { list_for_each_entry(desc, &pch->work_list , node) {
desc->status = DONE; desc->status = FREE;
list_move_tail(&desc->node, &list); dma_cookie_complete(&desc->txd);
} }
list_splice_tail_init(&list, &pdmac->desc_pool); list_for_each_entry(desc, &pch->completed_list , node) {
desc->status = FREE;
dma_cookie_complete(&desc->txd);
}
list_splice_tail_init(&pch->work_list, &pdmac->desc_pool);
list_splice_tail_init(&pch->completed_list, &pdmac->desc_pool);
spin_unlock_irqrestore(&pch->lock, flags); spin_unlock_irqrestore(&pch->lock, flags);
break; break;
case DMA_SLAVE_CONFIG: case DMA_SLAVE_CONFIG:
...@@ -2814,6 +2780,28 @@ pl330_prep_dma_memcpy(struct dma_chan *chan, dma_addr_t dst, ...@@ -2814,6 +2780,28 @@ pl330_prep_dma_memcpy(struct dma_chan *chan, dma_addr_t dst,
return &desc->txd; return &desc->txd;
} }
static void __pl330_giveback_desc(struct dma_pl330_dmac *pdmac,
struct dma_pl330_desc *first)
{
unsigned long flags;
struct dma_pl330_desc *desc;
if (!first)
return;
spin_lock_irqsave(&pdmac->pool_lock, flags);
while (!list_empty(&first->node)) {
desc = list_entry(first->node.next,
struct dma_pl330_desc, node);
list_move_tail(&desc->node, &pdmac->desc_pool);
}
list_move_tail(&first->node, &pdmac->desc_pool);
spin_unlock_irqrestore(&pdmac->pool_lock, flags);
}
static struct dma_async_tx_descriptor * static struct dma_async_tx_descriptor *
pl330_prep_slave_sg(struct dma_chan *chan, struct scatterlist *sgl, pl330_prep_slave_sg(struct dma_chan *chan, struct scatterlist *sgl,
unsigned int sg_len, enum dma_transfer_direction direction, unsigned int sg_len, enum dma_transfer_direction direction,
...@@ -2822,7 +2810,6 @@ pl330_prep_slave_sg(struct dma_chan *chan, struct scatterlist *sgl, ...@@ -2822,7 +2810,6 @@ pl330_prep_slave_sg(struct dma_chan *chan, struct scatterlist *sgl,
struct dma_pl330_desc *first, *desc = NULL; struct dma_pl330_desc *first, *desc = NULL;
struct dma_pl330_chan *pch = to_pchan(chan); struct dma_pl330_chan *pch = to_pchan(chan);
struct scatterlist *sg; struct scatterlist *sg;
unsigned long flags;
int i; int i;
dma_addr_t addr; dma_addr_t addr;
...@@ -2842,20 +2829,7 @@ pl330_prep_slave_sg(struct dma_chan *chan, struct scatterlist *sgl, ...@@ -2842,20 +2829,7 @@ pl330_prep_slave_sg(struct dma_chan *chan, struct scatterlist *sgl,
dev_err(pch->dmac->pif.dev, dev_err(pch->dmac->pif.dev,
"%s:%d Unable to fetch desc\n", "%s:%d Unable to fetch desc\n",
__func__, __LINE__); __func__, __LINE__);
if (!first) __pl330_giveback_desc(pdmac, first);
return NULL;
spin_lock_irqsave(&pdmac->pool_lock, flags);
while (!list_empty(&first->node)) {
desc = list_entry(first->node.next,
struct dma_pl330_desc, node);
list_move_tail(&desc->node, &pdmac->desc_pool);
}
list_move_tail(&first->node, &pdmac->desc_pool);
spin_unlock_irqrestore(&pdmac->pool_lock, flags);
return NULL; return NULL;
} }
...@@ -2896,6 +2870,25 @@ static irqreturn_t pl330_irq_handler(int irq, void *data) ...@@ -2896,6 +2870,25 @@ static irqreturn_t pl330_irq_handler(int irq, void *data)
return IRQ_NONE; return IRQ_NONE;
} }
#define PL330_DMA_BUSWIDTHS \
BIT(DMA_SLAVE_BUSWIDTH_UNDEFINED) | \
BIT(DMA_SLAVE_BUSWIDTH_1_BYTE) | \
BIT(DMA_SLAVE_BUSWIDTH_2_BYTES) | \
BIT(DMA_SLAVE_BUSWIDTH_4_BYTES) | \
BIT(DMA_SLAVE_BUSWIDTH_8_BYTES)
static int pl330_dma_device_slave_caps(struct dma_chan *dchan,
struct dma_slave_caps *caps)
{
caps->src_addr_widths = PL330_DMA_BUSWIDTHS;
caps->dstn_addr_widths = PL330_DMA_BUSWIDTHS;
caps->directions = BIT(DMA_DEV_TO_MEM) | BIT(DMA_MEM_TO_DEV);
caps->cmd_pause = false;
caps->cmd_terminate = true;
return 0;
}
static int static int
pl330_probe(struct amba_device *adev, const struct amba_id *id) pl330_probe(struct amba_device *adev, const struct amba_id *id)
{ {
...@@ -2908,7 +2901,7 @@ pl330_probe(struct amba_device *adev, const struct amba_id *id) ...@@ -2908,7 +2901,7 @@ pl330_probe(struct amba_device *adev, const struct amba_id *id)
int i, ret, irq; int i, ret, irq;
int num_chan; int num_chan;
pdat = adev->dev.platform_data; pdat = dev_get_platdata(&adev->dev);
/* Allocate a new DMAC and its Channels */ /* Allocate a new DMAC and its Channels */
pdmac = devm_kzalloc(&adev->dev, sizeof(*pdmac), GFP_KERNEL); pdmac = devm_kzalloc(&adev->dev, sizeof(*pdmac), GFP_KERNEL);
...@@ -2971,6 +2964,7 @@ pl330_probe(struct amba_device *adev, const struct amba_id *id) ...@@ -2971,6 +2964,7 @@ pl330_probe(struct amba_device *adev, const struct amba_id *id)
pch->chan.private = adev->dev.of_node; pch->chan.private = adev->dev.of_node;
INIT_LIST_HEAD(&pch->work_list); INIT_LIST_HEAD(&pch->work_list);
INIT_LIST_HEAD(&pch->completed_list);
spin_lock_init(&pch->lock); spin_lock_init(&pch->lock);
pch->pl330_chid = NULL; pch->pl330_chid = NULL;
pch->chan.device = pd; pch->chan.device = pd;
...@@ -3000,6 +2994,7 @@ pl330_probe(struct amba_device *adev, const struct amba_id *id) ...@@ -3000,6 +2994,7 @@ pl330_probe(struct amba_device *adev, const struct amba_id *id)
pd->device_prep_slave_sg = pl330_prep_slave_sg; pd->device_prep_slave_sg = pl330_prep_slave_sg;
pd->device_control = pl330_control; pd->device_control = pl330_control;
pd->device_issue_pending = pl330_issue_pending; pd->device_issue_pending = pl330_issue_pending;
pd->device_slave_caps = pl330_dma_device_slave_caps;
ret = dma_async_device_register(pd); ret = dma_async_device_register(pd);
if (ret) { if (ret) {
...@@ -3015,6 +3010,14 @@ pl330_probe(struct amba_device *adev, const struct amba_id *id) ...@@ -3015,6 +3010,14 @@ pl330_probe(struct amba_device *adev, const struct amba_id *id)
"unable to register DMA to the generic DT DMA helpers\n"); "unable to register DMA to the generic DT DMA helpers\n");
} }
} }
/*
* This is the limit for transfers with a buswidth of 1, larger
* buswidths will have larger limits.
*/
ret = dma_set_max_seg_size(&adev->dev, 1900800);
if (ret)
dev_err(&adev->dev, "unable to set the seg size\n");
dev_info(&adev->dev, dev_info(&adev->dev,
"Loaded driver for PL330 DMAC-%d\n", adev->periphid); "Loaded driver for PL330 DMAC-%d\n", adev->periphid);
......
...@@ -22,3 +22,13 @@ config SUDMAC ...@@ -22,3 +22,13 @@ config SUDMAC
depends on SH_DMAE_BASE depends on SH_DMAE_BASE
help help
Enable support for the Renesas SUDMAC controllers. Enable support for the Renesas SUDMAC controllers.
config RCAR_HPB_DMAE
tristate "Renesas R-Car HPB DMAC support"
depends on SH_DMAE_BASE
help
Enable support for the Renesas R-Car series DMA controllers.
config SHDMA_R8A73A4
def_bool y
depends on ARCH_R8A73A4 && SH_DMAE != n
obj-$(CONFIG_SH_DMAE_BASE) += shdma-base.o shdma-of.o obj-$(CONFIG_SH_DMAE_BASE) += shdma-base.o shdma-of.o
obj-$(CONFIG_SH_DMAE) += shdma.o obj-$(CONFIG_SH_DMAE) += shdma.o
shdma-y := shdmac.o
ifeq ($(CONFIG_OF),y)
shdma-$(CONFIG_SHDMA_R8A73A4) += shdma-r8a73a4.o
endif
shdma-objs := $(shdma-y)
obj-$(CONFIG_SUDMAC) += sudmac.o obj-$(CONFIG_SUDMAC) += sudmac.o
obj-$(CONFIG_RCAR_HPB_DMAE) += rcar-hpbdma.o
This diff is collapsed.
/*
* Renesas SuperH DMA Engine support
*
* Copyright (C) 2013 Renesas Electronics, Inc.
*
* This is free software; you can redistribute it and/or modify it under the
* terms of version 2 the GNU General Public License as published by the Free
* Software Foundation.
*/
#ifndef SHDMA_ARM_H
#define SHDMA_ARM_H
#include "shdma.h"
/* Transmit sizes and respective CHCR register values */
enum {
XMIT_SZ_8BIT = 0,
XMIT_SZ_16BIT = 1,
XMIT_SZ_32BIT = 2,
XMIT_SZ_64BIT = 7,
XMIT_SZ_128BIT = 3,
XMIT_SZ_256BIT = 4,
XMIT_SZ_512BIT = 5,
};
/* log2(size / 8) - used to calculate number of transfers */
#define SH_DMAE_TS_SHIFT { \
[XMIT_SZ_8BIT] = 0, \
[XMIT_SZ_16BIT] = 1, \
[XMIT_SZ_32BIT] = 2, \
[XMIT_SZ_64BIT] = 3, \
[XMIT_SZ_128BIT] = 4, \
[XMIT_SZ_256BIT] = 5, \
[XMIT_SZ_512BIT] = 6, \
}
#define TS_LOW_BIT 0x3 /* --xx */
#define TS_HI_BIT 0xc /* xx-- */
#define TS_LOW_SHIFT (3)
#define TS_HI_SHIFT (20 - 2) /* 2 bits for shifted low TS */
#define TS_INDEX2VAL(i) \
((((i) & TS_LOW_BIT) << TS_LOW_SHIFT) |\
(((i) & TS_HI_BIT) << TS_HI_SHIFT))
#define CHCR_TX(xmit_sz) (DM_FIX | SM_INC | 0x800 | TS_INDEX2VAL((xmit_sz)))
#define CHCR_RX(xmit_sz) (DM_INC | SM_FIX | 0x800 | TS_INDEX2VAL((xmit_sz)))
#endif
...@@ -171,7 +171,8 @@ static struct shdma_desc *shdma_get_desc(struct shdma_chan *schan) ...@@ -171,7 +171,8 @@ static struct shdma_desc *shdma_get_desc(struct shdma_chan *schan)
return NULL; return NULL;
} }
static int shdma_setup_slave(struct shdma_chan *schan, int slave_id) static int shdma_setup_slave(struct shdma_chan *schan, int slave_id,
dma_addr_t slave_addr)
{ {
struct shdma_dev *sdev = to_shdma_dev(schan->dma_chan.device); struct shdma_dev *sdev = to_shdma_dev(schan->dma_chan.device);
const struct shdma_ops *ops = sdev->ops; const struct shdma_ops *ops = sdev->ops;
...@@ -179,7 +180,7 @@ static int shdma_setup_slave(struct shdma_chan *schan, int slave_id) ...@@ -179,7 +180,7 @@ static int shdma_setup_slave(struct shdma_chan *schan, int slave_id)
if (schan->dev->of_node) { if (schan->dev->of_node) {
match = schan->hw_req; match = schan->hw_req;
ret = ops->set_slave(schan, match, true); ret = ops->set_slave(schan, match, slave_addr, true);
if (ret < 0) if (ret < 0)
return ret; return ret;
...@@ -194,7 +195,7 @@ static int shdma_setup_slave(struct shdma_chan *schan, int slave_id) ...@@ -194,7 +195,7 @@ static int shdma_setup_slave(struct shdma_chan *schan, int slave_id)
if (test_and_set_bit(slave_id, shdma_slave_used)) if (test_and_set_bit(slave_id, shdma_slave_used))
return -EBUSY; return -EBUSY;
ret = ops->set_slave(schan, match, false); ret = ops->set_slave(schan, match, slave_addr, false);
if (ret < 0) { if (ret < 0) {
clear_bit(slave_id, shdma_slave_used); clear_bit(slave_id, shdma_slave_used);
return ret; return ret;
...@@ -236,7 +237,7 @@ bool shdma_chan_filter(struct dma_chan *chan, void *arg) ...@@ -236,7 +237,7 @@ bool shdma_chan_filter(struct dma_chan *chan, void *arg)
if (!schan->dev->of_node && match >= slave_num) if (!schan->dev->of_node && match >= slave_num)
return false; return false;
ret = ops->set_slave(schan, match, true); ret = ops->set_slave(schan, match, 0, true);
if (ret < 0) if (ret < 0)
return false; return false;
...@@ -259,7 +260,7 @@ static int shdma_alloc_chan_resources(struct dma_chan *chan) ...@@ -259,7 +260,7 @@ static int shdma_alloc_chan_resources(struct dma_chan *chan)
*/ */
if (slave) { if (slave) {
/* Legacy mode: .private is set in filter */ /* Legacy mode: .private is set in filter */
ret = shdma_setup_slave(schan, slave->slave_id); ret = shdma_setup_slave(schan, slave->slave_id, 0);
if (ret < 0) if (ret < 0)
goto esetslave; goto esetslave;
} else { } else {
...@@ -680,7 +681,9 @@ static int shdma_control(struct dma_chan *chan, enum dma_ctrl_cmd cmd, ...@@ -680,7 +681,9 @@ static int shdma_control(struct dma_chan *chan, enum dma_ctrl_cmd cmd,
* channel, while using it... * channel, while using it...
*/ */
config = (struct dma_slave_config *)arg; config = (struct dma_slave_config *)arg;
ret = shdma_setup_slave(schan, config->slave_id); ret = shdma_setup_slave(schan, config->slave_id,
config->direction == DMA_DEV_TO_MEM ?
config->src_addr : config->dst_addr);
if (ret < 0) if (ret < 0)
return ret; return ret;
break; break;
...@@ -831,8 +834,8 @@ static irqreturn_t chan_irqt(int irq, void *dev) ...@@ -831,8 +834,8 @@ static irqreturn_t chan_irqt(int irq, void *dev)
int shdma_request_irq(struct shdma_chan *schan, int irq, int shdma_request_irq(struct shdma_chan *schan, int irq,
unsigned long flags, const char *name) unsigned long flags, const char *name)
{ {
int ret = request_threaded_irq(irq, chan_irq, chan_irqt, int ret = devm_request_threaded_irq(schan->dev, irq, chan_irq,
flags, name, schan); chan_irqt, flags, name, schan);
schan->irq = ret < 0 ? ret : irq; schan->irq = ret < 0 ? ret : irq;
...@@ -840,13 +843,6 @@ int shdma_request_irq(struct shdma_chan *schan, int irq, ...@@ -840,13 +843,6 @@ int shdma_request_irq(struct shdma_chan *schan, int irq,
} }
EXPORT_SYMBOL(shdma_request_irq); EXPORT_SYMBOL(shdma_request_irq);
void shdma_free_irq(struct shdma_chan *schan)
{
if (schan->irq >= 0)
free_irq(schan->irq, schan);
}
EXPORT_SYMBOL(shdma_free_irq);
void shdma_chan_probe(struct shdma_dev *sdev, void shdma_chan_probe(struct shdma_dev *sdev,
struct shdma_chan *schan, int id) struct shdma_chan *schan, int id)
{ {
......
...@@ -42,12 +42,9 @@ static struct dma_chan *shdma_of_xlate(struct of_phandle_args *dma_spec, ...@@ -42,12 +42,9 @@ static struct dma_chan *shdma_of_xlate(struct of_phandle_args *dma_spec,
static int shdma_of_probe(struct platform_device *pdev) static int shdma_of_probe(struct platform_device *pdev)
{ {
const struct of_dev_auxdata *lookup = pdev->dev.platform_data; const struct of_dev_auxdata *lookup = dev_get_platdata(&pdev->dev);
int ret; int ret;
if (!lookup)
return -EINVAL;
ret = of_dma_controller_register(pdev->dev.of_node, ret = of_dma_controller_register(pdev->dev.of_node,
shdma_of_xlate, pdev); shdma_of_xlate, pdev);
if (ret < 0) if (ret < 0)
......
/*
* Renesas SuperH DMA Engine support for r8a73a4 (APE6) SoCs
*
* Copyright (C) 2013 Renesas Electronics, Inc.
*
* This is free software; you can redistribute it and/or modify it under the
* terms of version 2 the GNU General Public License as published by the Free
* Software Foundation.
*/
#include <linux/sh_dma.h>
#include "shdma-arm.h"
const unsigned int dma_ts_shift[] = SH_DMAE_TS_SHIFT;
static const struct sh_dmae_slave_config dma_slaves[] = {
{
.chcr = CHCR_TX(XMIT_SZ_32BIT),
.mid_rid = 0xd1, /* MMC0 Tx */
}, {
.chcr = CHCR_RX(XMIT_SZ_32BIT),
.mid_rid = 0xd2, /* MMC0 Rx */
}, {
.chcr = CHCR_TX(XMIT_SZ_32BIT),
.mid_rid = 0xe1, /* MMC1 Tx */
}, {
.chcr = CHCR_RX(XMIT_SZ_32BIT),
.mid_rid = 0xe2, /* MMC1 Rx */
},
};
#define DMAE_CHANNEL(a, b) \
{ \
.offset = (a) - 0x20, \
.dmars = (a) - 0x20 + 0x40, \
.chclr_bit = (b), \
.chclr_offset = 0x80 - 0x20, \
}
static const struct sh_dmae_channel dma_channels[] = {
DMAE_CHANNEL(0x8000, 0),
DMAE_CHANNEL(0x8080, 1),
DMAE_CHANNEL(0x8100, 2),
DMAE_CHANNEL(0x8180, 3),
DMAE_CHANNEL(0x8200, 4),
DMAE_CHANNEL(0x8280, 5),
DMAE_CHANNEL(0x8300, 6),
DMAE_CHANNEL(0x8380, 7),
DMAE_CHANNEL(0x8400, 8),
DMAE_CHANNEL(0x8480, 9),
DMAE_CHANNEL(0x8500, 10),
DMAE_CHANNEL(0x8580, 11),
DMAE_CHANNEL(0x8600, 12),
DMAE_CHANNEL(0x8680, 13),
DMAE_CHANNEL(0x8700, 14),
DMAE_CHANNEL(0x8780, 15),
DMAE_CHANNEL(0x8800, 16),
DMAE_CHANNEL(0x8880, 17),
DMAE_CHANNEL(0x8900, 18),
DMAE_CHANNEL(0x8980, 19),
};
const struct sh_dmae_pdata r8a73a4_dma_pdata = {
.slave = dma_slaves,
.slave_num = ARRAY_SIZE(dma_slaves),
.channel = dma_channels,
.channel_num = ARRAY_SIZE(dma_channels),
.ts_low_shift = TS_LOW_SHIFT,
.ts_low_mask = TS_LOW_BIT << TS_LOW_SHIFT,
.ts_high_shift = TS_HI_SHIFT,
.ts_high_mask = TS_HI_BIT << TS_HI_SHIFT,
.ts_shift = dma_ts_shift,
.ts_shift_num = ARRAY_SIZE(dma_ts_shift),
.dmaor_init = DMAOR_DME,
.chclr_present = 1,
.chclr_bitwise = 1,
};
...@@ -28,18 +28,19 @@ struct sh_dmae_chan { ...@@ -28,18 +28,19 @@ struct sh_dmae_chan {
struct shdma_chan shdma_chan; struct shdma_chan shdma_chan;
const struct sh_dmae_slave_config *config; /* Slave DMA configuration */ const struct sh_dmae_slave_config *config; /* Slave DMA configuration */
int xmit_shift; /* log_2(bytes_per_xfer) */ int xmit_shift; /* log_2(bytes_per_xfer) */
u32 __iomem *base; void __iomem *base;
char dev_id[16]; /* unique name per DMAC of channel */ char dev_id[16]; /* unique name per DMAC of channel */
int pm_error; int pm_error;
dma_addr_t slave_addr;
}; };
struct sh_dmae_device { struct sh_dmae_device {
struct shdma_dev shdma_dev; struct shdma_dev shdma_dev;
struct sh_dmae_chan *chan[SH_DMAE_MAX_CHANNELS]; struct sh_dmae_chan *chan[SH_DMAE_MAX_CHANNELS];
struct sh_dmae_pdata *pdata; const struct sh_dmae_pdata *pdata;
struct list_head node; struct list_head node;
u32 __iomem *chan_reg; void __iomem *chan_reg;
u16 __iomem *dmars; void __iomem *dmars;
unsigned int chcr_offset; unsigned int chcr_offset;
u32 chcr_ie_bit; u32 chcr_ie_bit;
}; };
...@@ -61,4 +62,11 @@ struct sh_dmae_desc { ...@@ -61,4 +62,11 @@ struct sh_dmae_desc {
#define to_sh_dev(chan) container_of(chan->shdma_chan.dma_chan.device,\ #define to_sh_dev(chan) container_of(chan->shdma_chan.dma_chan.device,\
struct sh_dmae_device, shdma_dev.dma_dev) struct sh_dmae_device, shdma_dev.dma_dev)
#ifdef CONFIG_SHDMA_R8A73A4
extern const struct sh_dmae_pdata r8a73a4_dma_pdata;
#define r8a73a4_shdma_devid (&r8a73a4_dma_pdata)
#else
#define r8a73a4_shdma_devid NULL
#endif
#endif /* __DMA_SHDMA_H */ #endif /* __DMA_SHDMA_H */
...@@ -150,7 +150,8 @@ static const struct sudmac_slave_config *sudmac_find_slave( ...@@ -150,7 +150,8 @@ static const struct sudmac_slave_config *sudmac_find_slave(
return NULL; return NULL;
} }
static int sudmac_set_slave(struct shdma_chan *schan, int slave_id, bool try) static int sudmac_set_slave(struct shdma_chan *schan, int slave_id,
dma_addr_t slave_addr, bool try)
{ {
struct sudmac_chan *sc = to_chan(schan); struct sudmac_chan *sc = to_chan(schan);
const struct sudmac_slave_config *cfg = sudmac_find_slave(sc, slave_id); const struct sudmac_slave_config *cfg = sudmac_find_slave(sc, slave_id);
...@@ -298,11 +299,8 @@ static void sudmac_chan_remove(struct sudmac_device *su_dev) ...@@ -298,11 +299,8 @@ static void sudmac_chan_remove(struct sudmac_device *su_dev)
int i; int i;
shdma_for_each_chan(schan, &su_dev->shdma_dev, i) { shdma_for_each_chan(schan, &su_dev->shdma_dev, i) {
struct sudmac_chan *sc = to_chan(schan);
BUG_ON(!schan); BUG_ON(!schan);
shdma_free_irq(&sc->shdma_chan);
shdma_chan_remove(schan); shdma_chan_remove(schan);
} }
dma_dev->chancnt = 0; dma_dev->chancnt = 0;
...@@ -335,7 +333,7 @@ static const struct shdma_ops sudmac_shdma_ops = { ...@@ -335,7 +333,7 @@ static const struct shdma_ops sudmac_shdma_ops = {
static int sudmac_probe(struct platform_device *pdev) static int sudmac_probe(struct platform_device *pdev)
{ {
struct sudmac_pdata *pdata = pdev->dev.platform_data; struct sudmac_pdata *pdata = dev_get_platdata(&pdev->dev);
int err, i; int err, i;
struct sudmac_device *su_dev; struct sudmac_device *su_dev;
struct dma_device *dma_dev; struct dma_device *dma_dev;
...@@ -345,9 +343,8 @@ static int sudmac_probe(struct platform_device *pdev) ...@@ -345,9 +343,8 @@ static int sudmac_probe(struct platform_device *pdev)
if (!pdata) if (!pdata)
return -ENODEV; return -ENODEV;
chan = platform_get_resource(pdev, IORESOURCE_MEM, 0);
irq_res = platform_get_resource(pdev, IORESOURCE_IRQ, 0); irq_res = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
if (!chan || !irq_res) if (!irq_res)
return -ENODEV; return -ENODEV;
err = -ENOMEM; err = -ENOMEM;
...@@ -360,9 +357,10 @@ static int sudmac_probe(struct platform_device *pdev) ...@@ -360,9 +357,10 @@ static int sudmac_probe(struct platform_device *pdev)
dma_dev = &su_dev->shdma_dev.dma_dev; dma_dev = &su_dev->shdma_dev.dma_dev;
su_dev->chan_reg = devm_request_and_ioremap(&pdev->dev, chan); chan = platform_get_resource(pdev, IORESOURCE_MEM, 0);
if (!su_dev->chan_reg) su_dev->chan_reg = devm_ioremap_resource(&pdev->dev, chan);
return err; if (IS_ERR(su_dev->chan_reg))
return PTR_ERR(su_dev->chan_reg);
dma_cap_set(DMA_SLAVE, dma_dev->cap_mask); dma_cap_set(DMA_SLAVE, dma_dev->cap_mask);
...@@ -373,7 +371,7 @@ static int sudmac_probe(struct platform_device *pdev) ...@@ -373,7 +371,7 @@ static int sudmac_probe(struct platform_device *pdev)
return err; return err;
/* platform data */ /* platform data */
su_dev->pdata = pdev->dev.platform_data; su_dev->pdata = dev_get_platdata(&pdev->dev);
platform_set_drvdata(pdev, su_dev); platform_set_drvdata(pdev, su_dev);
...@@ -393,7 +391,6 @@ static int sudmac_probe(struct platform_device *pdev) ...@@ -393,7 +391,6 @@ static int sudmac_probe(struct platform_device *pdev)
chan_probe_err: chan_probe_err:
sudmac_chan_remove(su_dev); sudmac_chan_remove(su_dev);
platform_set_drvdata(pdev, NULL);
shdma_cleanup(&su_dev->shdma_dev); shdma_cleanup(&su_dev->shdma_dev);
return err; return err;
...@@ -407,7 +404,6 @@ static int sudmac_remove(struct platform_device *pdev) ...@@ -407,7 +404,6 @@ static int sudmac_remove(struct platform_device *pdev)
dma_async_device_unregister(dma_dev); dma_async_device_unregister(dma_dev);
sudmac_chan_remove(su_dev); sudmac_chan_remove(su_dev);
shdma_cleanup(&su_dev->shdma_dev); shdma_cleanup(&su_dev->shdma_dev);
platform_set_drvdata(pdev, NULL);
return 0; return 0;
} }
......
This diff is collapsed.
This diff is collapsed.
...@@ -767,13 +767,11 @@ static enum dma_status tegra_dma_tx_status(struct dma_chan *dc, ...@@ -767,13 +767,11 @@ static enum dma_status tegra_dma_tx_status(struct dma_chan *dc,
unsigned long flags; unsigned long flags;
unsigned int residual; unsigned int residual;
spin_lock_irqsave(&tdc->lock, flags);
ret = dma_cookie_status(dc, cookie, txstate); ret = dma_cookie_status(dc, cookie, txstate);
if (ret == DMA_SUCCESS) { if (ret == DMA_SUCCESS)
spin_unlock_irqrestore(&tdc->lock, flags);
return ret; return ret;
}
spin_lock_irqsave(&tdc->lock, flags);
/* Check on wait_ack desc status */ /* Check on wait_ack desc status */
list_for_each_entry(dma_desc, &tdc->free_dma_desc, node) { list_for_each_entry(dma_desc, &tdc->free_dma_desc, node) {
......
...@@ -669,7 +669,7 @@ static irqreturn_t td_irq(int irq, void *devid) ...@@ -669,7 +669,7 @@ static irqreturn_t td_irq(int irq, void *devid)
static int td_probe(struct platform_device *pdev) static int td_probe(struct platform_device *pdev)
{ {
struct timb_dma_platform_data *pdata = pdev->dev.platform_data; struct timb_dma_platform_data *pdata = dev_get_platdata(&pdev->dev);
struct timb_dma *td; struct timb_dma *td;
struct resource *iomem; struct resource *iomem;
int irq; int irq;
......
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment