Commit 7650862f authored by Hans de Goede's avatar Hans de Goede Committed by Mark Brown

ASoC: Intel: bytcht_es8316: Determine quirks/routing with codec-dev ACPI DSM

Add support for querying the same ACPI Device-Specific-Method (DSM) as
Windows uses to determine things like speaker and mic routing.

This avoids the need to add DMI quirks for each new ESS8316 tablet model.

This has been tested on the following devices:

1. Chuwi Hi12 CHT with stereo speakers and IN2-mic-map,
   this avoids the need to add a DMI quirk for this model.

2. Nanote UMPC-01 CHT with stereo speakers and IN1-mic-map,
   the existing DMI quirk is still necessary because of a bug
   in the DSM return values for the speakers (it returns mono).

3. Onda V80 plus CHT with mono speaker and IN1-mic-map,
   DSM set quirks match the previously used defaults.

4. GP-electronic T701 BYT with mono speaker and IN2-mic-map,
   DSM set quirks match the previously used defaults.
Signed-off-by: default avatarHans de Goede <hdegoede@redhat.com>
Reviewed-by: default avatarPierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Link: https://lore.kernel.org/r/20231202123946.54347-5-hdegoede@redhat.comSigned-off-by: default avatarMark Brown <broonie@kernel.org>
parent e8acf91a
...@@ -462,6 +462,66 @@ static const struct dmi_system_id byt_cht_es8316_quirk_table[] = { ...@@ -462,6 +462,66 @@ static const struct dmi_system_id byt_cht_es8316_quirk_table[] = {
{} {}
}; };
static int byt_cht_es8316_get_quirks_from_dsm(struct byt_cht_es8316_private *priv,
bool is_bytcr)
{
int ret, val1, val2, dsm_quirk = 0;
if (is_bytcr)
dsm_quirk |= BYT_CHT_ES8316_SSP0;
ret = es83xx_dsm(priv->codec_dev, PLATFORM_MAINMIC_TYPE_ARG, &val1);
if (ret < 0)
return ret;
ret = es83xx_dsm(priv->codec_dev, PLATFORM_HPMIC_TYPE_ARG, &val2);
if (ret < 0)
return ret;
if (val1 == PLATFORM_MIC_AMIC_LIN1RIN1 && val2 == PLATFORM_MIC_AMIC_LIN2RIN2) {
dsm_quirk |= BYT_CHT_ES8316_INTMIC_IN1_MAP;
} else if (val1 == PLATFORM_MIC_AMIC_LIN2RIN2 && val2 == PLATFORM_MIC_AMIC_LIN1RIN1) {
dsm_quirk |= BYT_CHT_ES8316_INTMIC_IN2_MAP;
} else {
dev_warn(priv->codec_dev, "Unknown mic settings mainmic 0x%02x hpmic 0x%02x\n",
val1, val2);
return -EINVAL;
}
ret = es83xx_dsm(priv->codec_dev, PLATFORM_SPK_TYPE_ARG, &val1);
if (ret < 0)
return ret;
switch (val1) {
case PLATFORM_SPK_MONO:
dsm_quirk |= BYT_CHT_ES8316_MONO_SPEAKER;
break;
case PLATFORM_SPK_STEREO:
break;
default:
dev_warn(priv->codec_dev, "Unknown speaker setting 0x%02x\n", val1);
return -EINVAL;
}
ret = es83xx_dsm(priv->codec_dev, PLATFORM_HPDET_INV_ARG, &val1);
if (ret < 0)
return ret;
switch (val1) {
case PLATFORM_HPDET_NORMAL:
break;
case PLATFORM_HPDET_INVERTED:
dsm_quirk |= BYT_CHT_ES8316_JD_INVERTED;
break;
default:
dev_warn(priv->codec_dev, "Unknown hpdet-inv setting 0x%02x\n", val1);
return -EINVAL;
}
quirk = dsm_quirk;
return 0;
}
static int snd_byt_cht_es8316_mc_probe(struct platform_device *pdev) static int snd_byt_cht_es8316_mc_probe(struct platform_device *pdev)
{ {
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
...@@ -528,6 +588,8 @@ static int snd_byt_cht_es8316_mc_probe(struct platform_device *pdev) ...@@ -528,6 +588,8 @@ static int snd_byt_cht_es8316_mc_probe(struct platform_device *pdev)
dmi_id = dmi_first_match(byt_cht_es8316_quirk_table); dmi_id = dmi_first_match(byt_cht_es8316_quirk_table);
if (dmi_id) { if (dmi_id) {
quirk = (unsigned long)dmi_id->driver_data; quirk = (unsigned long)dmi_id->driver_data;
} else if (!byt_cht_es8316_get_quirks_from_dsm(priv, is_bytcr)) {
dev_info(dev, "Using ACPI DSM info for quirks\n");
} else if (is_bytcr) { } else if (is_bytcr) {
/* On BYTCR default to SSP0, internal-mic-in2-map, mono-spk */ /* On BYTCR default to SSP0, internal-mic-in2-map, mono-spk */
quirk = BYT_CHT_ES8316_SSP0 | BYT_CHT_ES8316_INTMIC_IN2_MAP | quirk = BYT_CHT_ES8316_SSP0 | BYT_CHT_ES8316_INTMIC_IN2_MAP |
......
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