Skip to content

Commit ba4c5fa

Browse files
AaronDotbroonie
authored andcommitted
ASoC: loongson: Add I2S controller driver as platform device
The Loongson I2S controller exists not only in PCI form (LS7A bridge chip), but also in platform device form (Loongson-2K1000 SoC). This patch adds support for platform device I2S controller. Signed-off-by: Binbin Zhou <zhoubinbin@loongson.cn> Link: https://patch.msgid.link/36c143358c7f48bc2e73c30e1d2009b2f2fc6498.1728459624.git.zhoubinbin@loongson.cn Signed-off-by: Mark Brown <broonie@kernel.org>
1 parent d4c2e9e commit ba4c5fa

File tree

3 files changed

+209
-10
lines changed

3 files changed

+209
-10
lines changed

sound/soc/loongson/Kconfig

Lines changed: 21 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -1,27 +1,38 @@
11
# SPDX-License-Identifier: GPL-2.0
22
menu "SoC Audio for Loongson CPUs"
3+
4+
config SND_SOC_LOONGSON_CARD
5+
tristate "Loongson Sound Card Driver"
36
depends on LOONGARCH || COMPILE_TEST
7+
select SND_SOC_LOONGSON_I2S_PCI if PCI
8+
select SND_SOC_LOONGSON_I2S_PLATFORM if OF
9+
help
10+
Say Y or M if you want to add support for SoC audio using
11+
loongson I2S controller.
12+
13+
The driver add support for ALSA SoC Audio support using
14+
loongson I2S controller.
415

516
config SND_SOC_LOONGSON_I2S_PCI
617
tristate "Loongson I2S-PCI Device Driver"
18+
depends on LOONGARCH || COMPILE_TEST
719
select REGMAP_MMIO
8-
depends on PCI
920
help
1021
Say Y or M if you want to add support for I2S driver for
1122
Loongson I2S controller.
1223

1324
The controller is found in loongson bridge chips or SoCs,
1425
and work as a PCI device.
1526

16-
config SND_SOC_LOONGSON_CARD
17-
tristate "Loongson Sound Card Driver"
18-
select SND_SOC_LOONGSON_I2S_PCI
19-
depends on PCI
27+
config SND_SOC_LOONGSON_I2S_PLATFORM
28+
tristate "Loongson I2S-PLAT Device Driver"
29+
depends on LOONGARCH || COMPILE_TEST
30+
select REGMAP_MMIO
31+
select SND_SOC_GENERIC_DMAENGINE_PCM
2032
help
21-
Say Y or M if you want to add support for SoC audio using
22-
loongson I2S controller.
23-
24-
The driver add support for ALSA SoC Audio support using
25-
loongson I2S controller.
33+
Say Y or M if you want to add support for I2S driver for
34+
Loongson I2S controller.
2635

36+
The controller work as a platform device, we can found it in
37+
Loongson-2K1000 SoCs.
2738
endmenu

sound/soc/loongson/Makefile

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,9 @@
33
snd-soc-loongson-i2s-pci-y := loongson_i2s_pci.o loongson_i2s.o loongson_dma.o
44
obj-$(CONFIG_SND_SOC_LOONGSON_I2S_PCI) += snd-soc-loongson-i2s-pci.o
55

