2014-11-19 12:43:27 +03:00
/*
* byt_cr_dpcm_rt5640 . c - ASoc Machine driver for Intel Byt CR platform
*
* Copyright ( C ) 2014 Intel Corp
* Author : Subhransu S . Prusty < subhransu . s . prusty @ intel . com >
* ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~
*
* 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>
# include <linux/module.h>
# include <linux/platform_device.h>
2015-12-18 05:35:41 +03:00
# include <linux/acpi.h>
2014-11-19 12:43:27 +03:00
# include <linux/device.h>
2015-12-18 05:35:41 +03:00
# include <linux/dmi.h>
2014-11-19 12:43:27 +03:00
# include <linux/slab.h>
2016-08-13 00:27:53 +03:00
# include <asm/cpu_device_id.h>
# include <asm/platform_sst_audio.h>
2016-08-13 00:27:57 +03:00
# include <linux/clk.h>
2014-11-19 12:43:27 +03:00
# include <sound/pcm.h>
# include <sound/pcm_params.h>
# include <sound/soc.h>
2015-12-18 05:35:41 +03:00
# include <sound/jack.h>
2015-04-02 10:37:02 +03:00
# include "../../codecs/rt5640.h"
2015-04-02 10:37:04 +03:00
# include "../atom/sst-atom-controls.h"
2016-01-05 02:20:28 +03:00
# include "../common/sst-acpi.h"
2016-08-13 00:27:53 +03:00
# include "../common/sst-dsp.h"
2014-11-19 12:43:27 +03:00
2016-03-04 06:36:37 +03:00
enum {
BYT_RT5640_DMIC1_MAP ,
BYT_RT5640_DMIC2_MAP ,
BYT_RT5640_IN1_MAP ,
2016-08-13 00:27:54 +03:00
BYT_RT5640_IN3_MAP ,
2016-03-04 06:36:37 +03:00
} ;
# define BYT_RT5640_MAP(quirk) ((quirk) & 0xff)
# define BYT_RT5640_DMIC_EN BIT(16)
2016-08-13 00:27:45 +03:00
# define BYT_RT5640_MONO_SPEAKER BIT(17)
2016-08-13 00:27:46 +03:00
# define BYT_RT5640_DIFF_MIC BIT(18) /* defaut is single-ended */
2016-08-13 00:27:50 +03:00
# define BYT_RT5640_SSP2_AIF2 BIT(19) /* default is using AIF1 */
2016-08-13 00:27:51 +03:00
# define BYT_RT5640_SSP0_AIF1 BIT(20)
# define BYT_RT5640_SSP0_AIF2 BIT(21)
2016-08-13 00:27:57 +03:00
# define BYT_RT5640_MCLK_EN BIT(22)
# define BYT_RT5640_MCLK_25MHZ BIT(23)
struct byt_rt5640_private {
struct clk * mclk ;
} ;
2016-03-04 06:36:37 +03:00
2016-11-13 03:07:46 +03:00
static unsigned long byt_rt5640_quirk = BYT_RT5640_MCLK_EN ;
2016-08-13 00:27:57 +03:00
2016-08-13 00:28:00 +03:00
static void log_quirks ( struct device * dev )
{
if ( BYT_RT5640_MAP ( byt_rt5640_quirk ) = = BYT_RT5640_DMIC1_MAP )
dev_info ( dev , " quirk DMIC1_MAP enabled " ) ;
if ( BYT_RT5640_MAP ( byt_rt5640_quirk ) = = BYT_RT5640_DMIC2_MAP )
dev_info ( dev , " quirk DMIC2_MAP enabled " ) ;
if ( BYT_RT5640_MAP ( byt_rt5640_quirk ) = = BYT_RT5640_IN1_MAP )
dev_info ( dev , " quirk IN1_MAP enabled " ) ;
if ( BYT_RT5640_MAP ( byt_rt5640_quirk ) = = BYT_RT5640_IN3_MAP )
dev_info ( dev , " quirk IN3_MAP enabled " ) ;
if ( byt_rt5640_quirk & BYT_RT5640_DMIC_EN )
dev_info ( dev , " quirk DMIC enabled " ) ;
if ( byt_rt5640_quirk & BYT_RT5640_MONO_SPEAKER )
dev_info ( dev , " quirk MONO_SPEAKER enabled " ) ;
if ( byt_rt5640_quirk & BYT_RT5640_DIFF_MIC )
dev_info ( dev , " quirk DIFF_MIC enabled " ) ;
if ( byt_rt5640_quirk & BYT_RT5640_SSP2_AIF2 )
dev_info ( dev , " quirk SSP2_AIF2 enabled " ) ;
if ( byt_rt5640_quirk & BYT_RT5640_SSP0_AIF1 )
dev_info ( dev , " quirk SSP0_AIF1 enabled " ) ;
if ( byt_rt5640_quirk & BYT_RT5640_SSP0_AIF2 )
dev_info ( dev , " quirk SSP0_AIF2 enabled " ) ;
if ( byt_rt5640_quirk & BYT_RT5640_MCLK_EN )
dev_info ( dev , " quirk MCLK_EN enabled " ) ;
if ( byt_rt5640_quirk & BYT_RT5640_MCLK_25MHZ )
dev_info ( dev , " quirk MCLK_25MHZ enabled " ) ;
}
2016-08-13 00:27:57 +03:00
# define BYT_CODEC_DAI1 "rt5640-aif1"
# define BYT_CODEC_DAI2 "rt5640-aif2"
static inline struct snd_soc_dai * byt_get_codec_dai ( struct snd_soc_card * card )
{
struct snd_soc_pcm_runtime * rtd ;
list_for_each_entry ( rtd , & card - > rtd_list , list ) {
if ( ! strncmp ( rtd - > codec_dai - > name , BYT_CODEC_DAI1 ,
strlen ( BYT_CODEC_DAI1 ) ) )
return rtd - > codec_dai ;
if ( ! strncmp ( rtd - > codec_dai - > name , BYT_CODEC_DAI2 ,
strlen ( BYT_CODEC_DAI2 ) ) )
return rtd - > codec_dai ;
}
return NULL ;
}
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_rt5640_private * priv = snd_soc_card_get_drvdata ( card ) ;
int ret ;
codec_dai = byt_get_codec_dai ( card ) ;
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_rt5640_quirk & BYT_RT5640_MCLK_EN ) & & priv - > mclk ) {
ret = clk_prepare_enable ( priv - > mclk ) ;
if ( ret < 0 ) {
dev_err ( card - > dev ,
" could not configure MCLK state " ) ;
return ret ;
}
}
ret = snd_soc_dai_set_sysclk ( codec_dai , RT5640_SCLK_S_PLL1 ,
48000 * 512 ,
SND_SOC_CLOCK_IN ) ;
} 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 , RT5640_SCLK_S_RCCLK ,
0 ,
SND_SOC_CLOCK_IN ) ;
if ( ! ret ) {
if ( ( byt_rt5640_quirk & BYT_RT5640_MCLK_EN ) & & priv - > mclk )
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-03-04 06:36:37 +03:00
2015-12-18 05:35:41 +03:00
static const struct snd_soc_dapm_widget byt_rt5640_widgets [ ] = {
2014-11-19 12:43:27 +03:00
SND_SOC_DAPM_HP ( " Headphone " , NULL ) ,
SND_SOC_DAPM_MIC ( " Headset Mic " , NULL ) ,
2015-12-18 05:35:40 +03:00
SND_SOC_DAPM_MIC ( " Internal Mic " , NULL ) ,
SND_SOC_DAPM_SPK ( " Speaker " , NULL ) ,
2016-08-13 00:27:57 +03: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 ) ,
2014-11-19 12:43:27 +03:00
} ;
2015-12-18 05:35:41 +03:00
static const struct snd_soc_dapm_route byt_rt5640_audio_map [ ] = {
2016-08-13 00:27:57 +03:00
{ " Headphone " , NULL , " Platform Clock " } ,
{ " Headset Mic " , NULL , " Platform Clock " } ,
{ " Internal Mic " , NULL , " Platform Clock " } ,
{ " Speaker " , NULL , " Platform Clock " } ,
2014-11-19 12:43:27 +03:00
{ " Headset Mic " , NULL , " MICBIAS1 " } ,
2015-12-18 05:35:40 +03:00
{ " IN2P " , NULL , " Headset Mic " } ,
2014-11-19 12:43:27 +03:00
{ " Headphone " , NULL , " HPOL " } ,
{ " Headphone " , NULL , " HPOR " } ,
} ;
2015-12-18 05:35:41 +03:00
static const struct snd_soc_dapm_route byt_rt5640_intmic_dmic1_map [ ] = {
{ " DMIC1 " , NULL , " Internal Mic " } ,
} ;
static const struct snd_soc_dapm_route byt_rt5640_intmic_dmic2_map [ ] = {
{ " DMIC2 " , NULL , " Internal Mic " } ,
} ;
static const struct snd_soc_dapm_route byt_rt5640_intmic_in1_map [ ] = {
{ " Internal Mic " , NULL , " MICBIAS1 " } ,
{ " IN1P " , NULL , " Internal Mic " } ,
} ;
2016-08-13 00:27:54 +03:00
static const struct snd_soc_dapm_route byt_rt5640_intmic_in3_map [ ] = {
{ " Internal Mic " , NULL , " MICBIAS1 " } ,
{ " IN3P " , NULL , " Internal Mic " } ,
} ;
2016-08-13 00:27:50 +03:00
static const struct snd_soc_dapm_route byt_rt5640_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_rt5640_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-08-13 00:27:51 +03:00
static const struct snd_soc_dapm_route byt_rt5640_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_rt5640_ssp0_aif2_map [ ] = {
{ " ssp0 Tx " , NULL , " modem_out " } ,
{ " modem_in " , NULL , " ssp0 Rx " } ,
{ " AIF2 Playback " , NULL , " ssp0 Tx " } ,
{ " ssp0 Rx " , NULL , " AIF2 Capture " } ,
} ;
2016-08-13 00:27:45 +03:00
static const struct snd_soc_dapm_route byt_rt5640_stereo_spk_map [ ] = {
{ " Speaker " , NULL , " SPOLP " } ,
{ " Speaker " , NULL , " SPOLN " } ,
{ " Speaker " , NULL , " SPORP " } ,
{ " Speaker " , NULL , " SPORN " } ,
} ;
static const struct snd_soc_dapm_route byt_rt5640_mono_spk_map [ ] = {
{ " Speaker " , NULL , " SPOLP " } ,
{ " Speaker " , NULL , " SPOLN " } ,
} ;
2015-12-18 05:35:41 +03:00
static const struct snd_kcontrol_new byt_rt5640_controls [ ] = {
2014-11-19 12:43:27 +03:00
SOC_DAPM_PIN_SWITCH ( " Headphone " ) ,
SOC_DAPM_PIN_SWITCH ( " Headset Mic " ) ,
2015-12-18 05:35:40 +03:00
SOC_DAPM_PIN_SWITCH ( " Internal Mic " ) ,
SOC_DAPM_PIN_SWITCH ( " Speaker " ) ,
2014-11-19 12:43:27 +03:00
} ;
2015-12-18 05:35:41 +03:00
static int byt_rt5640_aif1_hw_params ( struct snd_pcm_substream * substream ,
2014-11-19 12:43:27 +03:00
struct snd_pcm_hw_params * params )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
struct snd_soc_dai * codec_dai = rtd - > codec_dai ;
int ret ;
ret = snd_soc_dai_set_sysclk ( codec_dai , RT5640_SCLK_S_PLL1 ,
params_rate ( params ) * 512 ,
SND_SOC_CLOCK_IN ) ;
2016-08-13 00:27:57 +03:00
2014-11-19 12:43:27 +03:00
if ( ret < 0 ) {
dev_err ( rtd - > dev , " can't set codec clock %d \n " , ret ) ;
return ret ;
}
2016-08-13 00:27:57 +03:00
if ( ! ( byt_rt5640_quirk & BYT_RT5640_MCLK_EN ) ) {
/* use bitclock as PLL input */
if ( ( byt_rt5640_quirk & BYT_RT5640_SSP0_AIF1 ) | |
( byt_rt5640_quirk & BYT_RT5640_SSP0_AIF2 ) ) {
/* 2x16 bit slots on SSP0 */
ret = snd_soc_dai_set_pll ( codec_dai , 0 ,
RT5640_PLL1_S_BCLK1 ,
params_rate ( params ) * 32 ,
params_rate ( params ) * 512 ) ;
} else {
/* 2x15 bit slots on SSP2 */
ret = snd_soc_dai_set_pll ( codec_dai , 0 ,
RT5640_PLL1_S_BCLK1 ,
params_rate ( params ) * 50 ,
params_rate ( params ) * 512 ) ;
}
2016-08-13 00:27:52 +03:00
} else {
2016-08-13 00:27:57 +03:00
if ( byt_rt5640_quirk & BYT_RT5640_MCLK_25MHZ ) {
ret = snd_soc_dai_set_pll ( codec_dai , 0 ,
RT5640_PLL1_S_MCLK ,
25000000 ,
params_rate ( params ) * 512 ) ;
} else {
ret = snd_soc_dai_set_pll ( codec_dai , 0 ,
RT5640_PLL1_S_MCLK ,
19200000 ,
params_rate ( params ) * 512 ) ;
}
2016-08-13 00:27:52 +03:00
}
2014-11-19 12:43:27 +03:00
if ( ret < 0 ) {
dev_err ( rtd - > dev , " can't set codec pll: %d \n " , ret ) ;
return ret ;
}
return 0 ;
}
2015-12-18 05:35:41 +03:00
static int byt_rt5640_quirk_cb ( const struct dmi_system_id * id )
{
byt_rt5640_quirk = ( unsigned long ) id - > driver_data ;
return 1 ;
}
static const struct dmi_system_id byt_rt5640_quirk_table [ ] = {
{
. callback = byt_rt5640_quirk_cb ,
. matches = {
2016-08-13 00:27:44 +03:00
DMI_EXACT_MATCH ( DMI_SYS_VENDOR , " ASUSTeK COMPUTER INC. " ) ,
DMI_EXACT_MATCH ( DMI_PRODUCT_NAME , " T100TA " ) ,
2015-12-18 05:35:41 +03:00
} ,
2016-08-13 00:27:57 +03:00
. driver_data = ( unsigned long * ) ( BYT_RT5640_IN1_MAP |
BYT_RT5640_MCLK_EN ) ,
2015-12-18 05:35:41 +03:00
} ,
{
. callback = byt_rt5640_quirk_cb ,
. matches = {
2016-08-13 00:27:44 +03:00
DMI_EXACT_MATCH ( DMI_SYS_VENDOR , " ASUSTeK COMPUTER INC. " ) ,
DMI_EXACT_MATCH ( DMI_PRODUCT_NAME , " T100TAF " ) ,
} ,
2016-08-13 00:27:45 +03:00
. driver_data = ( unsigned long * ) ( BYT_RT5640_IN1_MAP |
2016-08-13 00:27:46 +03:00
BYT_RT5640_MONO_SPEAKER |
2016-08-13 00:27:53 +03:00
BYT_RT5640_DIFF_MIC |
2016-08-13 00:27:57 +03:00
BYT_RT5640_SSP0_AIF2 |
BYT_RT5640_MCLK_EN
2016-08-13 00:27:45 +03:00
) ,
2016-08-13 00:27:44 +03:00
} ,
{
. callback = byt_rt5640_quirk_cb ,
. matches = {
DMI_EXACT_MATCH ( DMI_SYS_VENDOR , " DellInc. " ) ,
DMI_EXACT_MATCH ( DMI_PRODUCT_NAME , " Venue 8 Pro 5830 " ) ,
2015-12-18 05:35:41 +03:00
} ,
. driver_data = ( unsigned long * ) ( BYT_RT5640_DMIC2_MAP |
2016-08-13 00:27:57 +03:00
BYT_RT5640_DMIC_EN |
BYT_RT5640_MCLK_EN ) ,
2015-12-18 05:35:41 +03:00
} ,
2016-01-05 02:20:30 +03:00
{
. callback = byt_rt5640_quirk_cb ,
. matches = {
2016-08-13 00:27:44 +03:00
DMI_EXACT_MATCH ( DMI_SYS_VENDOR , " Hewlett-Packard " ) ,
DMI_EXACT_MATCH ( DMI_PRODUCT_NAME , " HP ElitePad 1000 G2 " ) ,
2016-01-05 02:20:30 +03:00
} ,
2016-08-13 00:27:57 +03:00
. driver_data = ( unsigned long * ) ( BYT_RT5640_IN1_MAP |
BYT_RT5640_MCLK_EN ) ,
2016-01-05 02:20:30 +03:00
} ,
2016-08-13 00:27:58 +03:00
{
. callback = byt_rt5640_quirk_cb ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Circuitco " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Minnowboard Max B3 PLATFORM " ) ,
} ,
. driver_data = ( unsigned long * ) ( BYT_RT5640_DMIC1_MAP |
BYT_RT5640_DMIC_EN ) ,
} ,
2016-08-13 00:27:59 +03:00
{
. callback = byt_rt5640_quirk_cb ,
. matches = {
DMI_MATCH ( DMI_BOARD_VENDOR , " TECLAST " ) ,
DMI_MATCH ( DMI_BOARD_NAME , " tPAD " ) ,
} ,
. driver_data = ( unsigned long * ) ( BYT_RT5640_IN3_MAP |
BYT_RT5640_MCLK_EN |
BYT_RT5640_SSP0_AIF1 ) ,
} ,
2016-08-17 02:28:22 +03:00
{
. callback = byt_rt5640_quirk_cb ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Acer " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Aspire SW5-012 " ) ,
} ,
. driver_data = ( unsigned long * ) ( BYT_RT5640_IN1_MAP |
BYT_RT5640_MCLK_EN |
BYT_RT5640_SSP0_AIF1 ) ,
} ,
2015-12-18 05:35:41 +03:00
{ }
} ;
2015-12-18 05:35:42 +03:00
static int byt_rt5640_init ( struct snd_soc_pcm_runtime * runtime )
{
int ret ;
struct snd_soc_codec * codec = runtime - > codec ;
struct snd_soc_card * card = runtime - > card ;
const struct snd_soc_dapm_route * custom_map ;
2016-08-13 00:27:57 +03:00
struct byt_rt5640_private * priv = snd_soc_card_get_drvdata ( card ) ;
2015-12-18 05:35:42 +03:00
int num_routes ;
card - > dapm . idle_bias_off = true ;
2016-01-05 02:20:27 +03:00
rt5640_sel_asrc_clk_src ( codec ,
RT5640_DA_STEREO_FILTER |
2016-08-13 00:27:57 +03:00
RT5640_DA_MONO_L_FILTER |
RT5640_DA_MONO_R_FILTER |
RT5640_AD_STEREO_FILTER |
RT5640_AD_MONO_L_FILTER |
RT5640_AD_MONO_R_FILTER ,
2016-01-05 02:20:27 +03:00
RT5640_CLK_SEL_ASRC ) ;
2015-12-18 05:35:42 +03:00
ret = snd_soc_add_card_controls ( card , byt_rt5640_controls ,
ARRAY_SIZE ( byt_rt5640_controls ) ) ;
if ( ret ) {
dev_err ( card - > dev , " unable to add card controls \n " ) ;
return ret ;
}
switch ( BYT_RT5640_MAP ( byt_rt5640_quirk ) ) {
case BYT_RT5640_IN1_MAP :
custom_map = byt_rt5640_intmic_in1_map ;
num_routes = ARRAY_SIZE ( byt_rt5640_intmic_in1_map ) ;
break ;
2016-08-13 00:27:54 +03:00
case BYT_RT5640_IN3_MAP :
custom_map = byt_rt5640_intmic_in3_map ;
num_routes = ARRAY_SIZE ( byt_rt5640_intmic_in3_map ) ;
break ;
2015-12-18 05:35:42 +03:00
case BYT_RT5640_DMIC2_MAP :
custom_map = byt_rt5640_intmic_dmic2_map ;
num_routes = ARRAY_SIZE ( byt_rt5640_intmic_dmic2_map ) ;
break ;
default :
custom_map = byt_rt5640_intmic_dmic1_map ;
num_routes = ARRAY_SIZE ( byt_rt5640_intmic_dmic1_map ) ;
}
ret = snd_soc_dapm_add_routes ( & card - > dapm , custom_map , num_routes ) ;
if ( ret )
return ret ;
2016-08-13 00:27:50 +03:00
if ( byt_rt5640_quirk & BYT_RT5640_SSP2_AIF2 ) {
ret = snd_soc_dapm_add_routes ( & card - > dapm ,
byt_rt5640_ssp2_aif2_map ,
ARRAY_SIZE ( byt_rt5640_ssp2_aif2_map ) ) ;
2016-08-13 00:27:51 +03:00
} else if ( byt_rt5640_quirk & BYT_RT5640_SSP0_AIF1 ) {
ret = snd_soc_dapm_add_routes ( & card - > dapm ,
byt_rt5640_ssp0_aif1_map ,
ARRAY_SIZE ( byt_rt5640_ssp0_aif1_map ) ) ;
} else if ( byt_rt5640_quirk & BYT_RT5640_SSP0_AIF2 ) {
ret = snd_soc_dapm_add_routes ( & card - > dapm ,
byt_rt5640_ssp0_aif2_map ,
ARRAY_SIZE ( byt_rt5640_ssp0_aif2_map ) ) ;
2016-08-13 00:27:50 +03:00
} else {
ret = snd_soc_dapm_add_routes ( & card - > dapm ,
byt_rt5640_ssp2_aif1_map ,
ARRAY_SIZE ( byt_rt5640_ssp2_aif1_map ) ) ;
}
if ( ret )
return ret ;
2016-08-13 00:27:45 +03:00
if ( byt_rt5640_quirk & BYT_RT5640_MONO_SPEAKER ) {
ret = snd_soc_dapm_add_routes ( & card - > dapm ,
byt_rt5640_mono_spk_map ,
ARRAY_SIZE ( byt_rt5640_mono_spk_map ) ) ;
} else {
ret = snd_soc_dapm_add_routes ( & card - > dapm ,
byt_rt5640_stereo_spk_map ,
ARRAY_SIZE ( byt_rt5640_stereo_spk_map ) ) ;
}
if ( ret )
return ret ;
2016-08-13 00:27:46 +03:00
if ( byt_rt5640_quirk & BYT_RT5640_DIFF_MIC ) {
snd_soc_update_bits ( codec , RT5640_IN1_IN2 , RT5640_IN_DF1 ,
RT5640_IN_DF1 ) ;
}
2015-12-18 05:35:42 +03:00
if ( byt_rt5640_quirk & BYT_RT5640_DMIC_EN ) {
ret = rt5640_dmic_enable ( codec , 0 , 0 ) ;
if ( ret )
return ret ;
}
snd_soc_dapm_ignore_suspend ( & card - > dapm , " Headphone " ) ;
snd_soc_dapm_ignore_suspend ( & card - > dapm , " Speaker " ) ;
2016-08-13 00:27:57 +03:00
if ( ( byt_rt5640_quirk & BYT_RT5640_MCLK_EN ) & & priv - > mclk ) {
/*
* 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_rt5640_quirk & BYT_RT5640_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 " ) ;
}
2015-12-18 05:35:42 +03:00
return ret ;
}
2015-12-18 05:35:41 +03:00
static const struct snd_soc_pcm_stream byt_rt5640_dai_params = {
2014-11-19 12:43:27 +03:00
. formats = SNDRV_PCM_FMTBIT_S24_LE ,
. rate_min = 48000 ,
. rate_max = 48000 ,
. channels_min = 2 ,
. channels_max = 2 ,
} ;
2015-12-18 05:35:41 +03:00
static int byt_rt5640_codec_fixup ( struct snd_soc_pcm_runtime * rtd ,
2014-11-19 12:43:27 +03:00
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 ) ;
2015-12-18 05:35:39 +03:00
int ret ;
2014-11-19 12:43:27 +03:00
2016-08-13 00:27:52 +03:00
/* The DSP will covert the FE rate to 48k, stereo */
2014-11-19 12:43:27 +03:00
rate - > min = rate - > max = 48000 ;
channels - > min = channels - > max = 2 ;
2016-08-13 00:27:52 +03:00
if ( ( byt_rt5640_quirk & BYT_RT5640_SSP0_AIF1 ) | |
( byt_rt5640_quirk & BYT_RT5640_SSP0_AIF2 ) ) {
2016-08-17 02:28:22 +03:00
/* set SSP0 to 16-bit */
2016-08-13 00:27:52 +03:00
params_set_format ( params , SNDRV_PCM_FORMAT_S16_LE ) ;
/*
* Default mode for SSP configuration is TDM 4 slot , override config
* with explicit setting to I2S 2 ch 16 - bit . The word length is set with
* dai_set_tdm_slot ( ) since there is no other API exposed
*/
ret = snd_soc_dai_set_fmt ( rtd - > cpu_dai ,
SND_SOC_DAIFMT_I2S |
SND_SOC_DAIFMT_NB_IF |
SND_SOC_DAIFMT_CBS_CFS
) ;
if ( ret < 0 ) {
dev_err ( rtd - > dev , " can't set format to I2S, err %d \n " , ret ) ;
return ret ;
}
2015-12-18 05:35:39 +03:00
2016-08-13 00:27:52 +03:00
ret = snd_soc_dai_set_tdm_slot ( rtd - > cpu_dai , 0x3 , 0x3 , 2 , 16 ) ;
if ( ret < 0 ) {
dev_err ( rtd - > dev , " can't set I2S config, err %d \n " , ret ) ;
return ret ;
}
} else {
2015-12-18 05:35:39 +03:00
2016-08-13 00:27:52 +03:00
/* set SSP2 to 24-bit */
params_set_format ( params , SNDRV_PCM_FORMAT_S24_LE ) ;
/*
* Default mode for SSP configuration is TDM 4 slot , override config
* with explicit setting to I2S 2 ch 24 - bit . The word length is set with
* dai_set_tdm_slot ( ) since there is no other API exposed
*/
ret = snd_soc_dai_set_fmt ( rtd - > cpu_dai ,
SND_SOC_DAIFMT_I2S |
SND_SOC_DAIFMT_NB_IF |
SND_SOC_DAIFMT_CBS_CFS
) ;
if ( ret < 0 ) {
dev_err ( rtd - > dev , " can't set format to I2S, err %d \n " , ret ) ;
return ret ;
}
ret = snd_soc_dai_set_tdm_slot ( rtd - > cpu_dai , 0x3 , 0x3 , 2 , 24 ) ;
if ( ret < 0 ) {
dev_err ( rtd - > dev , " can't set I2S config, err %d \n " , ret ) ;
return ret ;
}
}
2014-11-19 12:43:27 +03:00
return 0 ;
}
2015-12-18 05:35:41 +03:00
static int byt_rt5640_aif1_startup ( struct snd_pcm_substream * substream )
2014-11-19 12:43:27 +03:00
{
2015-10-18 16:39:30 +03:00
return snd_pcm_hw_constraint_single ( substream - > runtime ,
SNDRV_PCM_HW_PARAM_RATE , 48000 ) ;
2014-11-19 12:43:27 +03:00
}
2016-10-15 17:55:49 +03:00
static const struct snd_soc_ops byt_rt5640_aif1_ops = {
2015-12-18 05:35:41 +03:00
. startup = byt_rt5640_aif1_startup ,
2014-11-19 12:43:27 +03:00
} ;
2016-10-15 17:55:49 +03:00
static const struct snd_soc_ops byt_rt5640_be_ssp2_ops = {
2015-12-18 05:35:41 +03:00
. hw_params = byt_rt5640_aif1_hw_params ,
2014-11-19 12:43:27 +03:00
} ;
2015-12-18 05:35:41 +03:00
static struct snd_soc_dai_link byt_rt5640_dais [ ] = {
2014-11-19 12:43:27 +03:00
[ MERR_DPCM_AUDIO ] = {
. name = " Baytrail Audio Port " ,
. stream_name = " Baytrail Audio " ,
. cpu_dai_name = " media-cpu-dai " ,
. codec_dai_name = " snd-soc-dummy-dai " ,
. codec_name = " snd-soc-dummy " ,
. platform_name = " sst-mfld-platform " ,
. ignore_suspend = 1 ,
. dynamic = 1 ,
. dpcm_playback = 1 ,
. dpcm_capture = 1 ,
2015-12-18 05:35:41 +03:00
. ops = & byt_rt5640_aif1_ops ,
2014-11-19 12:43:27 +03:00
} ,
2015-12-18 05:35:45 +03:00
[ 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 " ,
. ignore_suspend = 1 ,
. nonatomic = true ,
. dynamic = 1 ,
. dpcm_playback = 1 ,
. ops = & byt_rt5640_aif1_ops ,
} ,
2014-11-19 12:43:27 +03:00
[ MERR_DPCM_COMPR ] = {
. name = " Baytrail Compressed Port " ,
. stream_name = " Baytrail Compress " ,
. cpu_dai_name = " compress-cpu-dai " ,
. codec_dai_name = " snd-soc-dummy-dai " ,
. codec_name = " snd-soc-dummy " ,
. platform_name = " sst-mfld-platform " ,
} ,
/* back ends */
{
. name = " SSP2-Codec " ,
2016-04-19 08:12:35 +03:00
. id = 1 ,
2016-08-13 00:27:51 +03:00
. cpu_dai_name = " ssp2-port " , /* overwritten for ssp0 routing */
2014-11-19 12:43:27 +03:00
. platform_name = " sst-mfld-platform " ,
. no_pcm = 1 ,
2016-08-13 00:27:50 +03:00
. codec_dai_name = " rt5640-aif1 " , /* changed w/ quirk */
2016-01-05 02:20:29 +03:00
. codec_name = " i2c-10EC5640:00 " , /* overwritten with HID */
2014-11-19 12:43:27 +03:00
. dai_fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF
| SND_SOC_DAIFMT_CBS_CFS ,
2015-12-18 05:35:41 +03:00
. be_hw_params_fixup = byt_rt5640_codec_fixup ,
2014-11-19 12:43:27 +03:00
. ignore_suspend = 1 ,
. dpcm_playback = 1 ,
. dpcm_capture = 1 ,
2015-12-18 05:35:42 +03:00
. init = byt_rt5640_init ,
2015-12-18 05:35:41 +03:00
. ops = & byt_rt5640_be_ssp2_ops ,
2014-11-19 12:43:27 +03:00
} ,
} ;
/* SoC card */
2015-12-18 05:35:42 +03:00
static struct snd_soc_card byt_rt5640_card = {
2015-12-18 05:35:41 +03:00
. name = " bytcr-rt5640 " ,
2015-08-21 15:59:21 +03:00
. owner = THIS_MODULE ,
2015-12-18 05:35:41 +03:00
. dai_link = byt_rt5640_dais ,
. num_links = ARRAY_SIZE ( byt_rt5640_dais ) ,
. dapm_widgets = byt_rt5640_widgets ,
. num_dapm_widgets = ARRAY_SIZE ( byt_rt5640_widgets ) ,
. dapm_routes = byt_rt5640_audio_map ,
. num_dapm_routes = ARRAY_SIZE ( byt_rt5640_audio_map ) ,
2015-12-18 05:35:42 +03:00
. fully_routed = true ,
2014-11-19 12:43:27 +03:00
} ;
2016-01-05 02:20:28 +03:00
static char byt_rt5640_codec_name [ 16 ] ; /* i2c-<HID>:00 with HID being 8 chars */
2016-08-13 00:27:50 +03:00
static char byt_rt5640_codec_aif_name [ 12 ] ; /* = "rt5640-aif[1|2]" */
2016-08-13 00:27:51 +03:00
static char byt_rt5640_cpu_dai_name [ 10 ] ; /* = "ssp[0|2]-port" */
2016-01-05 02:20:28 +03:00
2016-08-13 00:27:53 +03:00
static bool is_valleyview ( void )
{
2016-08-17 02:28:21 +03:00
static const struct x86_cpu_id cpu_ids [ ] = {
2016-08-13 00:27:53 +03:00
{ X86_VENDOR_INTEL , 6 , 55 } , /* Valleyview, Bay Trail */
{ }
} ;
if ( ! x86_match_cpu ( cpu_ids ) )
return false ;
return true ;
}
2016-11-13 03:07:45 +03: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 */
} ;
2016-08-13 00:27:57 +03:00
2015-12-18 05:35:41 +03:00
static int snd_byt_rt5640_mc_probe ( struct platform_device * pdev )
2014-11-19 12:43:27 +03:00
{
int ret_val = 0 ;
2016-01-05 02:20:28 +03:00
struct sst_acpi_mach * mach ;
2016-03-04 06:36:38 +03:00
const char * i2c_name = NULL ;
int i ;
int dai_index ;
2016-08-13 00:27:57 +03:00
struct byt_rt5640_private * priv ;
2016-11-13 03:07:45 +03:00
bool is_bytcr = false ;
2016-08-13 00:27:57 +03:00
priv = devm_kzalloc ( & pdev - > dev , sizeof ( * priv ) , GFP_ATOMIC ) ;
if ( ! priv )
return - ENOMEM ;
2014-11-19 12:43:27 +03:00
/* register the soc card */
2015-12-18 05:35:42 +03:00
byt_rt5640_card . dev = & pdev - > dev ;
2016-01-05 02:20:28 +03:00
mach = byt_rt5640_card . dev - > platform_data ;
2016-08-13 00:27:57 +03:00
snd_soc_card_set_drvdata ( & byt_rt5640_card , priv ) ;
2016-01-05 02:20:28 +03:00
2016-03-04 06:36:38 +03:00
/* fix index of codec dai */
dai_index = MERR_DPCM_COMPR + 1 ;
for ( i = 0 ; i < ARRAY_SIZE ( byt_rt5640_dais ) ; i + + ) {
if ( ! strcmp ( byt_rt5640_dais [ i ] . codec_name , " i2c-10EC5640:00 " ) ) {
dai_index = i ;
break ;
}
}
2016-01-05 02:20:28 +03:00
/* fixup codec name based on HID */
2016-03-04 06:36:38 +03:00
i2c_name = sst_acpi_find_name_from_hid ( mach - > id ) ;
if ( i2c_name ! = NULL ) {
snprintf ( byt_rt5640_codec_name , sizeof ( byt_rt5640_codec_name ) ,
" %s%s " , " i2c- " , i2c_name ) ;
byt_rt5640_dais [ dai_index ] . codec_name = byt_rt5640_codec_name ;
}
2015-12-18 05:35:42 +03:00
2016-08-13 00:27:53 +03:00
/*
* swap SSP0 if bytcr is detected
* ( will be overridden if DMI quirk is detected )
*/
if ( is_valleyview ( ) ) {
struct sst_platform_info * p_info = mach - > pdata ;
const struct sst_res_info * res_info = p_info - > res_info ;
2016-11-13 03:07:45 +03:00
if ( res_info - > acpi_ipc_irq_index = = 0 )
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 sst_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 = sst_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_rt5640_quirk | = BYT_RT5640_SSP0_AIF1 ;
} else if ( chan_package . aif_value = = 2 ) {
dev_info ( & pdev - > dev , " BIOS Routing: AIF2 connected \n " ) ;
byt_rt5640_quirk | = BYT_RT5640_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 */
2016-08-13 00:27:53 +03:00
byt_rt5640_quirk | = BYT_RT5640_SSP0_AIF2 ;
}
2016-11-13 03:07:46 +03:00
/* change defaults for Baytrail-CR capture */
byt_rt5640_quirk | = BYT_RT5640_IN1_MAP ;
byt_rt5640_quirk | = BYT_RT5640_DIFF_MIC ;
} else {
byt_rt5640_quirk | = ( BYT_RT5640_DMIC1_MAP |
BYT_RT5640_DMIC_EN ) ;
2016-08-13 00:27:53 +03:00
}
2016-03-04 06:36:37 +03:00
/* check quirks before creating card */
dmi_check_system ( byt_rt5640_quirk_table ) ;
2016-08-13 00:28:00 +03:00
log_quirks ( & pdev - > dev ) ;
2016-03-04 06:36:37 +03:00
2016-08-13 00:27:51 +03:00
if ( ( byt_rt5640_quirk & BYT_RT5640_SSP2_AIF2 ) | |
( byt_rt5640_quirk & BYT_RT5640_SSP0_AIF2 ) ) {
2016-08-13 00:27:50 +03:00
/* fixup codec aif name */
snprintf ( byt_rt5640_codec_aif_name ,
sizeof ( byt_rt5640_codec_aif_name ) ,
" %s " , " rt5640-aif2 " ) ;
byt_rt5640_dais [ dai_index ] . codec_dai_name =
byt_rt5640_codec_aif_name ;
}
2016-08-13 00:27:51 +03:00
if ( ( byt_rt5640_quirk & BYT_RT5640_SSP0_AIF1 ) | |
( byt_rt5640_quirk & BYT_RT5640_SSP0_AIF2 ) ) {
/* fixup cpu dai name name */
snprintf ( byt_rt5640_cpu_dai_name ,
sizeof ( byt_rt5640_cpu_dai_name ) ,
" %s " , " ssp0-port " ) ;
byt_rt5640_dais [ dai_index ] . cpu_dai_name =
byt_rt5640_cpu_dai_name ;
}
2016-08-13 00:27:57 +03:00
if ( ( byt_rt5640_quirk & BYT_RT5640_MCLK_EN ) & & ( is_valleyview ( ) ) ) {
priv - > mclk = devm_clk_get ( & pdev - > dev , " pmc_plt_clk_3 " ) ;
if ( IS_ERR ( priv - > mclk ) ) {
dev_err ( & pdev - > dev ,
" Failed to get MCLK from pmc_plt_clk_3: %ld \n " ,
PTR_ERR ( priv - > mclk ) ) ;
return PTR_ERR ( priv - > mclk ) ;
}
}
2015-12-18 05:35:42 +03:00
ret_val = devm_snd_soc_register_card ( & pdev - > dev , & byt_rt5640_card ) ;
2014-11-19 12:43:27 +03:00
if ( ret_val ) {
2015-12-18 05:35:41 +03:00
dev_err ( & pdev - > dev , " devm_snd_soc_register_card failed %d \n " ,
ret_val ) ;
2014-11-19 12:43:27 +03:00
return ret_val ;
}
2015-12-18 05:35:42 +03:00
platform_set_drvdata ( pdev , & byt_rt5640_card ) ;
2014-11-19 12:43:27 +03:00
return ret_val ;
}
2015-12-18 05:35:41 +03:00
static struct platform_driver snd_byt_rt5640_mc_driver = {
2014-11-19 12:43:27 +03:00
. driver = {
2015-12-18 05:35:41 +03:00
. name = " bytcr_rt5640 " ,
2014-11-19 12:43:27 +03:00
} ,
2015-12-18 05:35:41 +03:00
. probe = snd_byt_rt5640_mc_probe ,
2014-11-19 12:43:27 +03:00
} ;
2015-12-18 05:35:41 +03:00
module_platform_driver ( snd_byt_rt5640_mc_driver ) ;
2014-11-19 12:43:27 +03:00
MODULE_DESCRIPTION ( " ASoC Intel(R) Baytrail CR Machine driver " ) ;
MODULE_AUTHOR ( " Subhransu S. Prusty <subhransu.s.prusty@intel.com> " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
2015-12-18 05:35:41 +03:00
MODULE_ALIAS ( " platform:bytcr_rt5640 " ) ;