Merge remote-tracking branch 'asoc/topic/samsung' into asoc-next

This commit is contained in:
Mark Brown 2018-03-28 10:26:30 +08:00
commit 848272e9a3
No known key found for this signature in database
GPG Key ID: 24D68B725D5487D0
13 changed files with 432 additions and 151 deletions

View File

@ -2,8 +2,10 @@ Samsung Exynos Odroid XU3/XU4 audio complex with MAX98090 codec
Required properties: Required properties:
- compatible - "samsung,odroidxu3-audio" - for Odroid XU3 board, - compatible - "hardkernel,odroid-xu3-audio" - for Odroid XU3 board,
"samsung,odroidxu4-audio" - for Odroid XU4 board "hardkernel,odroid-xu4-audio" - for Odroid XU4 board (deprecated),
"samsung,odroid-xu3-audio" - for Odroid XU3 board (deprecated),
"samsung,odroid-xu4-audio" - for Odroid XU4 board (deprecated)
- model - the user-visible name of this sound complex - model - the user-visible name of this sound complex
- clocks - should contain entries matching clock names in the clock-names - clocks - should contain entries matching clock names in the clock-names
property property
@ -35,7 +37,7 @@ Required sub-nodes:
Example: Example:
sound { sound {
compatible = "samsung,odroidxu3-audio"; compatible = "hardkernel,odroid-xu3-audio";
model = "Odroid-XU3"; model = "Odroid-XU3";
samsung,audio-routing = samsung,audio-routing =
"Headphone Jack", "HPL", "Headphone Jack", "HPL",

View File

@ -4,9 +4,13 @@ Required properties:
- compatible : "samsung,tm2-audio" - compatible : "samsung,tm2-audio"
- model : the user-visible name of this sound complex - model : the user-visible name of this sound complex
- audio-codec : the phandle of the wm5110 audio codec node, - audio-codec : the first entry should be phandle of the wm5110 audio
as described in ../mfd/arizona.txt codec node, as described in ../mfd/arizona.txt;
- i2s-controller : the phandle of the I2S controller the second entry should be phandle of the HDMI
transmitter node
- i2s-controller : the list of phandle and argument tuples pointing to
I2S controllers, the first entry should be I2S0 and
the second one I2S1
- audio-amplifier : the phandle of the MAX98504 amplifier - audio-amplifier : the phandle of the MAX98504 amplifier
- samsung,audio-routing : a list of the connections between audio components; - samsung,audio-routing : a list of the connections between audio components;
each entry is a pair of strings, the first being the each entry is a pair of strings, the first being the
@ -22,8 +26,8 @@ Example:
sound { sound {
compatible = "samsung,tm2-audio"; compatible = "samsung,tm2-audio";
audio-codec = <&wm5110>; audio-codec = <&wm5110>, <&hdmi>;
i2s-controller = <&i2s0>; i2s-controller = <&i2s0 0>, <&i2s1 0>;
audio-amplifier = <&max98504>; audio-amplifier = <&max98504>;
mic-bias-gpios = <&gpr3 2 0>; mic-bias-gpios = <&gpr3 2 0>;
model = "wm5110"; model = "wm5110";

View File

@ -7,7 +7,7 @@ Required SoC Specific Properties:
- samsung,s5pv210-i2s: for 8/16/24bit multichannel(5.1) I2S with - samsung,s5pv210-i2s: for 8/16/24bit multichannel(5.1) I2S with
secondary fifo, s/w reset control and internal mux for root clk src. secondary fifo, s/w reset control and internal mux for root clk src.
- samsung,exynos5420-i2s: for 8/16/24bit multichannel(5.1) I2S for - samsung,exynos5420-i2s: for 8/16/24bit multichannel(5.1) I2S for
playback, sterio channel capture, secondary fifo using internal playback, stereo channel capture, secondary fifo using internal
or external dma, s/w reset control, internal mux for root clk src or external dma, s/w reset control, internal mux for root clk src
and 7.1 channel TDM support for playback. TDM (Time division multiplexing) and 7.1 channel TDM support for playback. TDM (Time division multiplexing)
is to allow transfer of multiple channel audio data on single data line. is to allow transfer of multiple channel audio data on single data line.
@ -25,7 +25,7 @@ Required SoC Specific Properties:
These strings correspond 1:1 with the ordered pairs in dmas. These strings correspond 1:1 with the ordered pairs in dmas.
- clocks: Handle to iis clock and RCLK source clk. - clocks: Handle to iis clock and RCLK source clk.
- clock-names: - clock-names:
i2s0 uses some base clks from CMU and some are from audio subsystem internal i2s0 uses some base clocks from CMU and some are from audio subsystem internal
clock controller. The clock names for i2s0 should be "iis", "i2s_opclk0" and clock controller. The clock names for i2s0 should be "iis", "i2s_opclk0" and
"i2s_opclk1" as shown in the example below. "i2s_opclk1" as shown in the example below.
i2s1 and i2s2 uses clocks from CMU. The clock names for i2s1 and i2s2 should i2s1 and i2s2 uses clocks from CMU. The clock names for i2s1 and i2s2 should
@ -36,9 +36,9 @@ Required SoC Specific Properties:
- #clock-cells: should be 1, this property must be present if the I2S device - #clock-cells: should be 1, this property must be present if the I2S device
is a clock provider in terms of the common clock bindings, described in is a clock provider in terms of the common clock bindings, described in
../clock/clock-bindings.txt. ../clock/clock-bindings.txt.
- clock-output-names: from the common clock bindings, names of the CDCLK - clock-output-names (deprecated): from the common clock bindings, names of
I2S output clocks, suggested values are "i2s_cdclk0", "i2s_cdclk1", the CDCLK I2S output clocks, suggested values are "i2s_cdclk0", "i2s_cdclk1",
"i2s_cdclk3" for the I2S0, I2S1, I2S2 devices recpectively. "i2s_cdclk3" for the I2S0, I2S1, I2S2 devices respectively.
There are following clocks available at the I2S device nodes: There are following clocks available at the I2S device nodes:
CLK_I2S_CDCLK - the CDCLK (CODECLKO) gate clock, CLK_I2S_CDCLK - the CDCLK (CODECLKO) gate clock,
@ -49,9 +49,10 @@ There are following clocks available at the I2S device nodes:
Refer to the SoC datasheet for availability of the above clocks. Refer to the SoC datasheet for availability of the above clocks.
The CLK_I2S_RCLK_PSR and CLK_I2S_RCLK_SRC clocks are usually only available The CLK_I2S_RCLK_PSR and CLK_I2S_RCLK_SRC clocks are usually only available
in the IIS Multi Audio Interface (I2S0). in the IIS Multi Audio Interface.
Note: Old DTs may not have the #clock-cells, clock-output-names properties
and then not use the I2S node as a clock supplier. Note: Old DTs may not have the #clock-cells property and then not use the I2S
node as a clock supplier.
Optional SoC Specific Properties: Optional SoC Specific Properties:
@ -59,6 +60,7 @@ Optional SoC Specific Properties:
sub system(used in secondary sound source). sub system(used in secondary sound source).
- pinctrl-0: Should specify pin control groups used for this controller. - pinctrl-0: Should specify pin control groups used for this controller.
- pinctrl-names: Should contain only one value - "default". - pinctrl-names: Should contain only one value - "default".
- #sound-dai-cells: should be 1.
Example: Example:
@ -74,9 +76,9 @@ i2s0: i2s@3830000 {
<&clock_audss EXYNOS_I2S_BUS>, <&clock_audss EXYNOS_I2S_BUS>,
<&clock_audss EXYNOS_SCLK_I2S>; <&clock_audss EXYNOS_SCLK_I2S>;
clock-names = "iis", "i2s_opclk0", "i2s_opclk1"; clock-names = "iis", "i2s_opclk0", "i2s_opclk1";
#clock-cells; #clock-cells = <1>;
clock-output-names = "i2s_cdclk0";
samsung,idma-addr = <0x03000000>; samsung,idma-addr = <0x03000000>;
pinctrl-names = "default"; pinctrl-names = "default";
pinctrl-0 = <&i2s0_bus>; pinctrl-0 = <&i2s0_bus>;
#sound-dai-cells = <1>;
}; };

View File

@ -5,8 +5,17 @@ Required properties:
"google,snow-audio-max98090" or "google,snow-audio-max98090" or
"google,snow-audio-max98091" or "google,snow-audio-max98091" or
"google,snow-audio-max98095" "google,snow-audio-max98095"
- samsung,i2s-controller: The phandle of the Samsung I2S controller - samsung,i2s-controller (deprecated): The phandle of the Samsung I2S controller
- samsung,audio-codec: The phandle of the audio codec - samsung,audio-codec (deprecated): The phandle of the audio codec
Required sub-nodes:
- 'cpu' subnode with a 'sound-dai' property containing the phandle of the I2S
controller
- 'codec' subnode with a 'sound-dai' property containing list of phandles
to the CODEC nodes, first entry must be the phandle of the MAX98090,
MAX98091 or MAX98095 CODEC (exact device type is indicated by the compatible
string) and the second entry must be the phandle of the HDMI IP block node
Optional: Optional:
- samsung,model: The name of the sound-card - samsung,model: The name of the sound-card

View File

@ -1807,6 +1807,7 @@ int snd_soc_of_get_dai_name(struct device_node *of_node,
int snd_soc_of_get_dai_link_codecs(struct device *dev, int snd_soc_of_get_dai_link_codecs(struct device *dev,
struct device_node *of_node, struct device_node *of_node,
struct snd_soc_dai_link *dai_link); struct snd_soc_dai_link *dai_link);
void snd_soc_of_put_dai_link_codecs(struct snd_soc_dai_link *dai_link);
int snd_soc_add_dai_link(struct snd_soc_card *card, int snd_soc_add_dai_link(struct snd_soc_card *card,
struct snd_soc_dai_link *dai_link); struct snd_soc_dai_link *dai_link);

View File

@ -21,8 +21,6 @@ obj-$(CONFIG_SND_SAMSUNG_I2S) += snd-soc-idma.o
# S3C24XX Machine Support # S3C24XX Machine Support
snd-soc-jive-wm8750-objs := jive_wm8750.o snd-soc-jive-wm8750-objs := jive_wm8750.o
snd-soc-neo1973-wm8753-objs := neo1973_wm8753.o snd-soc-neo1973-wm8753-objs := neo1973_wm8753.o
snd-soc-smdk2443-wm9710-objs := smdk2443_wm9710.o
snd-soc-ln2440sbc-alc650-objs := ln2440sbc_alc650.o
snd-soc-s3c24xx-uda134x-objs := s3c24xx_uda134x.o snd-soc-s3c24xx-uda134x-objs := s3c24xx_uda134x.o
snd-soc-s3c24xx-simtec-objs := s3c24xx_simtec.o snd-soc-s3c24xx-simtec-objs := s3c24xx_simtec.o
snd-soc-s3c24xx-simtec-hermes-objs := s3c24xx_simtec_hermes.o snd-soc-s3c24xx-simtec-hermes-objs := s3c24xx_simtec_hermes.o
@ -32,7 +30,6 @@ snd-soc-rx1950-uda1380-objs := rx1950_uda1380.o
snd-soc-smdk-wm8580-objs := smdk_wm8580.o snd-soc-smdk-wm8580-objs := smdk_wm8580.o
snd-soc-smdk-wm8994-objs := smdk_wm8994.o snd-soc-smdk-wm8994-objs := smdk_wm8994.o
snd-soc-snow-objs := snow.o snd-soc-snow-objs := snow.o
snd-soc-smdk-wm9713-objs := smdk_wm9713.o
snd-soc-s3c64xx-smartq-wm8987-objs := smartq_wm8987.o snd-soc-s3c64xx-smartq-wm8987-objs := smartq_wm8987.o
snd-soc-smdk-spdif-objs := smdk_spdif.o snd-soc-smdk-spdif-objs := smdk_spdif.o
snd-soc-smdk-wm8994pcm-objs := smdk_wm8994pcm.o snd-soc-smdk-wm8994pcm-objs := smdk_wm8994pcm.o

View File

@ -65,11 +65,12 @@
#define CON_RXDMA_ACTIVE (1 << 1) #define CON_RXDMA_ACTIVE (1 << 1)
#define CON_ACTIVE (1 << 0) #define CON_ACTIVE (1 << 0)
#define MOD_OPCLK_CDCLK_OUT (0 << 30) #define MOD_OPCLK_SHIFT 30
#define MOD_OPCLK_CDCLK_IN (1 << 30) #define MOD_OPCLK_CDCLK_OUT (0 << MOD_OPCLK_SHIFT)
#define MOD_OPCLK_BCLK_OUT (2 << 30) #define MOD_OPCLK_CDCLK_IN (1 << MOD_OPCLK_SHIFT)
#define MOD_OPCLK_PCLK (3 << 30) #define MOD_OPCLK_BCLK_OUT (2 << MOD_OPCLK_SHIFT)
#define MOD_OPCLK_MASK (3 << 30) #define MOD_OPCLK_PCLK (3 << MOD_OPCLK_SHIFT)
#define MOD_OPCLK_MASK (3 << MOD_OPCLK_SHIFT)
#define MOD_TXS_IDMA (1 << 28) /* Sec_TXFIFO use I-DMA */ #define MOD_TXS_IDMA (1 << 28) /* Sec_TXFIFO use I-DMA */
#define MOD_BLCS_SHIFT 26 #define MOD_BLCS_SHIFT 26

View File

@ -489,7 +489,7 @@ static int i2s_set_sysclk(struct snd_soc_dai *dai,
switch (clk_id) { switch (clk_id) {
case SAMSUNG_I2S_OPCLK: case SAMSUNG_I2S_OPCLK:
mask = MOD_OPCLK_MASK; mask = MOD_OPCLK_MASK;
val = dir; val = (dir << MOD_OPCLK_SHIFT) & MOD_OPCLK_MASK;
break; break;
case SAMSUNG_I2S_CDCLK: case SAMSUNG_I2S_CDCLK:
mask = 1 << i2s_regs->cdclkcon_off; mask = 1 << i2s_regs->cdclkcon_off;
@ -656,8 +656,12 @@ static int i2s_set_fmt(struct snd_soc_dai *dai,
tmp |= mod_slave; tmp |= mod_slave;
break; break;
case SND_SOC_DAIFMT_CBS_CFS: case SND_SOC_DAIFMT_CBS_CFS:
/* Set default source clock in Master mode */ /*
if (i2s->rclk_srcrate == 0) * Set default source clock in Master mode, only when the
* CLK_I2S_RCLK_SRC clock is not exposed so we ensure any
* clock configuration assigned in DT is not overwritten.
*/
if (i2s->rclk_srcrate == 0 && i2s->clk_data.clks == NULL)
i2s_set_sysclk(dai, SAMSUNG_I2S_RCLKSRC_0, i2s_set_sysclk(dai, SAMSUNG_I2S_RCLKSRC_0,
0, SND_SOC_CLOCK_IN); 0, SND_SOC_CLOCK_IN);
break; break;
@ -881,6 +885,11 @@ static int config_setup(struct i2s_dai *i2s)
return 0; return 0;
if (!(i2s->quirks & QUIRK_NO_MUXPSR)) { if (!(i2s->quirks & QUIRK_NO_MUXPSR)) {
struct clk *rclksrc = i2s->clk_table[CLK_I2S_RCLK_SRC];
if (rclksrc && !IS_ERR(rclksrc))
i2s->rclk_srcrate = clk_get_rate(rclksrc);
psr = i2s->rclk_srcrate / i2s->frmclk / rfs; psr = i2s->rclk_srcrate / i2s->frmclk / rfs;
writel(((psr - 1) << 8) | PSR_PSREN, i2s->addr + I2SPSR); writel(((psr - 1) << 8) | PSR_PSREN, i2s->addr + I2SPSR);
dev_dbg(&i2s->pdev->dev, dev_dbg(&i2s->pdev->dev,
@ -1184,11 +1193,13 @@ static void i2s_unregister_clock_provider(struct platform_device *pdev)
static int i2s_register_clock_provider(struct platform_device *pdev) static int i2s_register_clock_provider(struct platform_device *pdev)
{ {
struct device *dev = &pdev->dev; const char * const i2s_clk_desc[] = { "cdclk", "rclk_src", "prescaler" };
struct i2s_dai *i2s = dev_get_drvdata(dev);
const char *clk_name[2] = { "i2s_opclk0", "i2s_opclk1" }; const char *clk_name[2] = { "i2s_opclk0", "i2s_opclk1" };
const char *p_names[2] = { NULL }; const char *p_names[2] = { NULL };
struct device *dev = &pdev->dev;
struct i2s_dai *i2s = dev_get_drvdata(dev);
const struct samsung_i2s_variant_regs *reg_info = i2s->variant_regs; const struct samsung_i2s_variant_regs *reg_info = i2s->variant_regs;
const char *i2s_clk_name[ARRAY_SIZE(i2s_clk_desc)];
struct clk *rclksrc; struct clk *rclksrc;
int ret, i; int ret, i;
@ -1205,30 +1216,38 @@ static int i2s_register_clock_provider(struct platform_device *pdev)
clk_put(rclksrc); clk_put(rclksrc);
} }
for (i = 0; i < ARRAY_SIZE(i2s_clk_desc); i++) {
i2s_clk_name[i] = devm_kasprintf(dev, GFP_KERNEL, "%s_%s",
dev_name(dev), i2s_clk_desc[i]);
if (!i2s_clk_name[i])
return -ENOMEM;
}
if (!(i2s->quirks & QUIRK_NO_MUXPSR)) { if (!(i2s->quirks & QUIRK_NO_MUXPSR)) {
/* Activate the prescaler */ /* Activate the prescaler */
u32 val = readl(i2s->addr + I2SPSR); u32 val = readl(i2s->addr + I2SPSR);
writel(val | PSR_PSREN, i2s->addr + I2SPSR); writel(val | PSR_PSREN, i2s->addr + I2SPSR);
i2s->clk_table[CLK_I2S_RCLK_SRC] = clk_register_mux(dev, i2s->clk_table[CLK_I2S_RCLK_SRC] = clk_register_mux(dev,
"i2s_rclksrc", p_names, ARRAY_SIZE(p_names), i2s_clk_name[CLK_I2S_RCLK_SRC], p_names,
ARRAY_SIZE(p_names),
CLK_SET_RATE_NO_REPARENT | CLK_SET_RATE_PARENT, CLK_SET_RATE_NO_REPARENT | CLK_SET_RATE_PARENT,
i2s->addr + I2SMOD, reg_info->rclksrc_off, i2s->addr + I2SMOD, reg_info->rclksrc_off,
1, 0, i2s->lock); 1, 0, i2s->lock);
i2s->clk_table[CLK_I2S_RCLK_PSR] = clk_register_divider(dev, i2s->clk_table[CLK_I2S_RCLK_PSR] = clk_register_divider(dev,
"i2s_presc", "i2s_rclksrc", i2s_clk_name[CLK_I2S_RCLK_PSR],
i2s_clk_name[CLK_I2S_RCLK_SRC],
CLK_SET_RATE_PARENT, CLK_SET_RATE_PARENT,
i2s->addr + I2SPSR, 8, 6, 0, i2s->lock); i2s->addr + I2SPSR, 8, 6, 0, i2s->lock);
p_names[0] = "i2s_presc"; p_names[0] = i2s_clk_name[CLK_I2S_RCLK_PSR];
i2s->clk_data.clk_num = 2; i2s->clk_data.clk_num = 2;
} }
of_property_read_string_index(dev->of_node,
"clock-output-names", 0, &clk_name[0]);
i2s->clk_table[CLK_I2S_CDCLK] = clk_register_gate(dev, clk_name[0], i2s->clk_table[CLK_I2S_CDCLK] = clk_register_gate(dev,
p_names[0], CLK_SET_RATE_PARENT, i2s_clk_name[CLK_I2S_CDCLK], p_names[0],
CLK_SET_RATE_PARENT,
i2s->addr + I2SMOD, reg_info->cdclkcon_off, i2s->addr + I2SMOD, reg_info->cdclkcon_off,
CLK_GATE_SET_TO_DISABLE, i2s->lock); CLK_GATE_SET_TO_DISABLE, i2s->lock);
@ -1385,9 +1404,14 @@ static int samsung_i2s_probe(struct platform_device *pdev)
pm_runtime_enable(&pdev->dev); pm_runtime_enable(&pdev->dev);
ret = i2s_register_clock_provider(pdev); ret = i2s_register_clock_provider(pdev);
if (!ret) if (ret < 0)
return 0; goto err_disable_pm;
pri_dai->op_clk = clk_get_parent(pri_dai->clk_table[CLK_I2S_RCLK_SRC]);
return 0;
err_disable_pm:
pm_runtime_disable(&pdev->dev); pm_runtime_disable(&pdev->dev);
err_disable_clk: err_disable_clk:
clk_disable_unprepare(pri_dai->clk); clk_disable_unprepare(pri_dai->clk);

View File

@ -16,11 +16,16 @@
#define SAMSUNG_I2S_DAI "samsung-i2s" #define SAMSUNG_I2S_DAI "samsung-i2s"
#define SAMSUNG_I2S_DAI_SEC "samsung-i2s-sec" #define SAMSUNG_I2S_DAI_SEC "samsung-i2s-sec"
#define SAMSUNG_I2S_DIV_BCLK 1 #define SAMSUNG_I2S_DIV_BCLK 1
#define SAMSUNG_I2S_RCLKSRC_0 0 #define SAMSUNG_I2S_RCLKSRC_0 0
#define SAMSUNG_I2S_RCLKSRC_1 1 #define SAMSUNG_I2S_RCLKSRC_1 1
#define SAMSUNG_I2S_CDCLK 2 #define SAMSUNG_I2S_CDCLK 2
/* Operation clock for IIS logic */
#define SAMSUNG_I2S_OPCLK 3 #define SAMSUNG_I2S_OPCLK 3
#define SAMSUNG_I2S_OPCLK_CDCLK_OUT 0 /* CODEC clock out */
#define SAMSUNG_I2S_OPCLK_CDCLK_IN 1 /* CODEC clock in */
#define SAMSUNG_I2S_OPCLK_BCLK_OUT 2 /* Bit clock out */
#define SAMSUNG_I2S_OPCLK_PCLK 3 /* Audio bus clock */
#endif /* __SND_SOC_SAMSUNG_I2S_H */ #endif /* __SND_SOC_SAMSUNG_I2S_H */

View File

@ -36,23 +36,24 @@ static int odroid_card_hw_params(struct snd_pcm_substream *substream,
{ {
struct snd_soc_pcm_runtime *rtd = substream->private_data; struct snd_soc_pcm_runtime *rtd = substream->private_data;
struct odroid_priv *priv = snd_soc_card_get_drvdata(rtd->card); struct odroid_priv *priv = snd_soc_card_get_drvdata(rtd->card);
unsigned int pll_freq, rclk_freq; unsigned int pll_freq, rclk_freq, rfs;
int ret; int ret;
switch (params_rate(params)) { switch (params_rate(params)) {
case 32000:
case 64000: case 64000:
pll_freq = 131072006U; pll_freq = 196608001U;
rfs = 384;
break; break;
case 44100: case 44100:
case 88200: case 88200:
case 176400:
pll_freq = 180633609U; pll_freq = 180633609U;
rfs = 512;
break; break;
case 32000:
case 48000: case 48000:
case 96000: case 96000:
case 192000:
pll_freq = 196608001U; pll_freq = 196608001U;
rfs = 512;
break; break;
default: default:
return -EINVAL; return -EINVAL;
@ -67,7 +68,7 @@ static int odroid_card_hw_params(struct snd_pcm_substream *substream,
* frequency values due to the EPLL output frequency not being exact * frequency values due to the EPLL output frequency not being exact
* multiple of the audio sampling rate. * multiple of the audio sampling rate.
*/ */
rclk_freq = params_rate(params) * 256 + 1; rclk_freq = params_rate(params) * rfs + 1;
ret = clk_set_rate(priv->sclk_i2s, rclk_freq); ret = clk_set_rate(priv->sclk_i2s, rclk_freq);
if (ret < 0) if (ret < 0)
@ -90,18 +91,6 @@ static const struct snd_soc_ops odroid_card_ops = {
.hw_params = odroid_card_hw_params, .hw_params = odroid_card_hw_params,
}; };
static void odroid_put_codec_of_nodes(struct snd_soc_dai_link *link)
{
struct snd_soc_dai_link_component *component = link->codecs;
int i;
for (i = 0; i < link->num_codecs; i++, component++) {
if (!component->of_node)
break;
of_node_put(component->of_node);
}
}
static int odroid_audio_probe(struct platform_device *pdev) static int odroid_audio_probe(struct platform_device *pdev)
{ {
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
@ -196,7 +185,7 @@ err_put_sclk:
err_put_i2s_n: err_put_i2s_n:
of_node_put(link->cpu_of_node); of_node_put(link->cpu_of_node);
err_put_codec_n: err_put_codec_n:
odroid_put_codec_of_nodes(link); snd_soc_of_put_dai_link_codecs(link);
return ret; return ret;
} }
@ -205,7 +194,7 @@ static int odroid_audio_remove(struct platform_device *pdev)
struct odroid_priv *priv = platform_get_drvdata(pdev); struct odroid_priv *priv = platform_get_drvdata(pdev);
of_node_put(priv->dai_link.cpu_of_node); of_node_put(priv->dai_link.cpu_of_node);
odroid_put_codec_of_nodes(&priv->dai_link); snd_soc_of_put_dai_link_codecs(&priv->dai_link);
clk_put(priv->sclk_i2s); clk_put(priv->sclk_i2s);
clk_put(priv->clk_i2s_bus); clk_put(priv->clk_i2s_bus);
@ -213,8 +202,10 @@ static int odroid_audio_remove(struct platform_device *pdev)
} }
static const struct of_device_id odroid_audio_of_match[] = { static const struct of_device_id odroid_audio_of_match[] = {
{ .compatible = "hardkernel,odroid-xu3-audio" },
{ .compatible = "hardkernel,odroid-xu4-audio" },
{ .compatible = "samsung,odroid-xu3-audio" }, { .compatible = "samsung,odroid-xu3-audio" },
{ .compatible = "samsung,odroid-xu4-audio"}, { .compatible = "samsung,odroid-xu4-audio" },
{ }, { },
}; };
MODULE_DEVICE_TABLE(of, odroid_audio_of_match); MODULE_DEVICE_TABLE(of, odroid_audio_of_match);

View File

@ -11,97 +11,207 @@
* General Public License for more details. * General Public License for more details.
*/ */
#include <linux/clk.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/of.h> #include <linux/of.h>
#include <linux/of_device.h> #include <linux/of_device.h>
#include <sound/pcm_params.h>
#include <sound/soc.h> #include <sound/soc.h>
#include "i2s.h" #include "i2s.h"
#define FIN_PLL_RATE 24000000 #define FIN_PLL_RATE 24000000
static struct snd_soc_dai_link snow_dai[] = { struct snow_priv {
{ struct snd_soc_dai_link dai_link;
.name = "Primary", struct clk *clk_i2s_bus;
.stream_name = "Primary", };
.codec_dai_name = "HiFi",
.dai_fmt = SND_SOC_DAIFMT_I2S | static int snow_card_hw_params(struct snd_pcm_substream *substream,
SND_SOC_DAIFMT_NB_NF | struct snd_pcm_hw_params *params)
SND_SOC_DAIFMT_CBS_CFS, {
}, static const unsigned int pll_rate[] = {
73728000U, 67737602U, 49152000U, 45158401U, 32768001U
};
struct snd_soc_pcm_runtime *rtd = substream->private_data;
struct snow_priv *priv = snd_soc_card_get_drvdata(rtd->card);
int bfs, psr, rfs, bitwidth;
unsigned long int rclk;
long int freq = -EINVAL;
int ret, i;
bitwidth = snd_pcm_format_width(params_format(params));
if (bitwidth < 0) {
dev_err(rtd->card->dev, "Invalid bit-width: %d\n", bitwidth);
return bitwidth;
}
if (bitwidth != 16 && bitwidth != 24) {
dev_err(rtd->card->dev, "Unsupported bit-width: %d\n", bitwidth);
return -EINVAL;
}
bfs = 2 * bitwidth;
switch (params_rate(params)) {
case 16000:
case 22050:
case 24000:
case 32000:
case 44100:
case 48000:
case 88200:
case 96000:
rfs = 8 * bfs;
break;
case 64000:
rfs = 384;
break;
case 8000:
case 11025:
case 12000:
rfs = 16 * bfs;
break;
default:
return -EINVAL;
}
rclk = params_rate(params) * rfs;
for (psr = 8; psr > 0; psr /= 2) {
for (i = 0; i < ARRAY_SIZE(pll_rate); i++) {
if ((pll_rate[i] - rclk * psr) <= 2) {
freq = pll_rate[i];
break;
}
}
}
if (freq < 0) {
dev_err(rtd->card->dev, "Unsupported RCLK rate: %lu\n", rclk);
return -EINVAL;
}
ret = clk_set_rate(priv->clk_i2s_bus, freq);
if (ret < 0) {
dev_err(rtd->card->dev, "I2S bus clock rate set failed\n");
return ret;
}
return 0;
}
static const struct snd_soc_ops snow_card_ops = {
.hw_params = snow_card_hw_params,
}; };
static int snow_late_probe(struct snd_soc_card *card) static int snow_late_probe(struct snd_soc_card *card)
{ {
struct snd_soc_pcm_runtime *rtd; struct snd_soc_pcm_runtime *rtd;
struct snd_soc_dai *codec_dai; struct snd_soc_dai *codec_dai;
struct snd_soc_dai *cpu_dai;
int ret;
rtd = snd_soc_get_pcm_runtime(card, card->dai_link[0].name); rtd = snd_soc_get_pcm_runtime(card, card->dai_link[0].name);
codec_dai = rtd->codec_dai;
cpu_dai = rtd->cpu_dai; /* In the multi-codec case codec_dais 0 is MAX98095 and 1 is HDMI. */
if (rtd->num_codecs > 1)
codec_dai = rtd->codec_dais[0];
else
codec_dai = rtd->codec_dai;
/* Set the MCLK rate for the codec */ /* Set the MCLK rate for the codec */
ret = snd_soc_dai_set_sysclk(codec_dai, 0, return snd_soc_dai_set_sysclk(codec_dai, 0,
FIN_PLL_RATE, SND_SOC_CLOCK_IN); FIN_PLL_RATE, SND_SOC_CLOCK_IN);
if (ret < 0)
return ret;
/* Select I2S Bus clock to set RCLK and BCLK */
ret = snd_soc_dai_set_sysclk(cpu_dai, SAMSUNG_I2S_RCLKSRC_0,
0, SND_SOC_CLOCK_IN);
if (ret < 0)
return ret;
return 0;
} }
static struct snd_soc_card snow_snd = { static struct snd_soc_card snow_snd = {
.name = "Snow-I2S", .name = "Snow-I2S",
.owner = THIS_MODULE, .owner = THIS_MODULE,
.dai_link = snow_dai,
.num_links = ARRAY_SIZE(snow_dai),
.late_probe = snow_late_probe, .late_probe = snow_late_probe,
}; };
static int snow_probe(struct platform_device *pdev) static int snow_probe(struct platform_device *pdev)
{ {
struct device *dev = &pdev->dev;
struct snd_soc_card *card = &snow_snd; struct snd_soc_card *card = &snow_snd;
struct device_node *i2s_node, *codec_node; struct device_node *cpu, *codec;
int i, ret; struct snd_soc_dai_link *link;
struct snow_priv *priv;
int ret;
i2s_node = of_parse_phandle(pdev->dev.of_node, priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
"samsung,i2s-controller", 0); if (!priv)
if (!i2s_node) { return -ENOMEM;
dev_err(&pdev->dev,
"Property 'i2s-controller' missing or invalid\n"); link = &priv->dai_link;
return -EINVAL;
link->dai_fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
SND_SOC_DAIFMT_CBS_CFS;
link->name = "Primary";
link->stream_name = link->name;
card->dai_link = link;
card->num_links = 1;
card->dev = dev;
/* Try new DT bindings with HDMI support first. */
cpu = of_get_child_by_name(dev->of_node, "cpu");
if (cpu) {
link->ops = &snow_card_ops;
link->cpu_of_node = of_parse_phandle(cpu, "sound-dai", 0);
of_node_put(cpu);
if (!link->cpu_of_node) {
dev_err(dev, "Failed parsing cpu/sound-dai property\n");
return -EINVAL;
}
codec = of_get_child_by_name(dev->of_node, "codec");
ret = snd_soc_of_get_dai_link_codecs(dev, codec, link);
of_node_put(codec);
if (ret < 0) {
of_node_put(link->cpu_of_node);
dev_err(dev, "Failed parsing codec node\n");
return ret;
}
priv->clk_i2s_bus = of_clk_get_by_name(link->cpu_of_node,
"i2s_opclk0");
if (IS_ERR(priv->clk_i2s_bus)) {
snd_soc_of_put_dai_link_codecs(link);
of_node_put(link->cpu_of_node);
return PTR_ERR(priv->clk_i2s_bus);
}
} else {
link->codec_dai_name = "HiFi",
link->cpu_of_node = of_parse_phandle(dev->of_node,
"samsung,i2s-controller", 0);
if (!link->cpu_of_node) {
dev_err(dev, "i2s-controller property parse error\n");
return -EINVAL;
}
link->codec_of_node = of_parse_phandle(dev->of_node,
"samsung,audio-codec", 0);
if (!link->codec_of_node) {
of_node_put(link->cpu_of_node);
dev_err(dev, "audio-codec property parse error\n");
return -EINVAL;
}
} }
codec_node = of_parse_phandle(pdev->dev.of_node, link->platform_of_node = link->cpu_of_node;
"samsung,audio-codec", 0);
if (!codec_node) {
dev_err(&pdev->dev,
"Property 'audio-codec' missing or invalid\n");
return -EINVAL;
}
for (i = 0; i < ARRAY_SIZE(snow_dai); i++) {
snow_dai[i].codec_of_node = codec_node;
snow_dai[i].cpu_of_node = i2s_node;
snow_dai[i].platform_of_node = i2s_node;
}
card->dev = &pdev->dev;
/* Update card-name if provided through DT, else use default name */ /* Update card-name if provided through DT, else use default name */
snd_soc_of_parse_card_name(card, "samsung,model"); snd_soc_of_parse_card_name(card, "samsung,model");
ret = devm_snd_soc_register_card(&pdev->dev, card); snd_soc_card_set_drvdata(card, priv);
ret = devm_snd_soc_register_card(dev, card);
if (ret) { if (ret) {
dev_err(&pdev->dev, "snd_soc_register_card failed (%d)\n", ret); dev_err(&pdev->dev, "snd_soc_register_card failed (%d)\n", ret);
return ret; return ret;
@ -110,6 +220,20 @@ static int snow_probe(struct platform_device *pdev)
return ret; return ret;
} }
static int snow_remove(struct platform_device *pdev)
{
struct snow_priv *priv = platform_get_drvdata(pdev);
struct snd_soc_dai_link *link = &priv->dai_link;
of_node_put(link->cpu_of_node);
of_node_put(link->codec_of_node);
snd_soc_of_put_dai_link_codecs(link);
clk_put(priv->clk_i2s_bus);
return 0;
}
static const struct of_device_id snow_of_match[] = { static const struct of_device_id snow_of_match[] = {
{ .compatible = "google,snow-audio-max98090", }, { .compatible = "google,snow-audio-max98090", },
{ .compatible = "google,snow-audio-max98091", }, { .compatible = "google,snow-audio-max98091", },
@ -125,6 +249,7 @@ static struct platform_driver snow_driver = {
.of_match_table = snow_of_match, .of_match_table = snow_of_match,
}, },
.probe = snow_probe, .probe = snow_probe,
.remove = snow_remove,
}; };
module_platform_driver(snow_driver); module_platform_driver(snow_driver);

View File

@ -210,6 +210,59 @@ static struct snd_soc_ops tm2_aif2_ops = {
.hw_free = tm2_aif2_hw_free, .hw_free = tm2_aif2_hw_free,
}; };
static int tm2_hdmi_hw_params(struct snd_pcm_substream *substream,
struct snd_pcm_hw_params *params)
{
struct snd_soc_pcm_runtime *rtd = substream->private_data;
struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
unsigned int bfs;
int bitwidth, ret;
bitwidth = snd_pcm_format_width(params_format(params));
if (bitwidth < 0) {
dev_err(rtd->card->dev, "Invalid bit-width: %d\n", bitwidth);
return bitwidth;
}
switch (bitwidth) {
case 48:
bfs = 64;
break;
case 16:
bfs = 32;
break;
default:
dev_err(rtd->card->dev, "Unsupported bit-width: %d\n", bitwidth);
return -EINVAL;
}
switch (params_rate(params)) {
case 48000:
case 96000:
case 192000:
break;
default:
dev_err(rtd->card->dev, "Unsupported sample rate: %d\n",
params_rate(params));
return -EINVAL;
}
ret = snd_soc_dai_set_sysclk(cpu_dai, SAMSUNG_I2S_OPCLK,
0, SAMSUNG_I2S_OPCLK_PCLK);
if (ret < 0)
return ret;
ret = snd_soc_dai_set_clkdiv(cpu_dai, SAMSUNG_I2S_DIV_BCLK, bfs);
if (ret < 0)
return ret;
return 0;
}
static struct snd_soc_ops tm2_hdmi_ops = {
.hw_params = tm2_hdmi_hw_params,
};
static int tm2_mic_bias(struct snd_soc_dapm_widget *w, static int tm2_mic_bias(struct snd_soc_dapm_widget *w,
struct snd_kcontrol *kcontrol, int event) struct snd_kcontrol *kcontrol, int event)
{ {
@ -405,6 +458,12 @@ static struct snd_soc_dai_link tm2_dai_links[] = {
.dai_fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF | .dai_fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
SND_SOC_DAIFMT_CBM_CFM, SND_SOC_DAIFMT_CBM_CFM,
.ignore_suspend = 1, .ignore_suspend = 1,
}, {
.name = "HDMI",
.stream_name = "i2s1",
.ops = &tm2_hdmi_ops,
.dai_fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
SND_SOC_DAIFMT_CBS_CFS,
} }
}; };
@ -412,7 +471,6 @@ static struct snd_soc_card tm2_card = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.dai_link = tm2_dai_links, .dai_link = tm2_dai_links,
.num_links = ARRAY_SIZE(tm2_dai_links),
.controls = tm2_controls, .controls = tm2_controls,
.num_controls = ARRAY_SIZE(tm2_controls), .num_controls = ARRAY_SIZE(tm2_controls),
.dapm_widgets = tm2_dapm_widgets, .dapm_widgets = tm2_dapm_widgets,
@ -426,11 +484,14 @@ static struct snd_soc_card tm2_card = {
static int tm2_probe(struct platform_device *pdev) static int tm2_probe(struct platform_device *pdev)
{ {
struct device_node *cpu_dai_node[2] = {};
struct device_node *codec_dai_node[2] = {};
const char *cells_name = NULL;
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct snd_soc_card *card = &tm2_card; struct snd_soc_card *card = &tm2_card;
struct tm2_machine_priv *priv; struct tm2_machine_priv *priv;
struct device_node *cpu_dai_node, *codec_dai_node; struct of_phandle_args args;
int ret, i; int num_codecs, ret, i;
priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL); priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
if (!priv) if (!priv)
@ -464,47 +525,92 @@ static int tm2_probe(struct platform_device *pdev)
return -EINVAL; return -EINVAL;
} }
cpu_dai_node = of_parse_phandle(dev->of_node, "i2s-controller", 0); num_codecs = of_count_phandle_with_args(dev->of_node, "audio-codec",
if (!cpu_dai_node) { NULL);
dev_err(dev, "i2s-controllers property invalid or missing\n");
ret = -EINVAL; /* Skip the HDMI link if not specified in DT */
goto amp_node_put; if (num_codecs > 1) {
card->num_links = ARRAY_SIZE(tm2_dai_links);
cells_name = "#sound-dai-cells";
} else {
card->num_links = ARRAY_SIZE(tm2_dai_links) - 1;
} }
codec_dai_node = of_parse_phandle(dev->of_node, "audio-codec", 0); for (i = 0; i < num_codecs; i++) {
if (!codec_dai_node) { struct of_phandle_args args;
dev_err(dev, "audio-codec property invalid or missing\n");
ret = -EINVAL; ret = of_parse_phandle_with_args(dev->of_node, "i2s-controller",
goto cpu_dai_node_put; cells_name, i, &args);
if (!args.np) {
dev_err(dev, "i2s-controller property parse error: %d\n", i);
ret = -EINVAL;
goto dai_node_put;
}
cpu_dai_node[i] = args.np;
codec_dai_node[i] = of_parse_phandle(dev->of_node,
"audio-codec", i);
if (!codec_dai_node[i]) {
dev_err(dev, "audio-codec property parse error\n");
ret = -EINVAL;
goto dai_node_put;
}
} }
/* Initialize WM5110 - I2S and HDMI - I2S1 DAI links */
for (i = 0; i < card->num_links; i++) { for (i = 0; i < card->num_links; i++) {
unsigned int dai_index = 0; /* WM5110 */
card->dai_link[i].cpu_name = NULL; card->dai_link[i].cpu_name = NULL;
card->dai_link[i].platform_name = NULL; card->dai_link[i].platform_name = NULL;
card->dai_link[i].codec_of_node = codec_dai_node;
card->dai_link[i].cpu_of_node = cpu_dai_node; if (num_codecs > 1 && i == card->num_links - 1)
card->dai_link[i].platform_of_node = cpu_dai_node; dai_index = 1; /* HDMI */
card->dai_link[i].codec_of_node = codec_dai_node[dai_index];
card->dai_link[i].cpu_of_node = cpu_dai_node[dai_index];
card->dai_link[i].platform_of_node = cpu_dai_node[dai_index];
}
if (num_codecs > 1) {
/* HDMI DAI link (I2S1) */
i = card->num_links - 1;
ret = of_parse_phandle_with_fixed_args(dev->of_node,
"audio-codec", 0, 1, &args);
if (ret) {
dev_err(dev, "audio-codec property parse error\n");
goto dai_node_put;
}
ret = snd_soc_get_dai_name(&args, &card->dai_link[i].codec_dai_name);
if (ret) {
dev_err(dev, "Unable to get codec_dai_name\n");
goto dai_node_put;
}
} }
ret = devm_snd_soc_register_component(dev, &tm2_component, ret = devm_snd_soc_register_component(dev, &tm2_component,
tm2_ext_dai, ARRAY_SIZE(tm2_ext_dai)); tm2_ext_dai, ARRAY_SIZE(tm2_ext_dai));
if (ret < 0) { if (ret < 0) {
dev_err(dev, "Failed to register component: %d\n", ret); dev_err(dev, "Failed to register component: %d\n", ret);
goto codec_dai_node_put; goto dai_node_put;
} }
ret = devm_snd_soc_register_card(dev, card); ret = devm_snd_soc_register_card(dev, card);
if (ret < 0) { if (ret < 0) {
dev_err(dev, "Failed to register card: %d\n", ret); dev_err(dev, "Failed to register card: %d\n", ret);
goto codec_dai_node_put; goto dai_node_put;
}
dai_node_put:
for (i = 0; i < num_codecs; i++) {
of_node_put(codec_dai_node[i]);
of_node_put(cpu_dai_node[i]);
} }
codec_dai_node_put:
of_node_put(codec_dai_node);
cpu_dai_node_put:
of_node_put(cpu_dai_node);
amp_node_put:
of_node_put(card->aux_dev[0].codec_of_node); of_node_put(card->aux_dev[0].codec_of_node);
return ret; return ret;
} }

View File

@ -4356,6 +4356,26 @@ int snd_soc_of_get_dai_name(struct device_node *of_node,
} }
EXPORT_SYMBOL_GPL(snd_soc_of_get_dai_name); EXPORT_SYMBOL_GPL(snd_soc_of_get_dai_name);
/*
* snd_soc_of_put_dai_link_codecs - Dereference device nodes in the codecs array
* @dai_link: DAI link
*
* Dereference device nodes acquired by snd_soc_of_get_dai_link_codecs().
*/
void snd_soc_of_put_dai_link_codecs(struct snd_soc_dai_link *dai_link)
{
struct snd_soc_dai_link_component *component = dai_link->codecs;
int index;
for (index = 0; index < dai_link->num_codecs; index++, component++) {
if (!component->of_node)
break;
of_node_put(component->of_node);
component->of_node = NULL;
}
}
EXPORT_SYMBOL_GPL(snd_soc_of_put_dai_link_codecs);
/* /*
* snd_soc_of_get_dai_link_codecs - Parse a list of CODECs in the devicetree * snd_soc_of_get_dai_link_codecs - Parse a list of CODECs in the devicetree
* @dev: Card device * @dev: Card device
@ -4365,7 +4385,8 @@ EXPORT_SYMBOL_GPL(snd_soc_of_get_dai_name);
* Builds an array of CODEC DAI components from the DAI link property * Builds an array of CODEC DAI components from the DAI link property
* 'sound-dai'. * 'sound-dai'.
* The array is set in the DAI link and the number of DAIs is set accordingly. * The array is set in the DAI link and the number of DAIs is set accordingly.
* The device nodes in the array (of_node) must be dereferenced by the caller. * The device nodes in the array (of_node) must be dereferenced by calling
* snd_soc_of_put_dai_link_codecs() on @dai_link.
* *
* Returns 0 for success * Returns 0 for success
*/ */
@ -4413,14 +4434,7 @@ int snd_soc_of_get_dai_link_codecs(struct device *dev,
} }
return 0; return 0;
err: err:
for (index = 0, component = dai_link->codecs; snd_soc_of_put_dai_link_codecs(dai_link);
index < dai_link->num_codecs;
index++, component++) {
if (!component->of_node)
break;
of_node_put(component->of_node);
component->of_node = NULL;
}
dai_link->codecs = NULL; dai_link->codecs = NULL;
dai_link->num_codecs = 0; dai_link->num_codecs = 0;
return ret; return ret;