Merge branch 'topic/jz' into for-linus
This commit is contained in:
commit
81c3ee027e
@ -2,8 +2,13 @@
|
|||||||
|
|
||||||
Required properties:
|
Required properties:
|
||||||
|
|
||||||
- compatible: Should be "ingenic,jz4780-dma"
|
- compatible: Should be one of:
|
||||||
- reg: Should contain the DMA controller registers location and length.
|
* ingenic,jz4740-dma
|
||||||
|
* ingenic,jz4725b-dma
|
||||||
|
* ingenic,jz4770-dma
|
||||||
|
* ingenic,jz4780-dma
|
||||||
|
- reg: Should contain the DMA channel registers location and length, followed
|
||||||
|
by the DMA controller registers location and length.
|
||||||
- interrupts: Should contain the interrupt specifier of the DMA controller.
|
- interrupts: Should contain the interrupt specifier of the DMA controller.
|
||||||
- clocks: Should contain a clock specifier for the JZ4780 PDMA clock.
|
- clocks: Should contain a clock specifier for the JZ4780 PDMA clock.
|
||||||
- #dma-cells: Must be <2>. Number of integer cells in the dmas property of
|
- #dma-cells: Must be <2>. Number of integer cells in the dmas property of
|
||||||
@ -19,9 +24,10 @@ Optional properties:
|
|||||||
|
|
||||||
Example:
|
Example:
|
||||||
|
|
||||||
dma: dma@13420000 {
|
dma: dma-controller@13420000 {
|
||||||
compatible = "ingenic,jz4780-dma";
|
compatible = "ingenic,jz4780-dma";
|
||||||
reg = <0x13420000 0x10000>;
|
reg = <0x13420000 0x400
|
||||||
|
0x13421000 0x40>;
|
||||||
|
|
||||||
interrupt-parent = <&intc>;
|
interrupt-parent = <&intc>;
|
||||||
interrupts = <10>;
|
interrupts = <10>;
|
||||||
|
@ -154,6 +154,21 @@
|
|||||||
clock-names = "baud", "module";
|
clock-names = "baud", "module";
|
||||||
};
|
};
|
||||||
|
|
||||||
|
dmac: dma-controller@13020000 {
|
||||||
|
compatible = "ingenic,jz4740-dma";
|
||||||
|
reg = <0x13020000 0xbc
|
||||||
|
0x13020300 0x14>;
|
||||||
|
#dma-cells = <2>;
|
||||||
|
|
||||||
|
interrupt-parent = <&intc>;
|
||||||
|
interrupts = <29>;
|
||||||
|
|
||||||
|
clocks = <&cgu JZ4740_CLK_DMA>;
|
||||||
|
|
||||||
|
/* Disable dmac until we have something that uses it */
|
||||||
|
status = "disabled";
|
||||||
|
};
|
||||||
|
|
||||||
uhc: uhc@13030000 {
|
uhc: uhc@13030000 {
|
||||||
compatible = "ingenic,jz4740-ohci", "generic-ohci";
|
compatible = "ingenic,jz4740-ohci", "generic-ohci";
|
||||||
reg = <0x13030000 0x1000>;
|
reg = <0x13030000 0x1000>;
|
||||||
|
@ -196,6 +196,36 @@
|
|||||||
status = "disabled";
|
status = "disabled";
|
||||||
};
|
};
|
||||||
|
|
||||||
|
dmac0: dma-controller@13420000 {
|
||||||
|
compatible = "ingenic,jz4770-dma";
|
||||||
|
reg = <0x13420000 0xC0
|
||||||
|
0x13420300 0x20>;
|
||||||
|
|
||||||
|
#dma-cells = <1>;
|
||||||
|
|
||||||
|
clocks = <&cgu JZ4770_CLK_DMA>;
|
||||||
|
interrupt-parent = <&intc>;
|
||||||
|
interrupts = <24>;
|
||||||
|
|
||||||
|
/* Disable dmac0 until we have something that uses it */
|
||||||
|
status = "disabled";
|
||||||
|
};
|
||||||
|
|
||||||
|
dmac1: dma-controller@13420100 {
|
||||||
|
compatible = "ingenic,jz4770-dma";
|
||||||
|
reg = <0x13420100 0xC0
|
||||||
|
0x13420400 0x20>;
|
||||||
|
|
||||||
|
#dma-cells = <1>;
|
||||||
|
|
||||||
|
clocks = <&cgu JZ4770_CLK_DMA>;
|
||||||
|
interrupt-parent = <&intc>;
|
||||||
|
interrupts = <23>;
|
||||||
|
|
||||||
|
/* Disable dmac1 until we have something that uses it */
|
||||||
|
status = "disabled";
|
||||||
|
};
|
||||||
|
|
||||||
uhc: uhc@13430000 {
|
uhc: uhc@13430000 {
|
||||||
compatible = "generic-ohci";
|
compatible = "generic-ohci";
|
||||||
reg = <0x13430000 0x1000>;
|
reg = <0x13430000 0x1000>;
|
||||||
|
@ -266,7 +266,8 @@
|
|||||||
|
|
||||||
dma: dma@13420000 {
|
dma: dma@13420000 {
|
||||||
compatible = "ingenic,jz4780-dma";
|
compatible = "ingenic,jz4780-dma";
|
||||||
reg = <0x13420000 0x10000>;
|
reg = <0x13420000 0x400
|
||||||
|
0x13421000 0x40>;
|
||||||
#dma-cells = <2>;
|
#dma-cells = <2>;
|
||||||
|
|
||||||
interrupt-parent = <&intc>;
|
interrupt-parent = <&intc>;
|
||||||
|
@ -143,7 +143,7 @@ config DMA_JZ4740
|
|||||||
|
|
||||||
config DMA_JZ4780
|
config DMA_JZ4780
|
||||||
tristate "JZ4780 DMA support"
|
tristate "JZ4780 DMA support"
|
||||||
depends on MACH_JZ4780 || COMPILE_TEST
|
depends on MIPS || COMPILE_TEST
|
||||||
select DMA_ENGINE
|
select DMA_ENGINE
|
||||||
select DMA_VIRTUAL_CHANNELS
|
select DMA_VIRTUAL_CHANNELS
|
||||||
help
|
help
|
||||||
|
@ -16,6 +16,7 @@
|
|||||||
#include <linux/interrupt.h>
|
#include <linux/interrupt.h>
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
#include <linux/of.h>
|
#include <linux/of.h>
|
||||||
|
#include <linux/of_device.h>
|
||||||
#include <linux/of_dma.h>
|
#include <linux/of_dma.h>
|
||||||
#include <linux/platform_device.h>
|
#include <linux/platform_device.h>
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
@ -23,33 +24,35 @@
|
|||||||
#include "dmaengine.h"
|
#include "dmaengine.h"
|
||||||
#include "virt-dma.h"
|
#include "virt-dma.h"
|
||||||
|
|
||||||
#define JZ_DMA_NR_CHANNELS 32
|
|
||||||
|
|
||||||
/* Global registers. */
|
/* Global registers. */
|
||||||
#define JZ_DMA_REG_DMAC 0x1000
|
#define JZ_DMA_REG_DMAC 0x00
|
||||||
#define JZ_DMA_REG_DIRQP 0x1004
|
#define JZ_DMA_REG_DIRQP 0x04
|
||||||
#define JZ_DMA_REG_DDR 0x1008
|
#define JZ_DMA_REG_DDR 0x08
|
||||||
#define JZ_DMA_REG_DDRS 0x100c
|
#define JZ_DMA_REG_DDRS 0x0c
|
||||||
#define JZ_DMA_REG_DMACP 0x101c
|
#define JZ_DMA_REG_DCKE 0x10
|
||||||
#define JZ_DMA_REG_DSIRQP 0x1020
|
#define JZ_DMA_REG_DCKES 0x14
|
||||||
#define JZ_DMA_REG_DSIRQM 0x1024
|
#define JZ_DMA_REG_DCKEC 0x18
|
||||||
#define JZ_DMA_REG_DCIRQP 0x1028
|
#define JZ_DMA_REG_DMACP 0x1c
|
||||||
#define JZ_DMA_REG_DCIRQM 0x102c
|
#define JZ_DMA_REG_DSIRQP 0x20
|
||||||
|
#define JZ_DMA_REG_DSIRQM 0x24
|
||||||
|
#define JZ_DMA_REG_DCIRQP 0x28
|
||||||
|
#define JZ_DMA_REG_DCIRQM 0x2c
|
||||||
|
|
||||||
/* Per-channel registers. */
|
/* Per-channel registers. */
|
||||||
#define JZ_DMA_REG_CHAN(n) (n * 0x20)
|
#define JZ_DMA_REG_CHAN(n) (n * 0x20)
|
||||||
#define JZ_DMA_REG_DSA(n) (0x00 + JZ_DMA_REG_CHAN(n))
|
#define JZ_DMA_REG_DSA 0x00
|
||||||
#define JZ_DMA_REG_DTA(n) (0x04 + JZ_DMA_REG_CHAN(n))
|
#define JZ_DMA_REG_DTA 0x04
|
||||||
#define JZ_DMA_REG_DTC(n) (0x08 + JZ_DMA_REG_CHAN(n))
|
#define JZ_DMA_REG_DTC 0x08
|
||||||
#define JZ_DMA_REG_DRT(n) (0x0c + JZ_DMA_REG_CHAN(n))
|
#define JZ_DMA_REG_DRT 0x0c
|
||||||
#define JZ_DMA_REG_DCS(n) (0x10 + JZ_DMA_REG_CHAN(n))
|
#define JZ_DMA_REG_DCS 0x10
|
||||||
#define JZ_DMA_REG_DCM(n) (0x14 + JZ_DMA_REG_CHAN(n))
|
#define JZ_DMA_REG_DCM 0x14
|
||||||
#define JZ_DMA_REG_DDA(n) (0x18 + JZ_DMA_REG_CHAN(n))
|
#define JZ_DMA_REG_DDA 0x18
|
||||||
#define JZ_DMA_REG_DSD(n) (0x1c + JZ_DMA_REG_CHAN(n))
|
#define JZ_DMA_REG_DSD 0x1c
|
||||||
|
|
||||||
#define JZ_DMA_DMAC_DMAE BIT(0)
|
#define JZ_DMA_DMAC_DMAE BIT(0)
|
||||||
#define JZ_DMA_DMAC_AR BIT(2)
|
#define JZ_DMA_DMAC_AR BIT(2)
|
||||||
#define JZ_DMA_DMAC_HLT BIT(3)
|
#define JZ_DMA_DMAC_HLT BIT(3)
|
||||||
|
#define JZ_DMA_DMAC_FAIC BIT(27)
|
||||||
#define JZ_DMA_DMAC_FMSC BIT(31)
|
#define JZ_DMA_DMAC_FMSC BIT(31)
|
||||||
|
|
||||||
#define JZ_DMA_DRT_AUTO 0x8
|
#define JZ_DMA_DRT_AUTO 0x8
|
||||||
@ -86,6 +89,14 @@
|
|||||||
BIT(DMA_SLAVE_BUSWIDTH_2_BYTES) | \
|
BIT(DMA_SLAVE_BUSWIDTH_2_BYTES) | \
|
||||||
BIT(DMA_SLAVE_BUSWIDTH_4_BYTES))
|
BIT(DMA_SLAVE_BUSWIDTH_4_BYTES))
|
||||||
|
|
||||||
|
#define JZ4780_DMA_CTRL_OFFSET 0x1000
|
||||||
|
|
||||||
|
/* macros for use with jz4780_dma_soc_data.flags */
|
||||||
|
#define JZ_SOC_DATA_ALLOW_LEGACY_DT BIT(0)
|
||||||
|
#define JZ_SOC_DATA_PROGRAMMABLE_DMA BIT(1)
|
||||||
|
#define JZ_SOC_DATA_PER_CHAN_PM BIT(2)
|
||||||
|
#define JZ_SOC_DATA_NO_DCKES_DCKEC BIT(3)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* struct jz4780_dma_hwdesc - descriptor structure read by the DMA controller.
|
* struct jz4780_dma_hwdesc - descriptor structure read by the DMA controller.
|
||||||
* @dcm: value for the DCM (channel command) register
|
* @dcm: value for the DCM (channel command) register
|
||||||
@ -94,17 +105,12 @@
|
|||||||
* @dtc: transfer count (number of blocks of the transfer size specified in DCM
|
* @dtc: transfer count (number of blocks of the transfer size specified in DCM
|
||||||
* to transfer) in the low 24 bits, offset of the next descriptor from the
|
* to transfer) in the low 24 bits, offset of the next descriptor from the
|
||||||
* descriptor base address in the upper 8 bits.
|
* descriptor base address in the upper 8 bits.
|
||||||
* @sd: target/source stride difference (in stride transfer mode).
|
|
||||||
* @drt: request type
|
|
||||||
*/
|
*/
|
||||||
struct jz4780_dma_hwdesc {
|
struct jz4780_dma_hwdesc {
|
||||||
uint32_t dcm;
|
uint32_t dcm;
|
||||||
uint32_t dsa;
|
uint32_t dsa;
|
||||||
uint32_t dta;
|
uint32_t dta;
|
||||||
uint32_t dtc;
|
uint32_t dtc;
|
||||||
uint32_t sd;
|
|
||||||
uint32_t drt;
|
|
||||||
uint32_t reserved[2];
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Size of allocations for hardware descriptor blocks. */
|
/* Size of allocations for hardware descriptor blocks. */
|
||||||
@ -135,14 +141,22 @@ struct jz4780_dma_chan {
|
|||||||
unsigned int curr_hwdesc;
|
unsigned int curr_hwdesc;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct jz4780_dma_soc_data {
|
||||||
|
unsigned int nb_channels;
|
||||||
|
unsigned int transfer_ord_max;
|
||||||
|
unsigned long flags;
|
||||||
|
};
|
||||||
|
|
||||||
struct jz4780_dma_dev {
|
struct jz4780_dma_dev {
|
||||||
struct dma_device dma_device;
|
struct dma_device dma_device;
|
||||||
void __iomem *base;
|
void __iomem *chn_base;
|
||||||
|
void __iomem *ctrl_base;
|
||||||
struct clk *clk;
|
struct clk *clk;
|
||||||
unsigned int irq;
|
unsigned int irq;
|
||||||
|
const struct jz4780_dma_soc_data *soc_data;
|
||||||
|
|
||||||
uint32_t chan_reserved;
|
uint32_t chan_reserved;
|
||||||
struct jz4780_dma_chan chan[JZ_DMA_NR_CHANNELS];
|
struct jz4780_dma_chan chan[];
|
||||||
};
|
};
|
||||||
|
|
||||||
struct jz4780_dma_filter_data {
|
struct jz4780_dma_filter_data {
|
||||||
@ -169,16 +183,51 @@ static inline struct jz4780_dma_dev *jz4780_dma_chan_parent(
|
|||||||
dma_device);
|
dma_device);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline uint32_t jz4780_dma_readl(struct jz4780_dma_dev *jzdma,
|
static inline uint32_t jz4780_dma_chn_readl(struct jz4780_dma_dev *jzdma,
|
||||||
unsigned int reg)
|
unsigned int chn, unsigned int reg)
|
||||||
{
|
{
|
||||||
return readl(jzdma->base + reg);
|
return readl(jzdma->chn_base + reg + JZ_DMA_REG_CHAN(chn));
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void jz4780_dma_writel(struct jz4780_dma_dev *jzdma,
|
static inline void jz4780_dma_chn_writel(struct jz4780_dma_dev *jzdma,
|
||||||
|
unsigned int chn, unsigned int reg, uint32_t val)
|
||||||
|
{
|
||||||
|
writel(val, jzdma->chn_base + reg + JZ_DMA_REG_CHAN(chn));
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline uint32_t jz4780_dma_ctrl_readl(struct jz4780_dma_dev *jzdma,
|
||||||
|
unsigned int reg)
|
||||||
|
{
|
||||||
|
return readl(jzdma->ctrl_base + reg);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void jz4780_dma_ctrl_writel(struct jz4780_dma_dev *jzdma,
|
||||||
unsigned int reg, uint32_t val)
|
unsigned int reg, uint32_t val)
|
||||||
{
|
{
|
||||||
writel(val, jzdma->base + reg);
|
writel(val, jzdma->ctrl_base + reg);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void jz4780_dma_chan_enable(struct jz4780_dma_dev *jzdma,
|
||||||
|
unsigned int chn)
|
||||||
|
{
|
||||||
|
if (jzdma->soc_data->flags & JZ_SOC_DATA_PER_CHAN_PM) {
|
||||||
|
unsigned int reg;
|
||||||
|
|
||||||
|
if (jzdma->soc_data->flags & JZ_SOC_DATA_NO_DCKES_DCKEC)
|
||||||
|
reg = JZ_DMA_REG_DCKE;
|
||||||
|
else
|
||||||
|
reg = JZ_DMA_REG_DCKES;
|
||||||
|
|
||||||
|
jz4780_dma_ctrl_writel(jzdma, reg, BIT(chn));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void jz4780_dma_chan_disable(struct jz4780_dma_dev *jzdma,
|
||||||
|
unsigned int chn)
|
||||||
|
{
|
||||||
|
if ((jzdma->soc_data->flags & JZ_SOC_DATA_PER_CHAN_PM) &&
|
||||||
|
!(jzdma->soc_data->flags & JZ_SOC_DATA_NO_DCKES_DCKEC))
|
||||||
|
jz4780_dma_ctrl_writel(jzdma, JZ_DMA_REG_DCKEC, BIT(chn));
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct jz4780_dma_desc *jz4780_dma_desc_alloc(
|
static struct jz4780_dma_desc *jz4780_dma_desc_alloc(
|
||||||
@ -215,8 +264,10 @@ static void jz4780_dma_desc_free(struct virt_dma_desc *vdesc)
|
|||||||
kfree(desc);
|
kfree(desc);
|
||||||
}
|
}
|
||||||
|
|
||||||
static uint32_t jz4780_dma_transfer_size(unsigned long val, uint32_t *shift)
|
static uint32_t jz4780_dma_transfer_size(struct jz4780_dma_chan *jzchan,
|
||||||
|
unsigned long val, uint32_t *shift)
|
||||||
{
|
{
|
||||||
|
struct jz4780_dma_dev *jzdma = jz4780_dma_chan_parent(jzchan);
|
||||||
int ord = ffs(val) - 1;
|
int ord = ffs(val) - 1;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -228,8 +279,8 @@ static uint32_t jz4780_dma_transfer_size(unsigned long val, uint32_t *shift)
|
|||||||
*/
|
*/
|
||||||
if (ord == 3)
|
if (ord == 3)
|
||||||
ord = 2;
|
ord = 2;
|
||||||
else if (ord > 7)
|
else if (ord > jzdma->soc_data->transfer_ord_max)
|
||||||
ord = 7;
|
ord = jzdma->soc_data->transfer_ord_max;
|
||||||
|
|
||||||
*shift = ord;
|
*shift = ord;
|
||||||
|
|
||||||
@ -262,7 +313,6 @@ static int jz4780_dma_setup_hwdesc(struct jz4780_dma_chan *jzchan,
|
|||||||
desc->dcm = JZ_DMA_DCM_SAI;
|
desc->dcm = JZ_DMA_DCM_SAI;
|
||||||
desc->dsa = addr;
|
desc->dsa = addr;
|
||||||
desc->dta = config->dst_addr;
|
desc->dta = config->dst_addr;
|
||||||
desc->drt = jzchan->transfer_type;
|
|
||||||
|
|
||||||
width = config->dst_addr_width;
|
width = config->dst_addr_width;
|
||||||
maxburst = config->dst_maxburst;
|
maxburst = config->dst_maxburst;
|
||||||
@ -270,7 +320,6 @@ static int jz4780_dma_setup_hwdesc(struct jz4780_dma_chan *jzchan,
|
|||||||
desc->dcm = JZ_DMA_DCM_DAI;
|
desc->dcm = JZ_DMA_DCM_DAI;
|
||||||
desc->dsa = config->src_addr;
|
desc->dsa = config->src_addr;
|
||||||
desc->dta = addr;
|
desc->dta = addr;
|
||||||
desc->drt = jzchan->transfer_type;
|
|
||||||
|
|
||||||
width = config->src_addr_width;
|
width = config->src_addr_width;
|
||||||
maxburst = config->src_maxburst;
|
maxburst = config->src_maxburst;
|
||||||
@ -283,7 +332,7 @@ static int jz4780_dma_setup_hwdesc(struct jz4780_dma_chan *jzchan,
|
|||||||
* divisible by the transfer size, and we must not use more than the
|
* divisible by the transfer size, and we must not use more than the
|
||||||
* maximum burst specified by the user.
|
* maximum burst specified by the user.
|
||||||
*/
|
*/
|
||||||
tsz = jz4780_dma_transfer_size(addr | len | (width * maxburst),
|
tsz = jz4780_dma_transfer_size(jzchan, addr | len | (width * maxburst),
|
||||||
&jzchan->transfer_shift);
|
&jzchan->transfer_shift);
|
||||||
|
|
||||||
switch (width) {
|
switch (width) {
|
||||||
@ -412,12 +461,13 @@ static struct dma_async_tx_descriptor *jz4780_dma_prep_dma_memcpy(
|
|||||||
if (!desc)
|
if (!desc)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
tsz = jz4780_dma_transfer_size(dest | src | len,
|
tsz = jz4780_dma_transfer_size(jzchan, dest | src | len,
|
||||||
&jzchan->transfer_shift);
|
&jzchan->transfer_shift);
|
||||||
|
|
||||||
|
jzchan->transfer_type = JZ_DMA_DRT_AUTO;
|
||||||
|
|
||||||
desc->desc[0].dsa = src;
|
desc->desc[0].dsa = src;
|
||||||
desc->desc[0].dta = dest;
|
desc->desc[0].dta = dest;
|
||||||
desc->desc[0].drt = JZ_DMA_DRT_AUTO;
|
|
||||||
desc->desc[0].dcm = JZ_DMA_DCM_TIE | JZ_DMA_DCM_SAI | JZ_DMA_DCM_DAI |
|
desc->desc[0].dcm = JZ_DMA_DCM_TIE | JZ_DMA_DCM_SAI | JZ_DMA_DCM_DAI |
|
||||||
tsz << JZ_DMA_DCM_TSZ_SHIFT |
|
tsz << JZ_DMA_DCM_TSZ_SHIFT |
|
||||||
JZ_DMA_WIDTH_32_BIT << JZ_DMA_DCM_SP_SHIFT |
|
JZ_DMA_WIDTH_32_BIT << JZ_DMA_DCM_SP_SHIFT |
|
||||||
@ -472,18 +522,34 @@ static void jz4780_dma_begin(struct jz4780_dma_chan *jzchan)
|
|||||||
(jzchan->curr_hwdesc + 1) % jzchan->desc->count;
|
(jzchan->curr_hwdesc + 1) % jzchan->desc->count;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Use 8-word descriptors. */
|
/* Enable the channel's clock. */
|
||||||
jz4780_dma_writel(jzdma, JZ_DMA_REG_DCS(jzchan->id), JZ_DMA_DCS_DES8);
|
jz4780_dma_chan_enable(jzdma, jzchan->id);
|
||||||
|
|
||||||
|
/* Use 4-word descriptors. */
|
||||||
|
jz4780_dma_chn_writel(jzdma, jzchan->id, JZ_DMA_REG_DCS, 0);
|
||||||
|
|
||||||
|
/* Set transfer type. */
|
||||||
|
jz4780_dma_chn_writel(jzdma, jzchan->id, JZ_DMA_REG_DRT,
|
||||||
|
jzchan->transfer_type);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Set the transfer count. This is redundant for a descriptor-driven
|
||||||
|
* transfer. However, there can be a delay between the transfer start
|
||||||
|
* time and when DTCn reg contains the new transfer count. Setting
|
||||||
|
* it explicitly ensures residue is computed correctly at all times.
|
||||||
|
*/
|
||||||
|
jz4780_dma_chn_writel(jzdma, jzchan->id, JZ_DMA_REG_DTC,
|
||||||
|
jzchan->desc->desc[jzchan->curr_hwdesc].dtc);
|
||||||
|
|
||||||
/* Write descriptor address and initiate descriptor fetch. */
|
/* Write descriptor address and initiate descriptor fetch. */
|
||||||
desc_phys = jzchan->desc->desc_phys +
|
desc_phys = jzchan->desc->desc_phys +
|
||||||
(jzchan->curr_hwdesc * sizeof(*jzchan->desc->desc));
|
(jzchan->curr_hwdesc * sizeof(*jzchan->desc->desc));
|
||||||
jz4780_dma_writel(jzdma, JZ_DMA_REG_DDA(jzchan->id), desc_phys);
|
jz4780_dma_chn_writel(jzdma, jzchan->id, JZ_DMA_REG_DDA, desc_phys);
|
||||||
jz4780_dma_writel(jzdma, JZ_DMA_REG_DDRS, BIT(jzchan->id));
|
jz4780_dma_ctrl_writel(jzdma, JZ_DMA_REG_DDRS, BIT(jzchan->id));
|
||||||
|
|
||||||
/* Enable the channel. */
|
/* Enable the channel. */
|
||||||
jz4780_dma_writel(jzdma, JZ_DMA_REG_DCS(jzchan->id),
|
jz4780_dma_chn_writel(jzdma, jzchan->id, JZ_DMA_REG_DCS,
|
||||||
JZ_DMA_DCS_DES8 | JZ_DMA_DCS_CTE);
|
JZ_DMA_DCS_CTE);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void jz4780_dma_issue_pending(struct dma_chan *chan)
|
static void jz4780_dma_issue_pending(struct dma_chan *chan)
|
||||||
@ -509,12 +575,14 @@ static int jz4780_dma_terminate_all(struct dma_chan *chan)
|
|||||||
spin_lock_irqsave(&jzchan->vchan.lock, flags);
|
spin_lock_irqsave(&jzchan->vchan.lock, flags);
|
||||||
|
|
||||||
/* Clear the DMA status and stop the transfer. */
|
/* Clear the DMA status and stop the transfer. */
|
||||||
jz4780_dma_writel(jzdma, JZ_DMA_REG_DCS(jzchan->id), 0);
|
jz4780_dma_chn_writel(jzdma, jzchan->id, JZ_DMA_REG_DCS, 0);
|
||||||
if (jzchan->desc) {
|
if (jzchan->desc) {
|
||||||
vchan_terminate_vdesc(&jzchan->desc->vdesc);
|
vchan_terminate_vdesc(&jzchan->desc->vdesc);
|
||||||
jzchan->desc = NULL;
|
jzchan->desc = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
jz4780_dma_chan_disable(jzdma, jzchan->id);
|
||||||
|
|
||||||
vchan_get_all_descriptors(&jzchan->vchan, &head);
|
vchan_get_all_descriptors(&jzchan->vchan, &head);
|
||||||
|
|
||||||
spin_unlock_irqrestore(&jzchan->vchan.lock, flags);
|
spin_unlock_irqrestore(&jzchan->vchan.lock, flags);
|
||||||
@ -526,8 +594,10 @@ static int jz4780_dma_terminate_all(struct dma_chan *chan)
|
|||||||
static void jz4780_dma_synchronize(struct dma_chan *chan)
|
static void jz4780_dma_synchronize(struct dma_chan *chan)
|
||||||
{
|
{
|
||||||
struct jz4780_dma_chan *jzchan = to_jz4780_dma_chan(chan);
|
struct jz4780_dma_chan *jzchan = to_jz4780_dma_chan(chan);
|
||||||
|
struct jz4780_dma_dev *jzdma = jz4780_dma_chan_parent(jzchan);
|
||||||
|
|
||||||
vchan_synchronize(&jzchan->vchan);
|
vchan_synchronize(&jzchan->vchan);
|
||||||
|
jz4780_dma_chan_disable(jzdma, jzchan->id);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int jz4780_dma_config(struct dma_chan *chan,
|
static int jz4780_dma_config(struct dma_chan *chan,
|
||||||
@ -549,21 +619,17 @@ static size_t jz4780_dma_desc_residue(struct jz4780_dma_chan *jzchan,
|
|||||||
struct jz4780_dma_desc *desc, unsigned int next_sg)
|
struct jz4780_dma_desc *desc, unsigned int next_sg)
|
||||||
{
|
{
|
||||||
struct jz4780_dma_dev *jzdma = jz4780_dma_chan_parent(jzchan);
|
struct jz4780_dma_dev *jzdma = jz4780_dma_chan_parent(jzchan);
|
||||||
unsigned int residue, count;
|
unsigned int count = 0;
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
|
|
||||||
residue = 0;
|
|
||||||
|
|
||||||
for (i = next_sg; i < desc->count; i++)
|
for (i = next_sg; i < desc->count; i++)
|
||||||
residue += desc->desc[i].dtc << jzchan->transfer_shift;
|
count += desc->desc[i].dtc & GENMASK(23, 0);
|
||||||
|
|
||||||
if (next_sg != 0) {
|
if (next_sg != 0)
|
||||||
count = jz4780_dma_readl(jzdma,
|
count += jz4780_dma_chn_readl(jzdma, jzchan->id,
|
||||||
JZ_DMA_REG_DTC(jzchan->id));
|
JZ_DMA_REG_DTC);
|
||||||
residue += count << jzchan->transfer_shift;
|
|
||||||
}
|
|
||||||
|
|
||||||
return residue;
|
return count << jzchan->transfer_shift;
|
||||||
}
|
}
|
||||||
|
|
||||||
static enum dma_status jz4780_dma_tx_status(struct dma_chan *chan,
|
static enum dma_status jz4780_dma_tx_status(struct dma_chan *chan,
|
||||||
@ -573,6 +639,7 @@ static enum dma_status jz4780_dma_tx_status(struct dma_chan *chan,
|
|||||||
struct virt_dma_desc *vdesc;
|
struct virt_dma_desc *vdesc;
|
||||||
enum dma_status status;
|
enum dma_status status;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
unsigned long residue = 0;
|
||||||
|
|
||||||
status = dma_cookie_status(chan, cookie, txstate);
|
status = dma_cookie_status(chan, cookie, txstate);
|
||||||
if ((status == DMA_COMPLETE) || (txstate == NULL))
|
if ((status == DMA_COMPLETE) || (txstate == NULL))
|
||||||
@ -583,13 +650,13 @@ static enum dma_status jz4780_dma_tx_status(struct dma_chan *chan,
|
|||||||
vdesc = vchan_find_desc(&jzchan->vchan, cookie);
|
vdesc = vchan_find_desc(&jzchan->vchan, cookie);
|
||||||
if (vdesc) {
|
if (vdesc) {
|
||||||
/* On the issued list, so hasn't been processed yet */
|
/* On the issued list, so hasn't been processed yet */
|
||||||
txstate->residue = jz4780_dma_desc_residue(jzchan,
|
residue = jz4780_dma_desc_residue(jzchan,
|
||||||
to_jz4780_dma_desc(vdesc), 0);
|
to_jz4780_dma_desc(vdesc), 0);
|
||||||
} else if (cookie == jzchan->desc->vdesc.tx.cookie) {
|
} else if (cookie == jzchan->desc->vdesc.tx.cookie) {
|
||||||
txstate->residue = jz4780_dma_desc_residue(jzchan, jzchan->desc,
|
residue = jz4780_dma_desc_residue(jzchan, jzchan->desc,
|
||||||
(jzchan->curr_hwdesc + 1) % jzchan->desc->count);
|
jzchan->curr_hwdesc + 1);
|
||||||
} else
|
}
|
||||||
txstate->residue = 0;
|
dma_set_residue(txstate, residue);
|
||||||
|
|
||||||
if (vdesc && jzchan->desc && vdesc == &jzchan->desc->vdesc
|
if (vdesc && jzchan->desc && vdesc == &jzchan->desc->vdesc
|
||||||
&& jzchan->desc->status & (JZ_DMA_DCS_AR | JZ_DMA_DCS_HLT))
|
&& jzchan->desc->status & (JZ_DMA_DCS_AR | JZ_DMA_DCS_HLT))
|
||||||
@ -606,8 +673,8 @@ static void jz4780_dma_chan_irq(struct jz4780_dma_dev *jzdma,
|
|||||||
|
|
||||||
spin_lock(&jzchan->vchan.lock);
|
spin_lock(&jzchan->vchan.lock);
|
||||||
|
|
||||||
dcs = jz4780_dma_readl(jzdma, JZ_DMA_REG_DCS(jzchan->id));
|
dcs = jz4780_dma_chn_readl(jzdma, jzchan->id, JZ_DMA_REG_DCS);
|
||||||
jz4780_dma_writel(jzdma, JZ_DMA_REG_DCS(jzchan->id), 0);
|
jz4780_dma_chn_writel(jzdma, jzchan->id, JZ_DMA_REG_DCS, 0);
|
||||||
|
|
||||||
if (dcs & JZ_DMA_DCS_AR) {
|
if (dcs & JZ_DMA_DCS_AR) {
|
||||||
dev_warn(&jzchan->vchan.chan.dev->device,
|
dev_warn(&jzchan->vchan.chan.dev->device,
|
||||||
@ -646,9 +713,9 @@ static irqreturn_t jz4780_dma_irq_handler(int irq, void *data)
|
|||||||
uint32_t pending, dmac;
|
uint32_t pending, dmac;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
pending = jz4780_dma_readl(jzdma, JZ_DMA_REG_DIRQP);
|
pending = jz4780_dma_ctrl_readl(jzdma, JZ_DMA_REG_DIRQP);
|
||||||
|
|
||||||
for (i = 0; i < JZ_DMA_NR_CHANNELS; i++) {
|
for (i = 0; i < jzdma->soc_data->nb_channels; i++) {
|
||||||
if (!(pending & (1<<i)))
|
if (!(pending & (1<<i)))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
@ -656,12 +723,12 @@ static irqreturn_t jz4780_dma_irq_handler(int irq, void *data)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Clear halt and address error status of all channels. */
|
/* Clear halt and address error status of all channels. */
|
||||||
dmac = jz4780_dma_readl(jzdma, JZ_DMA_REG_DMAC);
|
dmac = jz4780_dma_ctrl_readl(jzdma, JZ_DMA_REG_DMAC);
|
||||||
dmac &= ~(JZ_DMA_DMAC_HLT | JZ_DMA_DMAC_AR);
|
dmac &= ~(JZ_DMA_DMAC_HLT | JZ_DMA_DMAC_AR);
|
||||||
jz4780_dma_writel(jzdma, JZ_DMA_REG_DMAC, dmac);
|
jz4780_dma_ctrl_writel(jzdma, JZ_DMA_REG_DMAC, dmac);
|
||||||
|
|
||||||
/* Clear interrupt pending status. */
|
/* Clear interrupt pending status. */
|
||||||
jz4780_dma_writel(jzdma, JZ_DMA_REG_DIRQP, 0);
|
jz4780_dma_ctrl_writel(jzdma, JZ_DMA_REG_DIRQP, 0);
|
||||||
|
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
@ -728,7 +795,7 @@ static struct dma_chan *jz4780_of_dma_xlate(struct of_phandle_args *dma_spec,
|
|||||||
data.channel = dma_spec->args[1];
|
data.channel = dma_spec->args[1];
|
||||||
|
|
||||||
if (data.channel > -1) {
|
if (data.channel > -1) {
|
||||||
if (data.channel >= JZ_DMA_NR_CHANNELS) {
|
if (data.channel >= jzdma->soc_data->nb_channels) {
|
||||||
dev_err(jzdma->dma_device.dev,
|
dev_err(jzdma->dma_device.dev,
|
||||||
"device requested non-existent channel %u\n",
|
"device requested non-existent channel %u\n",
|
||||||
data.channel);
|
data.channel);
|
||||||
@ -755,16 +822,29 @@ static struct dma_chan *jz4780_of_dma_xlate(struct of_phandle_args *dma_spec,
|
|||||||
static int jz4780_dma_probe(struct platform_device *pdev)
|
static int jz4780_dma_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct device *dev = &pdev->dev;
|
struct device *dev = &pdev->dev;
|
||||||
|
const struct jz4780_dma_soc_data *soc_data;
|
||||||
struct jz4780_dma_dev *jzdma;
|
struct jz4780_dma_dev *jzdma;
|
||||||
struct jz4780_dma_chan *jzchan;
|
struct jz4780_dma_chan *jzchan;
|
||||||
struct dma_device *dd;
|
struct dma_device *dd;
|
||||||
struct resource *res;
|
struct resource *res;
|
||||||
int i, ret;
|
int i, ret;
|
||||||
|
|
||||||
jzdma = devm_kzalloc(dev, sizeof(*jzdma), GFP_KERNEL);
|
if (!dev->of_node) {
|
||||||
|
dev_err(dev, "This driver must be probed from devicetree\n");
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
soc_data = device_get_match_data(dev);
|
||||||
|
if (!soc_data)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
jzdma = devm_kzalloc(dev, sizeof(*jzdma)
|
||||||
|
+ sizeof(*jzdma->chan) * soc_data->nb_channels,
|
||||||
|
GFP_KERNEL);
|
||||||
if (!jzdma)
|
if (!jzdma)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
jzdma->soc_data = soc_data;
|
||||||
platform_set_drvdata(pdev, jzdma);
|
platform_set_drvdata(pdev, jzdma);
|
||||||
|
|
||||||
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
||||||
@ -773,9 +853,26 @@ static int jz4780_dma_probe(struct platform_device *pdev)
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
jzdma->base = devm_ioremap_resource(dev, res);
|
jzdma->chn_base = devm_ioremap_resource(dev, res);
|
||||||
if (IS_ERR(jzdma->base))
|
if (IS_ERR(jzdma->chn_base))
|
||||||
return PTR_ERR(jzdma->base);
|
return PTR_ERR(jzdma->chn_base);
|
||||||
|
|
||||||
|
res = platform_get_resource(pdev, IORESOURCE_MEM, 1);
|
||||||
|
if (res) {
|
||||||
|
jzdma->ctrl_base = devm_ioremap_resource(dev, res);
|
||||||
|
if (IS_ERR(jzdma->ctrl_base))
|
||||||
|
return PTR_ERR(jzdma->ctrl_base);
|
||||||
|
} else if (soc_data->flags & JZ_SOC_DATA_ALLOW_LEGACY_DT) {
|
||||||
|
/*
|
||||||
|
* On JZ4780, if the second memory resource was not supplied,
|
||||||
|
* assume we're using an old devicetree, and calculate the
|
||||||
|
* offset to the control registers.
|
||||||
|
*/
|
||||||
|
jzdma->ctrl_base = jzdma->chn_base + JZ4780_DMA_CTRL_OFFSET;
|
||||||
|
} else {
|
||||||
|
dev_err(dev, "failed to get I/O memory\n");
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
ret = platform_get_irq(pdev, 0);
|
ret = platform_get_irq(pdev, 0);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
@ -833,13 +930,15 @@ static int jz4780_dma_probe(struct platform_device *pdev)
|
|||||||
* Also set the FMSC bit - it increases MSC performance, so it makes
|
* Also set the FMSC bit - it increases MSC performance, so it makes
|
||||||
* little sense not to enable it.
|
* little sense not to enable it.
|
||||||
*/
|
*/
|
||||||
jz4780_dma_writel(jzdma, JZ_DMA_REG_DMAC,
|
jz4780_dma_ctrl_writel(jzdma, JZ_DMA_REG_DMAC, JZ_DMA_DMAC_DMAE |
|
||||||
JZ_DMA_DMAC_DMAE | JZ_DMA_DMAC_FMSC);
|
JZ_DMA_DMAC_FAIC | JZ_DMA_DMAC_FMSC);
|
||||||
jz4780_dma_writel(jzdma, JZ_DMA_REG_DMACP, 0);
|
|
||||||
|
if (soc_data->flags & JZ_SOC_DATA_PROGRAMMABLE_DMA)
|
||||||
|
jz4780_dma_ctrl_writel(jzdma, JZ_DMA_REG_DMACP, 0);
|
||||||
|
|
||||||
INIT_LIST_HEAD(&dd->channels);
|
INIT_LIST_HEAD(&dd->channels);
|
||||||
|
|
||||||
for (i = 0; i < JZ_DMA_NR_CHANNELS; i++) {
|
for (i = 0; i < soc_data->nb_channels; i++) {
|
||||||
jzchan = &jzdma->chan[i];
|
jzchan = &jzdma->chan[i];
|
||||||
jzchan->id = i;
|
jzchan->id = i;
|
||||||
|
|
||||||
@ -881,14 +980,40 @@ static int jz4780_dma_remove(struct platform_device *pdev)
|
|||||||
|
|
||||||
free_irq(jzdma->irq, jzdma);
|
free_irq(jzdma->irq, jzdma);
|
||||||
|
|
||||||
for (i = 0; i < JZ_DMA_NR_CHANNELS; i++)
|
for (i = 0; i < jzdma->soc_data->nb_channels; i++)
|
||||||
tasklet_kill(&jzdma->chan[i].vchan.task);
|
tasklet_kill(&jzdma->chan[i].vchan.task);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct jz4780_dma_soc_data jz4740_dma_soc_data = {
|
||||||
|
.nb_channels = 6,
|
||||||
|
.transfer_ord_max = 5,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct jz4780_dma_soc_data jz4725b_dma_soc_data = {
|
||||||
|
.nb_channels = 6,
|
||||||
|
.transfer_ord_max = 5,
|
||||||
|
.flags = JZ_SOC_DATA_PER_CHAN_PM | JZ_SOC_DATA_NO_DCKES_DCKEC,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct jz4780_dma_soc_data jz4770_dma_soc_data = {
|
||||||
|
.nb_channels = 6,
|
||||||
|
.transfer_ord_max = 6,
|
||||||
|
.flags = JZ_SOC_DATA_PER_CHAN_PM,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct jz4780_dma_soc_data jz4780_dma_soc_data = {
|
||||||
|
.nb_channels = 32,
|
||||||
|
.transfer_ord_max = 7,
|
||||||
|
.flags = JZ_SOC_DATA_ALLOW_LEGACY_DT | JZ_SOC_DATA_PROGRAMMABLE_DMA,
|
||||||
|
};
|
||||||
|
|
||||||
static const struct of_device_id jz4780_dma_dt_match[] = {
|
static const struct of_device_id jz4780_dma_dt_match[] = {
|
||||||
{ .compatible = "ingenic,jz4780-dma", .data = NULL },
|
{ .compatible = "ingenic,jz4740-dma", .data = &jz4740_dma_soc_data },
|
||||||
|
{ .compatible = "ingenic,jz4725b-dma", .data = &jz4725b_dma_soc_data },
|
||||||
|
{ .compatible = "ingenic,jz4770-dma", .data = &jz4770_dma_soc_data },
|
||||||
|
{ .compatible = "ingenic,jz4780-dma", .data = &jz4780_dma_soc_data },
|
||||||
{},
|
{},
|
||||||
};
|
};
|
||||||
MODULE_DEVICE_TABLE(of, jz4780_dma_dt_match);
|
MODULE_DEVICE_TABLE(of, jz4780_dma_dt_match);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user