2016-01-04 17:20:24 -06:00
/*
* bytcr_rt5651 . c - ASoc Machine driver for Intel Byt CR platform
* ( derived from bytcr_rt5640 . c )
*
* Copyright ( C ) 2015 Intel Corp
* ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; version 2 of the License .
*
* This program is distributed in the hope that it will be useful , but
* WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
* General Public License for more details .
*
* ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~
*/
# include <linux/init.h>
2018-03-04 15:35:59 +01:00
# include <linux/i2c.h>
2016-01-04 17:20:24 -06:00
# include <linux/module.h>
# include <linux/platform_device.h>
2018-03-04 15:35:59 +01:00
# include <linux/property.h>
2016-01-04 17:20:24 -06:00
# include <linux/acpi.h>
2017-10-12 18:37:57 -05:00
# include <linux/clk.h>
2016-01-04 17:20:24 -06:00
# include <linux/device.h>
# include <linux/dmi.h>
2018-07-05 00:59:35 +02:00
# include <linux/input.h>
2018-07-01 20:36:31 +02:00
# include <linux/gpio/consumer.h>
# include <linux/gpio/machine.h>
2016-01-04 17:20:24 -06:00
# include <linux/slab.h>
2018-03-04 15:36:07 +01:00
# include <asm/cpu_device_id.h>
2018-07-01 20:36:29 +02:00
# include <asm/intel-family.h>
2016-01-04 17:20:24 -06:00
# include <sound/pcm.h>
# include <sound/pcm_params.h>
# include <sound/soc.h>
# include <sound/jack.h>
2017-10-12 18:49:38 -05:00
# include <sound/soc-acpi.h>
2016-01-04 17:20:24 -06:00
# include "../../codecs/rt5651.h"
# include "../atom/sst-atom-controls.h"
2017-10-12 18:37:57 -05:00
enum {
BYT_RT5651_DMIC_MAP ,
BYT_RT5651_IN1_MAP ,
2018-07-18 22:55:40 +02:00
BYT_RT5651_IN2_MAP ,
2017-11-21 17:15:45 +00:00
BYT_RT5651_IN1_IN2_MAP ,
2017-10-12 18:37:57 -05:00
} ;
2018-03-04 15:35:59 +01:00
enum {
BYT_RT5651_JD_NULL = ( RT5651_JD_NULL < < 4 ) ,
BYT_RT5651_JD1_1 = ( RT5651_JD1_1 < < 4 ) ,
BYT_RT5651_JD1_2 = ( RT5651_JD1_2 < < 4 ) ,
BYT_RT5651_JD2 = ( RT5651_JD2 < < 4 ) ,
} ;
2018-03-04 15:36:02 +01:00
enum {
BYT_RT5651_OVCD_TH_600UA = ( 6 < < 8 ) ,
BYT_RT5651_OVCD_TH_1500UA = ( 15 < < 8 ) ,
BYT_RT5651_OVCD_TH_2000UA = ( 20 < < 8 ) ,
} ;
enum {
BYT_RT5651_OVCD_SF_0P5 = ( RT5651_OVCD_SF_0P5 < < 13 ) ,
BYT_RT5651_OVCD_SF_0P75 = ( RT5651_OVCD_SF_0P75 < < 13 ) ,
BYT_RT5651_OVCD_SF_1P0 = ( RT5651_OVCD_SF_1P0 < < 13 ) ,
BYT_RT5651_OVCD_SF_1P5 = ( RT5651_OVCD_SF_1P5 < < 13 ) ,
} ;
2018-03-04 15:35:59 +01:00
# define BYT_RT5651_MAP(quirk) ((quirk) & GENMASK(3, 0))
# define BYT_RT5651_JDSRC(quirk) (((quirk) & GENMASK(7, 4)) >> 4)
2018-03-04 15:36:02 +01:00
# define BYT_RT5651_OVCD_TH(quirk) (((quirk) & GENMASK(12, 8)) >> 8)
# define BYT_RT5651_OVCD_SF(quirk) (((quirk) & GENMASK(14, 13)) >> 13)
2018-03-04 15:35:59 +01:00
# define BYT_RT5651_DMIC_EN BIT(16)
# define BYT_RT5651_MCLK_EN BIT(17)
# define BYT_RT5651_MCLK_25MHZ BIT(18)
2018-03-04 15:36:07 +01:00
# define BYT_RT5651_SSP2_AIF2 BIT(19) /* default is using AIF1 */
# define BYT_RT5651_SSP0_AIF1 BIT(20)
# define BYT_RT5651_SSP0_AIF2 BIT(21)
2018-06-24 16:06:33 +02:00
# define BYT_RT5651_HP_LR_SWAPPED BIT(22)
2018-07-18 22:55:41 +02:00
# define BYT_RT5651_MONO_SPEAKER BIT(23)
2018-03-04 15:35:59 +01:00
2018-06-24 16:06:26 +02:00
# define BYT_RT5651_DEFAULT_QUIRKS (BYT_RT5651_MCLK_EN | \
BYT_RT5651_JD1_1 | \
BYT_RT5651_OVCD_TH_2000UA | \
BYT_RT5651_OVCD_SF_0P75 )
2018-03-04 15:36:02 +01:00
/* jack-detect-source + dmic-en + ovcd-th + -sf + terminating empty entry */
# define MAX_NO_PROPS 5
2017-10-12 18:37:57 -05:00
struct byt_rt5651_private {
struct clk * mclk ;
2018-07-01 20:36:31 +02:00
struct gpio_desc * ext_amp_gpio ;
2017-10-20 12:18:56 +01:00
struct snd_soc_jack jack ;
2017-10-12 18:37:57 -05:00
} ;
2018-07-18 22:55:40 +02:00
/* Default: jack-detect on JD1_1, internal mic on in2, headsetmic on in3 */
2018-06-24 16:06:26 +02:00
static unsigned long byt_rt5651_quirk = BYT_RT5651_DEFAULT_QUIRKS |
2018-07-18 22:55:40 +02:00
BYT_RT5651_IN2_MAP ;
2017-10-12 18:37:57 -05:00
static void log_quirks ( struct device * dev )
{
if ( BYT_RT5651_MAP ( byt_rt5651_quirk ) = = BYT_RT5651_DMIC_MAP )
dev_info ( dev , " quirk DMIC_MAP enabled " ) ;
if ( BYT_RT5651_MAP ( byt_rt5651_quirk ) = = BYT_RT5651_IN1_MAP )
dev_info ( dev , " quirk IN1_MAP enabled " ) ;
2018-07-18 22:55:40 +02:00
if ( BYT_RT5651_MAP ( byt_rt5651_quirk ) = = BYT_RT5651_IN2_MAP )
dev_info ( dev , " quirk IN2_MAP enabled " ) ;
2018-06-24 16:06:28 +02:00
if ( BYT_RT5651_MAP ( byt_rt5651_quirk ) = = BYT_RT5651_IN1_IN2_MAP )
dev_info ( dev , " quirk IN1_IN2_MAP enabled " ) ;
2018-03-04 15:36:02 +01:00
if ( BYT_RT5651_JDSRC ( byt_rt5651_quirk ) ) {
2018-03-04 15:35:59 +01:00
dev_info ( dev , " quirk realtek,jack-detect-source %ld \n " ,
BYT_RT5651_JDSRC ( byt_rt5651_quirk ) ) ;
2018-03-04 15:36:02 +01:00
dev_info ( dev , " quirk realtek,over-current-threshold-microamp %ld \n " ,
BYT_RT5651_OVCD_TH ( byt_rt5651_quirk ) * 100 ) ;
dev_info ( dev , " quirk realtek,over-current-scale-factor %ld \n " ,
BYT_RT5651_OVCD_SF ( byt_rt5651_quirk ) ) ;
}
2017-10-12 18:37:57 -05:00
if ( byt_rt5651_quirk & BYT_RT5651_DMIC_EN )
dev_info ( dev , " quirk DMIC enabled " ) ;
if ( byt_rt5651_quirk & BYT_RT5651_MCLK_EN )
dev_info ( dev , " quirk MCLK_EN enabled " ) ;
if ( byt_rt5651_quirk & BYT_RT5651_MCLK_25MHZ )
dev_info ( dev , " quirk MCLK_25MHZ enabled " ) ;
2018-03-04 15:36:07 +01:00
if ( byt_rt5651_quirk & BYT_RT5651_SSP2_AIF2 )
dev_info ( dev , " quirk SSP2_AIF2 enabled \n " ) ;
if ( byt_rt5651_quirk & BYT_RT5651_SSP0_AIF1 )
dev_info ( dev , " quirk SSP0_AIF1 enabled \n " ) ;
if ( byt_rt5651_quirk & BYT_RT5651_SSP0_AIF2 )
dev_info ( dev , " quirk SSP0_AIF2 enabled \n " ) ;
2018-07-18 22:55:41 +02:00
if ( byt_rt5651_quirk & BYT_RT5651_MONO_SPEAKER )
dev_info ( dev , " quirk MONO_SPEAKER enabled \n " ) ;
2017-10-12 18:37:57 -05:00
}
# define BYT_CODEC_DAI1 "rt5651-aif1"
2018-03-04 15:36:07 +01:00
# define BYT_CODEC_DAI2 "rt5651-aif2"
2017-10-12 18:37:57 -05:00
2018-03-04 15:36:03 +01:00
static int byt_rt5651_prepare_and_enable_pll1 ( struct snd_soc_dai * codec_dai ,
int rate , int bclk_ratio )
{
int clk_id , clk_freq , ret ;
/* Configure the PLL before selecting it */
if ( ! ( byt_rt5651_quirk & BYT_RT5651_MCLK_EN ) ) {
clk_id = RT5651_PLL1_S_BCLK1 ,
clk_freq = rate * bclk_ratio ;
} else {
clk_id = RT5651_PLL1_S_MCLK ;
if ( byt_rt5651_quirk & BYT_RT5651_MCLK_25MHZ )
clk_freq = 25000000 ;
else
clk_freq = 19200000 ;
}
ret = snd_soc_dai_set_pll ( codec_dai , 0 , clk_id , clk_freq , rate * 512 ) ;
if ( ret < 0 ) {
2018-03-28 01:49:36 +00:00
dev_err ( codec_dai - > component - > dev , " can't set pll: %d \n " , ret ) ;
2018-03-04 15:36:03 +01:00
return ret ;
}
ret = snd_soc_dai_set_sysclk ( codec_dai , RT5651_SCLK_S_PLL1 ,
rate * 512 , SND_SOC_CLOCK_IN ) ;
if ( ret < 0 ) {
2018-03-28 01:49:36 +00:00
dev_err ( codec_dai - > component - > dev , " can't set clock %d \n " , ret ) ;
2018-03-04 15:36:03 +01:00
return ret ;
}
return 0 ;
}
2017-10-12 18:37:57 -05:00
static int platform_clock_control ( struct snd_soc_dapm_widget * w ,
struct snd_kcontrol * k , int event )
{
struct snd_soc_dapm_context * dapm = w - > dapm ;
struct snd_soc_card * card = dapm - > card ;
struct snd_soc_dai * codec_dai ;
struct byt_rt5651_private * priv = snd_soc_card_get_drvdata ( card ) ;
int ret ;
2017-10-12 18:38:03 -05:00
codec_dai = snd_soc_card_get_codec_dai ( card , BYT_CODEC_DAI1 ) ;
2018-03-04 15:36:07 +01:00
if ( ! codec_dai )
codec_dai = snd_soc_card_get_codec_dai ( card , BYT_CODEC_DAI2 ) ;
2017-10-12 18:37:57 -05:00
if ( ! codec_dai ) {
dev_err ( card - > dev ,
" Codec dai not found; Unable to set platform clock \n " ) ;
return - EIO ;
}
if ( SND_SOC_DAPM_EVENT_ON ( event ) ) {
if ( byt_rt5651_quirk & BYT_RT5651_MCLK_EN ) {
ret = clk_prepare_enable ( priv - > mclk ) ;
if ( ret < 0 ) {
dev_err ( card - > dev ,
" could not configure MCLK state " ) ;
return ret ;
}
}
2018-03-04 15:36:03 +01:00
ret = byt_rt5651_prepare_and_enable_pll1 ( codec_dai , 48000 , 50 ) ;
2017-10-12 18:37:57 -05:00
} else {
/*
* Set codec clock source to internal clock before
* turning off the platform clock . Codec needs clock
* for Jack detection and button press
*/
ret = snd_soc_dai_set_sysclk ( codec_dai , RT5651_SCLK_S_RCCLK ,
48000 * 512 ,
SND_SOC_CLOCK_IN ) ;
if ( ! ret )
if ( byt_rt5651_quirk & BYT_RT5651_MCLK_EN )
clk_disable_unprepare ( priv - > mclk ) ;
}
if ( ret < 0 ) {
dev_err ( card - > dev , " can't set codec sysclk: %d \n " , ret ) ;
return ret ;
}
return 0 ;
}
2016-01-04 17:20:24 -06:00
2018-07-01 20:36:31 +02:00
static int rt5651_ext_amp_power_event ( struct snd_soc_dapm_widget * w ,
struct snd_kcontrol * kcontrol , int event )
{
struct snd_soc_card * card = w - > dapm - > card ;
struct byt_rt5651_private * priv = snd_soc_card_get_drvdata ( card ) ;
if ( SND_SOC_DAPM_EVENT_ON ( event ) )
gpiod_set_value_cansleep ( priv - > ext_amp_gpio , 1 ) ;
else
gpiod_set_value_cansleep ( priv - > ext_amp_gpio , 0 ) ;
return 0 ;
}
2016-01-04 17:20:24 -06:00
static const struct snd_soc_dapm_widget byt_rt5651_widgets [ ] = {
SND_SOC_DAPM_HP ( " Headphone " , NULL ) ,
SND_SOC_DAPM_MIC ( " Headset Mic " , NULL ) ,
SND_SOC_DAPM_MIC ( " Internal Mic " , NULL ) ,
SND_SOC_DAPM_SPK ( " Speaker " , NULL ) ,
2018-01-05 13:20:23 -06:00
SND_SOC_DAPM_LINE ( " Line In " , NULL ) ,
2017-10-12 18:37:57 -05:00
SND_SOC_DAPM_SUPPLY ( " Platform Clock " , SND_SOC_NOPM , 0 , 0 ,
platform_clock_control , SND_SOC_DAPM_PRE_PMU |
SND_SOC_DAPM_POST_PMD ) ,
2018-07-01 20:36:31 +02:00
SND_SOC_DAPM_SUPPLY ( " Ext Amp Power " , SND_SOC_NOPM , 0 , 0 ,
rt5651_ext_amp_power_event ,
SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMU ) ,
2016-01-04 17:20:24 -06:00
} ;
static const struct snd_soc_dapm_route byt_rt5651_audio_map [ ] = {
2017-10-12 18:37:57 -05:00
{ " Headphone " , NULL , " Platform Clock " } ,
{ " Headset Mic " , NULL , " Platform Clock " } ,
{ " Internal Mic " , NULL , " Platform Clock " } ,
{ " Speaker " , NULL , " Platform Clock " } ,
2018-07-01 20:36:31 +02:00
{ " Speaker " , NULL , " Ext Amp Power " } ,
2018-01-05 13:20:23 -06:00
{ " Line In " , NULL , " Platform Clock " } ,
2017-10-12 18:37:57 -05:00
2016-01-04 17:20:24 -06:00
{ " Headset Mic " , NULL , " micbias1 " } , /* lowercase for rt5651 */
{ " Headphone " , NULL , " HPOL " } ,
{ " Headphone " , NULL , " HPOR " } ,
{ " Speaker " , NULL , " LOUTL " } ,
{ " Speaker " , NULL , " LOUTR " } ,
2018-01-05 13:20:23 -06:00
{ " IN2P " , NULL , " Line In " } ,
{ " IN2N " , NULL , " Line In " } ,
2016-01-04 17:20:24 -06:00
} ;
2017-09-08 12:43:53 -05:00
static const struct snd_soc_dapm_route byt_rt5651_intmic_dmic_map [ ] = {
{ " DMIC L1 " , NULL , " Internal Mic " } ,
{ " DMIC R1 " , NULL , " Internal Mic " } ,
2018-06-24 16:06:31 +02:00
{ " IN3P " , NULL , " Headset Mic " } ,
2016-01-04 17:20:24 -06:00
} ;
static const struct snd_soc_dapm_route byt_rt5651_intmic_in1_map [ ] = {
{ " Internal Mic " , NULL , " micbias1 " } ,
{ " IN1P " , NULL , " Internal Mic " } ,
2018-06-24 16:06:30 +02:00
{ " IN3P " , NULL , " Headset Mic " } ,
2016-01-04 17:20:24 -06:00
} ;
2018-07-18 22:55:40 +02:00
static const struct snd_soc_dapm_route byt_rt5651_intmic_in2_map [ ] = {
{ " Internal Mic " , NULL , " micbias1 " } ,
{ " IN2P " , NULL , " Internal Mic " } ,
{ " IN3P " , NULL , " Headset Mic " } ,
} ;
2017-11-21 17:15:45 +00:00
static const struct snd_soc_dapm_route byt_rt5651_intmic_in1_in2_map [ ] = {
{ " Internal Mic " , NULL , " micbias1 " } ,
{ " IN1P " , NULL , " Internal Mic " } ,
{ " IN2P " , NULL , " Internal Mic " } ,
{ " IN3P " , NULL , " Headset Mic " } ,
} ;
2018-03-04 15:36:07 +01:00
static const struct snd_soc_dapm_route byt_rt5651_ssp0_aif1_map [ ] = {
{ " ssp0 Tx " , NULL , " modem_out " } ,
{ " modem_in " , NULL , " ssp0 Rx " } ,
{ " AIF1 Playback " , NULL , " ssp0 Tx " } ,
{ " ssp0 Rx " , NULL , " AIF1 Capture " } ,
} ;
static const struct snd_soc_dapm_route byt_rt5651_ssp0_aif2_map [ ] = {
{ " ssp0 Tx " , NULL , " modem_out " } ,
{ " modem_in " , NULL , " ssp0 Rx " } ,
{ " AIF2 Playback " , NULL , " ssp0 Tx " } ,
{ " ssp0 Rx " , NULL , " AIF2 Capture " } ,
} ;
static const struct snd_soc_dapm_route byt_rt5651_ssp2_aif1_map [ ] = {
{ " ssp2 Tx " , NULL , " codec_out0 " } ,
{ " ssp2 Tx " , NULL , " codec_out1 " } ,
{ " codec_in0 " , NULL , " ssp2 Rx " } ,
{ " codec_in1 " , NULL , " ssp2 Rx " } ,
{ " AIF1 Playback " , NULL , " ssp2 Tx " } ,
{ " ssp2 Rx " , NULL , " AIF1 Capture " } ,
} ;
static const struct snd_soc_dapm_route byt_rt5651_ssp2_aif2_map [ ] = {
{ " ssp2 Tx " , NULL , " codec_out0 " } ,
{ " ssp2 Tx " , NULL , " codec_out1 " } ,
{ " codec_in0 " , NULL , " ssp2 Rx " } ,
{ " codec_in1 " , NULL , " ssp2 Rx " } ,
{ " AIF2 Playback " , NULL , " ssp2 Tx " } ,
{ " ssp2 Rx " , NULL , " AIF2 Capture " } ,
} ;
2016-01-04 17:20:24 -06:00
static const struct snd_kcontrol_new byt_rt5651_controls [ ] = {
SOC_DAPM_PIN_SWITCH ( " Headphone " ) ,
SOC_DAPM_PIN_SWITCH ( " Headset Mic " ) ,
SOC_DAPM_PIN_SWITCH ( " Internal Mic " ) ,
SOC_DAPM_PIN_SWITCH ( " Speaker " ) ,
2018-01-05 13:20:23 -06:00
SOC_DAPM_PIN_SWITCH ( " Line In " ) ,
2016-01-04 17:20:24 -06:00
} ;
2017-10-20 12:18:56 +01:00
static struct snd_soc_jack_pin bytcr_jack_pins [ ] = {
{
. pin = " Headphone " ,
. mask = SND_JACK_HEADPHONE ,
} ,
{
. pin = " Headset Mic " ,
. mask = SND_JACK_MICROPHONE ,
} ,
} ;
2016-01-04 17:20:24 -06:00
static int byt_rt5651_aif1_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 * codec_dai = rtd - > codec_dai ;
2018-03-04 15:36:07 +01:00
snd_pcm_format_t format = params_format ( params ) ;
2018-03-04 15:36:03 +01:00
int rate = params_rate ( params ) ;
2018-03-04 15:36:07 +01:00
int bclk_ratio ;
2016-01-04 17:20:24 -06:00
2018-03-04 15:36:07 +01:00
if ( format = = SNDRV_PCM_FORMAT_S16_LE )
bclk_ratio = 32 ;
else
bclk_ratio = 50 ;
return byt_rt5651_prepare_and_enable_pll1 ( codec_dai , rate , bclk_ratio ) ;
2016-01-04 17:20:24 -06:00
}
2017-10-12 18:37:57 -05:00
static int byt_rt5651_quirk_cb ( const struct dmi_system_id * id )
{
byt_rt5651_quirk = ( unsigned long ) id - > driver_data ;
return 1 ;
}
2016-01-04 17:20:24 -06:00
static const struct dmi_system_id byt_rt5651_quirk_table [ ] = {
2017-10-12 18:37:57 -05:00
{
2018-06-24 16:06:34 +02:00
/* Chuwi Hi8 Pro (CWI513) */
2017-10-12 18:37:57 -05:00
. callback = byt_rt5651_quirk_cb ,
. matches = {
2018-06-24 16:06:34 +02:00
DMI_MATCH ( DMI_SYS_VENDOR , " Hampoo " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " X1D3_C806N " ) ,
2017-10-12 18:37:57 -05:00
} ,
2018-06-24 16:06:34 +02:00
. driver_data = ( void * ) ( BYT_RT5651_DEFAULT_QUIRKS |
2018-07-18 22:55:40 +02:00
BYT_RT5651_IN2_MAP |
2018-07-18 22:55:41 +02:00
BYT_RT5651_HP_LR_SWAPPED |
BYT_RT5651_MONO_SPEAKER ) ,
2017-10-12 18:37:57 -05:00
} ,
2018-01-05 13:20:20 -06:00
{
2018-06-24 16:06:34 +02:00
/* Chuwi Vi8 Plus (CWI519) */
2018-01-05 13:20:20 -06:00
. callback = byt_rt5651_quirk_cb ,
. matches = {
2018-06-24 16:06:34 +02:00
DMI_MATCH ( DMI_SYS_VENDOR , " Hampoo " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " D2D3_Vi8A1 " ) ,
2018-01-05 13:20:20 -06:00
} ,
2018-06-24 16:06:34 +02:00
. driver_data = ( void * ) ( BYT_RT5651_DEFAULT_QUIRKS |
2018-07-18 22:55:40 +02:00
BYT_RT5651_IN2_MAP |
2018-07-18 22:55:41 +02:00
BYT_RT5651_HP_LR_SWAPPED |
BYT_RT5651_MONO_SPEAKER ) ,
2018-01-05 13:20:20 -06:00
} ,
2018-07-18 22:55:42 +02:00
{
/* I.T.Works TW701, Ployer Momo7w and Trekstor ST70416-6
* ( these all use the same mainboard ) */
. callback = byt_rt5651_quirk_cb ,
. matches = {
DMI_MATCH ( DMI_BIOS_VENDOR , " INSYDE Corp. " ) ,
/* Partial match for all of itWORKS.G.WI71C.JGBMRBA,
* TREK . G . WI71C . JGBMRBA0x and MOMO . G . WI71C . MABMRBA02 */
DMI_MATCH ( DMI_BIOS_VERSION , " .G.WI71C. " ) ,
} ,
. driver_data = ( void * ) ( BYT_RT5651_DEFAULT_QUIRKS |
BYT_RT5651_IN2_MAP |
BYT_RT5651_SSP0_AIF1 |
BYT_RT5651_MONO_SPEAKER ) ,
} ,
2017-10-20 12:18:57 +01:00
{
2018-06-24 16:06:34 +02:00
/* KIANO SlimNote 14.2 */
2017-10-20 12:18:57 +01:00
. callback = byt_rt5651_quirk_cb ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " KIANO " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " KIANO SlimNote 14.2 " ) ,
} ,
2018-06-24 16:06:26 +02:00
. driver_data = ( void * ) ( BYT_RT5651_DEFAULT_QUIRKS |
2018-01-05 13:20:21 -06:00
BYT_RT5651_IN1_IN2_MAP ) ,
2017-10-20 12:18:57 +01:00
} ,
2018-06-24 16:06:33 +02:00
{
2018-06-24 16:06:34 +02:00
/* Minnowboard Max B3 */
2018-06-24 16:06:33 +02:00
. callback = byt_rt5651_quirk_cb ,
. matches = {
2018-06-24 16:06:34 +02:00
DMI_MATCH ( DMI_SYS_VENDOR , " Circuitco " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Minnowboard Max B3 PLATFORM " ) ,
2018-06-24 16:06:33 +02:00
} ,
2018-06-24 16:06:34 +02:00
. driver_data = ( void * ) ( BYT_RT5651_IN1_MAP ) ,
2018-06-24 16:06:33 +02:00
} ,
2018-03-04 15:36:06 +01:00
{
2018-06-24 16:06:34 +02:00
/* Minnowboard Turbot */
2018-03-04 15:36:06 +01:00
. callback = byt_rt5651_quirk_cb ,
. matches = {
2018-06-24 16:06:34 +02:00
DMI_MATCH ( DMI_SYS_VENDOR , " ADI " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Minnowboard Turbot " ) ,
2018-03-04 15:36:06 +01:00
} ,
2018-06-24 16:06:34 +02:00
. driver_data = ( void * ) ( BYT_RT5651_MCLK_EN |
BYT_RT5651_IN1_MAP ) ,
2018-03-04 15:36:06 +01:00
} ,
2018-03-04 15:36:08 +01:00
{
/* VIOS LTH17 */
. callback = byt_rt5651_quirk_cb ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " VIOS " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " LTH17 " ) ,
} ,
2018-07-18 22:55:39 +02:00
. driver_data = ( void * ) ( BYT_RT5651_IN1_IN2_MAP |
BYT_RT5651_JD1_1 |
BYT_RT5651_OVCD_TH_2000UA |
BYT_RT5651_OVCD_SF_1P0 |
BYT_RT5651_MCLK_EN ) ,
2018-03-04 15:36:08 +01:00
} ,
2018-07-18 22:55:42 +02:00
{
/* Yours Y8W81 (and others using the same mainboard) */
. callback = byt_rt5651_quirk_cb ,
. matches = {
DMI_MATCH ( DMI_BIOS_VENDOR , " INSYDE Corp. " ) ,
/* Partial match for all devs with a W86C mainboard */
DMI_MATCH ( DMI_BIOS_VERSION , " .F.W86C. " ) ,
} ,
. driver_data = ( void * ) ( BYT_RT5651_DEFAULT_QUIRKS |
BYT_RT5651_IN2_MAP |
BYT_RT5651_SSP0_AIF1 |
BYT_RT5651_MONO_SPEAKER ) ,
} ,
2016-01-04 17:20:24 -06:00
{ }
} ;
2018-03-04 15:35:59 +01:00
/*
* Note this MUST be called before snd_soc_register_card ( ) , so that the props
* are in place before the codec component driver ' s probe function parses them .
*/
2018-07-01 20:36:30 +02:00
static int byt_rt5651_add_codec_device_props ( struct device * i2c_dev )
2018-03-04 15:35:59 +01:00
{
struct property_entry props [ MAX_NO_PROPS ] = { } ;
2018-07-01 20:36:30 +02:00
int cnt = 0 ;
2018-03-04 15:35:59 +01:00
props [ cnt + + ] = PROPERTY_ENTRY_U32 ( " realtek,jack-detect-source " ,
BYT_RT5651_JDSRC ( byt_rt5651_quirk ) ) ;
2018-03-04 15:36:02 +01:00
props [ cnt + + ] = PROPERTY_ENTRY_U32 ( " realtek,over-current-threshold-microamp " ,
BYT_RT5651_OVCD_TH ( byt_rt5651_quirk ) * 100 ) ;
props [ cnt + + ] = PROPERTY_ENTRY_U32 ( " realtek,over-current-scale-factor " ,
BYT_RT5651_OVCD_SF ( byt_rt5651_quirk ) ) ;
2018-03-04 15:36:00 +01:00
if ( byt_rt5651_quirk & BYT_RT5651_DMIC_EN )
props [ cnt + + ] = PROPERTY_ENTRY_BOOL ( " realtek,dmic-en " ) ;
2018-07-01 20:36:30 +02:00
return device_add_properties ( i2c_dev , props ) ;
2018-03-04 15:35:59 +01:00
}
2016-01-04 17:20:24 -06:00
static int byt_rt5651_init ( struct snd_soc_pcm_runtime * runtime )
{
struct snd_soc_card * card = runtime - > card ;
2018-01-29 03:44:39 +00:00
struct snd_soc_component * codec = runtime - > codec_dai - > component ;
2017-10-12 18:37:57 -05:00
struct byt_rt5651_private * priv = snd_soc_card_get_drvdata ( card ) ;
2016-01-04 17:20:24 -06:00
const struct snd_soc_dapm_route * custom_map ;
int num_routes ;
2017-10-12 18:37:57 -05:00
int ret ;
2016-01-04 17:20:24 -06:00
card - > dapm . idle_bias_off = true ;
2018-03-04 15:36:10 +01:00
/* Start with RC clk for jack-detect (we disable MCLK below) */
if ( byt_rt5651_quirk & BYT_RT5651_MCLK_EN )
snd_soc_component_update_bits ( codec , RT5651_GLB_CLK ,
RT5651_SCLK_SRC_MASK , RT5651_SCLK_SRC_RCCLK ) ;
2016-01-04 17:20:24 -06:00
switch ( BYT_RT5651_MAP ( byt_rt5651_quirk ) ) {
case BYT_RT5651_IN1_MAP :
custom_map = byt_rt5651_intmic_in1_map ;
num_routes = ARRAY_SIZE ( byt_rt5651_intmic_in1_map ) ;
break ;
2018-07-18 22:55:40 +02:00
case BYT_RT5651_IN2_MAP :
custom_map = byt_rt5651_intmic_in2_map ;
num_routes = ARRAY_SIZE ( byt_rt5651_intmic_in2_map ) ;
break ;
2017-11-21 17:15:45 +00:00
case BYT_RT5651_IN1_IN2_MAP :
custom_map = byt_rt5651_intmic_in1_in2_map ;
num_routes = ARRAY_SIZE ( byt_rt5651_intmic_in1_in2_map ) ;
break ;
2016-01-04 17:20:24 -06:00
default :
2017-09-08 12:43:53 -05:00
custom_map = byt_rt5651_intmic_dmic_map ;
num_routes = ARRAY_SIZE ( byt_rt5651_intmic_dmic_map ) ;
2016-01-04 17:20:24 -06:00
}
2017-09-08 12:43:53 -05:00
ret = snd_soc_dapm_add_routes ( & card - > dapm , custom_map , num_routes ) ;
if ( ret )
return ret ;
2016-01-04 17:20:24 -06:00
2018-03-04 15:36:07 +01:00
if ( byt_rt5651_quirk & BYT_RT5651_SSP2_AIF2 ) {
ret = snd_soc_dapm_add_routes ( & card - > dapm ,
byt_rt5651_ssp2_aif2_map ,
ARRAY_SIZE ( byt_rt5651_ssp2_aif2_map ) ) ;
} else if ( byt_rt5651_quirk & BYT_RT5651_SSP0_AIF1 ) {
ret = snd_soc_dapm_add_routes ( & card - > dapm ,
byt_rt5651_ssp0_aif1_map ,
ARRAY_SIZE ( byt_rt5651_ssp0_aif1_map ) ) ;
} else if ( byt_rt5651_quirk & BYT_RT5651_SSP0_AIF2 ) {
ret = snd_soc_dapm_add_routes ( & card - > dapm ,
byt_rt5651_ssp0_aif2_map ,
ARRAY_SIZE ( byt_rt5651_ssp0_aif2_map ) ) ;
} else {
ret = snd_soc_dapm_add_routes ( & card - > dapm ,
byt_rt5651_ssp2_aif1_map ,
ARRAY_SIZE ( byt_rt5651_ssp2_aif1_map ) ) ;
}
if ( ret )
return ret ;
2016-01-04 17:20:24 -06:00
ret = snd_soc_add_card_controls ( card , byt_rt5651_controls ,
ARRAY_SIZE ( byt_rt5651_controls ) ) ;
if ( ret ) {
dev_err ( card - > dev , " unable to add card controls \n " ) ;
return ret ;
}
snd_soc_dapm_ignore_suspend ( & card - > dapm , " Headphone " ) ;
snd_soc_dapm_ignore_suspend ( & card - > dapm , " Speaker " ) ;
2017-10-12 18:37:57 -05:00
if ( byt_rt5651_quirk & BYT_RT5651_MCLK_EN ) {
/*
* The firmware might enable the clock at
* boot ( this information may or may not
* be reflected in the enable clock register ) .
* To change the rate we must disable the clock
* first to cover these cases . Due to common
* clock framework restrictions that do not allow
* to disable a clock that has not been enabled ,
* we need to enable the clock first .
*/
ret = clk_prepare_enable ( priv - > mclk ) ;
if ( ! ret )
clk_disable_unprepare ( priv - > mclk ) ;
if ( byt_rt5651_quirk & BYT_RT5651_MCLK_25MHZ )
ret = clk_set_rate ( priv - > mclk , 25000000 ) ;
else
ret = clk_set_rate ( priv - > mclk , 19200000 ) ;
if ( ret )
dev_err ( card - > dev , " unable to set MCLK rate \n " ) ;
}
2018-03-04 15:36:01 +01:00
if ( BYT_RT5651_JDSRC ( byt_rt5651_quirk ) ) {
ret = snd_soc_card_jack_new ( runtime - > card , " Headset " ,
2018-07-05 00:59:35 +02:00
SND_JACK_HEADSET | SND_JACK_BTN_0 ,
& priv - > jack , bytcr_jack_pins ,
ARRAY_SIZE ( bytcr_jack_pins ) ) ;
2018-03-04 15:36:01 +01:00
if ( ret ) {
dev_err ( runtime - > dev , " jack creation failed %d \n " , ret ) ;
return ret ;
}
2017-10-20 12:18:56 +01:00
2018-07-05 00:59:35 +02:00
snd_jack_set_key ( priv - > jack . jack , SND_JACK_BTN_0 ,
KEY_PLAYPAUSE ) ;
2018-03-04 15:36:01 +01:00
ret = snd_soc_component_set_jack ( codec , & priv - > jack , NULL ) ;
if ( ret )
return ret ;
}
2017-10-20 12:18:56 +01:00
2018-03-04 15:36:01 +01:00
return 0 ;
2016-01-04 17:20:24 -06:00
}
static const struct snd_soc_pcm_stream byt_rt5651_dai_params = {
. formats = SNDRV_PCM_FMTBIT_S24_LE ,
. rate_min = 48000 ,
. rate_max = 48000 ,
. channels_min = 2 ,
. channels_max = 2 ,
} ;
static int byt_rt5651_codec_fixup ( struct snd_soc_pcm_runtime * rtd ,
struct snd_pcm_hw_params * params )
{
struct snd_interval * rate = hw_param_interval ( params ,
SNDRV_PCM_HW_PARAM_RATE ) ;
struct snd_interval * channels = hw_param_interval ( params ,
SNDRV_PCM_HW_PARAM_CHANNELS ) ;
2018-03-04 15:36:07 +01:00
int ret , bits ;
2016-01-04 17:20:24 -06:00
2018-03-04 15:36:07 +01:00
/* The DSP will covert the FE rate to 48k, stereo */
2016-01-04 17:20:24 -06:00
rate - > min = rate - > max = 48000 ;
channels - > min = channels - > max = 2 ;
2018-03-04 15:36:07 +01:00
if ( ( byt_rt5651_quirk & BYT_RT5651_SSP0_AIF1 ) | |
( byt_rt5651_quirk & BYT_RT5651_SSP0_AIF2 ) ) {
/* set SSP0 to 16-bit */
params_set_format ( params , SNDRV_PCM_FORMAT_S16_LE ) ;
bits = 16 ;
} else {
/* set SSP2 to 24-bit */
params_set_format ( params , SNDRV_PCM_FORMAT_S24_LE ) ;
bits = 24 ;
}
2016-01-04 17:20:24 -06:00
/*
* Default mode for SSP configuration is TDM 4 slot , override config
2018-03-04 15:36:07 +01:00
* with explicit setting to I2S 2 ch . The word length is set with
2016-01-04 17:20:24 -06:00
* dai_set_tdm_slot ( ) since there is no other API exposed
*/
ret = snd_soc_dai_set_fmt ( rtd - > cpu_dai ,
SND_SOC_DAIFMT_I2S |
2017-01-26 14:09:29 -06:00
SND_SOC_DAIFMT_NB_NF |
2016-01-04 17:20:24 -06:00
SND_SOC_DAIFMT_CBS_CFS
) ;
if ( ret < 0 ) {
dev_err ( rtd - > dev , " can't set format to I2S, err %d \n " , ret ) ;
return ret ;
}
2018-03-04 15:36:07 +01:00
ret = snd_soc_dai_set_tdm_slot ( rtd - > cpu_dai , 0x3 , 0x3 , 2 , bits ) ;
2016-01-04 17:20:24 -06:00
if ( ret < 0 ) {
dev_err ( rtd - > dev , " can't set I2S config, err %d \n " , ret ) ;
return ret ;
}
return 0 ;
}
2017-06-08 23:37:22 +02:00
static const unsigned int rates_48000 [ ] = {
2016-01-04 17:20:24 -06:00
48000 ,
} ;
2017-06-08 23:37:22 +02:00
static const struct snd_pcm_hw_constraint_list constraints_48000 = {
2016-01-04 17:20:24 -06:00
. count = ARRAY_SIZE ( rates_48000 ) ,
. list = rates_48000 ,
} ;
static int byt_rt5651_aif1_startup ( struct snd_pcm_substream * substream )
{
return snd_pcm_hw_constraint_list ( substream - > runtime , 0 ,
SNDRV_PCM_HW_PARAM_RATE ,
& constraints_48000 ) ;
}
2016-10-15 16:55:49 +02:00
static const struct snd_soc_ops byt_rt5651_aif1_ops = {
2016-01-04 17:20:24 -06:00
. startup = byt_rt5651_aif1_startup ,
} ;
2016-10-15 16:55:49 +02:00
static const struct snd_soc_ops byt_rt5651_be_ssp2_ops = {
2016-01-04 17:20:24 -06:00
. hw_params = byt_rt5651_aif1_hw_params ,
} ;
static struct snd_soc_dai_link byt_rt5651_dais [ ] = {
[ MERR_DPCM_AUDIO ] = {
. name = " Audio Port " ,
. stream_name = " Audio " ,
. cpu_dai_name = " media-cpu-dai " ,
. codec_dai_name = " snd-soc-dummy-dai " ,
. codec_name = " snd-soc-dummy " ,
. platform_name = " sst-mfld-platform " ,
. nonatomic = true ,
. dynamic = 1 ,
. dpcm_playback = 1 ,
. dpcm_capture = 1 ,
. ops = & byt_rt5651_aif1_ops ,
} ,
[ MERR_DPCM_DEEP_BUFFER ] = {
. name = " Deep-Buffer Audio Port " ,
. stream_name = " Deep-Buffer Audio " ,
. cpu_dai_name = " deepbuffer-cpu-dai " ,
. codec_dai_name = " snd-soc-dummy-dai " ,
. codec_name = " snd-soc-dummy " ,
. platform_name = " sst-mfld-platform " ,
. nonatomic = true ,
. dynamic = 1 ,
. dpcm_playback = 1 ,
. ops = & byt_rt5651_aif1_ops ,
} ,
/* CODEC<->CODEC link */
/* back ends */
{
. name = " SSP2-Codec " ,
2017-10-12 18:38:04 -05:00
. id = 0 ,
2016-01-04 17:20:24 -06:00
. cpu_dai_name = " ssp2-port " ,
. platform_name = " sst-mfld-platform " ,
. no_pcm = 1 ,
. codec_dai_name = " rt5651-aif1 " ,
. codec_name = " i2c-10EC5651:00 " ,
. dai_fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF
| SND_SOC_DAIFMT_CBS_CFS ,
. be_hw_params_fixup = byt_rt5651_codec_fixup ,
. ignore_suspend = 1 ,
. nonatomic = true ,
. dpcm_playback = 1 ,
. dpcm_capture = 1 ,
. init = byt_rt5651_init ,
. ops = & byt_rt5651_be_ssp2_ops ,
} ,
} ;
/* SoC card */
2018-07-05 00:59:34 +02:00
static char byt_rt5651_codec_name [ SND_ACPI_I2C_ID_LEN ] ;
static char byt_rt5651_codec_aif_name [ 12 ] ; /* = "rt5651-aif[1|2]" */
static char byt_rt5651_cpu_dai_name [ 10 ] ; /* = "ssp[0|2]-port" */
2018-07-18 22:55:41 +02:00
static char byt_rt5651_long_name [ 50 ] ; /* = "bytcr-rt5651-*-spk-*-mic[-swapped-hp]" */
2018-07-05 00:59:34 +02:00
static int byt_rt5651_suspend ( struct snd_soc_card * card )
{
struct snd_soc_component * component ;
if ( ! BYT_RT5651_JDSRC ( byt_rt5651_quirk ) )
return 0 ;
2018-09-18 01:29:55 +00:00
for_each_card_components ( card , component ) {
2018-07-05 00:59:34 +02:00
if ( ! strcmp ( component - > name , byt_rt5651_codec_name ) ) {
dev_dbg ( component - > dev , " disabling jack detect before suspend \n " ) ;
snd_soc_component_set_jack ( component , NULL , NULL ) ;
break ;
}
}
return 0 ;
}
static int byt_rt5651_resume ( struct snd_soc_card * card )
{
struct byt_rt5651_private * priv = snd_soc_card_get_drvdata ( card ) ;
struct snd_soc_component * component ;
if ( ! BYT_RT5651_JDSRC ( byt_rt5651_quirk ) )
return 0 ;
2018-09-18 01:29:55 +00:00
for_each_card_components ( card , component ) {
2018-07-05 00:59:34 +02:00
if ( ! strcmp ( component - > name , byt_rt5651_codec_name ) ) {
dev_dbg ( component - > dev , " re-enabling jack detect after resume \n " ) ;
snd_soc_component_set_jack ( component , & priv - > jack , NULL ) ;
break ;
}
}
return 0 ;
}
2016-01-04 17:20:24 -06:00
static struct snd_soc_card byt_rt5651_card = {
. name = " bytcr-rt5651 " ,
. owner = THIS_MODULE ,
. dai_link = byt_rt5651_dais ,
. num_links = ARRAY_SIZE ( byt_rt5651_dais ) ,
. dapm_widgets = byt_rt5651_widgets ,
. num_dapm_widgets = ARRAY_SIZE ( byt_rt5651_widgets ) ,
. dapm_routes = byt_rt5651_audio_map ,
. num_dapm_routes = ARRAY_SIZE ( byt_rt5651_audio_map ) ,
. fully_routed = true ,
2018-07-05 00:59:34 +02:00
. suspend_pre = byt_rt5651_suspend ,
. resume_post = byt_rt5651_resume ,
2016-01-04 17:20:24 -06:00
} ;
2018-07-01 20:36:29 +02:00
static const struct x86_cpu_id baytrail_cpu_ids [ ] = {
2018-08-07 10:17:27 -07:00
{ X86_VENDOR_INTEL , 6 , INTEL_FAM6_ATOM_SILVERMONT } , /* Valleyview */
2018-07-01 20:36:29 +02:00
{ }
} ;
2018-03-04 15:36:07 +01:00
2018-07-01 20:36:31 +02:00
static const struct x86_cpu_id cherrytrail_cpu_ids [ ] = {
{ X86_VENDOR_INTEL , 6 , INTEL_FAM6_ATOM_AIRMONT } , /* Braswell */
{ }
} ;
2018-07-18 22:55:38 +02:00
static const struct acpi_gpio_params first_gpio = { 0 , 0 , false } ;
static const struct acpi_gpio_params second_gpio = { 1 , 0 , false } ;
2018-07-01 20:36:31 +02:00
2018-07-18 22:55:38 +02:00
static const struct acpi_gpio_mapping byt_rt5651_amp_en_first [ ] = {
{ " ext-amp-enable-gpios " , & first_gpio , 1 } ,
2018-07-01 20:36:31 +02:00
{ } ,
} ;
2018-07-18 22:55:38 +02:00
static const struct acpi_gpio_mapping byt_rt5651_amp_en_second [ ] = {
{ " ext-amp-enable-gpios " , & second_gpio , 1 } ,
{ } ,
} ;
/*
* Some boards have I2cSerialBusV2 , GpioIo , GpioInt as ACPI resources , other
* boards may have I2cSerialBusV2 , GpioInt , GpioIo instead . We want the
* GpioIo one for the ext - amp - enable - gpio and both count for the index in
* acpi_gpio_params index . So we have 2 different mappings and the code
* below figures out which one to use .
*/
struct byt_rt5651_acpi_resource_data {
int gpio_count ;
int gpio_int_idx ;
} ;
static int snd_byt_rt5651_acpi_resource ( struct acpi_resource * ares , void * arg )
{
struct byt_rt5651_acpi_resource_data * data = arg ;
if ( ares - > type ! = ACPI_RESOURCE_TYPE_GPIO )
return 0 ;
if ( ares - > data . gpio . connection_type = = ACPI_RESOURCE_GPIO_TYPE_INT )
data - > gpio_int_idx = data - > gpio_count ;
data - > gpio_count + + ;
return 0 ;
}
static void snd_byt_rt5651_mc_add_amp_en_gpio_mapping ( struct device * codec )
{
struct byt_rt5651_acpi_resource_data data = { 0 , - 1 } ;
LIST_HEAD ( resources ) ;
int ret ;
ret = acpi_dev_get_resources ( ACPI_COMPANION ( codec ) , & resources ,
snd_byt_rt5651_acpi_resource , & data ) ;
if ( ret < 0 ) {
dev_warn ( codec , " Failed to get ACPI resources, not adding external amplifier GPIO mapping \n " ) ;
return ;
}
/* All info we need is gathered during the walk */
acpi_dev_free_resource_list ( & resources ) ;
switch ( data . gpio_int_idx ) {
case 0 :
devm_acpi_dev_add_driver_gpios ( codec , byt_rt5651_amp_en_second ) ;
break ;
case 1 :
devm_acpi_dev_add_driver_gpios ( codec , byt_rt5651_amp_en_first ) ;
break ;
default :
dev_warn ( codec , " Unknown GpioInt index %d, not adding external amplifier GPIO mapping \n " ,
data . gpio_int_idx ) ;
}
}
2018-03-04 15:36:07 +01:00
struct acpi_chan_package { /* ACPICA seems to require 64 bit integers */
u64 aif_value ; /* 1: AIF1, 2: AIF2 */
u64 mclock_value ; /* usually 25MHz (0x17d7940), ignored */
} ;
2017-10-12 18:37:57 -05:00
2016-01-04 17:20:24 -06:00
static int snd_byt_rt5651_mc_probe ( struct platform_device * pdev )
{
2018-07-18 22:55:40 +02:00
const char * const mic_name [ ] = { " dmic " , " in1 " , " in2 " , " in12 " } ;
2017-10-12 18:37:57 -05:00
struct byt_rt5651_private * priv ;
2017-10-12 18:49:38 -05:00
struct snd_soc_acpi_mach * mach ;
2018-07-01 20:36:30 +02:00
struct device * codec_dev ;
2017-10-12 18:37:57 -05:00
const char * i2c_name = NULL ;
2018-06-24 16:06:33 +02:00
const char * hp_swapped ;
2018-03-04 15:36:07 +01:00
bool is_bytcr = false ;
2016-01-04 17:20:24 -06:00
int ret_val = 0 ;
2017-10-12 18:38:05 -05:00
int dai_index = 0 ;
2017-10-12 18:37:57 -05:00
int i ;
2018-04-20 16:14:28 +05:30
priv = devm_kzalloc ( & pdev - > dev , sizeof ( * priv ) , GFP_KERNEL ) ;
2017-10-12 18:37:57 -05:00
if ( ! priv )
return - ENOMEM ;
2016-01-04 17:20:24 -06:00
/* register the soc card */
byt_rt5651_card . dev = & pdev - > dev ;
2017-10-12 18:37:57 -05:00
mach = byt_rt5651_card . dev - > platform_data ;
snd_soc_card_set_drvdata ( & byt_rt5651_card , priv ) ;
/* fix index of codec dai */
for ( i = 0 ; i < ARRAY_SIZE ( byt_rt5651_dais ) ; i + + ) {
if ( ! strcmp ( byt_rt5651_dais [ i ] . codec_name , " i2c-10EC5651:00 " ) ) {
dai_index = i ;
break ;
}
}
/* fixup codec name based on HID */
2018-01-12 23:16:17 +02:00
i2c_name = acpi_dev_get_first_match_name ( mach - > id , NULL , - 1 ) ;
2018-03-04 15:35:58 +01:00
if ( ! i2c_name ) {
dev_err ( & pdev - > dev , " Error cannot find '%s' dev \n " , mach - > id ) ;
return - ENODEV ;
2017-10-12 18:37:57 -05:00
}
2018-03-04 15:35:58 +01:00
snprintf ( byt_rt5651_codec_name , sizeof ( byt_rt5651_codec_name ) ,
" %s%s " , " i2c- " , i2c_name ) ;
byt_rt5651_dais [ dai_index ] . codec_name = byt_rt5651_codec_name ;
2017-10-12 18:37:57 -05:00
2018-07-01 20:36:30 +02:00
codec_dev = bus_find_device_by_name ( & i2c_bus_type , NULL ,
byt_rt5651_codec_name ) ;
if ( ! codec_dev )
return - EPROBE_DEFER ;
2018-03-04 15:36:07 +01:00
/*
* swap SSP0 if bytcr is detected
* ( will be overridden if DMI quirk is detected )
*/
2018-07-01 20:36:29 +02:00
if ( x86_match_cpu ( baytrail_cpu_ids ) ) {
2018-11-01 16:34:50 -05:00
if ( mach - > mach_params . acpi_ipc_irq_index = = 0 )
2018-03-04 15:36:07 +01:00
is_bytcr = true ;
}
if ( is_bytcr ) {
/*
* Baytrail CR platforms may have CHAN package in BIOS , try
* to find relevant routing quirk based as done on Windows
* platforms . We have to read the information directly from the
* BIOS , at this stage the card is not created and the links
* with the codec driver / pdata are non - existent
*/
struct acpi_chan_package chan_package ;
/* format specified: 2 64-bit integers */
struct acpi_buffer format = { sizeof ( " NN " ) , " NN " } ;
struct acpi_buffer state = { 0 , NULL } ;
struct snd_soc_acpi_package_context pkg_ctx ;
bool pkg_found = false ;
state . length = sizeof ( chan_package ) ;
state . pointer = & chan_package ;
pkg_ctx . name = " CHAN " ;
pkg_ctx . length = 2 ;
pkg_ctx . format = & format ;
pkg_ctx . state = & state ;
pkg_ctx . data_valid = false ;
pkg_found = snd_soc_acpi_find_package_from_hid ( mach - > id ,
& pkg_ctx ) ;
if ( pkg_found ) {
if ( chan_package . aif_value = = 1 ) {
dev_info ( & pdev - > dev , " BIOS Routing: AIF1 connected \n " ) ;
byt_rt5651_quirk | = BYT_RT5651_SSP0_AIF1 ;
} else if ( chan_package . aif_value = = 2 ) {
dev_info ( & pdev - > dev , " BIOS Routing: AIF2 connected \n " ) ;
byt_rt5651_quirk | = BYT_RT5651_SSP0_AIF2 ;
} else {
dev_info ( & pdev - > dev , " BIOS Routing isn't valid, ignored \n " ) ;
pkg_found = false ;
}
}
if ( ! pkg_found ) {
/* no BIOS indications, assume SSP0-AIF2 connection */
byt_rt5651_quirk | = BYT_RT5651_SSP0_AIF2 ;
}
}
2017-10-12 18:37:57 -05:00
/* check quirks before creating card */
dmi_check_system ( byt_rt5651_quirk_table ) ;
2018-03-04 15:35:59 +01:00
/* Must be called before register_card, also see declaration comment. */
2018-07-01 20:36:30 +02:00
ret_val = byt_rt5651_add_codec_device_props ( codec_dev ) ;
2018-07-01 20:36:31 +02:00
if ( ret_val ) {
put_device ( codec_dev ) ;
2018-03-04 15:35:59 +01:00
return ret_val ;
2018-07-01 20:36:31 +02:00
}
/* Cherry Trail devices use an external amplifier enable gpio */
if ( x86_match_cpu ( cherrytrail_cpu_ids ) ) {
2018-07-18 22:55:38 +02:00
snd_byt_rt5651_mc_add_amp_en_gpio_mapping ( codec_dev ) ;
2018-07-01 20:36:31 +02:00
priv - > ext_amp_gpio = devm_fwnode_get_index_gpiod_from_child (
& pdev - > dev , " ext-amp-enable " , 0 ,
codec_dev - > fwnode ,
GPIOD_OUT_LOW , " speaker-amp " ) ;
if ( IS_ERR ( priv - > ext_amp_gpio ) ) {
ret_val = PTR_ERR ( priv - > ext_amp_gpio ) ;
switch ( ret_val ) {
case - ENOENT :
priv - > ext_amp_gpio = NULL ;
break ;
default :
dev_err ( & pdev - > dev , " Failed to get ext-amp-enable GPIO: %d \n " ,
ret_val ) ;
/* fall through */
case - EPROBE_DEFER :
put_device ( codec_dev ) ;
return ret_val ;
}
}
}
put_device ( codec_dev ) ;
2018-03-04 15:35:59 +01:00
2017-10-12 18:37:57 -05:00
log_quirks ( & pdev - > dev ) ;
2018-03-04 15:36:07 +01:00
if ( ( byt_rt5651_quirk & BYT_RT5651_SSP2_AIF2 ) | |
( byt_rt5651_quirk & BYT_RT5651_SSP0_AIF2 ) ) {
/* fixup codec aif name */
snprintf ( byt_rt5651_codec_aif_name ,
sizeof ( byt_rt5651_codec_aif_name ) ,
" %s " , " rt5651-aif2 " ) ;
byt_rt5651_dais [ dai_index ] . codec_dai_name =
byt_rt5651_codec_aif_name ;
}
if ( ( byt_rt5651_quirk & BYT_RT5651_SSP0_AIF1 ) | |
( byt_rt5651_quirk & BYT_RT5651_SSP0_AIF2 ) ) {
/* fixup cpu dai name name */
snprintf ( byt_rt5651_cpu_dai_name ,
sizeof ( byt_rt5651_cpu_dai_name ) ,
" %s " , " ssp0-port " ) ;
byt_rt5651_dais [ dai_index ] . cpu_dai_name =
byt_rt5651_cpu_dai_name ;
}
2017-10-12 18:37:57 -05:00
if ( byt_rt5651_quirk & BYT_RT5651_MCLK_EN ) {
priv - > mclk = devm_clk_get ( & pdev - > dev , " pmc_plt_clk_3 " ) ;
if ( IS_ERR ( priv - > mclk ) ) {
2018-05-02 15:29:45 +01:00
ret_val = PTR_ERR ( priv - > mclk ) ;
2017-10-12 18:37:57 -05:00
dev_err ( & pdev - > dev ,
2018-05-02 15:29:45 +01:00
" Failed to get MCLK from pmc_plt_clk_3: %d \n " ,
ret_val ) ;
2017-10-12 18:37:57 -05:00
/*
* Fall back to bit clock usage for - ENOENT ( clock not
* available likely due to missing dependencies ) , bail
* for all other errors , including - EPROBE_DEFER
*/
if ( ret_val ! = - ENOENT )
return ret_val ;
byt_rt5651_quirk & = ~ BYT_RT5651_MCLK_EN ;
}
}
2018-06-24 16:06:33 +02:00
if ( byt_rt5651_quirk & BYT_RT5651_HP_LR_SWAPPED )
hp_swapped = " -hp-swapped " ;
else
hp_swapped = " " ;
2018-05-18 21:35:06 +02:00
snprintf ( byt_rt5651_long_name , sizeof ( byt_rt5651_long_name ) ,
2018-07-18 22:55:41 +02:00
" bytcr-rt5651-%s-spk-%s-mic%s " ,
( byt_rt5651_quirk & BYT_RT5651_MONO_SPEAKER ) ?
" mono " : " stereo " ,
2018-06-24 16:06:33 +02:00
mic_name [ BYT_RT5651_MAP ( byt_rt5651_quirk ) ] , hp_swapped ) ;
2018-05-18 21:35:06 +02:00
byt_rt5651_card . long_name = byt_rt5651_long_name ;
2016-01-04 17:20:24 -06:00
ret_val = devm_snd_soc_register_card ( & pdev - > dev , & byt_rt5651_card ) ;
if ( ret_val ) {
dev_err ( & pdev - > dev , " devm_snd_soc_register_card failed %d \n " ,
ret_val ) ;
return ret_val ;
}
platform_set_drvdata ( pdev , & byt_rt5651_card ) ;
return ret_val ;
}
static struct platform_driver snd_byt_rt5651_mc_driver = {
. driver = {
. name = " bytcr_rt5651 " ,
} ,
. probe = snd_byt_rt5651_mc_probe ,
} ;
module_platform_driver ( snd_byt_rt5651_mc_driver ) ;
MODULE_DESCRIPTION ( " ASoC Intel(R) Baytrail CR Machine driver for RT5651 " ) ;
MODULE_AUTHOR ( " Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com> " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_ALIAS ( " platform:bytcr_rt5651 " ) ;