Commit da1b6c05 authored by Tomasz Figa's avatar Tomasz Figa Committed by Vinod Koul

dmaengine: PL08x: Add support for PL080S variant

PL080S is a modified version of PL080 that can be found on Samsung SoCs,
such as S3C6400 and S3C6410.

It has different offset of CONFIG register, separate CONTROL1 register
that holds transfer size and larger maximum transfer size.
Signed-off-by: default avatarTomasz Figa <tomasz.figa@gmail.com>
Acked-by: default avatarLinus Walleij <linus.walleij@linaro.org>
Signed-off-by: default avatarVinod Koul <vinod.koul@intel.com>
parent 48924e42
...@@ -24,6 +24,7 @@ ...@@ -24,6 +24,7 @@
* *
* Documentation: ARM DDI 0196G == PL080 * Documentation: ARM DDI 0196G == PL080
* Documentation: ARM DDI 0218E == PL081 * Documentation: ARM DDI 0218E == PL081
* Documentation: S3C6410 User's Manual == PL080S
* *
* PL080 & PL081 both have 16 sets of DMA signals that can be routed to any * PL080 & PL081 both have 16 sets of DMA signals that can be routed to any
* channel. * channel.
...@@ -36,6 +37,14 @@ ...@@ -36,6 +37,14 @@
* *
* The PL080 has a dual bus master, PL081 has a single master. * The PL080 has a dual bus master, PL081 has a single master.
* *
* PL080S is a version modified by Samsung and used in S3C64xx SoCs.
* It differs in following aspects:
* - CH_CONFIG register at different offset,
* - separate CH_CONTROL2 register for transfer size,
* - bigger maximum transfer size,
* - 8-word aligned LLI, instead of 4-word, due to extra CCTL2 word,
* - no support for peripheral flow control.
*
* Memory to peripheral transfer may be visualized as * Memory to peripheral transfer may be visualized as
* Get data from memory to DMAC * Get data from memory to DMAC
* Until no data left * Until no data left
...@@ -64,10 +73,7 @@ ...@@ -64,10 +73,7 @@
* - Peripheral flow control: the transfer size is ignored (and should be * - Peripheral flow control: the transfer size is ignored (and should be
* zero). The data is transferred from the current LLI entry, until * zero). The data is transferred from the current LLI entry, until
* after the final transfer signalled by LBREQ or LSREQ. The DMAC * after the final transfer signalled by LBREQ or LSREQ. The DMAC
* will then move to the next LLI entry. * will then move to the next LLI entry. Unsupported by PL080S.
*
* Global TODO:
* - Break out common code from arch/arm/mach-s3c64xx and share
*/ */
#include <linux/amba/bus.h> #include <linux/amba/bus.h>
#include <linux/amba/pl08x.h> #include <linux/amba/pl08x.h>
...@@ -100,12 +106,15 @@ struct pl08x_driver_data; ...@@ -100,12 +106,15 @@ struct pl08x_driver_data;
* @nomadik: whether the channels have Nomadik security extension bits * @nomadik: whether the channels have Nomadik security extension bits
* that need to be checked for permission before use and some registers are * that need to be checked for permission before use and some registers are
* missing * missing
* @pl080s: whether this version is a PL080S, which has separate register and
* LLI word for transfer size.
*/ */
struct vendor_data { struct vendor_data {
u8 config_offset; u8 config_offset;
u8 channels; u8 channels;
bool dualmaster; bool dualmaster;
bool nomadik; bool nomadik;
bool pl080s;
}; };
/** /**
...@@ -264,9 +273,11 @@ struct pl08x_driver_data { ...@@ -264,9 +273,11 @@ struct pl08x_driver_data {
#define PL080_LLI_DST 1 #define PL080_LLI_DST 1
#define PL080_LLI_LLI 2 #define PL080_LLI_LLI 2
#define PL080_LLI_CCTL 3 #define PL080_LLI_CCTL 3
#define PL080S_LLI_CCTL2 4
/* Total words in an LLI. */ /* Total words in an LLI. */
#define PL080_LLI_WORDS 4 #define PL080_LLI_WORDS 4
#define PL080S_LLI_WORDS 8
/* /*
* Number of LLIs in each LLI buffer allocated for one transfer * Number of LLIs in each LLI buffer allocated for one transfer
...@@ -340,6 +351,14 @@ static int pl08x_phy_channel_busy(struct pl08x_phy_chan *ch) ...@@ -340,6 +351,14 @@ static int pl08x_phy_channel_busy(struct pl08x_phy_chan *ch)
static void pl08x_write_lli(struct pl08x_driver_data *pl08x, static void pl08x_write_lli(struct pl08x_driver_data *pl08x,
struct pl08x_phy_chan *phychan, const u32 *lli, u32 ccfg) struct pl08x_phy_chan *phychan, const u32 *lli, u32 ccfg)
{ {
if (pl08x->vd->pl080s)
dev_vdbg(&pl08x->adev->dev,
"WRITE channel %d: csrc=0x%08x, cdst=0x%08x, "
"clli=0x%08x, cctl=0x%08x, cctl2=0x%08x, ccfg=0x%08x\n",
phychan->id, lli[PL080_LLI_SRC], lli[PL080_LLI_DST],
lli[PL080_LLI_LLI], lli[PL080_LLI_CCTL],
lli[PL080S_LLI_CCTL2], ccfg);
else
dev_vdbg(&pl08x->adev->dev, dev_vdbg(&pl08x->adev->dev,
"WRITE channel %d: csrc=0x%08x, cdst=0x%08x, " "WRITE channel %d: csrc=0x%08x, cdst=0x%08x, "
"clli=0x%08x, cctl=0x%08x, ccfg=0x%08x\n", "clli=0x%08x, cctl=0x%08x, ccfg=0x%08x\n",
...@@ -351,6 +370,10 @@ static void pl08x_write_lli(struct pl08x_driver_data *pl08x, ...@@ -351,6 +370,10 @@ static void pl08x_write_lli(struct pl08x_driver_data *pl08x,
writel_relaxed(lli[PL080_LLI_LLI], phychan->base + PL080_CH_LLI); writel_relaxed(lli[PL080_LLI_LLI], phychan->base + PL080_CH_LLI);
writel_relaxed(lli[PL080_LLI_CCTL], phychan->base + PL080_CH_CONTROL); writel_relaxed(lli[PL080_LLI_CCTL], phychan->base + PL080_CH_CONTROL);
if (pl08x->vd->pl080s)
writel_relaxed(lli[PL080S_LLI_CCTL2],
phychan->base + PL080S_CH_CONTROL2);
writel(ccfg, phychan->reg_config); writel(ccfg, phychan->reg_config);
} }
...@@ -469,6 +492,24 @@ static inline u32 get_bytes_in_cctl(u32 cctl) ...@@ -469,6 +492,24 @@ static inline u32 get_bytes_in_cctl(u32 cctl)
return bytes; return bytes;
} }
static inline u32 get_bytes_in_cctl_pl080s(u32 cctl, u32 cctl1)
{
/* The source width defines the number of bytes */
u32 bytes = cctl1 & PL080S_CONTROL_TRANSFER_SIZE_MASK;
switch (cctl >> PL080_CONTROL_SWIDTH_SHIFT) {
case PL080_WIDTH_8BIT:
break;
case PL080_WIDTH_16BIT:
bytes *= 2;
break;
case PL080_WIDTH_32BIT:
bytes *= 4;
break;
}
return bytes;
}
/* The channel should be paused when calling this */ /* The channel should be paused when calling this */
static u32 pl08x_getbytes_chan(struct pl08x_dma_chan *plchan) static u32 pl08x_getbytes_chan(struct pl08x_dma_chan *plchan)
{ {
...@@ -494,6 +535,11 @@ static u32 pl08x_getbytes_chan(struct pl08x_dma_chan *plchan) ...@@ -494,6 +535,11 @@ static u32 pl08x_getbytes_chan(struct pl08x_dma_chan *plchan)
clli = readl(ch->base + PL080_CH_LLI) & ~PL080_LLI_LM_AHB2; clli = readl(ch->base + PL080_CH_LLI) & ~PL080_LLI_LM_AHB2;
/* First get the remaining bytes in the active transfer */ /* First get the remaining bytes in the active transfer */
if (pl08x->vd->pl080s)
bytes = get_bytes_in_cctl_pl080s(
readl(ch->base + PL080_CH_CONTROL),
readl(ch->base + PL080S_CH_CONTROL2));
else
bytes = get_bytes_in_cctl(readl(ch->base + PL080_CH_CONTROL)); bytes = get_bytes_in_cctl(readl(ch->base + PL080_CH_CONTROL));
if (!clli) if (!clli)
...@@ -515,6 +561,11 @@ static u32 pl08x_getbytes_chan(struct pl08x_dma_chan *plchan) ...@@ -515,6 +561,11 @@ static u32 pl08x_getbytes_chan(struct pl08x_dma_chan *plchan)
llis_va_limit = llis_va + llis_max_words; llis_va_limit = llis_va + llis_max_words;
for (; llis_va < llis_va_limit; llis_va += pl08x->lli_words) { for (; llis_va < llis_va_limit; llis_va += pl08x->lli_words) {
if (pl08x->vd->pl080s)
bytes += get_bytes_in_cctl_pl080s(
llis_va[PL080_LLI_CCTL],
llis_va[PL080S_LLI_CCTL2]);
else
bytes += get_bytes_in_cctl(llis_va[PL080_LLI_CCTL]); bytes += get_bytes_in_cctl(llis_va[PL080_LLI_CCTL]);
/* /*
...@@ -778,7 +829,7 @@ static void pl08x_choose_master_bus(struct pl08x_lli_build_data *bd, ...@@ -778,7 +829,7 @@ static void pl08x_choose_master_bus(struct pl08x_lli_build_data *bd,
*/ */
static void pl08x_fill_lli_for_desc(struct pl08x_driver_data *pl08x, static void pl08x_fill_lli_for_desc(struct pl08x_driver_data *pl08x,
struct pl08x_lli_build_data *bd, struct pl08x_lli_build_data *bd,
int num_llis, int len, u32 cctl) int num_llis, int len, u32 cctl, u32 cctl2)
{ {
u32 offset = num_llis * pl08x->lli_words; u32 offset = num_llis * pl08x->lli_words;
u32 *llis_va = bd->txd->llis_va + offset; u32 *llis_va = bd->txd->llis_va + offset;
...@@ -794,6 +845,8 @@ static void pl08x_fill_lli_for_desc(struct pl08x_driver_data *pl08x, ...@@ -794,6 +845,8 @@ static void pl08x_fill_lli_for_desc(struct pl08x_driver_data *pl08x,
llis_va[PL080_LLI_LLI] = (llis_bus + sizeof(u32) * offset); llis_va[PL080_LLI_LLI] = (llis_bus + sizeof(u32) * offset);
llis_va[PL080_LLI_LLI] |= bd->lli_bus; llis_va[PL080_LLI_LLI] |= bd->lli_bus;
llis_va[PL080_LLI_CCTL] = cctl; llis_va[PL080_LLI_CCTL] = cctl;
if (pl08x->vd->pl080s)
llis_va[PL080S_LLI_CCTL2] = cctl2;
if (cctl & PL080_CONTROL_SRC_INCR) if (cctl & PL080_CONTROL_SRC_INCR)
bd->srcbus.addr += len; bd->srcbus.addr += len;
...@@ -810,7 +863,7 @@ static inline void prep_byte_width_lli(struct pl08x_driver_data *pl08x, ...@@ -810,7 +863,7 @@ static inline void prep_byte_width_lli(struct pl08x_driver_data *pl08x,
int num_llis, size_t *total_bytes) int num_llis, size_t *total_bytes)
{ {
*cctl = pl08x_cctl_bits(*cctl, 1, 1, len); *cctl = pl08x_cctl_bits(*cctl, 1, 1, len);
pl08x_fill_lli_for_desc(pl08x, bd, num_llis, len, *cctl); pl08x_fill_lli_for_desc(pl08x, bd, num_llis, len, *cctl, len);
(*total_bytes) += len; (*total_bytes) += len;
} }
...@@ -820,6 +873,20 @@ static void pl08x_dump_lli(struct pl08x_driver_data *pl08x, ...@@ -820,6 +873,20 @@ static void pl08x_dump_lli(struct pl08x_driver_data *pl08x,
{ {
int i; int i;
if (pl08x->vd->pl080s) {
dev_vdbg(&pl08x->adev->dev,
"%-3s %-9s %-10s %-10s %-10s %-10s %s\n",
"lli", "", "csrc", "cdst", "clli", "cctl", "cctl2");
for (i = 0; i < num_llis; i++) {
dev_vdbg(&pl08x->adev->dev,
"%3d @%p: 0x%08x 0x%08x 0x%08x 0x%08x 0x%08x\n",
i, llis_va, llis_va[PL080_LLI_SRC],
llis_va[PL080_LLI_DST], llis_va[PL080_LLI_LLI],
llis_va[PL080_LLI_CCTL],
llis_va[PL080S_LLI_CCTL2]);
llis_va += pl08x->lli_words;
}
} else {
dev_vdbg(&pl08x->adev->dev, dev_vdbg(&pl08x->adev->dev,
"%-3s %-9s %-10s %-10s %-10s %s\n", "%-3s %-9s %-10s %-10s %-10s %s\n",
"lli", "", "csrc", "cdst", "clli", "cctl"); "lli", "", "csrc", "cdst", "clli", "cctl");
...@@ -831,6 +898,7 @@ static void pl08x_dump_lli(struct pl08x_driver_data *pl08x, ...@@ -831,6 +898,7 @@ static void pl08x_dump_lli(struct pl08x_driver_data *pl08x,
llis_va[PL080_LLI_CCTL]); llis_va[PL080_LLI_CCTL]);
llis_va += pl08x->lli_words; llis_va += pl08x->lli_words;
} }
}
} }
#else #else
static inline void pl08x_dump_lli(struct pl08x_driver_data *pl08x, static inline void pl08x_dump_lli(struct pl08x_driver_data *pl08x,
...@@ -938,7 +1006,7 @@ static int pl08x_fill_llis_for_desc(struct pl08x_driver_data *pl08x, ...@@ -938,7 +1006,7 @@ static int pl08x_fill_llis_for_desc(struct pl08x_driver_data *pl08x,
cctl = pl08x_cctl_bits(cctl, bd.srcbus.buswidth, cctl = pl08x_cctl_bits(cctl, bd.srcbus.buswidth,
bd.dstbus.buswidth, 0); bd.dstbus.buswidth, 0);
pl08x_fill_lli_for_desc(pl08x, &bd, num_llis++, pl08x_fill_lli_for_desc(pl08x, &bd, num_llis++,
0, cctl); 0, cctl, 0);
break; break;
} }
...@@ -1018,7 +1086,7 @@ static int pl08x_fill_llis_for_desc(struct pl08x_driver_data *pl08x, ...@@ -1018,7 +1086,7 @@ static int pl08x_fill_llis_for_desc(struct pl08x_driver_data *pl08x,
cctl = pl08x_cctl_bits(cctl, bd.srcbus.buswidth, cctl = pl08x_cctl_bits(cctl, bd.srcbus.buswidth,
bd.dstbus.buswidth, tsize); bd.dstbus.buswidth, tsize);
pl08x_fill_lli_for_desc(pl08x, &bd, num_llis++, pl08x_fill_lli_for_desc(pl08x, &bd, num_llis++,
lli_len, cctl); lli_len, cctl, tsize);
total_bytes += lli_len; total_bytes += lli_len;
} }
...@@ -1332,6 +1400,7 @@ static int dma_set_runtime_config(struct dma_chan *chan, ...@@ -1332,6 +1400,7 @@ static int dma_set_runtime_config(struct dma_chan *chan,
struct dma_slave_config *config) struct dma_slave_config *config)
{ {
struct pl08x_dma_chan *plchan = to_pl08x_chan(chan); struct pl08x_dma_chan *plchan = to_pl08x_chan(chan);
struct pl08x_driver_data *pl08x = plchan->host;
if (!plchan->slave) if (!plchan->slave)
return -EINVAL; return -EINVAL;
...@@ -1341,6 +1410,13 @@ static int dma_set_runtime_config(struct dma_chan *chan, ...@@ -1341,6 +1410,13 @@ static int dma_set_runtime_config(struct dma_chan *chan,
config->dst_addr_width == DMA_SLAVE_BUSWIDTH_8_BYTES) config->dst_addr_width == DMA_SLAVE_BUSWIDTH_8_BYTES)
return -EINVAL; return -EINVAL;
if (config->device_fc && pl08x->vd->pl080s) {
dev_err(&pl08x->adev->dev,
"%s: PL080S does not support peripheral flow control\n",
__func__);
return -EINVAL;
}
plchan->cfg = *config; plchan->cfg = *config;
return 0; return 0;
...@@ -1930,6 +2006,9 @@ static int pl08x_probe(struct amba_device *adev, const struct amba_id *id) ...@@ -1930,6 +2006,9 @@ static int pl08x_probe(struct amba_device *adev, const struct amba_id *id)
pl08x->mem_buses = pl08x->pd->mem_buses; pl08x->mem_buses = pl08x->pd->mem_buses;
} }
if (vd->pl080s)
pl08x->lli_words = PL080S_LLI_WORDS;
else
pl08x->lli_words = PL080_LLI_WORDS; pl08x->lli_words = PL080_LLI_WORDS;
tsfr_size = MAX_NUM_TSFR_LLIS * pl08x->lli_words * sizeof(u32); tsfr_size = MAX_NUM_TSFR_LLIS * pl08x->lli_words * sizeof(u32);
...@@ -2040,8 +2119,8 @@ static int pl08x_probe(struct amba_device *adev, const struct amba_id *id) ...@@ -2040,8 +2119,8 @@ static int pl08x_probe(struct amba_device *adev, const struct amba_id *id)
amba_set_drvdata(adev, pl08x); amba_set_drvdata(adev, pl08x);
init_pl08x_debugfs(pl08x); init_pl08x_debugfs(pl08x);
dev_info(&pl08x->adev->dev, "DMA: PL%03x rev%u at 0x%08llx irq %d\n", dev_info(&pl08x->adev->dev, "DMA: PL%03x%s rev%u at 0x%08llx irq %d\n",
amba_part(adev), amba_rev(adev), amba_part(adev), pl08x->vd->pl080s ? "s" : "", amba_rev(adev),
(unsigned long long)adev->res.start, adev->irq[0]); (unsigned long long)adev->res.start, adev->irq[0]);
return 0; return 0;
...@@ -2082,6 +2161,12 @@ static struct vendor_data vendor_nomadik = { ...@@ -2082,6 +2161,12 @@ static struct vendor_data vendor_nomadik = {
.nomadik = true, .nomadik = true,
}; };
static struct vendor_data vendor_pl080s = {
.config_offset = PL080S_CH_CONFIG,
.channels = 8,
.pl080s = true,
};
static struct vendor_data vendor_pl081 = { static struct vendor_data vendor_pl081 = {
.config_offset = PL080_CH_CONFIG, .config_offset = PL080_CH_CONFIG,
.channels = 2, .channels = 2,
...@@ -2089,6 +2174,12 @@ static struct vendor_data vendor_pl081 = { ...@@ -2089,6 +2174,12 @@ static struct vendor_data vendor_pl081 = {
}; };
static struct amba_id pl08x_ids[] = { static struct amba_id pl08x_ids[] = {
/* Samsung PL080S variant */
{
.id = 0x0a141080,
.mask = 0xffffffff,
.data = &vendor_pl080s,
},
/* PL080 */ /* PL080 */
{ {
.id = 0x00041080, .id = 0x00041080,
......
...@@ -87,6 +87,7 @@ ...@@ -87,6 +87,7 @@
#define PL080_CONTROL_SB_SIZE_MASK (0x7 << 12) #define PL080_CONTROL_SB_SIZE_MASK (0x7 << 12)
#define PL080_CONTROL_SB_SIZE_SHIFT (12) #define PL080_CONTROL_SB_SIZE_SHIFT (12)
#define PL080_CONTROL_TRANSFER_SIZE_MASK (0xfff << 0) #define PL080_CONTROL_TRANSFER_SIZE_MASK (0xfff << 0)
#define PL080S_CONTROL_TRANSFER_SIZE_MASK (0x1ffffff << 0)
#define PL080_CONTROL_TRANSFER_SIZE_SHIFT (0) #define PL080_CONTROL_TRANSFER_SIZE_SHIFT (0)
#define PL080_BSIZE_1 (0x0) #define PL080_BSIZE_1 (0x0)
......
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