6+
snd-soc-loongson-i2s-plat-y := loongson_i2s_plat.o loongson_i2s.o
7+
obj-$(CONFIG_SND_SOC_LOONGSON_I2S_PLATFORM) += snd-soc-loongson-i2s-plat.o
8+
69
#Machine Support
710
snd-soc-loongson-card-y := loongson_card.o
811
obj-$(CONFIG_SND_SOC_LOONGSON_CARD) += snd-soc-loongson-card.o
Lines changed: 185 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,185 @@
1+
// SPDX-License-Identifier: GPL-2.0
2+
//
3+
// Loongson I2S controller master mode dirver(platform device)
4+
//
5+
// Copyright (C) 2023-2024 Loongson Technology Corporation Limited
6+
//
7+
// Author: Yingkun Meng <mengyingkun@loongson.cn>
8+
// Binbin Zhou <zhoubinbin@loongson.cn>
9+
10+
#include <linux/clk.h>
11+
#include <linux/dma-mapping.h>
12+
#include <linux/module.h>
13+
#include <linux/of_dma.h>
14+
#include <linux/platform_device.h>
15+
#include <linux/pm_runtime.h>
16+
#include <sound/dmaengine_pcm.h>
17+
#include <sound/pcm.h>
18+
#include <sound/pcm_params.h>
19+
#include <sound/soc.h>
20+
21+
#include "loongson_i2s.h"
22+
23+
#define LOONGSON_I2S_RX_DMA_OFFSET 21
24+
#define LOONGSON_I2S_TX_DMA_OFFSET 18
25+
26+
#define LOONGSON_DMA0_CONF 0x0
27+
#define LOONGSON_DMA1_CONF 0x1
28+
#define LOONGSON_DMA2_CONF 0x2
29+
#define LOONGSON_DMA3_CONF 0x3
30+
#define LOONGSON_DMA4_CONF 0x4
31+
32+
/* periods_max = PAGE_SIZE / sizeof(struct ls_dma_chan_reg) */
33+
static const struct snd_pcm_hardware loongson_pcm_hardware = {
34+
.info = SNDRV_PCM_INFO_MMAP |
35+
SNDRV_PCM_INFO_INTERLEAVED |
36+
SNDRV_PCM_INFO_MMAP_VALID |
37+
SNDRV_PCM_INFO_RESUME |
38+
SNDRV_PCM_INFO_PAUSE,
39+
.formats = SNDRV_PCM_FMTBIT_S16_LE |
40+
SNDRV_PCM_FMTBIT_S20_3LE |
41+
SNDRV_PCM_FMTBIT_S24_LE,
42+
.period_bytes_min = 128,
43+
.period_bytes_max = 128 * 1024,
44+
.periods_min = 1,
45+
.periods_max = 64,
46+
.buffer_bytes_max = 1024 * 1024,
47+
};
48+
49+
static const struct snd_dmaengine_pcm_config loongson_dmaengine_pcm_config = {
50+
.pcm_hardware = &loongson_pcm_hardware,
51+
.prepare_slave_config = snd_dmaengine_pcm_prepare_slave_config,
52+
.prealloc_buffer_size = 128 * 1024,
53+
};
54+
55+
static int loongson_pcm_open(struct snd_soc_component *component,
56+
struct snd_pcm_substream *substream)
57+
{
58+
struct snd_pcm_runtime *runtime = substream->runtime;
59+
60+
if (substream->pcm->device & 1) {
61+
runtime->hw.info &= ~SNDRV_PCM_INFO_INTERLEAVED;
62+
runtime->hw.info |= SNDRV_PCM_INFO_NONINTERLEAVED;
63+
}
64+
65+
if (substream->pcm->device & 2)
66+
runtime->hw.info &= ~(SNDRV_PCM_INFO_MMAP |
67+
SNDRV_PCM_INFO_MMAP_VALID);
68+
/*
69+
* For mysterious reasons (and despite what the manual says)
70+
* playback samples are lost if the DMA count is not a multiple
71+
* of the DMA burst size. Let's add a rule to enforce that.
72+
*/
73+
snd_pcm_hw_constraint_step(runtime, 0,
74+
SNDRV_PCM_HW_PARAM_PERIOD_BYTES, 128);
75+
snd_pcm_hw_constraint_step(runtime, 0,
76+
SNDRV_PCM_HW_PARAM_BUFFER_BYTES, 128);
77+
snd_pcm_hw_constraint_integer(substream->runtime,
78+
SNDRV_PCM_HW_PARAM_PERIODS);
79+
80+
return 0;
81+
}
82+
83+
static const struct snd_soc_component_driver loongson_i2s_component_driver = {
84+
.name = LS_I2S_DRVNAME,
85+
.open = loongson_pcm_open,
86+
};
87+
88+
static const struct regmap_config loongson_i2s_regmap_config = {
89+
.reg_bits = 32,
90+
.reg_stride = 4,
91+
.val_bits = 32,
92+
.max_register = 0x14,
93+
.cache_type = REGCACHE_FLAT,
94+
};
95+
96+
static int loongson_i2s_apbdma_config(struct platform_device *pdev)
97+
{
98+
int val;
99+
void __iomem *regs;
100+
101+
regs = devm_platform_ioremap_resource(pdev, 1);
102+
if (IS_ERR(regs))
103+
return PTR_ERR(regs);
104+
105+
val = readl(regs);
106+
val |= LOONGSON_DMA2_CONF << LOONGSON_I2S_TX_DMA_OFFSET;
107+
val |= LOONGSON_DMA3_CONF << LOONGSON_I2S_RX_DMA_OFFSET;
108+
writel(val, regs);
109+
110+
return 0;
111+
}
112+
113+
static int loongson_i2s_plat_probe(struct platform_device *pdev)
114+
{
115+
struct device *dev = &pdev->dev;
116+
struct loongson_i2s *i2s;
117+
struct resource *res;
118+
struct clk *i2s_clk;
119+
int ret;
120+
121+
i2s = devm_kzalloc(dev, sizeof(*i2s), GFP_KERNEL);
122+
if (!i2s)
123+
return -ENOMEM;
124+
125+
ret = loongson_i2s_apbdma_config(pdev);
126+
if (ret)
127+
return ret;
128+
129+
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
130+
i2s->reg_base = devm_ioremap_resource(&pdev->dev, res);
131+
if (IS_ERR(i2s->reg_base))
132+
return dev_err_probe(dev, PTR_ERR(i2s->reg_base),
133+
"devm_ioremap_resource failed\n");
134+
135+
i2s->regmap = devm_regmap_init_mmio(dev, i2s->reg_base,
136+
&loongson_i2s_regmap_config);
137+
if (IS_ERR(i2s->regmap))
138+
return dev_err_probe(dev, PTR_ERR(i2s->regmap),
139+
"devm_regmap_init_mmio failed\n");
140+
141+
i2s->playback_dma_data.addr = res->start + LS_I2S_TX_DATA;
142+
i2s->playback_dma_data.addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
143+
i2s->playback_dma_data.maxburst = 4;
144+
145+
i2s->capture_dma_data.addr = res->start + LS_I2S_RX_DATA;
146+
i2s->capture_dma_data.addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
147+
i2s->capture_dma_data.maxburst = 4;
148+
149+
i2s_clk = devm_clk_get_enabled(dev, NULL);
150+
if (IS_ERR(i2s_clk))
151+
return dev_err_probe(dev, PTR_ERR(i2s_clk), "clock property invalid\n");
152+
i2s->clk_rate = clk_get_rate(i2s_clk);
153+
154+
dma_set_mask_and_coherent(dev, DMA_BIT_MASK(64));
155+
dev_set_name(dev, LS_I2S_DRVNAME);
156+
dev_set_drvdata(dev, i2s);
157+
158+
ret = devm_snd_soc_register_component(dev, &loongson_i2s_component_driver,
159+
&loongson_i2s_dai, 1);
160+
if (ret)
161+
return dev_err_probe(dev, ret, "failed to register DAI\n");
162+
163+
return devm_snd_dmaengine_pcm_register(dev, &loongson_dmaengine_pcm_config,
164+
SND_DMAENGINE_PCM_FLAG_COMPAT);
165+
}
166+
167+
static const struct of_device_id loongson_i2s_ids[] = {
168+
{ .compatible = "loongson,ls2k1000-i2s" },
169+
{ /* sentinel */ },
170+
};
171+
MODULE_DEVICE_TABLE(of, loongson_i2s_ids);
172+
173+
static struct platform_driver loongson_i2s_driver = {
174+
.probe = loongson_i2s_plat_probe,
175+
.driver = {
176+
.name = "loongson-i2s-plat",
177+
.pm = pm_sleep_ptr(&loongson_i2s_pm),
178+
.of_match_table = loongson_i2s_ids,
179+
},
180+
};
181+
module_platform_driver(loongson_i2s_driver);
182+
183+
MODULE_DESCRIPTION("Loongson I2S Master Mode ASoC Driver");
184+
MODULE_AUTHOR("Loongson Technology Corporation Limited");
185+
MODULE_LICENSE("GPL");

0 commit comments

Comments
 (0)