ASoC: rt5670: add rt5672 codec support
authorBard Liao <bardliao@realtek.com>
Tue, 11 Nov 2014 09:59:50 +0000 (17:59 +0800)
committerMark Brown <broonie@kernel.org>
Tue, 11 Nov 2014 12:07:13 +0000 (12:07 +0000)
rt5672 is very similar to rt5670. Therefore we use one codec driver
to support both codecs. The difference between rt5670 and rt5672 is
there is some difference in their dapm routing table.

Signed-off-by: Bard Liao <bardliao@realtek.com>
Signed-off-by: Mark Brown <broonie@kernel.org>
sound/soc/codecs/rt5670.c
sound/soc/codecs/rt5670.h

index ba9d9b4d485783c80efa30e13e3218f1fec7742e..066b58317c24ac38a29ad039f968eacaaec16025 100644 (file)
@@ -1595,29 +1595,40 @@ static const struct snd_soc_dapm_widget rt5670_dapm_widgets[] = {
        /* PDM */
        SND_SOC_DAPM_SUPPLY("PDM1 Power", RT5670_PWR_DIG2,
                RT5670_PWR_PDM1_BIT, 0, NULL, 0),
-       SND_SOC_DAPM_SUPPLY("PDM2 Power", RT5670_PWR_DIG2,
-               RT5670_PWR_PDM2_BIT, 0, NULL, 0),
 
        SND_SOC_DAPM_MUX("PDM1 L Mux", RT5670_PDM_OUT_CTRL,
                         RT5670_M_PDM1_L_SFT, 1, &rt5670_pdm1_l_mux),
        SND_SOC_DAPM_MUX("PDM1 R Mux", RT5670_PDM_OUT_CTRL,
                         RT5670_M_PDM1_R_SFT, 1, &rt5670_pdm1_r_mux),
-       SND_SOC_DAPM_MUX("PDM2 L Mux", RT5670_PDM_OUT_CTRL,
-                        RT5670_M_PDM2_L_SFT, 1, &rt5670_pdm2_l_mux),
-       SND_SOC_DAPM_MUX("PDM2 R Mux", RT5670_PDM_OUT_CTRL,
-                        RT5670_M_PDM2_R_SFT, 1, &rt5670_pdm2_r_mux),
 
        /* Output Lines */
        SND_SOC_DAPM_OUTPUT("HPOL"),
        SND_SOC_DAPM_OUTPUT("HPOR"),
        SND_SOC_DAPM_OUTPUT("LOUTL"),
        SND_SOC_DAPM_OUTPUT("LOUTR"),
+};
+
+static const struct snd_soc_dapm_widget rt5670_specific_dapm_widgets[] = {
+       SND_SOC_DAPM_SUPPLY("PDM2 Power", RT5670_PWR_DIG2,
+               RT5670_PWR_PDM2_BIT, 0, NULL, 0),
+       SND_SOC_DAPM_MUX("PDM2 L Mux", RT5670_PDM_OUT_CTRL,
+                        RT5670_M_PDM2_L_SFT, 1, &rt5670_pdm2_l_mux),
+       SND_SOC_DAPM_MUX("PDM2 R Mux", RT5670_PDM_OUT_CTRL,
+                        RT5670_M_PDM2_R_SFT, 1, &rt5670_pdm2_r_mux),
        SND_SOC_DAPM_OUTPUT("PDM1L"),
        SND_SOC_DAPM_OUTPUT("PDM1R"),
        SND_SOC_DAPM_OUTPUT("PDM2L"),
        SND_SOC_DAPM_OUTPUT("PDM2R"),
 };
 
