Path: blob/master/sound/soc/intel/boards/bytcr_rt5651.c
53976 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_card *card = snd_soc_dapm_to_card(w->dapm);191struct snd_soc_dai *codec_dai;192struct byt_rt5651_private *priv = snd_soc_card_get_drvdata(card);193int ret;194195codec_dai = snd_soc_card_get_codec_dai(card, BYT_CODEC_DAI1);196if (!codec_dai)197codec_dai = snd_soc_card_get_codec_dai(card, BYT_CODEC_DAI2);198if (!codec_dai) {199dev_err(card->dev,200"Codec dai not found; Unable to set platform clock\n");201return -EIO;202}203204if (SND_SOC_DAPM_EVENT_ON(event)) {205ret = clk_prepare_enable(priv->mclk);206if (ret < 0) {207dev_err(card->dev, "could not configure MCLK state");208return ret;209}210ret = byt_rt5651_prepare_and_enable_pll1(codec_dai, 48000, 50);211} else {212/*213* Set codec clock source to internal clock before214* turning off the platform clock. Codec needs clock215* for Jack detection and button press216*/217ret = snd_soc_dai_set_sysclk(codec_dai, RT5651_SCLK_S_RCCLK,21848000 * 512,219SND_SOC_CLOCK_IN);220if (!ret)221clk_disable_unprepare(priv->mclk);222}223224if (ret < 0) {225dev_err(card->dev, "can't set codec sysclk: %d\n", ret);226return ret;227}228229return 0;230}231232static int rt5651_ext_amp_power_event(struct snd_soc_dapm_widget *w,233struct snd_kcontrol *kcontrol, int event)234{235struct snd_soc_card *card = snd_soc_dapm_to_card(w->dapm);236struct byt_rt5651_private *priv = snd_soc_card_get_drvdata(card);237238if (SND_SOC_DAPM_EVENT_ON(event))239gpiod_set_value_cansleep(priv->ext_amp_gpio, 1);240else241gpiod_set_value_cansleep(priv->ext_amp_gpio, 0);242243return 0;244}245246static const struct snd_soc_dapm_widget byt_rt5651_widgets[] = {247SND_SOC_DAPM_HP("Headphone", NULL),248SND_SOC_DAPM_MIC("Headset Mic", NULL),249SND_SOC_DAPM_MIC("Internal Mic", NULL),250SND_SOC_DAPM_SPK("Speaker", NULL),251SND_SOC_DAPM_LINE("Line In", NULL),252SND_SOC_DAPM_SUPPLY("Platform Clock", SND_SOC_NOPM, 0, 0,253platform_clock_control, SND_SOC_DAPM_PRE_PMU |254SND_SOC_DAPM_POST_PMD),255SND_SOC_DAPM_SUPPLY("Ext Amp Power", SND_SOC_NOPM, 0, 0,256rt5651_ext_amp_power_event,257SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMU),258};259260static const struct snd_soc_dapm_route byt_rt5651_audio_map[] = {261{"Headphone", NULL, "Platform Clock"},262{"Headset Mic", NULL, "Platform Clock"},263{"Internal Mic", NULL, "Platform Clock"},264{"Speaker", NULL, "Platform Clock"},265{"Speaker", NULL, "Ext Amp Power"},266{"Line In", NULL, "Platform Clock"},267268{"Headset Mic", NULL, "micbias1"}, /* lowercase for rt5651 */269{"Headphone", NULL, "HPOL"},270{"Headphone", NULL, "HPOR"},271{"Speaker", NULL, "LOUTL"},272{"Speaker", NULL, "LOUTR"},273{"IN2P", NULL, "Line In"},274{"IN2N", NULL, "Line In"},275276};277278static const struct snd_soc_dapm_route byt_rt5651_intmic_dmic_map[] = {279{"DMIC L1", NULL, "Internal Mic"},280{"DMIC R1", NULL, "Internal Mic"},281{"IN2P", NULL, "Headset Mic"},282};283284static const struct snd_soc_dapm_route byt_rt5651_intmic_in1_map[] = {285{"Internal Mic", NULL, "micbias1"},286{"IN1P", NULL, "Internal Mic"},287{"IN3P", NULL, "Headset Mic"},288};289290static const struct snd_soc_dapm_route byt_rt5651_intmic_in2_map[] = {291{"Internal Mic", NULL, "micbias1"},292{"IN2P", NULL, "Internal Mic"},293{"IN3P", NULL, "Headset Mic"},294};295296static const struct snd_soc_dapm_route byt_rt5651_intmic_in1_in2_map[] = {297{"Internal Mic", NULL, "micbias1"},298{"IN1P", NULL, "Internal Mic"},299{"IN2P", NULL, "Internal Mic"},300{"IN3P", NULL, "Headset Mic"},301};302303static const struct snd_soc_dapm_route byt_rt5651_ssp0_aif1_map[] = {304{"ssp0 Tx", NULL, "modem_out"},305{"modem_in", NULL, "ssp0 Rx"},306307{"AIF1 Playback", NULL, "ssp0 Tx"},308{"ssp0 Rx", NULL, "AIF1 Capture"},309};310311static const struct snd_soc_dapm_route byt_rt5651_ssp0_aif2_map[] = {312{"ssp0 Tx", NULL, "modem_out"},313{"modem_in", NULL, "ssp0 Rx"},314315{"AIF2 Playback", NULL, "ssp0 Tx"},316{"ssp0 Rx", NULL, "AIF2 Capture"},317};318319static const struct snd_soc_dapm_route byt_rt5651_ssp2_aif1_map[] = {320{"ssp2 Tx", NULL, "codec_out0"},321{"ssp2 Tx", NULL, "codec_out1"},322{"codec_in0", NULL, "ssp2 Rx"},323{"codec_in1", NULL, "ssp2 Rx"},324325{"AIF1 Playback", NULL, "ssp2 Tx"},326{"ssp2 Rx", NULL, "AIF1 Capture"},327};328329static const struct snd_soc_dapm_route byt_rt5651_ssp2_aif2_map[] = {330{"ssp2 Tx", NULL, "codec_out0"},331{"ssp2 Tx", NULL, "codec_out1"},332{"codec_in0", NULL, "ssp2 Rx"},333{"codec_in1", NULL, "ssp2 Rx"},334335{"AIF2 Playback", NULL, "ssp2 Tx"},336{"ssp2 Rx", NULL, "AIF2 Capture"},337};338339static const struct snd_kcontrol_new byt_rt5651_controls[] = {340SOC_DAPM_PIN_SWITCH("Headphone"),341SOC_DAPM_PIN_SWITCH("Headset Mic"),342SOC_DAPM_PIN_SWITCH("Internal Mic"),343SOC_DAPM_PIN_SWITCH("Speaker"),344SOC_DAPM_PIN_SWITCH("Line In"),345};346347static struct snd_soc_jack_pin bytcr_jack_pins[] = {348{349.pin = "Headphone",350.mask = SND_JACK_HEADPHONE,351},352{353.pin = "Headset Mic",354.mask = SND_JACK_MICROPHONE,355},356};357358static int byt_rt5651_aif1_hw_params(struct snd_pcm_substream *substream,359struct snd_pcm_hw_params *params)360{361struct snd_soc_pcm_runtime *rtd = snd_soc_substream_to_rtd(substream);362struct snd_soc_dai *codec_dai = snd_soc_rtd_to_codec(rtd, 0);363snd_pcm_format_t format = params_format(params);364int rate = params_rate(params);365int bclk_ratio;366367if (format == SNDRV_PCM_FORMAT_S16_LE)368bclk_ratio = 32;369else370bclk_ratio = 50;371372return byt_rt5651_prepare_and_enable_pll1(codec_dai, rate, bclk_ratio);373}374375static const struct acpi_gpio_params pov_p1006w_hp_detect = { 1, 0, false };376static const struct acpi_gpio_params pov_p1006w_ext_amp_en = { 2, 0, true };377378static const struct acpi_gpio_mapping byt_rt5651_pov_p1006w_gpios[] = {379{ "hp-detect-gpios", &pov_p1006w_hp_detect, 1, },380{ "ext-amp-enable-gpios", &pov_p1006w_ext_amp_en, 1, },381{ },382};383384static int byt_rt5651_pov_p1006w_quirk_cb(const struct dmi_system_id *id)385{386byt_rt5651_quirk = (unsigned long)id->driver_data;387byt_rt5651_gpios = byt_rt5651_pov_p1006w_gpios;388return 1;389}390391static int byt_rt5651_quirk_cb(const struct dmi_system_id *id)392{393byt_rt5651_quirk = (unsigned long)id->driver_data;394return 1;395}396397static const struct dmi_system_id byt_rt5651_quirk_table[] = {398{399/* Chuwi Hi8 Pro (CWI513) */400.callback = byt_rt5651_quirk_cb,401.matches = {402DMI_MATCH(DMI_SYS_VENDOR, "Hampoo"),403DMI_MATCH(DMI_PRODUCT_NAME, "X1D3_C806N"),404},405.driver_data = (void *)(BYT_RT5651_DEFAULT_QUIRKS |406BYT_RT5651_IN2_MAP |407BYT_RT5651_HP_LR_SWAPPED |408BYT_RT5651_MONO_SPEAKER),409},410{411/* Chuwi Vi8 Plus (CWI519) */412.callback = byt_rt5651_quirk_cb,413.matches = {414DMI_MATCH(DMI_SYS_VENDOR, "Hampoo"),415DMI_MATCH(DMI_PRODUCT_NAME, "D2D3_Vi8A1"),416},417.driver_data = (void *)(BYT_RT5651_DEFAULT_QUIRKS |418BYT_RT5651_IN2_MAP |419BYT_RT5651_HP_LR_SWAPPED |420BYT_RT5651_MONO_SPEAKER),421},422{423/* Complet Electro Serv MY8307 */424.callback = byt_rt5651_quirk_cb,425.matches = {426DMI_MATCH(DMI_SYS_VENDOR, "Complet Electro Serv"),427DMI_MATCH(DMI_PRODUCT_NAME, "MY8307"),428},429.driver_data = (void *)(BYT_RT5651_DEFAULT_QUIRKS |430BYT_RT5651_IN2_MAP |431BYT_RT5651_MONO_SPEAKER |432BYT_RT5651_JD_NOT_INV),433},434{435/* I.T.Works TW701, Ployer Momo7w and Trekstor ST70416-6436* (these all use the same mainboard) */437.callback = byt_rt5651_quirk_cb,438.matches = {439DMI_MATCH(DMI_BIOS_VENDOR, "INSYDE Corp."),440/* Partial match for all of itWORKS.G.WI71C.JGBMRBA,441* TREK.G.WI71C.JGBMRBA0x and MOMO.G.WI71C.MABMRBA02 */442DMI_MATCH(DMI_BIOS_VERSION, ".G.WI71C."),443},444.driver_data = (void *)(BYT_RT5651_DEFAULT_QUIRKS |445BYT_RT5651_IN2_MAP |446BYT_RT5651_SSP0_AIF1 |447BYT_RT5651_MONO_SPEAKER),448},449{450/* Jumper EZpad 7 */451.callback = byt_rt5651_quirk_cb,452.matches = {453DMI_MATCH(DMI_SYS_VENDOR, "Jumper"),454DMI_MATCH(DMI_PRODUCT_NAME, "EZpad"),455/* Jumper12x.WJ2012.bsBKRCP05 with the version dropped */456DMI_MATCH(DMI_BIOS_VERSION, "Jumper12x.WJ2012.bsBKRCP"),457},458.driver_data = (void *)(BYT_RT5651_DEFAULT_QUIRKS |459BYT_RT5651_IN2_MAP |460BYT_RT5651_JD_NOT_INV),461},462{463/* KIANO SlimNote 14.2 */464.callback = byt_rt5651_quirk_cb,465.matches = {466DMI_MATCH(DMI_SYS_VENDOR, "KIANO"),467DMI_MATCH(DMI_PRODUCT_NAME, "KIANO SlimNote 14.2"),468},469.driver_data = (void *)(BYT_RT5651_DEFAULT_QUIRKS |470BYT_RT5651_IN1_IN2_MAP),471},472{473/* Minnowboard Max B3 */474.callback = byt_rt5651_quirk_cb,475.matches = {476DMI_MATCH(DMI_SYS_VENDOR, "Circuitco"),477DMI_MATCH(DMI_PRODUCT_NAME, "Minnowboard Max B3 PLATFORM"),478},479.driver_data = (void *)(BYT_RT5651_IN1_MAP),480},481{482/* Minnowboard Turbot */483.callback = byt_rt5651_quirk_cb,484.matches = {485DMI_MATCH(DMI_SYS_VENDOR, "ADI"),486DMI_MATCH(DMI_PRODUCT_NAME, "Minnowboard Turbot"),487},488.driver_data = (void *)(BYT_RT5651_MCLK_EN |489BYT_RT5651_IN1_MAP),490},491{492/* Point of View mobii wintab p1006w (v1.0) */493.callback = byt_rt5651_pov_p1006w_quirk_cb,494.matches = {495DMI_EXACT_MATCH(DMI_SYS_VENDOR, "Insyde"),496DMI_EXACT_MATCH(DMI_PRODUCT_NAME, "BayTrail"),497/* Note 105b is Foxcon's USB/PCI vendor id */498DMI_EXACT_MATCH(DMI_BOARD_VENDOR, "105B"),499DMI_EXACT_MATCH(DMI_BOARD_NAME, "0E57"),500},501.driver_data = (void *)(BYT_RT5651_DMIC_MAP |502BYT_RT5651_OVCD_TH_2000UA |503BYT_RT5651_OVCD_SF_0P75 |504BYT_RT5651_DMIC_EN |505BYT_RT5651_MCLK_EN |506BYT_RT5651_SSP0_AIF1),507},508{509/* VIOS LTH17 */510.callback = byt_rt5651_quirk_cb,511.matches = {512DMI_MATCH(DMI_SYS_VENDOR, "VIOS"),513DMI_MATCH(DMI_PRODUCT_NAME, "LTH17"),514},515.driver_data = (void *)(BYT_RT5651_IN1_IN2_MAP |516BYT_RT5651_JD1_1 |517BYT_RT5651_OVCD_TH_2000UA |518BYT_RT5651_OVCD_SF_1P0 |519BYT_RT5651_MCLK_EN),520},521{522/* Yours Y8W81 (and others using the same mainboard) */523.callback = byt_rt5651_quirk_cb,524.matches = {525DMI_MATCH(DMI_BIOS_VENDOR, "INSYDE Corp."),526/* Partial match for all devs with a W86C mainboard */527DMI_MATCH(DMI_BIOS_VERSION, ".F.W86C."),528},529.driver_data = (void *)(BYT_RT5651_DEFAULT_QUIRKS |530BYT_RT5651_IN2_MAP |531BYT_RT5651_SSP0_AIF1 |532BYT_RT5651_MONO_SPEAKER),533},534{}535};536537/*538* Note this MUST be called before snd_soc_register_card(), so that the props539* are in place before the codec component driver's probe function parses them.540*/541static int byt_rt5651_add_codec_device_props(struct device *i2c_dev,542struct byt_rt5651_private *priv)543{544struct property_entry props[MAX_NO_PROPS] = {};545struct fwnode_handle *fwnode;546int cnt = 0;547int ret;548549props[cnt++] = PROPERTY_ENTRY_U32("realtek,jack-detect-source",550BYT_RT5651_JDSRC(byt_rt5651_quirk));551552props[cnt++] = PROPERTY_ENTRY_U32("realtek,over-current-threshold-microamp",553BYT_RT5651_OVCD_TH(byt_rt5651_quirk) * 100);554555props[cnt++] = PROPERTY_ENTRY_U32("realtek,over-current-scale-factor",556BYT_RT5651_OVCD_SF(byt_rt5651_quirk));557558if (byt_rt5651_quirk & BYT_RT5651_DMIC_EN)559props[cnt++] = PROPERTY_ENTRY_BOOL("realtek,dmic-en");560561if (byt_rt5651_quirk & BYT_RT5651_JD_NOT_INV)562props[cnt++] = PROPERTY_ENTRY_BOOL("realtek,jack-detect-not-inverted");563564fwnode = fwnode_create_software_node(props, NULL);565if (IS_ERR(fwnode)) {566/* put_device(i2c_dev) is handled in caller */567return PTR_ERR(fwnode);568}569570ret = device_add_software_node(i2c_dev, to_software_node(fwnode));571572fwnode_handle_put(fwnode);573574return ret;575}576577static int byt_rt5651_init(struct snd_soc_pcm_runtime *runtime)578{579struct snd_soc_card *card = runtime->card;580struct snd_soc_dapm_context *dapm = snd_soc_card_to_dapm(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;587588snd_soc_dapm_set_idle_bias(dapm, 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(dapm, custom_map, num_routes);613if (ret)614return ret;615616if (byt_rt5651_quirk & BYT_RT5651_SSP2_AIF2) {617ret = snd_soc_dapm_add_routes(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(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(dapm,626byt_rt5651_ssp0_aif2_map,627ARRAY_SIZE(byt_rt5651_ssp0_aif2_map));628} else {629ret = snd_soc_dapm_add_routes(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