提交 3c68565b 编写于 作者: K Kuninori Morimoto 提交者: Mark Brown

ASoC: rsnd: enable to care 1st / 2nd DMAC on rsnd_dma_xxx()

rsnd driver needs to care about Audio DAMC (via DMAEngine),
Audio DMAC peri peri (via local method) on rsnd driver.
This patch adds new rsnd_dma_ops and care it.
Signed-off-by: NKuninori Morimoto <kuninori.morimoto.gx@renesas.com>
Signed-off-by: NMark Brown <broonie@kernel.org>
上级 747c71b1
...@@ -10,7 +10,7 @@ ...@@ -10,7 +10,7 @@
*/ */
#include "rsnd.h" #include "rsnd.h"
static void rsnd_dma_complete(void *data) static void rsnd_dmaen_complete(void *data)
{ {
struct rsnd_dma *dma = (struct rsnd_dma *)data; struct rsnd_dma *dma = (struct rsnd_dma *)data;
struct rsnd_mod *mod = rsnd_dma_to_mod(dma); struct rsnd_mod *mod = rsnd_dma_to_mod(dma);
...@@ -32,7 +32,7 @@ static void rsnd_dma_complete(void *data) ...@@ -32,7 +32,7 @@ static void rsnd_dma_complete(void *data)
} }
#define DMA_NAME_SIZE 16 #define DMA_NAME_SIZE 16
static int _rsnd_dma_of_name(char *dma_name, struct rsnd_mod *mod) static int _rsnd_dmaen_of_name(char *dma_name, struct rsnd_mod *mod)
{ {
if (mod) if (mod)
return snprintf(dma_name, DMA_NAME_SIZE / 2, "%s%d", return snprintf(dma_name, DMA_NAME_SIZE / 2, "%s%d",
...@@ -42,23 +42,23 @@ static int _rsnd_dma_of_name(char *dma_name, struct rsnd_mod *mod) ...@@ -42,23 +42,23 @@ static int _rsnd_dma_of_name(char *dma_name, struct rsnd_mod *mod)
} }
static void rsnd_dma_of_name(struct rsnd_mod *mod_from, static void rsnd_dmaen_of_name(struct rsnd_mod *mod_from,
struct rsnd_mod *mod_to, struct rsnd_mod *mod_to,
char *dma_name) char *dma_name)
{ {
int index = 0; int index = 0;
index = _rsnd_dma_of_name(dma_name + index, mod_from); index = _rsnd_dmaen_of_name(dma_name + index, mod_from);
*(dma_name + index++) = '_'; *(dma_name + index++) = '_';
index = _rsnd_dma_of_name(dma_name + index, mod_to); index = _rsnd_dmaen_of_name(dma_name + index, mod_to);
} }
void rsnd_dma_stop(struct rsnd_dma *dma) static void rsnd_dmaen_stop(struct rsnd_dma *dma)
{ {
dmaengine_terminate_all(dma->chan); dmaengine_terminate_all(dma->chan);
} }
void rsnd_dma_start(struct rsnd_dma *dma) static void rsnd_dmaen_start(struct rsnd_dma *dma)
{ {
struct rsnd_mod *mod = rsnd_dma_to_mod(dma); struct rsnd_mod *mod = rsnd_dma_to_mod(dma);
struct rsnd_priv *priv = rsnd_mod_to_priv(mod); struct rsnd_priv *priv = rsnd_mod_to_priv(mod);
...@@ -80,7 +80,7 @@ void rsnd_dma_start(struct rsnd_dma *dma) ...@@ -80,7 +80,7 @@ void rsnd_dma_start(struct rsnd_dma *dma)
return; return;
} }
desc->callback = rsnd_dma_complete; desc->callback = rsnd_dmaen_complete;
desc->callback_param = dma; desc->callback_param = dma;
if (dmaengine_submit(desc) < 0) { if (dmaengine_submit(desc) < 0) {
...@@ -91,22 +91,12 @@ void rsnd_dma_start(struct rsnd_dma *dma) ...@@ -91,22 +91,12 @@ void rsnd_dma_start(struct rsnd_dma *dma)
dma_async_issue_pending(dma->chan); dma_async_issue_pending(dma->chan);
} }
static void rsnd_dma_of_path(struct rsnd_dma *dma, static int rsnd_dmaen_init(struct rsnd_priv *priv, struct rsnd_dma *dma, int id,
int is_play, struct rsnd_mod *mod_from, struct rsnd_mod *mod_to)
struct rsnd_mod **mod_from,
struct rsnd_mod **mod_to);
static dma_addr_t rsnd_dma_addr(struct rsnd_priv *priv,
struct rsnd_mod *mod,
int is_play, int is_from);
int rsnd_dma_init(struct rsnd_priv *priv, struct rsnd_dma *dma, int id)
{ {
struct device *dev = rsnd_priv_to_dev(priv); struct device *dev = rsnd_priv_to_dev(priv);
struct dma_slave_config cfg = {}; struct dma_slave_config cfg = {};
struct rsnd_mod *mod = rsnd_dma_to_mod(dma); struct rsnd_mod *mod = rsnd_dma_to_mod(dma);
struct rsnd_mod *mod_from;
struct rsnd_mod *mod_to;
struct rsnd_dai_stream *io = rsnd_mod_to_io(mod); struct rsnd_dai_stream *io = rsnd_mod_to_io(mod);
int is_play = rsnd_io_is_play(io); int is_play = rsnd_io_is_play(io);
char dma_name[DMA_NAME_SIZE]; char dma_name[DMA_NAME_SIZE];
...@@ -121,12 +111,11 @@ int rsnd_dma_init(struct rsnd_priv *priv, struct rsnd_dma *dma, int id) ...@@ -121,12 +111,11 @@ int rsnd_dma_init(struct rsnd_priv *priv, struct rsnd_dma *dma, int id)
dma_cap_zero(mask); dma_cap_zero(mask);
dma_cap_set(DMA_SLAVE, mask); dma_cap_set(DMA_SLAVE, mask);
rsnd_dma_of_path(dma, is_play, &mod_from, &mod_to); rsnd_dmaen_of_name(mod_from, mod_to, dma_name);
rsnd_dma_of_name(mod_from, mod_to, dma_name);
cfg.direction = is_play ? DMA_MEM_TO_DEV : DMA_DEV_TO_MEM; cfg.direction = is_play ? DMA_MEM_TO_DEV : DMA_DEV_TO_MEM;
cfg.src_addr = rsnd_dma_addr(priv, mod_from, is_play, 1); cfg.src_addr = dma->src_addr;
cfg.dst_addr = rsnd_dma_addr(priv, mod_to, is_play, 0); cfg.dst_addr = dma->dst_addr;
cfg.src_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES; cfg.src_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
cfg.dst_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES; cfg.dst_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
...@@ -163,7 +152,7 @@ int rsnd_dma_init(struct rsnd_priv *priv, struct rsnd_dma *dma, int id) ...@@ -163,7 +152,7 @@ int rsnd_dma_init(struct rsnd_priv *priv, struct rsnd_dma *dma, int id)
return -EAGAIN; return -EAGAIN;
} }
void rsnd_dma_quit(struct rsnd_dma *dma) static void rsnd_dmaen_quit(struct rsnd_dma *dma)
{ {
if (dma->chan) if (dma->chan)
dma_release_channel(dma->chan); dma_release_channel(dma->chan);
...@@ -171,6 +160,13 @@ void rsnd_dma_quit(struct rsnd_dma *dma) ...@@ -171,6 +160,13 @@ void rsnd_dma_quit(struct rsnd_dma *dma)
dma->chan = NULL; dma->chan = NULL;
} }
static struct rsnd_dma_ops rsnd_dmaen_ops = {
.start = rsnd_dmaen_start,
.stop = rsnd_dmaen_stop,
.init = rsnd_dmaen_init,
.quit = rsnd_dmaen_quit,
};
/* /*
* DMA read/write register offset * DMA read/write register offset
* *
...@@ -343,3 +339,35 @@ static void rsnd_dma_of_path(struct rsnd_dma *dma, ...@@ -343,3 +339,35 @@ static void rsnd_dma_of_path(struct rsnd_dma *dma,
} }
} }
void rsnd_dma_stop(struct rsnd_dma *dma)
{
dma->ops->stop(dma);
}
void rsnd_dma_start(struct rsnd_dma *dma)
{
dma->ops->start(dma);
}
void rsnd_dma_quit(struct rsnd_dma *dma)
{
dma->ops->quit(dma);
}
int rsnd_dma_init(struct rsnd_priv *priv, struct rsnd_dma *dma, int id)
{
struct rsnd_mod *mod = rsnd_dma_to_mod(dma);
struct rsnd_mod *mod_from;
struct rsnd_mod *mod_to;
struct rsnd_dai_stream *io = rsnd_mod_to_io(mod);
int is_play = rsnd_io_is_play(io);
rsnd_dma_of_path(dma, is_play, &mod_from, &mod_to);
dma->src_addr = rsnd_dma_addr(priv, mod_from, is_play, 1);
dma->dst_addr = rsnd_dma_addr(priv, mod_to, is_play, 0);
dma->ops = &rsnd_dmaen_ops;
return dma->ops->init(priv, dma, id, mod_from, mod_to);
}
...@@ -170,10 +170,22 @@ u32 rsnd_get_adinr(struct rsnd_mod *mod); ...@@ -170,10 +170,22 @@ u32 rsnd_get_adinr(struct rsnd_mod *mod);
/* /*
* R-Car DMA * R-Car DMA
*/ */
struct rsnd_dma;
struct rsnd_dma_ops {
void (*start)(struct rsnd_dma *dma);
void (*stop)(struct rsnd_dma *dma);
int (*init)(struct rsnd_priv *priv, struct rsnd_dma *dma, int id,
struct rsnd_mod *mod_from, struct rsnd_mod *mod_to);
void (*quit)(struct rsnd_dma *dma);
};
struct rsnd_dma { struct rsnd_dma {
struct dma_chan *chan; struct dma_chan *chan;
struct rsnd_dma_ops *ops;
enum dma_transfer_direction dir; enum dma_transfer_direction dir;
dma_addr_t addr; dma_addr_t addr;
dma_addr_t src_addr;
dma_addr_t dst_addr;
}; };
void rsnd_dma_start(struct rsnd_dma *dma); void rsnd_dma_start(struct rsnd_dma *dma);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册