+static const struct snd_soc_dapm_widget rt5672_specific_dapm_widgets[] = {
+       SND_SOC_DAPM_PGA("SPO Amp", SND_SOC_NOPM, 0, 0, NULL, 0),
+       SND_SOC_DAPM_OUTPUT("SPOLP"),
+       SND_SOC_DAPM_OUTPUT("SPOLN"),
+       SND_SOC_DAPM_OUTPUT("SPORP"),
+       SND_SOC_DAPM_OUTPUT("SPORN"),
+};
+
 static const struct snd_soc_dapm_route rt5670_dapm_routes[] = {
        { "ADC Stereo1 Filter", NULL, "ADC STO1 ASRC", is_using_asrc },
        { "ADC Stereo2 Filter", NULL, "ADC STO2 ASRC", is_using_asrc },
@@ -1970,12 +1981,6 @@ static const struct snd_soc_dapm_route rt5670_dapm_routes[] = {
        { "PDM1 R Mux", "Stereo DAC", "Stereo DAC MIXR" },
        { "PDM1 R Mux", "Mono DAC", "Mono DAC MIXR" },
        { "PDM1 R Mux", NULL, "PDM1 Power" },
-       { "PDM2 L Mux", "Stereo DAC", "Stereo DAC MIXL" },
-       { "PDM2 L Mux", "Mono DAC", "Mono DAC MIXL" },
-       { "PDM2 L Mux", NULL, "PDM2 Power" },
-       { "PDM2 R Mux", "Stereo DAC", "Stereo DAC MIXR" },
-       { "PDM2 R Mux", "Mono DAC", "Mono DAC MIXR" },
-       { "PDM2 R Mux", NULL, "PDM2 Power" },
 
        { "HP Amp", NULL, "HPO MIX" },
        { "HP Amp", NULL, "Mic Det Power" },
@@ -1993,13 +1998,30 @@ static const struct snd_soc_dapm_route rt5670_dapm_routes[] = {
        { "LOUTR", NULL, "LOUT R Playback" },
        { "LOUTL", NULL, "Improve HP Amp Drv" },
        { "LOUTR", NULL, "Improve HP Amp Drv" },
+};
 
+static const struct snd_soc_dapm_route rt5670_specific_dapm_routes[] = {
+       { "PDM2 L Mux", "Stereo DAC", "Stereo DAC MIXL" },
+       { "PDM2 L Mux", "Mono DAC", "Mono DAC MIXL" },
+       { "PDM2 L Mux", NULL, "PDM2 Power" },
+       { "PDM2 R Mux", "Stereo DAC", "Stereo DAC MIXR" },
+       { "PDM2 R Mux", "Mono DAC", "Mono DAC MIXR" },
+       { "PDM2 R Mux", NULL, "PDM2 Power" },
        { "PDM1L", NULL, "PDM1 L Mux" },
        { "PDM1R", NULL, "PDM1 R Mux" },
        { "PDM2L", NULL, "PDM2 L Mux" },
        { "PDM2R", NULL, "PDM2 R Mux" },
 };
 
+static const struct snd_soc_dapm_route rt5672_specific_dapm_routes[] = {
+       { "SPO Amp", NULL, "PDM1 L Mux" },
+       { "SPO Amp", NULL, "PDM1 R Mux" },
+       { "SPOLP", NULL, "SPO Amp" },
+       { "SPOLN", NULL, "SPO Amp" },
+       { "SPORP", NULL, "SPO Amp" },
+       { "SPORN", NULL, "SPO Amp" },
+};
+
 static int rt5670_hw_params(struct snd_pcm_substream *substream,
        struct snd_pcm_hw_params *params, struct snd_soc_dai *dai)
 {
@@ -2331,6 +2353,29 @@ static int rt5670_probe(struct snd_soc_codec *codec)
 {
        struct rt5670_priv *rt5670 = snd_soc_codec_get_drvdata(codec);
 
+       switch (snd_soc_read(codec, RT5670_RESET) & RT5670_ID_MASK) {
+       case RT5670_ID_5670:
+       case RT5670_ID_5671:
+               snd_soc_dapm_new_controls(&codec->dapm,
+                       rt5670_specific_dapm_widgets,
+                       ARRAY_SIZE(rt5670_specific_dapm_widgets));
+               snd_soc_dapm_add_routes(&codec->dapm,
+                       rt5670_specific_dapm_routes,
+                       ARRAY_SIZE(rt5670_specific_dapm_routes));
+               break;
+       case RT5670_ID_5672:
+               snd_soc_dapm_new_controls(&codec->dapm,
+                       rt5672_specific_dapm_widgets,
+                       ARRAY_SIZE(rt5672_specific_dapm_widgets));
+               snd_soc_dapm_add_routes(&codec->dapm,
+                       rt5672_specific_dapm_routes,
+                       ARRAY_SIZE(rt5672_specific_dapm_routes));
+               break;
+       default:
+               dev_err(codec->dev,
+                       "The driver is for RT5670 RT5671 or RT5672 only\n");
+               return -ENODEV;
+       }
        rt5670->codec = codec;
 
        return 0;
@@ -2452,6 +2497,8 @@ static const struct regmap_config rt5670_regmap = {
 
 static const struct i2c_device_id rt5670_i2c_id[] = {
        { "rt5670", 0 },
+       { "rt5671", 0 },
+       { "rt5672", 0 },
        { }
 };
 MODULE_DEVICE_TABLE(i2c, rt5670_i2c_id);
index a0b5c855b49291f9b649a83056071fe06181a584..d11b9c207e26799ba6cbb85cd3979ace7c76ea4d 100644 (file)
 #define RT5670_R_VOL_MASK                      (0x3f)
 #define RT5670_R_VOL_SFT                       0
 
+/* SW Reset & Device ID (0x00) */
+#define RT5670_ID_MASK                         (0x3 << 1)
+#define RT5670_ID_5670                         (0x0 << 1)
+#define RT5670_ID_5672                         (0x1 << 1)
+#define RT5670_ID_5671                         (0x2 << 1)
+
 /* Combo Jack Control 1 (0x0a) */
 #define RT5670_CBJ_BST1_MASK                   (0xf << 12)
 #define RT5670_CBJ_BST1_SFT                    (12)