Commit fc2c8067 authored by Vijendar Mukunda's avatar Vijendar Mukunda Committed by Mark Brown

ASoC: amd: irq handler changes for ACP5x PCM dma driver

Whenever audio data equal to the I2S FIFO watermark level are
produced/consumed, interrupt is generated.
Acknowledge the interrupt.
Reviewed-by: default avatarPierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Signed-off-by: default avatarVijendar Mukunda <Vijendar.Mukunda@amd.com>
Link: https://lore.kernel.org/r/20210721180430.11571-7-Vijendar.Mukunda@amd.comSigned-off-by: default avatarMark Brown <broonie@kernel.org>
parent 77f61444
...@@ -21,10 +21,55 @@ static const struct snd_soc_component_driver acp5x_i2s_component = { ...@@ -21,10 +21,55 @@ static const struct snd_soc_component_driver acp5x_i2s_component = {
.name = DRV_NAME, .name = DRV_NAME,
}; };
static irqreturn_t i2s_irq_handler(int irq, void *dev_id)
{
struct i2s_dev_data *vg_i2s_data;
u16 irq_flag;
u32 val;
vg_i2s_data = dev_id;
if (!vg_i2s_data)
return IRQ_NONE;
irq_flag = 0;
val = acp_readl(vg_i2s_data->acp5x_base + ACP_EXTERNAL_INTR_STAT);
if ((val & BIT(HS_TX_THRESHOLD)) && vg_i2s_data->play_stream) {
acp_writel(BIT(HS_TX_THRESHOLD), vg_i2s_data->acp5x_base +
ACP_EXTERNAL_INTR_STAT);
snd_pcm_period_elapsed(vg_i2s_data->play_stream);
irq_flag = 1;
}
if ((val & BIT(I2S_TX_THRESHOLD)) && vg_i2s_data->i2ssp_play_stream) {
acp_writel(BIT(I2S_TX_THRESHOLD),
vg_i2s_data->acp5x_base + ACP_EXTERNAL_INTR_STAT);
snd_pcm_period_elapsed(vg_i2s_data->i2ssp_play_stream);
irq_flag = 1;
}
if ((val & BIT(HS_RX_THRESHOLD)) && vg_i2s_data->capture_stream) {
acp_writel(BIT(HS_RX_THRESHOLD), vg_i2s_data->acp5x_base +
ACP_EXTERNAL_INTR_STAT);
snd_pcm_period_elapsed(vg_i2s_data->capture_stream);
irq_flag = 1;
}
if ((val & BIT(I2S_RX_THRESHOLD)) && vg_i2s_data->i2ssp_capture_stream) {
acp_writel(BIT(I2S_RX_THRESHOLD),
vg_i2s_data->acp5x_base + ACP_EXTERNAL_INTR_STAT);
snd_pcm_period_elapsed(vg_i2s_data->i2ssp_capture_stream);
irq_flag = 1;
}
if (irq_flag)
return IRQ_HANDLED;
else
return IRQ_NONE;
}
static int acp5x_audio_probe(struct platform_device *pdev) static int acp5x_audio_probe(struct platform_device *pdev)
{ {
struct resource *res; struct resource *res;
struct i2s_dev_data *adata; struct i2s_dev_data *adata;
unsigned int irqflags;
int status; int status;
if (!pdev->dev.platform_data) { if (!pdev->dev.platform_data) {
...@@ -47,12 +92,26 @@ static int acp5x_audio_probe(struct platform_device *pdev) ...@@ -47,12 +92,26 @@ static int acp5x_audio_probe(struct platform_device *pdev)
resource_size(res)); resource_size(res));
if (!adata->acp5x_base) if (!adata->acp5x_base)
return -ENOMEM; return -ENOMEM;
res = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
if (!res) {
dev_err(&pdev->dev, "IORESOURCE_IRQ FAILED\n");
return -ENODEV;
}
adata->i2s_irq = res->start;
dev_set_drvdata(&pdev->dev, adata); dev_set_drvdata(&pdev->dev, adata);
status = devm_snd_soc_register_component(&pdev->dev, status = devm_snd_soc_register_component(&pdev->dev,
&acp5x_i2s_component, &acp5x_i2s_component,
NULL, 0); NULL, 0);
if (status) if (status) {
dev_err(&pdev->dev, "Fail to register acp i2s component\n"); dev_err(&pdev->dev, "Fail to register acp i2s component\n");
return status;
}
status = devm_request_irq(&pdev->dev, adata->i2s_irq, i2s_irq_handler,
irqflags, "ACP5x_I2S_IRQ", adata);
if (status)
dev_err(&pdev->dev, "ACP5x I2S IRQ request failed\n");
return status; return status;
} }
......
...@@ -32,9 +32,18 @@ ...@@ -32,9 +32,18 @@
#define I2S_MODE 0 #define I2S_MODE 0
#define ACP5x_I2S_MODE 1 #define ACP5x_I2S_MODE 1
#define ACP5x_RES 4 #define ACP5x_RES 4
#define I2S_RX_THRESHOLD 27
#define I2S_TX_THRESHOLD 28
#define HS_TX_THRESHOLD 24
#define HS_RX_THRESHOLD 23
struct i2s_dev_data { struct i2s_dev_data {
unsigned int i2s_irq;
void __iomem *acp5x_base; void __iomem *acp5x_base;
struct snd_pcm_substream *play_stream;
struct snd_pcm_substream *capture_stream;
struct snd_pcm_substream *i2ssp_play_stream;
struct snd_pcm_substream *i2ssp_capture_stream;
}; };
/* common header file uses exact offset rather than relative /* common header file uses exact offset rather than relative
......
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