2023-08-04 11:45:59 +01:00
// SPDX-License-Identifier: GPL-2.0
/*
* CS42L43 core driver
*
* Copyright ( C ) 2022 - 2023 Cirrus Logic , Inc . and
* Cirrus Logic International Semiconductor Ltd .
*/
# include <linux/bitops.h>
# include <linux/build_bug.h>
# include <linux/delay.h>
# include <linux/err.h>
# include <linux/errno.h>
# include <linux/firmware.h>
# include <linux/jiffies.h>
# include <linux/mfd/core.h>
# include <linux/mfd/cs42l43-regs.h>
# include <linux/module.h>
# include <linux/pm_runtime.h>
# include <linux/soundwire/sdw.h>
# include "cs42l43.h"
# define CS42L43_RESET_DELAY 20
# define CS42L43_SDW_ATTACH_TIMEOUT 500
# define CS42L43_SDW_DETACH_TIMEOUT 100
# define CS42L43_MCU_BOOT_STAGE1 1
# define CS42L43_MCU_BOOT_STAGE2 2
# define CS42L43_MCU_BOOT_STAGE3 3
# define CS42L43_MCU_BOOT_STAGE4 4
# define CS42L43_MCU_POLL 5000
# define CS42L43_MCU_CMD_TIMEOUT 20000
# define CS42L43_MCU_UPDATE_FORMAT 3
# define CS42L43_MCU_UPDATE_OFFSET 0x100000
# define CS42L43_MCU_UPDATE_TIMEOUT 500000
# define CS42L43_MCU_UPDATE_RETRIES 5
# define CS42L43_MCU_SUPPORTED_REV 0x2105
# define CS42L43_MCU_SHADOW_REGS_REQUIRED_REV 0x2200
# define CS42L43_MCU_SUPPORTED_BIOS_REV 0x0001
# define CS42L43_VDDP_DELAY 50
# define CS42L43_VDDD_DELAY 1000
# define CS42L43_AUTOSUSPEND_TIME 250
struct cs42l43_patch_header {
__le16 version ;
__le16 size ;
u8 reserved ;
u8 secure ;
__le16 bss_size ;
__le32 apply_addr ;
__le32 checksum ;
__le32 sha ;
__le16 swrev ;
__le16 patchid ;
__le16 ipxid ;
__le16 romver ;
__le32 load_addr ;
} __packed ;
static const struct reg_sequence cs42l43_reva_patch [ ] = {
{ 0x4000 , 0x00000055 } ,
{ 0x4000 , 0x000000AA } ,
{ 0x10084 , 0x00000000 } ,
{ 0x1741C , 0x00CD2000 } ,
{ 0x1718C , 0x00000003 } ,
{ 0x4000 , 0x00000000 } ,
{ CS42L43_CCM_BLK_CLK_CONTROL , 0x00000002 } ,
{ CS42L43_HPPATHVOL , 0x011B011B } ,
{ CS42L43_OSC_DIV_SEL , 0x00000001 } ,
{ CS42L43_DACCNFG2 , 0x00000005 } ,
{ CS42L43_MIC_DETECT_CONTROL_ANDROID , 0x80790079 } ,
{ CS42L43_RELID , 0x0000000F } ,
} ;
const struct reg_default cs42l43_reg_default [ CS42L43_N_DEFAULTS ] = {
{ CS42L43_DRV_CTRL1 , 0x000186C0 } ,
{ CS42L43_DRV_CTRL3 , 0x286DB018 } ,
{ CS42L43_DRV_CTRL4 , 0x000006D8 } ,
{ CS42L43_DRV_CTRL_5 , 0x136C00C0 } ,
{ CS42L43_GPIO_CTRL1 , 0x00000707 } ,
{ CS42L43_GPIO_CTRL2 , 0x00000000 } ,
{ CS42L43_GPIO_FN_SEL , 0x00000000 } ,
{ CS42L43_MCLK_SRC_SEL , 0x00000000 } ,
{ CS42L43_SAMPLE_RATE1 , 0x00000003 } ,
{ CS42L43_SAMPLE_RATE2 , 0x00000003 } ,
{ CS42L43_SAMPLE_RATE3 , 0x00000003 } ,
{ CS42L43_SAMPLE_RATE4 , 0x00000003 } ,
{ CS42L43_PLL_CONTROL , 0x00000000 } ,
{ CS42L43_FS_SELECT1 , 0x00000000 } ,
{ CS42L43_FS_SELECT2 , 0x00000000 } ,
{ CS42L43_FS_SELECT3 , 0x00000000 } ,
{ CS42L43_FS_SELECT4 , 0x00000000 } ,
{ CS42L43_PDM_CONTROL , 0x00000000 } ,
{ CS42L43_ASP_CLK_CONFIG1 , 0x00010001 } ,
{ CS42L43_ASP_CLK_CONFIG2 , 0x00000000 } ,
{ CS42L43_OSC_DIV_SEL , 0x00000001 } ,
{ CS42L43_ADC_B_CTRL1 , 0x00000000 } ,
{ CS42L43_ADC_B_CTRL2 , 0x00000000 } ,
{ CS42L43_DECIM_HPF_WNF_CTRL1 , 0x00000001 } ,
{ CS42L43_DECIM_HPF_WNF_CTRL2 , 0x00000001 } ,
{ CS42L43_DECIM_HPF_WNF_CTRL3 , 0x00000001 } ,
{ CS42L43_DECIM_HPF_WNF_CTRL4 , 0x00000001 } ,
{ CS42L43_DMIC_PDM_CTRL , 0x00000000 } ,
{ CS42L43_DECIM_VOL_CTRL_CH1_CH2 , 0x20122012 } ,
{ CS42L43_DECIM_VOL_CTRL_CH3_CH4 , 0x20122012 } ,
{ CS42L43_INTP_VOLUME_CTRL1 , 0x00000180 } ,
{ CS42L43_INTP_VOLUME_CTRL2 , 0x00000180 } ,
{ CS42L43_AMP1_2_VOL_RAMP , 0x00000022 } ,
{ CS42L43_ASP_CTRL , 0x00000004 } ,
{ CS42L43_ASP_FSYNC_CTRL1 , 0x000000FA } ,
{ CS42L43_ASP_FSYNC_CTRL2 , 0x00000001 } ,
{ CS42L43_ASP_FSYNC_CTRL3 , 0x00000000 } ,
{ CS42L43_ASP_FSYNC_CTRL4 , 0x000001F4 } ,
{ CS42L43_ASP_DATA_CTRL , 0x0000003A } ,
{ CS42L43_ASP_RX_EN , 0x00000000 } ,
{ CS42L43_ASP_TX_EN , 0x00000000 } ,
{ CS42L43_ASP_RX_CH1_CTRL , 0x00170001 } ,
{ CS42L43_ASP_RX_CH2_CTRL , 0x00170031 } ,
{ CS42L43_ASP_RX_CH3_CTRL , 0x00170061 } ,
{ CS42L43_ASP_RX_CH4_CTRL , 0x00170091 } ,
{ CS42L43_ASP_RX_CH5_CTRL , 0x001700C1 } ,
{ CS42L43_ASP_RX_CH6_CTRL , 0x001700F1 } ,
{ CS42L43_ASP_TX_CH1_CTRL , 0x00170001 } ,
{ CS42L43_ASP_TX_CH2_CTRL , 0x00170031 } ,
{ CS42L43_ASP_TX_CH3_CTRL , 0x00170061 } ,
{ CS42L43_ASP_TX_CH4_CTRL , 0x00170091 } ,
{ CS42L43_ASP_TX_CH5_CTRL , 0x001700C1 } ,
{ CS42L43_ASP_TX_CH6_CTRL , 0x001700F1 } ,
{ CS42L43_ASPTX1_INPUT , 0x00800000 } ,
{ CS42L43_ASPTX2_INPUT , 0x00800000 } ,
{ CS42L43_ASPTX3_INPUT , 0x00800000 } ,
{ CS42L43_ASPTX4_INPUT , 0x00800000 } ,
{ CS42L43_ASPTX5_INPUT , 0x00800000 } ,
{ CS42L43_ASPTX6_INPUT , 0x00800000 } ,
{ CS42L43_SWIRE_DP1_CH1_INPUT , 0x00800000 } ,
{ CS42L43_SWIRE_DP1_CH2_INPUT , 0x00800000 } ,
{ CS42L43_SWIRE_DP1_CH3_INPUT , 0x00800000 } ,
{ CS42L43_SWIRE_DP1_CH4_INPUT , 0x00800000 } ,
{ CS42L43_SWIRE_DP2_CH1_INPUT , 0x00800000 } ,
{ CS42L43_SWIRE_DP2_CH2_INPUT , 0x00800000 } ,
{ CS42L43_SWIRE_DP3_CH1_INPUT , 0x00800000 } ,
{ CS42L43_SWIRE_DP3_CH2_INPUT , 0x00800000 } ,
{ CS42L43_SWIRE_DP4_CH1_INPUT , 0x00800000 } ,
{ CS42L43_SWIRE_DP4_CH2_INPUT , 0x00800000 } ,
{ CS42L43_ASRC_INT1_INPUT1 , 0x00800000 } ,
{ CS42L43_ASRC_INT2_INPUT1 , 0x00800000 } ,
{ CS42L43_ASRC_INT3_INPUT1 , 0x00800000 } ,
{ CS42L43_ASRC_INT4_INPUT1 , 0x00800000 } ,
{ CS42L43_ASRC_DEC1_INPUT1 , 0x00800000 } ,
{ CS42L43_ASRC_DEC2_INPUT1 , 0x00800000 } ,
{ CS42L43_ASRC_DEC3_INPUT1 , 0x00800000 } ,
{ CS42L43_ASRC_DEC4_INPUT1 , 0x00800000 } ,
{ CS42L43_ISRC1INT1_INPUT1 , 0x00800000 } ,
{ CS42L43_ISRC1INT2_INPUT1 , 0x00800000 } ,
{ CS42L43_ISRC1DEC1_INPUT1 , 0x00800000 } ,
{ CS42L43_ISRC1DEC2_INPUT1 , 0x00800000 } ,
{ CS42L43_ISRC2INT1_INPUT1 , 0x00800000 } ,
{ CS42L43_ISRC2INT2_INPUT1 , 0x00800000 } ,
{ CS42L43_ISRC2DEC1_INPUT1 , 0x00800000 } ,
{ CS42L43_ISRC2DEC2_INPUT1 , 0x00800000 } ,
{ CS42L43_EQ1MIX_INPUT1 , 0x00800000 } ,
{ CS42L43_EQ1MIX_INPUT2 , 0x00800000 } ,
{ CS42L43_EQ1MIX_INPUT3 , 0x00800000 } ,
{ CS42L43_EQ1MIX_INPUT4 , 0x00800000 } ,
{ CS42L43_EQ2MIX_INPUT1 , 0x00800000 } ,
{ CS42L43_EQ2MIX_INPUT2 , 0x00800000 } ,
{ CS42L43_EQ2MIX_INPUT3 , 0x00800000 } ,
{ CS42L43_EQ2MIX_INPUT4 , 0x00800000 } ,
{ CS42L43_SPDIF1_INPUT1 , 0x00800000 } ,
{ CS42L43_SPDIF2_INPUT1 , 0x00800000 } ,
{ CS42L43_AMP1MIX_INPUT1 , 0x00800000 } ,
{ CS42L43_AMP1MIX_INPUT2 , 0x00800000 } ,
{ CS42L43_AMP1MIX_INPUT3 , 0x00800000 } ,
{ CS42L43_AMP1MIX_INPUT4 , 0x00800000 } ,
{ CS42L43_AMP2MIX_INPUT1 , 0x00800000 } ,
{ CS42L43_AMP2MIX_INPUT2 , 0x00800000 } ,
{ CS42L43_AMP2MIX_INPUT3 , 0x00800000 } ,
{ CS42L43_AMP2MIX_INPUT4 , 0x00800000 } ,
{ CS42L43_AMP3MIX_INPUT1 , 0x00800000 } ,
{ CS42L43_AMP3MIX_INPUT2 , 0x00800000 } ,
{ CS42L43_AMP3MIX_INPUT3 , 0x00800000 } ,
{ CS42L43_AMP3MIX_INPUT4 , 0x00800000 } ,
{ CS42L43_AMP4MIX_INPUT1 , 0x00800000 } ,
{ CS42L43_AMP4MIX_INPUT2 , 0x00800000 } ,
{ CS42L43_AMP4MIX_INPUT3 , 0x00800000 } ,
{ CS42L43_AMP4MIX_INPUT4 , 0x00800000 } ,
{ CS42L43_ASRC_INT_ENABLES , 0x00000100 } ,
{ CS42L43_ASRC_DEC_ENABLES , 0x00000100 } ,
{ CS42L43_PDNCNTL , 0x00000000 } ,
{ CS42L43_RINGSENSE_DEB_CTRL , 0x0000001B } ,
{ CS42L43_TIPSENSE_DEB_CTRL , 0x0000001B } ,
{ CS42L43_HS2 , 0x050106F3 } ,
{ CS42L43_STEREO_MIC_CTRL , 0x00000000 } ,
{ CS42L43_STEREO_MIC_CLAMP_CTRL , 0x00000001 } ,
{ CS42L43_BLOCK_EN2 , 0x00000000 } ,
{ CS42L43_BLOCK_EN3 , 0x00000000 } ,
{ CS42L43_BLOCK_EN4 , 0x00000000 } ,
{ CS42L43_BLOCK_EN5 , 0x00000000 } ,
{ CS42L43_BLOCK_EN6 , 0x00000000 } ,
{ CS42L43_BLOCK_EN7 , 0x00000000 } ,
{ CS42L43_BLOCK_EN8 , 0x00000000 } ,
{ CS42L43_BLOCK_EN9 , 0x00000000 } ,
{ CS42L43_BLOCK_EN10 , 0x00000000 } ,
{ CS42L43_BLOCK_EN11 , 0x00000000 } ,
{ CS42L43_TONE_CH1_CTRL , 0x00000000 } ,
{ CS42L43_TONE_CH2_CTRL , 0x00000000 } ,
{ CS42L43_MIC_DETECT_CONTROL_1 , 0x00000003 } ,
{ CS42L43_HS_BIAS_SENSE_AND_CLAMP_AUTOCONTROL , 0x02000003 } ,
{ CS42L43_MIC_DETECT_CONTROL_ANDROID , 0x80790079 } ,
{ CS42L43_ISRC1_CTRL , 0x00000000 } ,
{ CS42L43_ISRC2_CTRL , 0x00000000 } ,
{ CS42L43_CTRL_REG , 0x00000006 } ,
{ CS42L43_FDIV_FRAC , 0x40000000 } ,
{ CS42L43_CAL_RATIO , 0x00000080 } ,
{ CS42L43_SPI_CLK_CONFIG1 , 0x00000000 } ,
{ CS42L43_SPI_CONFIG1 , 0x00000000 } ,
{ CS42L43_SPI_CONFIG2 , 0x00000000 } ,
{ CS42L43_SPI_CONFIG3 , 0x00000001 } ,
{ CS42L43_SPI_CONFIG4 , 0x00000000 } ,
{ CS42L43_TRAN_CONFIG3 , 0x00000000 } ,
{ CS42L43_TRAN_CONFIG4 , 0x00000000 } ,
{ CS42L43_TRAN_CONFIG5 , 0x00000000 } ,
{ CS42L43_TRAN_CONFIG6 , 0x00000000 } ,
{ CS42L43_TRAN_CONFIG7 , 0x00000000 } ,
{ CS42L43_TRAN_CONFIG8 , 0x00000000 } ,
{ CS42L43_DACCNFG1 , 0x00000008 } ,
{ CS42L43_DACCNFG2 , 0x00000005 } ,
{ CS42L43_HPPATHVOL , 0x011B011B } ,
{ CS42L43_PGAVOL , 0x00003470 } ,
{ CS42L43_LOADDETENA , 0x00000000 } ,
{ CS42L43_CTRL , 0x00000037 } ,
{ CS42L43_COEFF_DATA_IN0 , 0x00000000 } ,
{ CS42L43_COEFF_RD_WR0 , 0x00000000 } ,
{ CS42L43_START_EQZ0 , 0x00000000 } ,
{ CS42L43_MUTE_EQ_IN0 , 0x00000000 } ,
{ CS42L43_DECIM_MASK , 0x0000000F } ,
{ CS42L43_EQ_MIX_MASK , 0x0000000F } ,
{ CS42L43_ASP_MASK , 0x000000FF } ,
{ CS42L43_PLL_MASK , 0x00000003 } ,
{ CS42L43_SOFT_MASK , 0x0000FFFF } ,
{ CS42L43_SWIRE_MASK , 0x00007FFF } ,
{ CS42L43_MSM_MASK , 0x00000FFF } ,
{ CS42L43_ACC_DET_MASK , 0x00000FFF } ,
{ CS42L43_I2C_TGT_MASK , 0x00000003 } ,
{ CS42L43_SPI_MSTR_MASK , 0x00000007 } ,
{ CS42L43_SW_TO_SPI_BRIDGE_MASK , 0x00000001 } ,
{ CS42L43_OTP_MASK , 0x00000007 } ,
{ CS42L43_CLASS_D_AMP_MASK , 0x00003FFF } ,
{ CS42L43_GPIO_INT_MASK , 0x0000003F } ,
{ CS42L43_ASRC_MASK , 0x0000000F } ,
{ CS42L43_HPOUT_MASK , 0x00000003 } ,
} ;
EXPORT_SYMBOL_NS_GPL ( cs42l43_reg_default , MFD_CS42L43 ) ;
bool cs42l43_readable_register ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case CS42L43_DEVID :
case CS42L43_REVID :
case CS42L43_RELID :
case CS42L43_SFT_RESET :
case CS42L43_DRV_CTRL1 :
case CS42L43_DRV_CTRL3 :
case CS42L43_DRV_CTRL4 :
case CS42L43_DRV_CTRL_5 :
case CS42L43_GPIO_CTRL1 :
case CS42L43_GPIO_CTRL2 :
case CS42L43_GPIO_STS :
case CS42L43_GPIO_FN_SEL :
case CS42L43_MCLK_SRC_SEL :
case CS42L43_SAMPLE_RATE1 . . . CS42L43_SAMPLE_RATE4 :
case CS42L43_PLL_CONTROL :
case CS42L43_FS_SELECT1 . . . CS42L43_FS_SELECT4 :
case CS42L43_PDM_CONTROL :
case CS42L43_ASP_CLK_CONFIG1 . . . CS42L43_ASP_CLK_CONFIG2 :
case CS42L43_OSC_DIV_SEL :
case CS42L43_ADC_B_CTRL1 . . . CS42L43_ADC_B_CTRL2 :
case CS42L43_DECIM_HPF_WNF_CTRL1 . . . CS42L43_DECIM_HPF_WNF_CTRL4 :
case CS42L43_DMIC_PDM_CTRL :
case CS42L43_DECIM_VOL_CTRL_CH1_CH2 . . . CS42L43_DECIM_VOL_CTRL_CH3_CH4 :
case CS42L43_INTP_VOLUME_CTRL1 . . . CS42L43_INTP_VOLUME_CTRL2 :
case CS42L43_AMP1_2_VOL_RAMP :
case CS42L43_ASP_CTRL :
case CS42L43_ASP_FSYNC_CTRL1 . . . CS42L43_ASP_FSYNC_CTRL4 :
case CS42L43_ASP_DATA_CTRL :
case CS42L43_ASP_RX_EN . . . CS42L43_ASP_TX_EN :
case CS42L43_ASP_RX_CH1_CTRL . . . CS42L43_ASP_RX_CH6_CTRL :
case CS42L43_ASP_TX_CH1_CTRL . . . CS42L43_ASP_TX_CH6_CTRL :
case CS42L43_OTP_REVISION_ID :
case CS42L43_ASPTX1_INPUT :
case CS42L43_ASPTX2_INPUT :
case CS42L43_ASPTX3_INPUT :
case CS42L43_ASPTX4_INPUT :
case CS42L43_ASPTX5_INPUT :
case CS42L43_ASPTX6_INPUT :
case CS42L43_SWIRE_DP1_CH1_INPUT :
case CS42L43_SWIRE_DP1_CH2_INPUT :
case CS42L43_SWIRE_DP1_CH3_INPUT :
case CS42L43_SWIRE_DP1_CH4_INPUT :
case CS42L43_SWIRE_DP2_CH1_INPUT :
case CS42L43_SWIRE_DP2_CH2_INPUT :
case CS42L43_SWIRE_DP3_CH1_INPUT :
case CS42L43_SWIRE_DP3_CH2_INPUT :
case CS42L43_SWIRE_DP4_CH1_INPUT :
case CS42L43_SWIRE_DP4_CH2_INPUT :
case CS42L43_ASRC_INT1_INPUT1 :
case CS42L43_ASRC_INT2_INPUT1 :
case CS42L43_ASRC_INT3_INPUT1 :
case CS42L43_ASRC_INT4_INPUT1 :
case CS42L43_ASRC_DEC1_INPUT1 :
case CS42L43_ASRC_DEC2_INPUT1 :
case CS42L43_ASRC_DEC3_INPUT1 :
case CS42L43_ASRC_DEC4_INPUT1 :
case CS42L43_ISRC1INT1_INPUT1 :
case CS42L43_ISRC1INT2_INPUT1 :
case CS42L43_ISRC1DEC1_INPUT1 :
case CS42L43_ISRC1DEC2_INPUT1 :
case CS42L43_ISRC2INT1_INPUT1 :
case CS42L43_ISRC2INT2_INPUT1 :
case CS42L43_ISRC2DEC1_INPUT1 :
case CS42L43_ISRC2DEC2_INPUT1 :
case CS42L43_EQ1MIX_INPUT1 . . . CS42L43_EQ1MIX_INPUT4 :
case CS42L43_EQ2MIX_INPUT1 . . . CS42L43_EQ2MIX_INPUT4 :
case CS42L43_SPDIF1_INPUT1 :
case CS42L43_SPDIF2_INPUT1 :
case CS42L43_AMP1MIX_INPUT1 . . . CS42L43_AMP1MIX_INPUT4 :
case CS42L43_AMP2MIX_INPUT1 . . . CS42L43_AMP2MIX_INPUT4 :
case CS42L43_AMP3MIX_INPUT1 . . . CS42L43_AMP3MIX_INPUT4 :
case CS42L43_AMP4MIX_INPUT1 . . . CS42L43_AMP4MIX_INPUT4 :
case CS42L43_ASRC_INT_ENABLES . . . CS42L43_ASRC_DEC_ENABLES :
case CS42L43_PDNCNTL :
case CS42L43_RINGSENSE_DEB_CTRL :
case CS42L43_TIPSENSE_DEB_CTRL :
case CS42L43_TIP_RING_SENSE_INTERRUPT_STATUS :
case CS42L43_HS2 :
case CS42L43_HS_STAT :
case CS42L43_MCU_SW_INTERRUPT :
case CS42L43_STEREO_MIC_CTRL :
case CS42L43_STEREO_MIC_CLAMP_CTRL :
case CS42L43_BLOCK_EN2 . . . CS42L43_BLOCK_EN11 :
case CS42L43_TONE_CH1_CTRL . . . CS42L43_TONE_CH2_CTRL :
case CS42L43_MIC_DETECT_CONTROL_1 :
case CS42L43_DETECT_STATUS_1 :
case CS42L43_HS_BIAS_SENSE_AND_CLAMP_AUTOCONTROL :
case CS42L43_MIC_DETECT_CONTROL_ANDROID :
case CS42L43_ISRC1_CTRL :
case CS42L43_ISRC2_CTRL :
case CS42L43_CTRL_REG :
case CS42L43_FDIV_FRAC :
case CS42L43_CAL_RATIO :
case CS42L43_SPI_CLK_CONFIG1 :
case CS42L43_SPI_CONFIG1 . . . CS42L43_SPI_CONFIG4 :
case CS42L43_SPI_STATUS1 . . . CS42L43_SPI_STATUS2 :
case CS42L43_TRAN_CONFIG1 . . . CS42L43_TRAN_CONFIG8 :
case CS42L43_TRAN_STATUS1 . . . CS42L43_TRAN_STATUS3 :
case CS42L43_TX_DATA :
case CS42L43_RX_DATA :
case CS42L43_DACCNFG1 . . . CS42L43_DACCNFG2 :
case CS42L43_HPPATHVOL :
case CS42L43_PGAVOL :
case CS42L43_LOADDETRESULTS :
case CS42L43_LOADDETENA :
case CS42L43_CTRL :
case CS42L43_COEFF_DATA_IN0 :
case CS42L43_COEFF_RD_WR0 :
case CS42L43_INIT_DONE0 :
case CS42L43_START_EQZ0 :
case CS42L43_MUTE_EQ_IN0 :
case CS42L43_DECIM_INT . . . CS42L43_HPOUT_INT :
case CS42L43_DECIM_MASK . . . CS42L43_HPOUT_MASK :
case CS42L43_DECIM_INT_SHADOW . . . CS42L43_HP_OUT_SHADOW :
case CS42L43_BOOT_CONTROL :
case CS42L43_BLOCK_EN :
case CS42L43_SHUTTER_CONTROL :
case CS42L43_MCU_SW_REV . . . CS42L43_MCU_RAM_MAX :
return true ;
default :
return false ;
}
}
EXPORT_SYMBOL_NS_GPL ( cs42l43_readable_register , MFD_CS42L43 ) ;
bool cs42l43_precious_register ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case CS42L43_SFT_RESET :
case CS42L43_TX_DATA :
case CS42L43_RX_DATA :
case CS42L43_DECIM_INT . . . CS42L43_HPOUT_INT :
case CS42L43_MCU_SW_REV . . . CS42L43_MCU_RAM_MAX :
return true ;
default :
return false ;
}
}
EXPORT_SYMBOL_NS_GPL ( cs42l43_precious_register , MFD_CS42L43 ) ;
bool cs42l43_volatile_register ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case CS42L43_DEVID :
case CS42L43_REVID :
case CS42L43_RELID :
case CS42L43_GPIO_STS :
case CS42L43_OTP_REVISION_ID :
case CS42L43_TIP_RING_SENSE_INTERRUPT_STATUS :
case CS42L43_HS_STAT :
case CS42L43_MCU_SW_INTERRUPT :
case CS42L43_DETECT_STATUS_1 :
case CS42L43_SPI_STATUS1 . . . CS42L43_SPI_STATUS2 :
case CS42L43_TRAN_CONFIG1 . . . CS42L43_TRAN_CONFIG2 :
case CS42L43_TRAN_CONFIG8 :
case CS42L43_TRAN_STATUS1 . . . CS42L43_TRAN_STATUS3 :
case CS42L43_LOADDETRESULTS :
case CS42L43_INIT_DONE0 :
case CS42L43_DECIM_INT_SHADOW . . . CS42L43_HP_OUT_SHADOW :
case CS42L43_BOOT_CONTROL :
case CS42L43_BLOCK_EN :
return true ;
default :
return cs42l43_precious_register ( dev , reg ) ;
}
}
EXPORT_SYMBOL_NS_GPL ( cs42l43_volatile_register , MFD_CS42L43 ) ;
# define CS42L43_IRQ_OFFSET(reg) ((CS42L43_##reg##_INT) - CS42L43_DECIM_INT)
# define CS42L43_IRQ_REG(name, reg) REGMAP_IRQ_REG(CS42L43_##name, \
CS42L43_IRQ_OFFSET ( reg ) , \
CS42L43_ # # name # # _INT_MASK )
static const struct regmap_irq cs42l43_regmap_irqs [ ] = {
CS42L43_IRQ_REG ( PLL_LOST_LOCK , PLL ) ,
CS42L43_IRQ_REG ( PLL_READY , PLL ) ,
CS42L43_IRQ_REG ( HP_STARTUP_DONE , MSM ) ,
CS42L43_IRQ_REG ( HP_SHUTDOWN_DONE , MSM ) ,
CS42L43_IRQ_REG ( HSDET_DONE , MSM ) ,
CS42L43_IRQ_REG ( TIPSENSE_UNPLUG_DB , MSM ) ,
CS42L43_IRQ_REG ( TIPSENSE_PLUG_DB , MSM ) ,
CS42L43_IRQ_REG ( RINGSENSE_UNPLUG_DB , MSM ) ,
CS42L43_IRQ_REG ( RINGSENSE_PLUG_DB , MSM ) ,
CS42L43_IRQ_REG ( TIPSENSE_UNPLUG_PDET , MSM ) ,
CS42L43_IRQ_REG ( TIPSENSE_PLUG_PDET , MSM ) ,
CS42L43_IRQ_REG ( RINGSENSE_UNPLUG_PDET , MSM ) ,
CS42L43_IRQ_REG ( RINGSENSE_PLUG_PDET , MSM ) ,
CS42L43_IRQ_REG ( HS2_BIAS_SENSE , ACC_DET ) ,
CS42L43_IRQ_REG ( HS1_BIAS_SENSE , ACC_DET ) ,
CS42L43_IRQ_REG ( DC_DETECT1_FALSE , ACC_DET ) ,
CS42L43_IRQ_REG ( DC_DETECT1_TRUE , ACC_DET ) ,
CS42L43_IRQ_REG ( HSBIAS_CLAMPED , ACC_DET ) ,
CS42L43_IRQ_REG ( HS3_4_BIAS_SENSE , ACC_DET ) ,
CS42L43_IRQ_REG ( AMP2_CLK_STOP_FAULT , CLASS_D_AMP ) ,
CS42L43_IRQ_REG ( AMP1_CLK_STOP_FAULT , CLASS_D_AMP ) ,
CS42L43_IRQ_REG ( AMP2_VDDSPK_FAULT , CLASS_D_AMP ) ,
CS42L43_IRQ_REG ( AMP1_VDDSPK_FAULT , CLASS_D_AMP ) ,
CS42L43_IRQ_REG ( AMP2_SHUTDOWN_DONE , CLASS_D_AMP ) ,
CS42L43_IRQ_REG ( AMP1_SHUTDOWN_DONE , CLASS_D_AMP ) ,
CS42L43_IRQ_REG ( AMP2_STARTUP_DONE , CLASS_D_AMP ) ,
CS42L43_IRQ_REG ( AMP1_STARTUP_DONE , CLASS_D_AMP ) ,
CS42L43_IRQ_REG ( AMP2_THERM_SHDN , CLASS_D_AMP ) ,
CS42L43_IRQ_REG ( AMP1_THERM_SHDN , CLASS_D_AMP ) ,
CS42L43_IRQ_REG ( AMP2_THERM_WARN , CLASS_D_AMP ) ,
CS42L43_IRQ_REG ( AMP1_THERM_WARN , CLASS_D_AMP ) ,
CS42L43_IRQ_REG ( AMP2_SCDET , CLASS_D_AMP ) ,
CS42L43_IRQ_REG ( AMP1_SCDET , CLASS_D_AMP ) ,
CS42L43_IRQ_REG ( GPIO3_FALL , GPIO ) ,
CS42L43_IRQ_REG ( GPIO3_RISE , GPIO ) ,
CS42L43_IRQ_REG ( GPIO2_FALL , GPIO ) ,
CS42L43_IRQ_REG ( GPIO2_RISE , GPIO ) ,
CS42L43_IRQ_REG ( GPIO1_FALL , GPIO ) ,
CS42L43_IRQ_REG ( GPIO1_RISE , GPIO ) ,
CS42L43_IRQ_REG ( HP_ILIMIT , HPOUT ) ,
CS42L43_IRQ_REG ( HP_LOADDET_DONE , HPOUT ) ,
} ;
static const struct regmap_irq_chip cs42l43_irq_chip = {
. name = " cs42l43 " ,
. status_base = CS42L43_DECIM_INT ,
. mask_base = CS42L43_DECIM_MASK ,
. num_regs = 16 ,
. irqs = cs42l43_regmap_irqs ,
. num_irqs = ARRAY_SIZE ( cs42l43_regmap_irqs ) ,
. runtime_pm = true ,
} ;
static const char * const cs42l43_core_supplies [ ] = {
" vdd-a " , " vdd-io " , " vdd-cp " ,
} ;
static const char * const cs42l43_parent_supplies [ ] = { " vdd-amp " } ;
static const struct mfd_cell cs42l43_devs [ ] = {
{ . name = " cs42l43-pinctrl " , } ,
{ . name = " cs42l43-spi " , } ,
{
. name = " cs42l43-codec " ,
. parent_supplies = cs42l43_parent_supplies ,
. num_parent_supplies = ARRAY_SIZE ( cs42l43_parent_supplies ) ,
} ,
} ;
/*
* If the device is connected over Soundwire , as well as soft resetting the
* device , this function will also way for the device to detach from the bus
* before returning .
*/
static int cs42l43_soft_reset ( struct cs42l43 * cs42l43 )
{
static const struct reg_sequence reset [ ] = {
{ CS42L43_SFT_RESET , CS42L43_SFT_RESET_VAL } ,
} ;
reinit_completion ( & cs42l43 - > device_detach ) ;
/*
* Apply cache only because the soft reset will cause the device to
* detach from the soundwire bus .
*/
regcache_cache_only ( cs42l43 - > regmap , true ) ;
regmap_multi_reg_write_bypassed ( cs42l43 - > regmap , reset , ARRAY_SIZE ( reset ) ) ;
msleep ( CS42L43_RESET_DELAY ) ;
if ( cs42l43 - > sdw ) {
unsigned long timeout = msecs_to_jiffies ( CS42L43_SDW_DETACH_TIMEOUT ) ;
unsigned long time ;
time = wait_for_completion_timeout ( & cs42l43 - > device_detach , timeout ) ;
if ( ! time ) {
dev_err ( cs42l43 - > dev , " Timed out waiting for device detach \n " ) ;
return - ETIMEDOUT ;
}
}
return - EAGAIN ;
}
/*
* This function is essentially a no - op on I2C , but will wait for the device to
* attach when the device is used on a SoundWire bus .
*/
static int cs42l43_wait_for_attach ( struct cs42l43 * cs42l43 )
{
if ( ! cs42l43 - > attached ) {
unsigned long timeout = msecs_to_jiffies ( CS42L43_SDW_ATTACH_TIMEOUT ) ;
unsigned long time ;
time = wait_for_completion_timeout ( & cs42l43 - > device_attach , timeout ) ;
if ( ! time ) {
dev_err ( cs42l43 - > dev , " Timed out waiting for device re-attach \n " ) ;
return - ETIMEDOUT ;
}
}
regcache_cache_only ( cs42l43 - > regmap , false ) ;
/* The hardware requires enabling OSC_DIV before doing any SoundWire reads. */
if ( cs42l43 - > sdw )
regmap_write ( cs42l43 - > regmap , CS42L43_OSC_DIV_SEL ,
CS42L43_OSC_DIV2_EN_MASK ) ;
return 0 ;
}
/*
* This function will advance the firmware into boot stage 3 from boot stage 2.
* Boot stage 3 is required to send commands to the firmware . This is achieved
* by setting the firmware NEED configuration register to zero , this indicates
* no configuration is required forcing the firmware to advance to boot stage 3.
*
* Later revisions of the firmware require the use of an alternative register
* for this purpose , which is indicated through the shadow flag .
*/
static int cs42l43_mcu_stage_2_3 ( struct cs42l43 * cs42l43 , bool shadow )
{
unsigned int need_reg = CS42L43_NEED_CONFIGS ;
unsigned int val ;
int ret ;
if ( shadow )
need_reg = CS42L43_FW_SH_BOOT_CFG_NEED_CONFIGS ;
regmap_write ( cs42l43 - > regmap , need_reg , 0 ) ;
ret = regmap_read_poll_timeout ( cs42l43 - > regmap , CS42L43_BOOT_STATUS ,
val , ( val = = CS42L43_MCU_BOOT_STAGE3 ) ,
CS42L43_MCU_POLL , CS42L43_MCU_CMD_TIMEOUT ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to move to stage 3: %d, 0x%x \n " , ret , val ) ;
return ret ;
}
return - EAGAIN ;
}
/*
* This function will return the firmware to boot stage 2 from boot stage 3.
* Boot stage 2 is required to apply updates to the firmware . This is achieved
* by setting the firmware NEED configuration register to FW_PATCH_NEED_CFG ,
* setting the HAVE configuration register to 0 , and soft resetting . The
* firmware will see it is missing a patch configuration and will pause in boot
* stage 2.
*
* Note : Unlike cs42l43_mcu_stage_2_3 there is no need to consider the shadow
* register here as the driver will only return to boot stage 2 if the firmware
* requires update which means the revision does not include shadow register
* support .
*/
static int cs42l43_mcu_stage_3_2 ( struct cs42l43 * cs42l43 )
{
regmap_write ( cs42l43 - > regmap , CS42L43_FW_MISSION_CTRL_NEED_CONFIGS ,
CS42L43_FW_PATCH_NEED_CFG_MASK ) ;
regmap_write ( cs42l43 - > regmap , CS42L43_FW_MISSION_CTRL_HAVE_CONFIGS , 0 ) ;
return cs42l43_soft_reset ( cs42l43 ) ;
}
/*
* Disable the firmware running on the device such that the driver can access
* the registers without fear of the MCU changing them under it .
*/
static int cs42l43_mcu_disable ( struct cs42l43 * cs42l43 )
{
unsigned int val ;
int ret ;
regmap_write ( cs42l43 - > regmap , CS42L43_FW_MISSION_CTRL_MM_MCU_CFG_REG ,
CS42L43_FW_MISSION_CTRL_MM_MCU_CFG_DISABLE_VAL ) ;
regmap_write ( cs42l43 - > regmap , CS42L43_FW_MISSION_CTRL_MM_CTRL_SELECTION ,
CS42L43_FW_MM_CTRL_MCU_SEL_MASK ) ;
regmap_write ( cs42l43 - > regmap , CS42L43_MCU_SW_INTERRUPT , CS42L43_CONTROL_IND_MASK ) ;
regmap_write ( cs42l43 - > regmap , CS42L43_MCU_SW_INTERRUPT , 0 ) ;
ret = regmap_read_poll_timeout ( cs42l43 - > regmap , CS42L43_SOFT_INT_SHADOW , val ,
( val & CS42L43_CONTROL_APPLIED_INT_MASK ) ,
CS42L43_MCU_POLL , CS42L43_MCU_CMD_TIMEOUT ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to disable firmware: %d, 0x%x \n " , ret , val ) ;
return ret ;
}
/* Soft reset to clear any register state the firmware left behind. */
return cs42l43_soft_reset ( cs42l43 ) ;
}
/*
* Callback to load firmware updates .
*/
static void cs42l43_mcu_load_firmware ( const struct firmware * firmware , void * context )
{
struct cs42l43 * cs42l43 = context ;
const struct cs42l43_patch_header * hdr ;
unsigned int loadaddr , val ;
int ret ;
if ( ! firmware ) {
dev_err ( cs42l43 - > dev , " Failed to load firmware \n " ) ;
cs42l43 - > firmware_error = - ENODEV ;
goto err ;
}
hdr = ( const struct cs42l43_patch_header * ) & firmware - > data [ 0 ] ;
loadaddr = le32_to_cpu ( hdr - > load_addr ) ;
if ( le16_to_cpu ( hdr - > version ) ! = CS42L43_MCU_UPDATE_FORMAT ) {
dev_err ( cs42l43 - > dev , " Bad firmware file format: %d \n " , hdr - > version ) ;
cs42l43 - > firmware_error = - EINVAL ;
goto err_release ;
}
regmap_write ( cs42l43 - > regmap , CS42L43_PATCH_START_ADDR , loadaddr ) ;
regmap_bulk_write ( cs42l43 - > regmap , loadaddr + CS42L43_MCU_UPDATE_OFFSET ,
& firmware - > data [ 0 ] , firmware - > size / sizeof ( u32 ) ) ;
regmap_write ( cs42l43 - > regmap , CS42L43_MCU_SW_INTERRUPT , CS42L43_PATCH_IND_MASK ) ;
regmap_write ( cs42l43 - > regmap , CS42L43_MCU_SW_INTERRUPT , 0 ) ;
ret = regmap_read_poll_timeout ( cs42l43 - > regmap , CS42L43_SOFT_INT_SHADOW , val ,
( val & CS42L43_PATCH_APPLIED_INT_MASK ) ,
CS42L43_MCU_POLL , CS42L43_MCU_UPDATE_TIMEOUT ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to update firmware: %d, 0x%x \n " , ret , val ) ;
cs42l43 - > firmware_error = ret ;
goto err_release ;
}
err_release :
release_firmware ( firmware ) ;
err :
complete ( & cs42l43 - > firmware_download ) ;
}
/*
* The process of updating the firmware is split into a series of steps , at the
* end of each step a soft reset of the device might be required which will
* require the driver to wait for the device to re - attach on the SoundWire bus ,
* if that control bus is being used .
*/
static int cs42l43_mcu_update_step ( struct cs42l43 * cs42l43 )
{
unsigned int mcu_rev , bios_rev , boot_status , secure_cfg ;
bool patched , shadow ;
int ret ;
/* Clear any stale software interrupt bits. */
regmap_read ( cs42l43 - > regmap , CS42L43_SOFT_INT , & mcu_rev ) ;
ret = regmap_read ( cs42l43 - > regmap , CS42L43_BOOT_STATUS , & boot_status ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to read boot status: %d \n " , ret ) ;
return ret ;
}
ret = regmap_read ( cs42l43 - > regmap , CS42L43_MCU_SW_REV , & mcu_rev ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to read firmware revision: %d \n " , ret ) ;
return ret ;
}
bios_rev = ( ( ( mcu_rev & CS42L43_BIOS_MAJOR_REV_MASK ) < < 12 ) |
( ( mcu_rev & CS42L43_BIOS_MINOR_REV_MASK ) < < 4 ) |
( ( mcu_rev & CS42L43_BIOS_SUBMINOR_REV_MASK ) > > 8 ) ) > >
CS42L43_BIOS_MAJOR_REV_SHIFT ;
mcu_rev = ( ( mcu_rev & CS42L43_FW_MAJOR_REV_MASK ) < < 12 ) |
( ( mcu_rev & CS42L43_FW_MINOR_REV_MASK ) < < 4 ) |
( ( mcu_rev & CS42L43_FW_SUBMINOR_REV_MASK ) > > 8 ) ;
/*
* The firmware has two revision numbers bringing either of them up to a
* supported version will provide the features the driver requires .
*/
patched = mcu_rev > = CS42L43_MCU_SUPPORTED_REV | |
bios_rev > = CS42L43_MCU_SUPPORTED_BIOS_REV ;
/*
* Later versions of the firmwware require the driver to access some
* features through a set of shadow registers .
*/
shadow = mcu_rev > = CS42L43_MCU_SHADOW_REGS_REQUIRED_REV ;
ret = regmap_read ( cs42l43 - > regmap , CS42L43_BOOT_CONTROL , & secure_cfg ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to read security settings: %d \n " , ret ) ;
return ret ;
}
cs42l43 - > hw_lock = secure_cfg & CS42L43_LOCK_HW_STS_MASK ;
if ( ! patched & & cs42l43 - > hw_lock ) {
dev_err ( cs42l43 - > dev , " Unpatched secure device \n " ) ;
return - EPERM ;
}
dev_dbg ( cs42l43 - > dev , " Firmware(0x%x, 0x%x) in boot stage %d \n " ,
mcu_rev , bios_rev , boot_status ) ;
switch ( boot_status ) {
case CS42L43_MCU_BOOT_STAGE2 :
if ( ! patched ) {
ret = request_firmware_nowait ( THIS_MODULE , FW_ACTION_UEVENT ,
" cs42l43.bin " , cs42l43 - > dev ,
GFP_KERNEL , cs42l43 ,
cs42l43_mcu_load_firmware ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to request firmware: %d \n " , ret ) ;
return ret ;
}
wait_for_completion ( & cs42l43 - > firmware_download ) ;
if ( cs42l43 - > firmware_error )
return cs42l43 - > firmware_error ;
return - EAGAIN ;
} else {
return cs42l43_mcu_stage_2_3 ( cs42l43 , shadow ) ;
}
case CS42L43_MCU_BOOT_STAGE3 :
if ( patched )
return cs42l43_mcu_disable ( cs42l43 ) ;
else
return cs42l43_mcu_stage_3_2 ( cs42l43 ) ;
case CS42L43_MCU_BOOT_STAGE4 :
return 0 ;
default :
dev_err ( cs42l43 - > dev , " Invalid boot status: %d \n " , boot_status ) ;
return - EINVAL ;
}
}
/*
* Update the firmware running on the device .
*/
static int cs42l43_mcu_update ( struct cs42l43 * cs42l43 )
{
int i , ret ;
for ( i = 0 ; i < CS42L43_MCU_UPDATE_RETRIES ; i + + ) {
ret = cs42l43_mcu_update_step ( cs42l43 ) ;
if ( ret ! = - EAGAIN )
return ret ;
ret = cs42l43_wait_for_attach ( cs42l43 ) ;
if ( ret )
return ret ;
}
dev_err ( cs42l43 - > dev , " Failed retrying update \n " ) ;
return - ETIMEDOUT ;
}
static int cs42l43_irq_config ( struct cs42l43 * cs42l43 )
{
struct irq_data * irq_data ;
unsigned long irq_flags ;
int ret ;
if ( cs42l43 - > sdw )
cs42l43 - > irq = cs42l43 - > sdw - > irq ;
cs42l43 - > irq_chip = cs42l43_irq_chip ;
cs42l43 - > irq_chip . irq_drv_data = cs42l43 ;
irq_data = irq_get_irq_data ( cs42l43 - > irq ) ;
if ( ! irq_data ) {
dev_err ( cs42l43 - > dev , " Invalid IRQ: %d \n " , cs42l43 - > irq ) ;
return - EINVAL ;
}
irq_flags = irqd_get_trigger_type ( irq_data ) ;
switch ( irq_flags ) {
case IRQF_TRIGGER_LOW :
case IRQF_TRIGGER_HIGH :
case IRQF_TRIGGER_RISING :
case IRQF_TRIGGER_FALLING :
break ;
case IRQ_TYPE_NONE :
default :
irq_flags = IRQF_TRIGGER_LOW ;
break ;
}
irq_flags | = IRQF_ONESHOT ;
ret = devm_regmap_add_irq_chip ( cs42l43 - > dev , cs42l43 - > regmap ,
cs42l43 - > irq , irq_flags , 0 ,
& cs42l43 - > irq_chip , & cs42l43 - > irq_data ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to add IRQ chip: %d \n " , ret ) ;
return ret ;
}
dev_dbg ( cs42l43 - > dev , " Configured IRQ %d with flags 0x%lx \n " ,
cs42l43 - > irq , irq_flags ) ;
return 0 ;
}
static void cs42l43_boot_work ( struct work_struct * work )
{
struct cs42l43 * cs42l43 = container_of ( work , struct cs42l43 , boot_work ) ;
unsigned int devid , revid , otp ;
int ret ;
ret = cs42l43_wait_for_attach ( cs42l43 ) ;
if ( ret )
goto err ;
ret = regmap_read ( cs42l43 - > regmap , CS42L43_DEVID , & devid ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to read devid: %d \n " , ret ) ;
goto err ;
}
switch ( devid ) {
case CS42L43_DEVID_VAL :
break ;
default :
dev_err ( cs42l43 - > dev , " Unrecognised devid: 0x%06x \n " , devid ) ;
goto err ;
}
ret = regmap_read ( cs42l43 - > regmap , CS42L43_REVID , & revid ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to read rev: %d \n " , ret ) ;
goto err ;
}
ret = regmap_read ( cs42l43 - > regmap , CS42L43_OTP_REVISION_ID , & otp ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to read otp rev: %d \n " , ret ) ;
goto err ;
}
dev_info ( cs42l43 - > dev ,
" devid: 0x%06x, rev: 0x%02x, otp: 0x%02x \n " , devid , revid , otp ) ;
ret = cs42l43_mcu_update ( cs42l43 ) ;
if ( ret )
goto err ;
ret = regmap_register_patch ( cs42l43 - > regmap , cs42l43_reva_patch ,
ARRAY_SIZE ( cs42l43_reva_patch ) ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to apply register patch: %d \n " , ret ) ;
goto err ;
}
ret = cs42l43_irq_config ( cs42l43 ) ;
if ( ret )
goto err ;
ret = devm_mfd_add_devices ( cs42l43 - > dev , PLATFORM_DEVID_NONE ,
cs42l43_devs , ARRAY_SIZE ( cs42l43_devs ) ,
NULL , 0 , NULL ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to add subdevices: %d \n " , ret ) ;
goto err ;
}
pm_runtime_mark_last_busy ( cs42l43 - > dev ) ;
pm_runtime_put_autosuspend ( cs42l43 - > dev ) ;
return ;
err :
pm_runtime_put_sync ( cs42l43 - > dev ) ;
cs42l43_dev_remove ( cs42l43 ) ;
}
static int cs42l43_power_up ( struct cs42l43 * cs42l43 )
{
int ret ;
ret = regulator_enable ( cs42l43 - > vdd_p ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to enable vdd-p: %d \n " , ret ) ;
return ret ;
}
/* vdd-p must be on for 50uS before any other supply */
usleep_range ( CS42L43_VDDP_DELAY , 2 * CS42L43_VDDP_DELAY ) ;
gpiod_set_value_cansleep ( cs42l43 - > reset , 1 ) ;
ret = regulator_bulk_enable ( CS42L43_N_SUPPLIES , cs42l43 - > core_supplies ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to enable core supplies: %d \n " , ret ) ;
goto err_reset ;
}
ret = regulator_enable ( cs42l43 - > vdd_d ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to enable vdd-d: %d \n " , ret ) ;
goto err_core_supplies ;
}
usleep_range ( CS42L43_VDDD_DELAY , 2 * CS42L43_VDDD_DELAY ) ;
return 0 ;
err_core_supplies :
regulator_bulk_disable ( CS42L43_N_SUPPLIES , cs42l43 - > core_supplies ) ;
err_reset :
gpiod_set_value_cansleep ( cs42l43 - > reset , 0 ) ;
regulator_disable ( cs42l43 - > vdd_p ) ;
return ret ;
}
static int cs42l43_power_down ( struct cs42l43 * cs42l43 )
{
int ret ;
ret = regulator_disable ( cs42l43 - > vdd_d ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to disable vdd-d: %d \n " , ret ) ;
return ret ;
}
ret = regulator_bulk_disable ( CS42L43_N_SUPPLIES , cs42l43 - > core_supplies ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to disable core supplies: %d \n " , ret ) ;
return ret ;
}
gpiod_set_value_cansleep ( cs42l43 - > reset , 0 ) ;
ret = regulator_disable ( cs42l43 - > vdd_p ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to disable vdd-p: %d \n " , ret ) ;
return ret ;
}
return 0 ;
}
int cs42l43_dev_probe ( struct cs42l43 * cs42l43 )
{
int i , ret ;
dev_set_drvdata ( cs42l43 - > dev , cs42l43 ) ;
mutex_init ( & cs42l43 - > pll_lock ) ;
init_completion ( & cs42l43 - > device_attach ) ;
init_completion ( & cs42l43 - > device_detach ) ;
init_completion ( & cs42l43 - > firmware_download ) ;
INIT_WORK ( & cs42l43 - > boot_work , cs42l43_boot_work ) ;
regcache_cache_only ( cs42l43 - > regmap , true ) ;
cs42l43 - > reset = devm_gpiod_get_optional ( cs42l43 - > dev , " reset " , GPIOD_OUT_LOW ) ;
if ( IS_ERR ( cs42l43 - > reset ) )
return dev_err_probe ( cs42l43 - > dev , PTR_ERR ( cs42l43 - > reset ) ,
" Failed to get reset \n " ) ;
cs42l43 - > vdd_p = devm_regulator_get ( cs42l43 - > dev , " vdd-p " ) ;
if ( IS_ERR ( cs42l43 - > vdd_p ) )
return dev_err_probe ( cs42l43 - > dev , PTR_ERR ( cs42l43 - > vdd_p ) ,
" Failed to get vdd-p \n " ) ;
cs42l43 - > vdd_d = devm_regulator_get ( cs42l43 - > dev , " vdd-d " ) ;
if ( IS_ERR ( cs42l43 - > vdd_d ) )
return dev_err_probe ( cs42l43 - > dev , PTR_ERR ( cs42l43 - > vdd_d ) ,
" Failed to get vdd-d \n " ) ;
BUILD_BUG_ON ( ARRAY_SIZE ( cs42l43_core_supplies ) ! = CS42L43_N_SUPPLIES ) ;
for ( i = 0 ; i < CS42L43_N_SUPPLIES ; i + + )
cs42l43 - > core_supplies [ i ] . supply = cs42l43_core_supplies [ i ] ;
ret = devm_regulator_bulk_get ( cs42l43 - > dev , CS42L43_N_SUPPLIES ,
cs42l43 - > core_supplies ) ;
if ( ret )
return dev_err_probe ( cs42l43 - > dev , ret ,
" Failed to get core supplies \n " ) ;
ret = cs42l43_power_up ( cs42l43 ) ;
if ( ret )
return ret ;
pm_runtime_set_autosuspend_delay ( cs42l43 - > dev , CS42L43_AUTOSUSPEND_TIME ) ;
pm_runtime_use_autosuspend ( cs42l43 - > dev ) ;
pm_runtime_set_active ( cs42l43 - > dev ) ;
/*
* The device is already powered up , but keep it from suspending until
* the boot work runs .
*/
pm_runtime_get_noresume ( cs42l43 - > dev ) ;
devm_pm_runtime_enable ( cs42l43 - > dev ) ;
queue_work ( system_long_wq , & cs42l43 - > boot_work ) ;
return 0 ;
}
EXPORT_SYMBOL_NS_GPL ( cs42l43_dev_probe , MFD_CS42L43 ) ;
void cs42l43_dev_remove ( struct cs42l43 * cs42l43 )
{
cs42l43_power_down ( cs42l43 ) ;
}
EXPORT_SYMBOL_NS_GPL ( cs42l43_dev_remove , MFD_CS42L43 ) ;
static int cs42l43_suspend ( struct device * dev )
{
struct cs42l43 * cs42l43 = dev_get_drvdata ( dev ) ;
int ret ;
/*
* Don ' t care about being resumed here , but the driver does want
* force_resume to always trigger an actual resume , so that register
* state for the MCU / GPIOs is returned as soon as possible after system
* resume . force_resume will resume if the reference count is resumed on
* suspend hence the get_noresume .
*/
pm_runtime_get_noresume ( dev ) ;
ret = pm_runtime_force_suspend ( dev ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to force suspend: %d \n " , ret ) ;
pm_runtime_put_noidle ( dev ) ;
return ret ;
}
pm_runtime_put_noidle ( dev ) ;
ret = cs42l43_power_down ( cs42l43 ) ;
if ( ret )
return ret ;
return 0 ;
}
static int cs42l43_resume ( struct device * dev )
{
struct cs42l43 * cs42l43 = dev_get_drvdata ( dev ) ;
int ret ;
ret = cs42l43_power_up ( cs42l43 ) ;
if ( ret )
return ret ;
ret = pm_runtime_force_resume ( dev ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to force resume: %d \n " , ret ) ;
return ret ;
}
return 0 ;
}
static int cs42l43_runtime_suspend ( struct device * dev )
{
struct cs42l43 * cs42l43 = dev_get_drvdata ( dev ) ;
/*
* Whilst the driver doesn ' t power the chip down here , going into runtime
* suspend lets the SoundWire bus power down , which means the driver
* can ' t communicate with the device any more .
*/
regcache_cache_only ( cs42l43 - > regmap , true ) ;
return 0 ;
}
static int cs42l43_runtime_resume ( struct device * dev )
{
struct cs42l43 * cs42l43 = dev_get_drvdata ( dev ) ;
unsigned int reset_canary ;
int ret ;
ret = cs42l43_wait_for_attach ( cs42l43 ) ;
if ( ret )
return ret ;
ret = regmap_read ( cs42l43 - > regmap , CS42L43_RELID , & reset_canary ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to check reset canary: %d \n " , ret ) ;
goto err ;
}
if ( ! reset_canary ) {
/*
* If the canary has cleared the chip has reset , re - handle the
* MCU and mark the cache as dirty to indicate the chip reset .
*/
ret = cs42l43_mcu_update ( cs42l43 ) ;
if ( ret )
goto err ;
regcache_mark_dirty ( cs42l43 - > regmap ) ;
}
ret = regcache_sync ( cs42l43 - > regmap ) ;
if ( ret ) {
dev_err ( cs42l43 - > dev , " Failed to restore register cache: %d \n " , ret ) ;
goto err ;
}
return 0 ;
err :
regcache_cache_only ( cs42l43 - > regmap , true ) ;
return ret ;
}
EXPORT_NS_GPL_DEV_PM_OPS ( cs42l43_pm_ops , MFD_CS42L43 ) = {
2023-09-19 13:03:20 +02:00
SYSTEM_SLEEP_PM_OPS ( cs42l43_suspend , cs42l43_resume )
RUNTIME_PM_OPS ( cs42l43_runtime_suspend , cs42l43_runtime_resume , NULL )
2023-08-04 11:45:59 +01:00
} ;
MODULE_DESCRIPTION ( " CS42L43 Core Driver " ) ;
MODULE_AUTHOR ( " Charles Keepax <ckeepax@opensource.cirrus.com> " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_FIRMWARE ( " cs42l43.bin " ) ;