Path: blob/master/sound/soc/intel/boards/bytcr_rt5651.c
29268 views
// SPDX-License-Identifier: GPL-2.0-only1/*2* bytcr_rt5651.c - ASoc Machine driver for Intel Byt CR platform3* (derived from bytcr_rt5640.c)4*5* Copyright (C) 2015 Intel Corp6* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~7*8* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~9*/1011#include <linux/init.h>12#include <linux/i2c.h>13#include <linux/module.h>14#include <linux/platform_device.h>15#include <linux/property.h>16#include <linux/acpi.h>17#include <linux/clk.h>18#include <linux/device.h>19#include <linux/dmi.h>20#include <linux/input.h>21#include <linux/gpio/consumer.h>22#include <linux/gpio/machine.h>23#include <linux/slab.h>24#include <sound/pcm.h>25#include <sound/pcm_params.h>26#include <sound/soc.h>27#include <sound/jack.h>28#include <sound/soc-acpi.h>29#include "../../codecs/rt5651.h"30#include "../atom/sst-atom-controls.h"31#include "../common/soc-intel-quirks.h"3233enum {34BYT_RT5651_DMIC_MAP,35BYT_RT5651_IN1_MAP,36BYT_RT5651_IN2_MAP,37BYT_RT5651_IN1_IN2_MAP,38};3940enum {41BYT_RT5651_JD_NULL = (RT5651_JD_NULL << 4),42BYT_RT5651_JD1_1 = (RT5651_JD1_1 << 4),43BYT_RT5651_JD1_2 = (RT5651_JD1_2 << 4),44BYT_RT5651_JD2 = (RT5651_JD2 << 4),45};4647enum {48BYT_RT5651_OVCD_TH_600UA = (6 << 8),49BYT_RT5651_OVCD_TH_1500UA = (15 << 8),50BYT_RT5651_OVCD_TH_2000UA = (20 << 8),51};5253enum {54BYT_RT5651_OVCD_SF_0P5 = (RT5651_OVCD_SF_0P5 << 13),55BYT_RT5651_OVCD_SF_0P75 = (RT5651_OVCD_SF_0P75 << 13),56BYT_RT5651_OVCD_SF_1P0 = (RT5651_OVCD_SF_1P0 << 13),57BYT_RT5651_OVCD_SF_1P5 = (RT5651_OVCD_SF_1P5 << 13),58};5960#define BYT_RT5651_MAP_MASK GENMASK(3, 0)61#define BYT_RT5651_MAP(quirk) ((quirk) & BYT_RT5651_MAP_MASK)62#define BYT_RT5651_JDSRC(quirk) (((quirk) & GENMASK(7, 4)) >> 4)63#define BYT_RT5651_OVCD_TH(quirk) (((quirk) & GENMASK(12, 8)) >> 8)64#define BYT_RT5651_OVCD_SF(quirk) (((quirk) & GENMASK(14, 13)) >> 13)65#define BYT_RT5651_DMIC_EN BIT(16)66#define BYT_RT5651_MCLK_EN BIT(17)67#define BYT_RT5651_MCLK_25MHZ BIT(18)68#define BYT_RT5651_SSP2_AIF2 BIT(19) /* default is using AIF1 */69#define BYT_RT5651_SSP0_AIF1 BIT(20)70#define BYT_RT5651_SSP0_AIF2 BIT(21)71#define BYT_RT5651_HP_LR_SWAPPED BIT(22)72#define BYT_RT5651_MONO_SPEAKER BIT(23)73#define BYT_RT5651_JD_NOT_INV BIT(24)7475#define BYT_RT5651_DEFAULT_QUIRKS (BYT_RT5651_MCLK_EN | \76BYT_RT5651_JD1_1 | \77BYT_RT5651_OVCD_TH_2000UA | \78BYT_RT5651_OVCD_SF_0P75)7980/* jack-detect-source + inv + dmic-en + ovcd-th + -sf + terminating entry */81#define MAX_NO_PROPS 68283struct byt_rt5651_private {84struct clk *mclk;85struct gpio_desc *ext_amp_gpio;86struct gpio_desc *hp_detect;87struct snd_soc_jack jack;88struct device *codec_dev;89};9091static const struct acpi_gpio_mapping *byt_rt5651_gpios;9293/* Default: jack-detect on JD1_1, internal mic on in2, headsetmic on in3 */94static unsigned long byt_rt5651_quirk = BYT_RT5651_DEFAULT_QUIRKS |95BYT_RT5651_IN2_MAP;9697static int quirk_override = -1;98module_param_named(quirk, quirk_override, int, 0444);99MODULE_PARM_DESC(quirk, "Board-specific quirk override");100101static void log_quirks(struct device *dev)102{103int map;104105map = BYT_RT5651_MAP(byt_rt5651_quirk);106switch (map) {107case BYT_RT5651_DMIC_MAP:108dev_info(dev, "quirk DMIC_MAP enabled");109break;110case BYT_RT5651_IN1_MAP:111dev_info(dev, "quirk IN1_MAP enabled");112break;113case BYT_RT5651_IN2_MAP:114dev_info(dev, "quirk IN2_MAP enabled");115break;116case BYT_RT5651_IN1_IN2_MAP:117dev_info(dev, "quirk IN1_IN2_MAP enabled");118break;119default:120dev_warn_once(dev, "quirk sets invalid input map: 0x%x, default to DMIC_MAP\n", map);121byt_rt5651_quirk &= ~BYT_RT5651_MAP_MASK;122byt_rt5651_quirk |= BYT_RT5651_DMIC_MAP;123break;124}125126if (BYT_RT5651_JDSRC(byt_rt5651_quirk)) {127dev_info(dev, "quirk realtek,jack-detect-source %ld\n",128BYT_RT5651_JDSRC(byt_rt5651_quirk));129dev_info(dev, "quirk realtek,over-current-threshold-microamp %ld\n",130BYT_RT5651_OVCD_TH(byt_rt5651_quirk) * 100);131dev_info(dev, "quirk realtek,over-current-scale-factor %ld\n",132BYT_RT5651_OVCD_SF(byt_rt5651_quirk));133}134if (byt_rt5651_quirk & BYT_RT5651_DMIC_EN)135dev_info(dev, "quirk DMIC enabled");136if (byt_rt5651_quirk & BYT_RT5651_MCLK_EN)137dev_info(dev, "quirk MCLK_EN enabled");138if (byt_rt5651_quirk & BYT_RT5651_MCLK_25MHZ)139dev_info(dev, "quirk MCLK_25MHZ enabled");140if (byt_rt5651_quirk & BYT_RT5651_SSP2_AIF2)141dev_info(dev, "quirk SSP2_AIF2 enabled\n");142if (byt_rt5651_quirk & BYT_RT5651_SSP0_AIF1)143dev_info(dev, "quirk SSP0_AIF1 enabled\n");144if (byt_rt5651_quirk & BYT_RT5651_SSP0_AIF2)145dev_info(dev, "quirk SSP0_AIF2 enabled\n");146if (byt_rt5651_quirk & BYT_RT5651_MONO_SPEAKER)147dev_info(dev, "quirk MONO_SPEAKER enabled\n");148if (byt_rt5651_quirk & BYT_RT5651_JD_NOT_INV)149dev_info(dev, "quirk JD_NOT_INV enabled\n");150}151152#define BYT_CODEC_DAI1 "rt5651-aif1"153#define BYT_CODEC_DAI2 "rt5651-aif2"154155static int byt_rt5651_prepare_and_enable_pll1(struct snd_soc_dai *codec_dai,156int rate, int bclk_ratio)157{158int clk_id, clk_freq, ret;159160/* Configure the PLL before selecting it */161if (!(byt_rt5651_quirk & BYT_RT5651_MCLK_EN)) {162clk_id = RT5651_PLL1_S_BCLK1;163clk_freq = rate * bclk_ratio;164} else {165clk_id = RT5651_PLL1_S_MCLK;166if (byt_rt5651_quirk & BYT_RT5651_MCLK_25MHZ)167clk_freq = 25000000;168else169clk_freq = 19200000;170}171ret = snd_soc_dai_set_pll(codec_dai, 0, clk_id, clk_freq, rate * 512);172if (ret < 0) {173dev_err(codec_dai->component->dev, "can't set pll: %d\n", ret);174return ret;175}176177ret = snd_soc_dai_set_sysclk(codec_dai, RT5651_SCLK_S_PLL1,178rate * 512, SND_SOC_CLOCK_IN);179if (ret < 0) {180dev_err(codec_dai->component->dev, "can't set clock %d\n", ret);181return ret;182}183184return 0;185}186187static int platform_clock_control(struct snd_soc_dapm_widget *w,188struct snd_kcontrol *k, int event)189{190struct snd_soc_dapm_context *dapm = w->dapm;191struct snd_soc_card *card = dapm->card;192struct snd_soc_dai *codec_dai;193struct byt_rt5651_private *priv = snd_soc_card_get_drvdata(card);194int ret;195196codec_dai = snd_soc_card_get_codec_dai(card, BYT_CODEC_DAI1);197if (!codec_dai)198codec_dai = snd_soc_card_get_codec_dai(card, BYT_CODEC_DAI2);199if (!codec_dai) {200dev_err(card->dev,201"Codec dai not found; Unable to set platform clock\n");202return -EIO;203}204205if (SND_SOC_DAPM_EVENT_ON(event)) {206ret = clk_prepare_enable(priv->mclk);207if (ret < 0) {208dev_err(card->dev, "could not configure MCLK state");209return ret;210}211ret = byt_rt5651_prepare_and_enable_pll1(codec_dai, 48000, 50);212} else {213/*214* Set codec clock source to internal clock before215* turning off the platform clock. Codec needs clock216* for Jack detection and button press217*/218ret = snd_soc_dai_set_sysclk(codec_dai, RT5651_SCLK_S_RCCLK,21948000 * 512,220SND_SOC_CLOCK_IN);221if (!ret)222clk_disable_unprepare(priv->mclk);223}224225if (ret < 0) {226dev_err(card->dev, "can't set codec sysclk: %d\n", ret);227return ret;228}229230return 0;231}232233static int rt5651_ext_amp_power_event(struct snd_soc_dapm_widget *w,234struct snd_kcontrol *kcontrol, int event)235{236struct snd_soc_card *card = w->dapm->card;237struct byt_rt5651_private *priv = snd_soc_card_get_drvdata(card);238239if (SND_SOC_DAPM_EVENT_ON(event))240gpiod_set_value_cansleep(priv->ext_amp_gpio, 1);241else242gpiod_set_value_cansleep(priv->ext_amp_gpio, 0);243244return 0;245}246247static const struct snd_soc_dapm_widget byt_rt5651_widgets[] = {248SND_SOC_DAPM_HP("Headphone", NULL),249SND_SOC_DAPM_MIC("Headset Mic", NULL),250SND_SOC_DAPM_MIC("Internal Mic", NULL),251SND_SOC_DAPM_SPK("Speaker", NULL),252SND_SOC_DAPM_LINE("Line In", NULL),253SND_SOC_DAPM_SUPPLY("Platform Clock", SND_SOC_NOPM, 0, 0,254platform_clock_control, SND_SOC_DAPM_PRE_PMU |255SND_SOC_DAPM_POST_PMD),256SND_SOC_DAPM_SUPPLY("Ext Amp Power", SND_SOC_NOPM, 0, 0,257rt5651_ext_amp_power_event,258SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMU),259};260261static const struct snd_soc_dapm_route byt_rt5651_audio_map[] = {262{"Headphone", NULL, "Platform Clock"},263{"Headset Mic", NULL, "Platform Clock"},264{"Internal Mic", NULL, "Platform Clock"},265{"Speaker", NULL, "Platform Clock"},266{"Speaker", NULL, "Ext Amp Power"},267{"Line In", NULL, "Platform Clock"},268269{"Headset Mic", NULL, "micbias1"}, /* lowercase for rt5651 */270{"Headphone", NULL, "HPOL"},271{"Headphone", NULL, "HPOR"},272{"Speaker", NULL, "LOUTL"},273{"Speaker", NULL, "LOUTR"},274{"IN2P", NULL, "Line In"},275{"IN2N", NULL, "Line In"},276277};278279static const struct snd_soc_dapm_route byt_rt5651_intmic_dmic_map[] = {280{"DMIC L1", NULL, "Internal Mic"},281{"DMIC R1", NULL, "Internal Mic"},282{"IN2P", NULL, "Headset Mic"},283};284285static const struct snd_soc_dapm_route byt_rt5651_intmic_in1_map[] = {286{"Internal Mic", NULL, "micbias1"},287{"IN1P", NULL, "Internal Mic"},288{"IN3P", NULL, "Headset Mic"},289};290291static const struct snd_soc_dapm_route byt_rt5651_intmic_in2_map[] = {292{"Internal Mic", NULL, "micbias1"},293{"IN2P", NULL, "Internal Mic"},294{"IN3P", NULL, "Headset Mic"},295};296297static const struct snd_soc_dapm_route byt_rt5651_intmic_in1_in2_map[] = {298{"Internal Mic", NULL, "micbias1"},299{"IN1P", NULL, "Internal Mic"},300{"IN2P", NULL, "Internal Mic"},301{"IN3P", NULL, "Headset Mic"},302};303304static const struct snd_soc_dapm_route byt_rt5651_ssp0_aif1_map[] = {305{"ssp0 Tx", NULL, "modem_out"},306{"modem_in", NULL, "ssp0 Rx"},307308{"AIF1 Playback", NULL, "ssp0 Tx"},309{"ssp0 Rx", NULL, "AIF1 Capture"},310};311312static const struct snd_soc_dapm_route byt_rt5651_ssp0_aif2_map[] = {313{"ssp0 Tx", NULL, "modem_out"},314{"modem_in", NULL, "ssp0 Rx"},315316{"AIF2 Playback", NULL, "ssp0 Tx"},317{"ssp0 Rx", NULL, "AIF2 Capture"},318};319320static const struct snd_soc_dapm_route byt_rt5651_ssp2_aif1_map[] = {321{"ssp2 Tx", NULL, "codec_out0"},322{"ssp2 Tx", NULL, "codec_out1"},323{"codec_in0", NULL, "ssp2 Rx"},324{"codec_in1", NULL, "ssp2 Rx"},325326{"AIF1 Playback", NULL, "ssp2 Tx"},327{"ssp2 Rx", NULL, "AIF1 Capture"},328};329330static const struct snd_soc_dapm_route byt_rt5651_ssp2_aif2_map[] = {331{"ssp2 Tx", NULL, "codec_out0"},332{"ssp2 Tx", NULL, "codec_out1"},333{"codec_in0", NULL, "ssp2 Rx"},334{"codec_in1", NULL, "ssp2 Rx"},335336{"AIF2 Playback", NULL, "ssp2 Tx"},337{"ssp2 Rx", NULL, "AIF2 Capture"},338};339340static const struct snd_kcontrol_new byt_rt5651_controls[] = {341SOC_DAPM_PIN_SWITCH("Headphone"),342SOC_DAPM_PIN_SWITCH("Headset Mic"),343SOC_DAPM_PIN_SWITCH("Internal Mic"),344SOC_DAPM_PIN_SWITCH("Speaker"),345SOC_DAPM_PIN_SWITCH("Line In"),346};347348static struct snd_soc_jack_pin bytcr_jack_pins[] = {349{350.pin = "Headphone",351.mask = SND_JACK_HEADPHONE,352},353{354.pin = "Headset Mic",355.mask = SND_JACK_MICROPHONE,356},357};358359static int byt_rt5651_aif1_hw_params(struct snd_pcm_substream *substream,360struct snd_pcm_hw_params *params)361{362struct snd_soc_pcm_runtime *rtd = snd_soc_substream_to_rtd(substream);363struct snd_soc_dai *codec_dai = snd_soc_rtd_to_codec(rtd, 0);364snd_pcm_format_t format = params_format(params);365int rate = params_rate(params);366int bclk_ratio;367368if (format == SNDRV_PCM_FORMAT_S16_LE)369bclk_ratio = 32;370else371bclk_ratio = 50;372373return byt_rt5651_prepare_and_enable_pll1(codec_dai, rate, bclk_ratio);374}375376static const struct acpi_gpio_params pov_p1006w_hp_detect = { 1, 0, false };377static const struct acpi_gpio_params pov_p1006w_ext_amp_en = { 2, 0, true };378379static const struct acpi_gpio_mapping byt_rt5651_pov_p1006w_gpios[] = {380{ "hp-detect-gpios", &pov_p1006w_hp_detect, 1, },381{ "ext-amp-enable-gpios", &pov_p1006w_ext_amp_en, 1, },382{ },383};384385static int byt_rt5651_pov_p1006w_quirk_cb(const struct dmi_system_id *id)386{387byt_rt5651_quirk = (unsigned long)id->driver_data;388byt_rt5651_gpios = byt_rt5651_pov_p1006w_gpios;389return 1;390}391392static int byt_rt5651_quirk_cb(const struct dmi_system_id *id)393{394byt_rt5651_quirk = (unsigned long)id->driver_data;395return 1;396}397398static const struct dmi_system_id byt_rt5651_quirk_table[] = {399{400/* Chuwi Hi8 Pro (CWI513) */401.callback = byt_rt5651_quirk_cb,402.matches = {403DMI_MATCH(DMI_SYS_VENDOR, "Hampoo"),404DMI_MATCH(DMI_PRODUCT_NAME, "X1D3_C806N"),405},406.driver_data = (void *)(BYT_RT5651_DEFAULT_QUIRKS |407BYT_RT5651_IN2_MAP |408BYT_RT5651_HP_LR_SWAPPED |409BYT_RT5651_MONO_SPEAKER),410},411{412/* Chuwi Vi8 Plus (CWI519) */413.callback = byt_rt5651_quirk_cb,414.matches = {415DMI_MATCH(DMI_SYS_VENDOR, "Hampoo"),416DMI_MATCH(DMI_PRODUCT_NAME, "D2D3_Vi8A1"),417},418.driver_data = (void *)(BYT_RT5651_DEFAULT_QUIRKS |419BYT_RT5651_IN2_MAP |420BYT_RT5651_HP_LR_SWAPPED |421BYT_RT5651_MONO_SPEAKER),422},423{424/* Complet Electro Serv MY8307 */425.callback = byt_rt5651_quirk_cb,426.matches = {427DMI_MATCH(DMI_SYS_VENDOR, "Complet Electro Serv"),428DMI_MATCH(DMI_PRODUCT_NAME, "MY8307"),429},430.driver_data = (void *)(BYT_RT5651_DEFAULT_QUIRKS |431BYT_RT5651_IN2_MAP |432BYT_RT5651_MONO_SPEAKER |433BYT_RT5651_JD_NOT_INV),434},435{436/* I.T.Works TW701, Ployer Momo7w and Trekstor ST70416-6437* (these all use the same mainboard) */438.callback = byt_rt5651_quirk_cb,439.matches = {440DMI_MATCH(DMI_BIOS_VENDOR, "INSYDE Corp."),441/* Partial match for all of itWORKS.G.WI71C.JGBMRBA,442* TREK.G.WI71C.JGBMRBA0x and MOMO.G.WI71C.MABMRBA02 */443DMI_MATCH(DMI_BIOS_VERSION, ".G.WI71C."),444},445.driver_data = (void *)(BYT_RT5651_DEFAULT_QUIRKS |446BYT_RT5651_IN2_MAP |447BYT_RT5651_SSP0_AIF1 |448BYT_RT5651_MONO_SPEAKER),449},450{451/* Jumper EZpad 7 */452.callback = byt_rt5651_quirk_cb,453.matches = {454DMI_MATCH(DMI_SYS_VENDOR, "Jumper"),455DMI_MATCH(DMI_PRODUCT_NAME, "EZpad"),456/* Jumper12x.WJ2012.bsBKRCP05 with the version dropped */457DMI_MATCH(DMI_BIOS_VERSION, "Jumper12x.WJ2012.bsBKRCP"),458},459.driver_data = (void *)(BYT_RT5651_DEFAULT_QUIRKS |460BYT_RT5651_IN2_MAP |461BYT_RT5651_JD_NOT_INV),462},463{464/* KIANO SlimNote 14.2 */465.callback = byt_rt5651_quirk_cb,466.matches = {467DMI_MATCH(DMI_SYS_VENDOR, "KIANO"),468DMI_MATCH(DMI_PRODUCT_NAME, "KIANO SlimNote 14.2"),469},470.driver_data = (void *)(BYT_RT5651_DEFAULT_QUIRKS |471BYT_RT5651_IN1_IN2_MAP),472},473{474/* Minnowboard Max B3 */475.callback = byt_rt5651_quirk_cb,476.matches = {477DMI_MATCH(DMI_SYS_VENDOR, "Circuitco"),478DMI_MATCH(DMI_PRODUCT_NAME, "Minnowboard Max B3 PLATFORM"),479},480.driver_data = (void *)(BYT_RT5651_IN1_MAP),481},482{483/* Minnowboard Turbot */484.callback = byt_rt5651_quirk_cb,485.matches = {486DMI_MATCH(DMI_SYS_VENDOR, "ADI"),487DMI_MATCH(DMI_PRODUCT_NAME, "Minnowboard Turbot"),488},489.driver_data = (void *)(BYT_RT5651_MCLK_EN |490BYT_RT5651_IN1_MAP),491},492{493/* Point of View mobii wintab p1006w (v1.0) */494.callback = byt_rt5651_pov_p1006w_quirk_cb,495.matches = {496DMI_EXACT_MATCH(DMI_SYS_VENDOR, "Insyde"),497DMI_EXACT_MATCH(DMI_PRODUCT_NAME, "BayTrail"),498/* Note 105b is Foxcon's USB/PCI vendor id */499DMI_EXACT_MATCH(DMI_BOARD_VENDOR, "105B"),500DMI_EXACT_MATCH(DMI_BOARD_NAME, "0E57"),501},502.driver_data = (void *)(BYT_RT5651_DMIC_MAP |503BYT_RT5651_OVCD_TH_2000UA |504BYT_RT5651_OVCD_SF_0P75 |505BYT_RT5651_DMIC_EN |506BYT_RT5651_MCLK_EN |507BYT_RT5651_SSP0_AIF1),508},509{510/* VIOS LTH17 */511.callback = byt_rt5651_quirk_cb,512.matches = {513DMI_MATCH(DMI_SYS_VENDOR, "VIOS"),514DMI_MATCH(DMI_PRODUCT_NAME, "LTH17"),515},516.driver_data = (void *)(BYT_RT5651_IN1_IN2_MAP |517BYT_RT5651_JD1_1 |518BYT_RT5651_OVCD_TH_2000UA |519BYT_RT5651_OVCD_SF_1P0 |520BYT_RT5651_MCLK_EN),521},522{523/* Yours Y8W81 (and others using the same mainboard) */524.callback = byt_rt5651_quirk_cb,525.matches = {526DMI_MATCH(DMI_BIOS_VENDOR, "INSYDE Corp."),527/* Partial match for all devs with a W86C mainboard */528DMI_MATCH(DMI_BIOS_VERSION, ".F.W86C."),529},530.driver_data = (void *)(BYT_RT5651_DEFAULT_QUIRKS |531BYT_RT5651_IN2_MAP |532BYT_RT5651_SSP0_AIF1 |533BYT_RT5651_MONO_SPEAKER),534},535{}536};537538/*539* Note this MUST be called before snd_soc_register_card(), so that the props540* are in place before the codec component driver's probe function parses them.541*/542static int byt_rt5651_add_codec_device_props(struct device *i2c_dev,543struct byt_rt5651_private *priv)544{545struct property_entry props[MAX_NO_PROPS] = {};546struct fwnode_handle *fwnode;547int cnt = 0;548int ret;549550props[cnt++] = PROPERTY_ENTRY_U32("realtek,jack-detect-source",551BYT_RT5651_JDSRC(byt_rt5651_quirk));552553props[cnt++] = PROPERTY_ENTRY_U32("realtek,over-current-threshold-microamp",554BYT_RT5651_OVCD_TH(byt_rt5651_quirk) * 100);555556props[cnt++] = PROPERTY_ENTRY_U32("realtek,over-current-scale-factor",557BYT_RT5651_OVCD_SF(byt_rt5651_quirk));558559if (byt_rt5651_quirk & BYT_RT5651_DMIC_EN)560props[cnt++] = PROPERTY_ENTRY_BOOL("realtek,dmic-en");561562if (byt_rt5651_quirk & BYT_RT5651_JD_NOT_INV)563props[cnt++] = PROPERTY_ENTRY_BOOL("realtek,jack-detect-not-inverted");564565fwnode = fwnode_create_software_node(props, NULL);566if (IS_ERR(fwnode)) {567/* put_device(i2c_dev) is handled in caller */568return PTR_ERR(fwnode);569}570571ret = device_add_software_node(i2c_dev, to_software_node(fwnode));572573fwnode_handle_put(fwnode);574575return ret;576}577578static int byt_rt5651_init(struct snd_soc_pcm_runtime *runtime)579{580struct snd_soc_card *card = runtime->card;581struct snd_soc_component *codec = snd_soc_rtd_to_codec(runtime, 0)->component;582struct byt_rt5651_private *priv = snd_soc_card_get_drvdata(card);583const struct snd_soc_dapm_route *custom_map;584int num_routes;585int report;586int ret;587588card->dapm.idle_bias = false;589590/* Start with RC clk for jack-detect (we disable MCLK below) */591if (byt_rt5651_quirk & BYT_RT5651_MCLK_EN)592snd_soc_component_update_bits(codec, RT5651_GLB_CLK,593RT5651_SCLK_SRC_MASK, RT5651_SCLK_SRC_RCCLK);594595switch (BYT_RT5651_MAP(byt_rt5651_quirk)) {596case BYT_RT5651_IN1_MAP:597custom_map = byt_rt5651_intmic_in1_map;598num_routes = ARRAY_SIZE(byt_rt5651_intmic_in1_map);599break;600case BYT_RT5651_IN2_MAP:601custom_map = byt_rt5651_intmic_in2_map;602num_routes = ARRAY_SIZE(byt_rt5651_intmic_in2_map);603break;604case BYT_RT5651_IN1_IN2_MAP:605custom_map = byt_rt5651_intmic_in1_in2_map;606num_routes = ARRAY_SIZE(byt_rt5651_intmic_in1_in2_map);607break;608default:609custom_map = byt_rt5651_intmic_dmic_map;610num_routes = ARRAY_SIZE(byt_rt5651_intmic_dmic_map);611}612ret = snd_soc_dapm_add_routes(&card->dapm, custom_map, num_routes);613if (ret)614return ret;615616if (byt_rt5651_quirk & BYT_RT5651_SSP2_AIF2) {617ret = snd_soc_dapm_add_routes(&card->dapm,618byt_rt5651_ssp2_aif2_map,619ARRAY_SIZE(byt_rt5651_ssp2_aif2_map));620} else if (byt_rt5651_quirk & BYT_RT5651_SSP0_AIF1) {621ret = snd_soc_dapm_add_routes(&card->dapm,622byt_rt5651_ssp0_aif1_map,623ARRAY_SIZE(byt_rt5651_ssp0_aif1_map));624} else if (byt_rt5651_quirk & BYT_RT5651_SSP0_AIF2) {625ret = snd_soc_dapm_add_routes(&card->dapm,626byt_rt5651_ssp0_aif2_map,627ARRAY_SIZE(byt_rt5651_ssp0_aif2_map));628} else {629ret = snd_soc_dapm_add_routes(&card->dapm,630byt_rt5651_ssp2_aif1_map,631ARRAY_SIZE(byt_rt5651_ssp2_aif1_map));632}633if (ret)634return ret;635636ret = snd_soc_add_card_controls(card, byt_rt5651_controls,637ARRAY_SIZE(byt_rt5651_controls));638if (ret) {639dev_err(card->dev, "unable to add card controls\n");640return ret;641}642643/*644* The firmware might enable the clock at boot (this information645* may or may not be reflected in the enable clock register).646* To change the rate we must disable the clock first to cover647* these cases. Due to common clock framework restrictions that648* do not allow to disable a clock that has not been enabled,649* we need to enable the clock first.650*/651ret = clk_prepare_enable(priv->mclk);652if (!ret)653clk_disable_unprepare(priv->mclk);654655if (byt_rt5651_quirk & BYT_RT5651_MCLK_25MHZ)656ret = clk_set_rate(priv->mclk, 25000000);657else658ret = clk_set_rate(priv->mclk, 19200000);659660if (ret)661dev_err(card->dev, "unable to set MCLK rate\n");662663report = 0;664if (BYT_RT5651_JDSRC(byt_rt5651_quirk))665report = SND_JACK_HEADSET | SND_JACK_BTN_0;666else if (priv->hp_detect)667report = SND_JACK_HEADSET;668669if (report) {670ret = snd_soc_card_jack_new_pins(runtime->card, "Headset",671report, &priv->jack,672bytcr_jack_pins,673ARRAY_SIZE(bytcr_jack_pins));674if (ret) {675dev_err(runtime->dev, "jack creation failed %d\n", ret);676return ret;677}678679if (report & SND_JACK_BTN_0)680snd_jack_set_key(priv->jack.jack, SND_JACK_BTN_0,681KEY_PLAYPAUSE);682683ret = snd_soc_component_set_jack(codec, &priv->jack,684priv->hp_detect);685if (ret)686return ret;687}688689return 0;690}691692static int byt_rt5651_codec_fixup(struct snd_soc_pcm_runtime *rtd,693struct snd_pcm_hw_params *params)694{695struct snd_interval *rate = hw_param_interval(params,696SNDRV_PCM_HW_PARAM_RATE);697struct snd_interval *channels = hw_param_interval(params,698SNDRV_PCM_HW_PARAM_CHANNELS);699int ret, bits;700701/* The DSP will convert the FE rate to 48k, stereo */702rate->min = rate->max = 48000;703channels->min = channels->max = 2;704705if ((byt_rt5651_quirk & BYT_RT5651_SSP0_AIF1) ||706(byt_rt5651_quirk & BYT_RT5651_SSP0_AIF2)) {707/* set SSP0 to 16-bit */708params_set_format(params, SNDRV_PCM_FORMAT_S16_LE);709bits = 16;710} else {711/* set SSP2 to 24-bit */712params_set_format(params, SNDRV_PCM_FORMAT_S24_LE);713bits = 24;714}715716/*717* Default mode for SSP configuration is TDM 4 slot, override config718* with explicit setting to I2S 2ch. The word length is set with719* dai_set_tdm_slot() since there is no other API exposed720*/721ret = snd_soc_dai_set_fmt(snd_soc_rtd_to_cpu(rtd, 0),722SND_SOC_DAIFMT_I2S |723SND_SOC_DAIFMT_NB_NF |724SND_SOC_DAIFMT_BP_FP725);726727if (ret < 0) {728dev_err(rtd->dev, "can't set format to I2S, err %d\n", ret);729return ret;730}731732ret = snd_soc_dai_set_tdm_slot(snd_soc_rtd_to_cpu(rtd, 0), 0x3, 0x3, 2, bits);733if (ret < 0) {734dev_err(rtd->dev, "can't set I2S config, err %d\n", ret);735return ret;736}737738return 0;739}740741static const unsigned int rates_48000[] = {74248000,743};744745static const struct snd_pcm_hw_constraint_list constraints_48000 = {746.count = ARRAY_SIZE(rates_48000),747.list = rates_48000,748};749750static int byt_rt5651_aif1_startup(struct snd_pcm_substream *substream)751{752return snd_pcm_hw_constraint_list(substream->runtime, 0,753SNDRV_PCM_HW_PARAM_RATE,754&constraints_48000);755}756757static const struct snd_soc_ops byt_rt5651_aif1_ops = {758.startup = byt_rt5651_aif1_startup,759};760761static const struct snd_soc_ops byt_rt5651_be_ssp2_ops = {762.hw_params = byt_rt5651_aif1_hw_params,763};764765SND_SOC_DAILINK_DEF(dummy,766DAILINK_COMP_ARRAY(COMP_DUMMY()));767768SND_SOC_DAILINK_DEF(media,769DAILINK_COMP_ARRAY(COMP_CPU("media-cpu-dai")));770771SND_SOC_DAILINK_DEF(deepbuffer,772DAILINK_COMP_ARRAY(COMP_CPU("deepbuffer-cpu-dai")));773774SND_SOC_DAILINK_DEF(ssp2_port,775DAILINK_COMP_ARRAY(COMP_CPU("ssp2-port")));776SND_SOC_DAILINK_DEF(ssp2_codec,777DAILINK_COMP_ARRAY(COMP_CODEC("i2c-10EC5651:00", "rt5651-aif1")));778779SND_SOC_DAILINK_DEF(platform,780DAILINK_COMP_ARRAY(COMP_PLATFORM("sst-mfld-platform")));781782static struct snd_soc_dai_link byt_rt5651_dais[] = {783[MERR_DPCM_AUDIO] = {784.name = "Audio Port",785.stream_name = "Audio",786.nonatomic = true,787.dynamic = 1,788.ops = &byt_rt5651_aif1_ops,789SND_SOC_DAILINK_REG(media, dummy, platform),790},791[MERR_DPCM_DEEP_BUFFER] = {792.name = "Deep-Buffer Audio Port",793.stream_name = "Deep-Buffer Audio",794.nonatomic = true,795.dynamic = 1,796.playback_only = 1,797.ops = &byt_rt5651_aif1_ops,798SND_SOC_DAILINK_REG(deepbuffer, dummy, platform),799},800/* CODEC<->CODEC link */801/* back ends */802{803.name = "SSP2-Codec",804.id = 0,805.no_pcm = 1,806.dai_fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF807| SND_SOC_DAIFMT_CBC_CFC,808.be_hw_params_fixup = byt_rt5651_codec_fixup,809.init = byt_rt5651_init,810.ops = &byt_rt5651_be_ssp2_ops,811SND_SOC_DAILINK_REG(ssp2_port, ssp2_codec, platform),812},813};814815/* SoC card */816static char byt_rt5651_codec_name[SND_ACPI_I2C_ID_LEN];817#if !IS_ENABLED(CONFIG_SND_SOC_INTEL_USER_FRIENDLY_LONG_NAMES)818static char byt_rt5651_long_name[50]; /* = "bytcr-rt5651-*-spk-*-mic[-swapped-hp]" */819#endif820static char byt_rt5651_components[50]; /* = "cfg-spk:* cfg-mic:*" */821822static int byt_rt5651_suspend(struct snd_soc_card *card)823{824struct snd_soc_component *component;825826if (!BYT_RT5651_JDSRC(byt_rt5651_quirk))827return 0;828829for_each_card_components(card, component) {830if (!strcmp(component->name, byt_rt5651_codec_name)) {831dev_dbg(component->dev, "disabling jack detect before suspend\n");832snd_soc_component_set_jack(component, NULL, NULL);833break;834}835}836837return 0;838}839840static int byt_rt5651_resume(struct snd_soc_card *card)841{842struct byt_rt5651_private *priv = snd_soc_card_get_drvdata(card);843struct snd_soc_component *component;844845if (!BYT_RT5651_JDSRC(byt_rt5651_quirk))846return 0;847848for_each_card_components(card, component) {849if (!strcmp(component->name, byt_rt5651_codec_name)) {850dev_dbg(component->dev, "re-enabling jack detect after resume\n");851snd_soc_component_set_jack(component, &priv->jack,852priv->hp_detect);853break;854}855}856857return 0;858}859860/* use space before codec name to simplify card ID, and simplify driver name */861#define SOF_CARD_NAME "bytcht rt5651" /* card name will be 'sof-bytcht rt5651' */862#define SOF_DRIVER_NAME "SOF"863864#define CARD_NAME "bytcr-rt5651"865#define DRIVER_NAME NULL /* card name will be used for driver name */866867static struct snd_soc_card byt_rt5651_card = {868.name = CARD_NAME,869.driver_name = DRIVER_NAME,870.owner = THIS_MODULE,871.dai_link = byt_rt5651_dais,872.num_links = ARRAY_SIZE(byt_rt5651_dais),873.dapm_widgets = byt_rt5651_widgets,874.num_dapm_widgets = ARRAY_SIZE(byt_rt5651_widgets),875.dapm_routes = byt_rt5651_audio_map,876.num_dapm_routes = ARRAY_SIZE(byt_rt5651_audio_map),877.fully_routed = true,878.suspend_pre = byt_rt5651_suspend,879.resume_post = byt_rt5651_resume,880};881882static const struct acpi_gpio_params ext_amp_enable_gpios = { 0, 0, false };883884static const struct acpi_gpio_mapping cht_rt5651_gpios[] = {885/*886* Some boards have I2cSerialBusV2, GpioIo, GpioInt as ACPI resources,887* other boards may have I2cSerialBusV2, GpioInt, GpioIo instead.888* We want the GpioIo one for the ext-amp-enable-gpio.889*/890{ "ext-amp-enable-gpios", &ext_amp_enable_gpios, 1, ACPI_GPIO_QUIRK_ONLY_GPIOIO },891{ },892};893894struct acpi_chan_package { /* ACPICA seems to require 64 bit integers */895u64 aif_value; /* 1: AIF1, 2: AIF2 */896u64 mclock_value; /* usually 25MHz (0x17d7940), ignored */897};898899static int snd_byt_rt5651_mc_probe(struct platform_device *pdev)900{901struct device *dev = &pdev->dev;902static const char * const mic_name[] = { "dmic", "in1", "in2", "in12" };903struct snd_soc_acpi_mach *mach = dev_get_platdata(dev);904struct byt_rt5651_private *priv;905const char *platform_name;906struct acpi_device *adev;907struct device *codec_dev;908bool sof_parent;909bool is_bytcr = false;910int ret_val = 0;911int dai_index = 0;912int i;913914priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);915if (!priv)916return -ENOMEM;917918/* register the soc card */919byt_rt5651_card.dev = dev;920snd_soc_card_set_drvdata(&byt_rt5651_card, priv);921922/* fix index of codec dai */923for (i = 0; i < ARRAY_SIZE(byt_rt5651_dais); i++) {924if (byt_rt5651_dais[i].num_codecs &&925!strcmp(byt_rt5651_dais[i].codecs->name,926"i2c-10EC5651:00")) {927dai_index = i;928break;929}930}931932/* fixup codec name based on HID */933adev = acpi_dev_get_first_match_dev(mach->id, NULL, -1);934if (adev) {935snprintf(byt_rt5651_codec_name, sizeof(byt_rt5651_codec_name),936"i2c-%s", acpi_dev_name(adev));937byt_rt5651_dais[dai_index].codecs->name = byt_rt5651_codec_name;938} else {939dev_err(dev, "Error cannot find '%s' dev\n", mach->id);940return -ENOENT;941}942943codec_dev = acpi_get_first_physical_node(adev);944acpi_dev_put(adev);945if (!codec_dev)946return -EPROBE_DEFER;947priv->codec_dev = get_device(codec_dev);948949/*950* swap SSP0 if bytcr is detected951* (will be overridden if DMI quirk is detected)952*/953if (soc_intel_is_byt()) {954if (mach->mach_params.acpi_ipc_irq_index == 0)955is_bytcr = true;956}957958if (is_bytcr) {959/*960* Baytrail CR platforms may have CHAN package in BIOS, try961* to find relevant routing quirk based as done on Windows962* platforms. We have to read the information directly from the963* BIOS, at this stage the card is not created and the links964* with the codec driver/pdata are non-existent965*/966967struct acpi_chan_package chan_package = { 0 };968969/* format specified: 2 64-bit integers */970struct acpi_buffer format = {sizeof("NN"), "NN"};971struct acpi_buffer state = {0, NULL};972struct snd_soc_acpi_package_context pkg_ctx;973bool pkg_found = false;974975state.length = sizeof(chan_package);976state.pointer = &chan_package;977978pkg_ctx.name = "CHAN";979pkg_ctx.length = 2;980pkg_ctx.format = &format;981pkg_ctx.state = &state;982pkg_ctx.data_valid = false;983984pkg_found = snd_soc_acpi_find_package_from_hid(mach->id,985&pkg_ctx);986if (pkg_found) {987if (chan_package.aif_value == 1) {988dev_info(dev, "BIOS Routing: AIF1 connected\n");989byt_rt5651_quirk |= BYT_RT5651_SSP0_AIF1;990} else if (chan_package.aif_value == 2) {991dev_info(dev, "BIOS Routing: AIF2 connected\n");992byt_rt5651_quirk |= BYT_RT5651_SSP0_AIF2;993} else {994dev_info(dev, "BIOS Routing isn't valid, ignored\n");995pkg_found = false;996}997}998999if (!pkg_found) {1000/* no BIOS indications, assume SSP0-AIF2 connection */1001byt_rt5651_quirk |= BYT_RT5651_SSP0_AIF2;1002}1003}10041005/* check quirks before creating card */1006dmi_check_system(byt_rt5651_quirk_table);10071008if (quirk_override != -1) {1009dev_info(dev, "Overriding quirk 0x%lx => 0x%x\n",1010byt_rt5651_quirk, quirk_override);1011byt_rt5651_quirk = quirk_override;1012}10131014/* Must be called before register_card, also see declaration comment. */1015ret_val = byt_rt5651_add_codec_device_props(codec_dev, priv);1016if (ret_val)1017goto err_device;10181019/* Cherry Trail devices use an external amplifier enable gpio */1020if (soc_intel_is_cht() && !byt_rt5651_gpios)1021byt_rt5651_gpios = cht_rt5651_gpios;10221023if (byt_rt5651_gpios) {1024devm_acpi_dev_add_driver_gpios(codec_dev, byt_rt5651_gpios);1025priv->ext_amp_gpio = devm_fwnode_gpiod_get(dev, codec_dev->fwnode,1026"ext-amp-enable",1027GPIOD_OUT_LOW,1028"speaker-amp");1029if (IS_ERR(priv->ext_amp_gpio)) {1030ret_val = PTR_ERR(priv->ext_amp_gpio);1031switch (ret_val) {1032case -ENOENT:1033priv->ext_amp_gpio = NULL;1034break;1035default:1036dev_err(dev, "Failed to get ext-amp-enable GPIO: %d\n", ret_val);1037fallthrough;1038case -EPROBE_DEFER:1039goto err;1040}1041}1042priv->hp_detect = devm_fwnode_gpiod_get(dev, codec_dev->fwnode,1043"hp-detect",1044GPIOD_IN,1045"hp-detect");1046if (IS_ERR(priv->hp_detect)) {1047ret_val = PTR_ERR(priv->hp_detect);1048switch (ret_val) {1049case -ENOENT:1050priv->hp_detect = NULL;1051break;1052default:1053dev_err(dev, "Failed to get hp-detect GPIO: %d\n", ret_val);1054fallthrough;1055case -EPROBE_DEFER:1056goto err;1057}1058}1059}10601061log_quirks(dev);10621063if ((byt_rt5651_quirk & BYT_RT5651_SSP2_AIF2) ||1064(byt_rt5651_quirk & BYT_RT5651_SSP0_AIF2))1065byt_rt5651_dais[dai_index].codecs->dai_name = "rt5651-aif2";10661067if ((byt_rt5651_quirk & BYT_RT5651_SSP0_AIF1) ||1068(byt_rt5651_quirk & BYT_RT5651_SSP0_AIF2))1069byt_rt5651_dais[dai_index].cpus->dai_name = "ssp0-port";10701071if (byt_rt5651_quirk & BYT_RT5651_MCLK_EN) {1072priv->mclk = devm_clk_get_optional(dev, "pmc_plt_clk_3");1073if (IS_ERR(priv->mclk)) {1074ret_val = dev_err_probe(dev, PTR_ERR(priv->mclk),1075"Failed to get MCLK from pmc_plt_clk_3\n");1076goto err;1077}1078/*1079* Fall back to bit clock usage when clock is not1080* available likely due to missing dependencies.1081*/1082if (!priv->mclk)1083byt_rt5651_quirk &= ~BYT_RT5651_MCLK_EN;1084}10851086snprintf(byt_rt5651_components, sizeof(byt_rt5651_components),1087"cfg-spk:%s cfg-mic:%s%s",1088(byt_rt5651_quirk & BYT_RT5651_MONO_SPEAKER) ? "1" : "2",1089mic_name[BYT_RT5651_MAP(byt_rt5651_quirk)],1090(byt_rt5651_quirk & BYT_RT5651_HP_LR_SWAPPED) ?1091" cfg-hp:lrswap" : "");1092byt_rt5651_card.components = byt_rt5651_components;1093#if !IS_ENABLED(CONFIG_SND_SOC_INTEL_USER_FRIENDLY_LONG_NAMES)1094snprintf(byt_rt5651_long_name, sizeof(byt_rt5651_long_name),1095"bytcr-rt5651-%s-spk-%s-mic%s",1096(byt_rt5651_quirk & BYT_RT5651_MONO_SPEAKER) ?1097"mono" : "stereo",1098mic_name[BYT_RT5651_MAP(byt_rt5651_quirk)],1099(byt_rt5651_quirk & BYT_RT5651_HP_LR_SWAPPED) ?1100"-hp-swapped" : "");1101byt_rt5651_card.long_name = byt_rt5651_long_name;1102#endif11031104/* override platform name, if required */1105platform_name = mach->mach_params.platform;11061107ret_val = snd_soc_fixup_dai_links_platform_name(&byt_rt5651_card,1108platform_name);1109if (ret_val)1110goto err;11111112sof_parent = snd_soc_acpi_sof_parent(dev);11131114/* set card and driver name */1115if (sof_parent) {1116byt_rt5651_card.name = SOF_CARD_NAME;1117byt_rt5651_card.driver_name = SOF_DRIVER_NAME;1118} else {1119byt_rt5651_card.name = CARD_NAME;1120byt_rt5651_card.driver_name = DRIVER_NAME;1121}11221123/* set pm ops */1124if (sof_parent)1125dev->driver->pm = &snd_soc_pm_ops;11261127ret_val = devm_snd_soc_register_card(dev, &byt_rt5651_card);1128if (ret_val) {1129dev_err(dev, "devm_snd_soc_register_card failed %d\n", ret_val);1130goto err;1131}1132platform_set_drvdata(pdev, &byt_rt5651_card);1133return ret_val;11341135err:1136device_remove_software_node(priv->codec_dev);1137err_device:1138put_device(priv->codec_dev);1139return ret_val;1140}11411142static void snd_byt_rt5651_mc_remove(struct platform_device *pdev)1143{1144struct snd_soc_card *card = platform_get_drvdata(pdev);1145struct byt_rt5651_private *priv = snd_soc_card_get_drvdata(card);11461147device_remove_software_node(priv->codec_dev);1148put_device(priv->codec_dev);1149}11501151static struct platform_driver snd_byt_rt5651_mc_driver = {1152.driver = {1153.name = "bytcr_rt5651",1154},1155.probe = snd_byt_rt5651_mc_probe,1156.remove = snd_byt_rt5651_mc_remove,1157};11581159module_platform_driver(snd_byt_rt5651_mc_driver);11601161MODULE_DESCRIPTION("ASoC Intel(R) Baytrail CR Machine driver for RT5651");1162MODULE_AUTHOR("Pierre-Louis Bossart <[email protected]>");1163MODULE_LICENSE("GPL v2");1164MODULE_ALIAS("platform:bytcr_rt5651");116511661167