162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * cs42l73.c -- CS42L73 ALSA Soc Audio driver 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright 2011 Cirrus Logic, Inc. 662306a36Sopenharmony_ci * 762306a36Sopenharmony_ci * Authors: Georgi Vlaev, Nucleus Systems Ltd, <joe@nucleusys.com> 862306a36Sopenharmony_ci * Brian Austin, Cirrus Logic Inc, <brian.austin@cirrus.com> 962306a36Sopenharmony_ci */ 1062306a36Sopenharmony_ci 1162306a36Sopenharmony_ci#include <linux/module.h> 1262306a36Sopenharmony_ci#include <linux/moduleparam.h> 1362306a36Sopenharmony_ci#include <linux/kernel.h> 1462306a36Sopenharmony_ci#include <linux/init.h> 1562306a36Sopenharmony_ci#include <linux/delay.h> 1662306a36Sopenharmony_ci#include <linux/of_gpio.h> 1762306a36Sopenharmony_ci#include <linux/pm.h> 1862306a36Sopenharmony_ci#include <linux/i2c.h> 1962306a36Sopenharmony_ci#include <linux/regmap.h> 2062306a36Sopenharmony_ci#include <linux/slab.h> 2162306a36Sopenharmony_ci#include <sound/core.h> 2262306a36Sopenharmony_ci#include <sound/pcm.h> 2362306a36Sopenharmony_ci#include <sound/pcm_params.h> 2462306a36Sopenharmony_ci#include <sound/soc.h> 2562306a36Sopenharmony_ci#include <sound/soc-dapm.h> 2662306a36Sopenharmony_ci#include <sound/initval.h> 2762306a36Sopenharmony_ci#include <sound/tlv.h> 2862306a36Sopenharmony_ci#include <sound/cs42l73.h> 2962306a36Sopenharmony_ci#include "cs42l73.h" 3062306a36Sopenharmony_ci#include "cirrus_legacy.h" 3162306a36Sopenharmony_ci 3262306a36Sopenharmony_cistruct sp_config { 3362306a36Sopenharmony_ci u8 spc, mmcc, spfs; 3462306a36Sopenharmony_ci u32 srate; 3562306a36Sopenharmony_ci}; 3662306a36Sopenharmony_cistruct cs42l73_private { 3762306a36Sopenharmony_ci struct cs42l73_platform_data pdata; 3862306a36Sopenharmony_ci struct sp_config config[3]; 3962306a36Sopenharmony_ci struct regmap *regmap; 4062306a36Sopenharmony_ci u32 sysclk; 4162306a36Sopenharmony_ci u8 mclksel; 4262306a36Sopenharmony_ci u32 mclk; 4362306a36Sopenharmony_ci int shutdwn_delay; 4462306a36Sopenharmony_ci}; 4562306a36Sopenharmony_ci 4662306a36Sopenharmony_cistatic const struct reg_default cs42l73_reg_defaults[] = { 4762306a36Sopenharmony_ci { 6, 0xF1 }, /* r06 - Power Ctl 1 */ 4862306a36Sopenharmony_ci { 7, 0xDF }, /* r07 - Power Ctl 2 */ 4962306a36Sopenharmony_ci { 8, 0x3F }, /* r08 - Power Ctl 3 */ 5062306a36Sopenharmony_ci { 9, 0x50 }, /* r09 - Charge Pump Freq */ 5162306a36Sopenharmony_ci { 10, 0x53 }, /* r0A - Output Load MicBias Short Detect */ 5262306a36Sopenharmony_ci { 11, 0x00 }, /* r0B - DMIC Master Clock Ctl */ 5362306a36Sopenharmony_ci { 12, 0x00 }, /* r0C - Aux PCM Ctl */ 5462306a36Sopenharmony_ci { 13, 0x15 }, /* r0D - Aux PCM Master Clock Ctl */ 5562306a36Sopenharmony_ci { 14, 0x00 }, /* r0E - Audio PCM Ctl */ 5662306a36Sopenharmony_ci { 15, 0x15 }, /* r0F - Audio PCM Master Clock Ctl */ 5762306a36Sopenharmony_ci { 16, 0x00 }, /* r10 - Voice PCM Ctl */ 5862306a36Sopenharmony_ci { 17, 0x15 }, /* r11 - Voice PCM Master Clock Ctl */ 5962306a36Sopenharmony_ci { 18, 0x00 }, /* r12 - Voice/Aux Sample Rate */ 6062306a36Sopenharmony_ci { 19, 0x06 }, /* r13 - Misc I/O Path Ctl */ 6162306a36Sopenharmony_ci { 20, 0x00 }, /* r14 - ADC Input Path Ctl */ 6262306a36Sopenharmony_ci { 21, 0x00 }, /* r15 - MICA Preamp, PGA Volume */ 6362306a36Sopenharmony_ci { 22, 0x00 }, /* r16 - MICB Preamp, PGA Volume */ 6462306a36Sopenharmony_ci { 23, 0x00 }, /* r17 - Input Path A Digital Volume */ 6562306a36Sopenharmony_ci { 24, 0x00 }, /* r18 - Input Path B Digital Volume */ 6662306a36Sopenharmony_ci { 25, 0x00 }, /* r19 - Playback Digital Ctl */ 6762306a36Sopenharmony_ci { 26, 0x00 }, /* r1A - HP/LO Left Digital Volume */ 6862306a36Sopenharmony_ci { 27, 0x00 }, /* r1B - HP/LO Right Digital Volume */ 6962306a36Sopenharmony_ci { 28, 0x00 }, /* r1C - Speakerphone Digital Volume */ 7062306a36Sopenharmony_ci { 29, 0x00 }, /* r1D - Ear/SPKLO Digital Volume */ 7162306a36Sopenharmony_ci { 30, 0x00 }, /* r1E - HP Left Analog Volume */ 7262306a36Sopenharmony_ci { 31, 0x00 }, /* r1F - HP Right Analog Volume */ 7362306a36Sopenharmony_ci { 32, 0x00 }, /* r20 - LO Left Analog Volume */ 7462306a36Sopenharmony_ci { 33, 0x00 }, /* r21 - LO Right Analog Volume */ 7562306a36Sopenharmony_ci { 34, 0x00 }, /* r22 - Stereo Input Path Advisory Volume */ 7662306a36Sopenharmony_ci { 35, 0x00 }, /* r23 - Aux PCM Input Advisory Volume */ 7762306a36Sopenharmony_ci { 36, 0x00 }, /* r24 - Audio PCM Input Advisory Volume */ 7862306a36Sopenharmony_ci { 37, 0x00 }, /* r25 - Voice PCM Input Advisory Volume */ 7962306a36Sopenharmony_ci { 38, 0x00 }, /* r26 - Limiter Attack Rate HP/LO */ 8062306a36Sopenharmony_ci { 39, 0x7F }, /* r27 - Limter Ctl, Release Rate HP/LO */ 8162306a36Sopenharmony_ci { 40, 0x00 }, /* r28 - Limter Threshold HP/LO */ 8262306a36Sopenharmony_ci { 41, 0x00 }, /* r29 - Limiter Attack Rate Speakerphone */ 8362306a36Sopenharmony_ci { 42, 0x3F }, /* r2A - Limter Ctl, Release Rate Speakerphone */ 8462306a36Sopenharmony_ci { 43, 0x00 }, /* r2B - Limter Threshold Speakerphone */ 8562306a36Sopenharmony_ci { 44, 0x00 }, /* r2C - Limiter Attack Rate Ear/SPKLO */ 8662306a36Sopenharmony_ci { 45, 0x3F }, /* r2D - Limter Ctl, Release Rate Ear/SPKLO */ 8762306a36Sopenharmony_ci { 46, 0x00 }, /* r2E - Limter Threshold Ear/SPKLO */ 8862306a36Sopenharmony_ci { 47, 0x00 }, /* r2F - ALC Enable, Attack Rate Left/Right */ 8962306a36Sopenharmony_ci { 48, 0x3F }, /* r30 - ALC Release Rate Left/Right */ 9062306a36Sopenharmony_ci { 49, 0x00 }, /* r31 - ALC Threshold Left/Right */ 9162306a36Sopenharmony_ci { 50, 0x00 }, /* r32 - Noise Gate Ctl Left/Right */ 9262306a36Sopenharmony_ci { 51, 0x00 }, /* r33 - ALC/NG Misc Ctl */ 9362306a36Sopenharmony_ci { 52, 0x18 }, /* r34 - Mixer Ctl */ 9462306a36Sopenharmony_ci { 53, 0x3F }, /* r35 - HP/LO Left Mixer Input Path Volume */ 9562306a36Sopenharmony_ci { 54, 0x3F }, /* r36 - HP/LO Right Mixer Input Path Volume */ 9662306a36Sopenharmony_ci { 55, 0x3F }, /* r37 - HP/LO Left Mixer Aux PCM Volume */ 9762306a36Sopenharmony_ci { 56, 0x3F }, /* r38 - HP/LO Right Mixer Aux PCM Volume */ 9862306a36Sopenharmony_ci { 57, 0x3F }, /* r39 - HP/LO Left Mixer Audio PCM Volume */ 9962306a36Sopenharmony_ci { 58, 0x3F }, /* r3A - HP/LO Right Mixer Audio PCM Volume */ 10062306a36Sopenharmony_ci { 59, 0x3F }, /* r3B - HP/LO Left Mixer Voice PCM Mono Volume */ 10162306a36Sopenharmony_ci { 60, 0x3F }, /* r3C - HP/LO Right Mixer Voice PCM Mono Volume */ 10262306a36Sopenharmony_ci { 61, 0x3F }, /* r3D - Aux PCM Left Mixer Input Path Volume */ 10362306a36Sopenharmony_ci { 62, 0x3F }, /* r3E - Aux PCM Right Mixer Input Path Volume */ 10462306a36Sopenharmony_ci { 63, 0x3F }, /* r3F - Aux PCM Left Mixer Volume */ 10562306a36Sopenharmony_ci { 64, 0x3F }, /* r40 - Aux PCM Left Mixer Volume */ 10662306a36Sopenharmony_ci { 65, 0x3F }, /* r41 - Aux PCM Left Mixer Audio PCM L Volume */ 10762306a36Sopenharmony_ci { 66, 0x3F }, /* r42 - Aux PCM Right Mixer Audio PCM R Volume */ 10862306a36Sopenharmony_ci { 67, 0x3F }, /* r43 - Aux PCM Left Mixer Voice PCM Volume */ 10962306a36Sopenharmony_ci { 68, 0x3F }, /* r44 - Aux PCM Right Mixer Voice PCM Volume */ 11062306a36Sopenharmony_ci { 69, 0x3F }, /* r45 - Audio PCM Left Input Path Volume */ 11162306a36Sopenharmony_ci { 70, 0x3F }, /* r46 - Audio PCM Right Input Path Volume */ 11262306a36Sopenharmony_ci { 71, 0x3F }, /* r47 - Audio PCM Left Mixer Aux PCM L Volume */ 11362306a36Sopenharmony_ci { 72, 0x3F }, /* r48 - Audio PCM Right Mixer Aux PCM R Volume */ 11462306a36Sopenharmony_ci { 73, 0x3F }, /* r49 - Audio PCM Left Mixer Volume */ 11562306a36Sopenharmony_ci { 74, 0x3F }, /* r4A - Audio PCM Right Mixer Volume */ 11662306a36Sopenharmony_ci { 75, 0x3F }, /* r4B - Audio PCM Left Mixer Voice PCM Volume */ 11762306a36Sopenharmony_ci { 76, 0x3F }, /* r4C - Audio PCM Right Mixer Voice PCM Volume */ 11862306a36Sopenharmony_ci { 77, 0x3F }, /* r4D - Voice PCM Left Input Path Volume */ 11962306a36Sopenharmony_ci { 78, 0x3F }, /* r4E - Voice PCM Right Input Path Volume */ 12062306a36Sopenharmony_ci { 79, 0x3F }, /* r4F - Voice PCM Left Mixer Aux PCM L Volume */ 12162306a36Sopenharmony_ci { 80, 0x3F }, /* r50 - Voice PCM Right Mixer Aux PCM R Volume */ 12262306a36Sopenharmony_ci { 81, 0x3F }, /* r51 - Voice PCM Left Mixer Audio PCM L Volume */ 12362306a36Sopenharmony_ci { 82, 0x3F }, /* r52 - Voice PCM Right Mixer Audio PCM R Volume */ 12462306a36Sopenharmony_ci { 83, 0x3F }, /* r53 - Voice PCM Left Mixer Voice PCM Volume */ 12562306a36Sopenharmony_ci { 84, 0x3F }, /* r54 - Voice PCM Right Mixer Voice PCM Volume */ 12662306a36Sopenharmony_ci { 85, 0xAA }, /* r55 - Mono Mixer Ctl */ 12762306a36Sopenharmony_ci { 86, 0x3F }, /* r56 - SPK Mono Mixer Input Path Volume */ 12862306a36Sopenharmony_ci { 87, 0x3F }, /* r57 - SPK Mono Mixer Aux PCM Mono/L/R Volume */ 12962306a36Sopenharmony_ci { 88, 0x3F }, /* r58 - SPK Mono Mixer Audio PCM Mono/L/R Volume */ 13062306a36Sopenharmony_ci { 89, 0x3F }, /* r59 - SPK Mono Mixer Voice PCM Mono Volume */ 13162306a36Sopenharmony_ci { 90, 0x3F }, /* r5A - SPKLO Mono Mixer Input Path Mono Volume */ 13262306a36Sopenharmony_ci { 91, 0x3F }, /* r5B - SPKLO Mono Mixer Aux Mono/L/R Volume */ 13362306a36Sopenharmony_ci { 92, 0x3F }, /* r5C - SPKLO Mono Mixer Audio Mono/L/R Volume */ 13462306a36Sopenharmony_ci { 93, 0x3F }, /* r5D - SPKLO Mono Mixer Voice Mono Volume */ 13562306a36Sopenharmony_ci { 94, 0x00 }, /* r5E - Interrupt Mask 1 */ 13662306a36Sopenharmony_ci { 95, 0x00 }, /* r5F - Interrupt Mask 2 */ 13762306a36Sopenharmony_ci}; 13862306a36Sopenharmony_ci 13962306a36Sopenharmony_cistatic bool cs42l73_volatile_register(struct device *dev, unsigned int reg) 14062306a36Sopenharmony_ci{ 14162306a36Sopenharmony_ci switch (reg) { 14262306a36Sopenharmony_ci case CS42L73_IS1: 14362306a36Sopenharmony_ci case CS42L73_IS2: 14462306a36Sopenharmony_ci return true; 14562306a36Sopenharmony_ci default: 14662306a36Sopenharmony_ci return false; 14762306a36Sopenharmony_ci } 14862306a36Sopenharmony_ci} 14962306a36Sopenharmony_ci 15062306a36Sopenharmony_cistatic bool cs42l73_readable_register(struct device *dev, unsigned int reg) 15162306a36Sopenharmony_ci{ 15262306a36Sopenharmony_ci switch (reg) { 15362306a36Sopenharmony_ci case CS42L73_DEVID_AB ... CS42L73_DEVID_E: 15462306a36Sopenharmony_ci case CS42L73_REVID ... CS42L73_IM2: 15562306a36Sopenharmony_ci return true; 15662306a36Sopenharmony_ci default: 15762306a36Sopenharmony_ci return false; 15862306a36Sopenharmony_ci } 15962306a36Sopenharmony_ci} 16062306a36Sopenharmony_ci 16162306a36Sopenharmony_cistatic const DECLARE_TLV_DB_RANGE(hpaloa_tlv, 16262306a36Sopenharmony_ci 0, 13, TLV_DB_SCALE_ITEM(-7600, 200, 0), 16362306a36Sopenharmony_ci 14, 75, TLV_DB_SCALE_ITEM(-4900, 100, 0) 16462306a36Sopenharmony_ci); 16562306a36Sopenharmony_ci 16662306a36Sopenharmony_cistatic DECLARE_TLV_DB_SCALE(adc_boost_tlv, 0, 2500, 0); 16762306a36Sopenharmony_ci 16862306a36Sopenharmony_cistatic DECLARE_TLV_DB_SCALE(hl_tlv, -10200, 50, 0); 16962306a36Sopenharmony_ci 17062306a36Sopenharmony_cistatic DECLARE_TLV_DB_SCALE(ipd_tlv, -9600, 100, 0); 17162306a36Sopenharmony_ci 17262306a36Sopenharmony_cistatic DECLARE_TLV_DB_SCALE(micpga_tlv, -600, 50, 0); 17362306a36Sopenharmony_ci 17462306a36Sopenharmony_cistatic const DECLARE_TLV_DB_RANGE(limiter_tlv, 17562306a36Sopenharmony_ci 0, 2, TLV_DB_SCALE_ITEM(-3000, 600, 0), 17662306a36Sopenharmony_ci 3, 7, TLV_DB_SCALE_ITEM(-1200, 300, 0) 17762306a36Sopenharmony_ci); 17862306a36Sopenharmony_ci 17962306a36Sopenharmony_cistatic const DECLARE_TLV_DB_SCALE(attn_tlv, -6300, 100, 1); 18062306a36Sopenharmony_ci 18162306a36Sopenharmony_cistatic const char * const cs42l73_pgaa_text[] = { "Line A", "Mic 1" }; 18262306a36Sopenharmony_cistatic const char * const cs42l73_pgab_text[] = { "Line B", "Mic 2" }; 18362306a36Sopenharmony_ci 18462306a36Sopenharmony_cistatic SOC_ENUM_SINGLE_DECL(pgaa_enum, 18562306a36Sopenharmony_ci CS42L73_ADCIPC, 3, 18662306a36Sopenharmony_ci cs42l73_pgaa_text); 18762306a36Sopenharmony_ci 18862306a36Sopenharmony_cistatic SOC_ENUM_SINGLE_DECL(pgab_enum, 18962306a36Sopenharmony_ci CS42L73_ADCIPC, 7, 19062306a36Sopenharmony_ci cs42l73_pgab_text); 19162306a36Sopenharmony_ci 19262306a36Sopenharmony_cistatic const struct snd_kcontrol_new pgaa_mux = 19362306a36Sopenharmony_ci SOC_DAPM_ENUM("Left Analog Input Capture Mux", pgaa_enum); 19462306a36Sopenharmony_ci 19562306a36Sopenharmony_cistatic const struct snd_kcontrol_new pgab_mux = 19662306a36Sopenharmony_ci SOC_DAPM_ENUM("Right Analog Input Capture Mux", pgab_enum); 19762306a36Sopenharmony_ci 19862306a36Sopenharmony_cistatic const struct snd_kcontrol_new input_left_mixer[] = { 19962306a36Sopenharmony_ci SOC_DAPM_SINGLE("ADC Left Input", CS42L73_PWRCTL1, 20062306a36Sopenharmony_ci 5, 1, 1), 20162306a36Sopenharmony_ci SOC_DAPM_SINGLE("DMIC Left Input", CS42L73_PWRCTL1, 20262306a36Sopenharmony_ci 4, 1, 1), 20362306a36Sopenharmony_ci}; 20462306a36Sopenharmony_ci 20562306a36Sopenharmony_cistatic const struct snd_kcontrol_new input_right_mixer[] = { 20662306a36Sopenharmony_ci SOC_DAPM_SINGLE("ADC Right Input", CS42L73_PWRCTL1, 20762306a36Sopenharmony_ci 7, 1, 1), 20862306a36Sopenharmony_ci SOC_DAPM_SINGLE("DMIC Right Input", CS42L73_PWRCTL1, 20962306a36Sopenharmony_ci 6, 1, 1), 21062306a36Sopenharmony_ci}; 21162306a36Sopenharmony_ci 21262306a36Sopenharmony_cistatic const char * const cs42l73_ng_delay_text[] = { 21362306a36Sopenharmony_ci "50ms", "100ms", "150ms", "200ms" }; 21462306a36Sopenharmony_ci 21562306a36Sopenharmony_cistatic SOC_ENUM_SINGLE_DECL(ng_delay_enum, 21662306a36Sopenharmony_ci CS42L73_NGCAB, 0, 21762306a36Sopenharmony_ci cs42l73_ng_delay_text); 21862306a36Sopenharmony_ci 21962306a36Sopenharmony_cistatic const char * const cs42l73_mono_mix_texts[] = { 22062306a36Sopenharmony_ci "Left", "Right", "Mono Mix"}; 22162306a36Sopenharmony_ci 22262306a36Sopenharmony_cistatic const unsigned int cs42l73_mono_mix_values[] = { 0, 1, 2 }; 22362306a36Sopenharmony_ci 22462306a36Sopenharmony_cistatic const struct soc_enum spk_asp_enum = 22562306a36Sopenharmony_ci SOC_VALUE_ENUM_SINGLE(CS42L73_MMIXCTL, 6, 3, 22662306a36Sopenharmony_ci ARRAY_SIZE(cs42l73_mono_mix_texts), 22762306a36Sopenharmony_ci cs42l73_mono_mix_texts, 22862306a36Sopenharmony_ci cs42l73_mono_mix_values); 22962306a36Sopenharmony_ci 23062306a36Sopenharmony_cistatic const struct snd_kcontrol_new spk_asp_mixer = 23162306a36Sopenharmony_ci SOC_DAPM_ENUM("Route", spk_asp_enum); 23262306a36Sopenharmony_ci 23362306a36Sopenharmony_cistatic const struct soc_enum spk_xsp_enum = 23462306a36Sopenharmony_ci SOC_VALUE_ENUM_SINGLE(CS42L73_MMIXCTL, 4, 3, 23562306a36Sopenharmony_ci ARRAY_SIZE(cs42l73_mono_mix_texts), 23662306a36Sopenharmony_ci cs42l73_mono_mix_texts, 23762306a36Sopenharmony_ci cs42l73_mono_mix_values); 23862306a36Sopenharmony_ci 23962306a36Sopenharmony_cistatic const struct snd_kcontrol_new spk_xsp_mixer = 24062306a36Sopenharmony_ci SOC_DAPM_ENUM("Route", spk_xsp_enum); 24162306a36Sopenharmony_ci 24262306a36Sopenharmony_cistatic const struct soc_enum esl_asp_enum = 24362306a36Sopenharmony_ci SOC_VALUE_ENUM_SINGLE(CS42L73_MMIXCTL, 2, 3, 24462306a36Sopenharmony_ci ARRAY_SIZE(cs42l73_mono_mix_texts), 24562306a36Sopenharmony_ci cs42l73_mono_mix_texts, 24662306a36Sopenharmony_ci cs42l73_mono_mix_values); 24762306a36Sopenharmony_ci 24862306a36Sopenharmony_cistatic const struct snd_kcontrol_new esl_asp_mixer = 24962306a36Sopenharmony_ci SOC_DAPM_ENUM("Route", esl_asp_enum); 25062306a36Sopenharmony_ci 25162306a36Sopenharmony_cistatic const struct soc_enum esl_xsp_enum = 25262306a36Sopenharmony_ci SOC_VALUE_ENUM_SINGLE(CS42L73_MMIXCTL, 0, 3, 25362306a36Sopenharmony_ci ARRAY_SIZE(cs42l73_mono_mix_texts), 25462306a36Sopenharmony_ci cs42l73_mono_mix_texts, 25562306a36Sopenharmony_ci cs42l73_mono_mix_values); 25662306a36Sopenharmony_ci 25762306a36Sopenharmony_cistatic const struct snd_kcontrol_new esl_xsp_mixer = 25862306a36Sopenharmony_ci SOC_DAPM_ENUM("Route", esl_xsp_enum); 25962306a36Sopenharmony_ci 26062306a36Sopenharmony_cistatic const char * const cs42l73_ip_swap_text[] = { 26162306a36Sopenharmony_ci "Stereo", "Mono A", "Mono B", "Swap A-B"}; 26262306a36Sopenharmony_ci 26362306a36Sopenharmony_cistatic SOC_ENUM_SINGLE_DECL(ip_swap_enum, 26462306a36Sopenharmony_ci CS42L73_MIOPC, 6, 26562306a36Sopenharmony_ci cs42l73_ip_swap_text); 26662306a36Sopenharmony_ci 26762306a36Sopenharmony_cistatic const char * const cs42l73_spo_mixer_text[] = {"Mono", "Stereo"}; 26862306a36Sopenharmony_ci 26962306a36Sopenharmony_cistatic SOC_ENUM_SINGLE_DECL(vsp_output_mux_enum, 27062306a36Sopenharmony_ci CS42L73_MIXERCTL, 5, 27162306a36Sopenharmony_ci cs42l73_spo_mixer_text); 27262306a36Sopenharmony_ci 27362306a36Sopenharmony_cistatic SOC_ENUM_SINGLE_DECL(xsp_output_mux_enum, 27462306a36Sopenharmony_ci CS42L73_MIXERCTL, 4, 27562306a36Sopenharmony_ci cs42l73_spo_mixer_text); 27662306a36Sopenharmony_ci 27762306a36Sopenharmony_cistatic const struct snd_kcontrol_new hp_amp_ctl = 27862306a36Sopenharmony_ci SOC_DAPM_SINGLE("Switch", CS42L73_PWRCTL3, 0, 1, 1); 27962306a36Sopenharmony_ci 28062306a36Sopenharmony_cistatic const struct snd_kcontrol_new lo_amp_ctl = 28162306a36Sopenharmony_ci SOC_DAPM_SINGLE("Switch", CS42L73_PWRCTL3, 1, 1, 1); 28262306a36Sopenharmony_ci 28362306a36Sopenharmony_cistatic const struct snd_kcontrol_new spk_amp_ctl = 28462306a36Sopenharmony_ci SOC_DAPM_SINGLE("Switch", CS42L73_PWRCTL3, 2, 1, 1); 28562306a36Sopenharmony_ci 28662306a36Sopenharmony_cistatic const struct snd_kcontrol_new spklo_amp_ctl = 28762306a36Sopenharmony_ci SOC_DAPM_SINGLE("Switch", CS42L73_PWRCTL3, 4, 1, 1); 28862306a36Sopenharmony_ci 28962306a36Sopenharmony_cistatic const struct snd_kcontrol_new ear_amp_ctl = 29062306a36Sopenharmony_ci SOC_DAPM_SINGLE("Switch", CS42L73_PWRCTL3, 3, 1, 1); 29162306a36Sopenharmony_ci 29262306a36Sopenharmony_cistatic const struct snd_kcontrol_new cs42l73_snd_controls[] = { 29362306a36Sopenharmony_ci SOC_DOUBLE_R_SX_TLV("Headphone Analog Playback Volume", 29462306a36Sopenharmony_ci CS42L73_HPAAVOL, CS42L73_HPBAVOL, 0, 29562306a36Sopenharmony_ci 0x41, 0x4B, hpaloa_tlv), 29662306a36Sopenharmony_ci 29762306a36Sopenharmony_ci SOC_DOUBLE_R_SX_TLV("LineOut Analog Playback Volume", CS42L73_LOAAVOL, 29862306a36Sopenharmony_ci CS42L73_LOBAVOL, 0, 0x41, 0x4B, hpaloa_tlv), 29962306a36Sopenharmony_ci 30062306a36Sopenharmony_ci SOC_DOUBLE_R_SX_TLV("Input PGA Analog Volume", CS42L73_MICAPREPGAAVOL, 30162306a36Sopenharmony_ci CS42L73_MICBPREPGABVOL, 0, 0x34, 30262306a36Sopenharmony_ci 0x24, micpga_tlv), 30362306a36Sopenharmony_ci 30462306a36Sopenharmony_ci SOC_DOUBLE_R("MIC Preamp Switch", CS42L73_MICAPREPGAAVOL, 30562306a36Sopenharmony_ci CS42L73_MICBPREPGABVOL, 6, 1, 1), 30662306a36Sopenharmony_ci 30762306a36Sopenharmony_ci SOC_DOUBLE_R_SX_TLV("Input Path Digital Volume", CS42L73_IPADVOL, 30862306a36Sopenharmony_ci CS42L73_IPBDVOL, 0, 0xA0, 0x6C, ipd_tlv), 30962306a36Sopenharmony_ci 31062306a36Sopenharmony_ci SOC_DOUBLE_R_SX_TLV("HL Digital Playback Volume", 31162306a36Sopenharmony_ci CS42L73_HLADVOL, CS42L73_HLBDVOL, 31262306a36Sopenharmony_ci 0, 0x34, 0xE4, hl_tlv), 31362306a36Sopenharmony_ci 31462306a36Sopenharmony_ci SOC_SINGLE_TLV("ADC A Boost Volume", 31562306a36Sopenharmony_ci CS42L73_ADCIPC, 2, 0x01, 1, adc_boost_tlv), 31662306a36Sopenharmony_ci 31762306a36Sopenharmony_ci SOC_SINGLE_TLV("ADC B Boost Volume", 31862306a36Sopenharmony_ci CS42L73_ADCIPC, 6, 0x01, 1, adc_boost_tlv), 31962306a36Sopenharmony_ci 32062306a36Sopenharmony_ci SOC_SINGLE_SX_TLV("Speakerphone Digital Volume", 32162306a36Sopenharmony_ci CS42L73_SPKDVOL, 0, 0x34, 0xE4, hl_tlv), 32262306a36Sopenharmony_ci 32362306a36Sopenharmony_ci SOC_SINGLE_SX_TLV("Ear Speaker Digital Volume", 32462306a36Sopenharmony_ci CS42L73_ESLDVOL, 0, 0x34, 0xE4, hl_tlv), 32562306a36Sopenharmony_ci 32662306a36Sopenharmony_ci SOC_DOUBLE_R("Headphone Analog Playback Switch", CS42L73_HPAAVOL, 32762306a36Sopenharmony_ci CS42L73_HPBAVOL, 7, 1, 1), 32862306a36Sopenharmony_ci 32962306a36Sopenharmony_ci SOC_DOUBLE_R("LineOut Analog Playback Switch", CS42L73_LOAAVOL, 33062306a36Sopenharmony_ci CS42L73_LOBAVOL, 7, 1, 1), 33162306a36Sopenharmony_ci SOC_DOUBLE("Input Path Digital Switch", CS42L73_ADCIPC, 0, 4, 1, 1), 33262306a36Sopenharmony_ci SOC_DOUBLE("HL Digital Playback Switch", CS42L73_PBDC, 0, 33362306a36Sopenharmony_ci 1, 1, 1), 33462306a36Sopenharmony_ci SOC_SINGLE("Speakerphone Digital Playback Switch", CS42L73_PBDC, 2, 1, 33562306a36Sopenharmony_ci 1), 33662306a36Sopenharmony_ci SOC_SINGLE("Ear Speaker Digital Playback Switch", CS42L73_PBDC, 3, 1, 33762306a36Sopenharmony_ci 1), 33862306a36Sopenharmony_ci 33962306a36Sopenharmony_ci SOC_SINGLE("PGA Soft-Ramp Switch", CS42L73_MIOPC, 3, 1, 0), 34062306a36Sopenharmony_ci SOC_SINGLE("Analog Zero Cross Switch", CS42L73_MIOPC, 2, 1, 0), 34162306a36Sopenharmony_ci SOC_SINGLE("Digital Soft-Ramp Switch", CS42L73_MIOPC, 1, 1, 0), 34262306a36Sopenharmony_ci SOC_SINGLE("Analog Output Soft-Ramp Switch", CS42L73_MIOPC, 0, 1, 0), 34362306a36Sopenharmony_ci 34462306a36Sopenharmony_ci SOC_DOUBLE("ADC Signal Polarity Switch", CS42L73_ADCIPC, 1, 5, 1, 34562306a36Sopenharmony_ci 0), 34662306a36Sopenharmony_ci 34762306a36Sopenharmony_ci SOC_SINGLE("HL Limiter Attack Rate", CS42L73_LIMARATEHL, 0, 0x3F, 34862306a36Sopenharmony_ci 0), 34962306a36Sopenharmony_ci SOC_SINGLE("HL Limiter Release Rate", CS42L73_LIMRRATEHL, 0, 35062306a36Sopenharmony_ci 0x3F, 0), 35162306a36Sopenharmony_ci 35262306a36Sopenharmony_ci 35362306a36Sopenharmony_ci SOC_SINGLE("HL Limiter Switch", CS42L73_LIMRRATEHL, 7, 1, 0), 35462306a36Sopenharmony_ci SOC_SINGLE("HL Limiter All Channels Switch", CS42L73_LIMRRATEHL, 6, 1, 35562306a36Sopenharmony_ci 0), 35662306a36Sopenharmony_ci 35762306a36Sopenharmony_ci SOC_SINGLE_TLV("HL Limiter Max Threshold Volume", CS42L73_LMAXHL, 5, 7, 35862306a36Sopenharmony_ci 1, limiter_tlv), 35962306a36Sopenharmony_ci 36062306a36Sopenharmony_ci SOC_SINGLE_TLV("HL Limiter Cushion Volume", CS42L73_LMAXHL, 2, 7, 1, 36162306a36Sopenharmony_ci limiter_tlv), 36262306a36Sopenharmony_ci 36362306a36Sopenharmony_ci SOC_SINGLE("SPK Limiter Attack Rate Volume", CS42L73_LIMARATESPK, 0, 36462306a36Sopenharmony_ci 0x3F, 0), 36562306a36Sopenharmony_ci SOC_SINGLE("SPK Limiter Release Rate Volume", CS42L73_LIMRRATESPK, 0, 36662306a36Sopenharmony_ci 0x3F, 0), 36762306a36Sopenharmony_ci SOC_SINGLE("SPK Limiter Switch", CS42L73_LIMRRATESPK, 7, 1, 0), 36862306a36Sopenharmony_ci SOC_SINGLE("SPK Limiter All Channels Switch", CS42L73_LIMRRATESPK, 36962306a36Sopenharmony_ci 6, 1, 0), 37062306a36Sopenharmony_ci SOC_SINGLE_TLV("SPK Limiter Max Threshold Volume", CS42L73_LMAXSPK, 5, 37162306a36Sopenharmony_ci 7, 1, limiter_tlv), 37262306a36Sopenharmony_ci 37362306a36Sopenharmony_ci SOC_SINGLE_TLV("SPK Limiter Cushion Volume", CS42L73_LMAXSPK, 2, 7, 1, 37462306a36Sopenharmony_ci limiter_tlv), 37562306a36Sopenharmony_ci 37662306a36Sopenharmony_ci SOC_SINGLE("ESL Limiter Attack Rate Volume", CS42L73_LIMARATEESL, 0, 37762306a36Sopenharmony_ci 0x3F, 0), 37862306a36Sopenharmony_ci SOC_SINGLE("ESL Limiter Release Rate Volume", CS42L73_LIMRRATEESL, 0, 37962306a36Sopenharmony_ci 0x3F, 0), 38062306a36Sopenharmony_ci SOC_SINGLE("ESL Limiter Switch", CS42L73_LIMRRATEESL, 7, 1, 0), 38162306a36Sopenharmony_ci SOC_SINGLE_TLV("ESL Limiter Max Threshold Volume", CS42L73_LMAXESL, 5, 38262306a36Sopenharmony_ci 7, 1, limiter_tlv), 38362306a36Sopenharmony_ci 38462306a36Sopenharmony_ci SOC_SINGLE_TLV("ESL Limiter Cushion Volume", CS42L73_LMAXESL, 2, 7, 1, 38562306a36Sopenharmony_ci limiter_tlv), 38662306a36Sopenharmony_ci 38762306a36Sopenharmony_ci SOC_SINGLE("ALC Attack Rate Volume", CS42L73_ALCARATE, 0, 0x3F, 0), 38862306a36Sopenharmony_ci SOC_SINGLE("ALC Release Rate Volume", CS42L73_ALCRRATE, 0, 0x3F, 0), 38962306a36Sopenharmony_ci SOC_DOUBLE("ALC Switch", CS42L73_ALCARATE, 6, 7, 1, 0), 39062306a36Sopenharmony_ci SOC_SINGLE_TLV("ALC Max Threshold Volume", CS42L73_ALCMINMAX, 5, 7, 0, 39162306a36Sopenharmony_ci limiter_tlv), 39262306a36Sopenharmony_ci SOC_SINGLE_TLV("ALC Min Threshold Volume", CS42L73_ALCMINMAX, 2, 7, 0, 39362306a36Sopenharmony_ci limiter_tlv), 39462306a36Sopenharmony_ci 39562306a36Sopenharmony_ci SOC_DOUBLE("NG Enable Switch", CS42L73_NGCAB, 6, 7, 1, 0), 39662306a36Sopenharmony_ci SOC_SINGLE("NG Boost Switch", CS42L73_NGCAB, 5, 1, 0), 39762306a36Sopenharmony_ci /* 39862306a36Sopenharmony_ci NG Threshold depends on NG_BOOTSAB, which selects 39962306a36Sopenharmony_ci between two threshold scales in decibels. 40062306a36Sopenharmony_ci Set linear values for now .. 40162306a36Sopenharmony_ci */ 40262306a36Sopenharmony_ci SOC_SINGLE("NG Threshold", CS42L73_NGCAB, 2, 7, 0), 40362306a36Sopenharmony_ci SOC_ENUM("NG Delay", ng_delay_enum), 40462306a36Sopenharmony_ci 40562306a36Sopenharmony_ci SOC_DOUBLE_R_TLV("XSP-IP Volume", 40662306a36Sopenharmony_ci CS42L73_XSPAIPAA, CS42L73_XSPBIPBA, 0, 0x3F, 1, 40762306a36Sopenharmony_ci attn_tlv), 40862306a36Sopenharmony_ci SOC_DOUBLE_R_TLV("XSP-XSP Volume", 40962306a36Sopenharmony_ci CS42L73_XSPAXSPAA, CS42L73_XSPBXSPBA, 0, 0x3F, 1, 41062306a36Sopenharmony_ci attn_tlv), 41162306a36Sopenharmony_ci SOC_DOUBLE_R_TLV("XSP-ASP Volume", 41262306a36Sopenharmony_ci CS42L73_XSPAASPAA, CS42L73_XSPAASPBA, 0, 0x3F, 1, 41362306a36Sopenharmony_ci attn_tlv), 41462306a36Sopenharmony_ci SOC_DOUBLE_R_TLV("XSP-VSP Volume", 41562306a36Sopenharmony_ci CS42L73_XSPAVSPMA, CS42L73_XSPBVSPMA, 0, 0x3F, 1, 41662306a36Sopenharmony_ci attn_tlv), 41762306a36Sopenharmony_ci 41862306a36Sopenharmony_ci SOC_DOUBLE_R_TLV("ASP-IP Volume", 41962306a36Sopenharmony_ci CS42L73_ASPAIPAA, CS42L73_ASPBIPBA, 0, 0x3F, 1, 42062306a36Sopenharmony_ci attn_tlv), 42162306a36Sopenharmony_ci SOC_DOUBLE_R_TLV("ASP-XSP Volume", 42262306a36Sopenharmony_ci CS42L73_ASPAXSPAA, CS42L73_ASPBXSPBA, 0, 0x3F, 1, 42362306a36Sopenharmony_ci attn_tlv), 42462306a36Sopenharmony_ci SOC_DOUBLE_R_TLV("ASP-ASP Volume", 42562306a36Sopenharmony_ci CS42L73_ASPAASPAA, CS42L73_ASPBASPBA, 0, 0x3F, 1, 42662306a36Sopenharmony_ci attn_tlv), 42762306a36Sopenharmony_ci SOC_DOUBLE_R_TLV("ASP-VSP Volume", 42862306a36Sopenharmony_ci CS42L73_ASPAVSPMA, CS42L73_ASPBVSPMA, 0, 0x3F, 1, 42962306a36Sopenharmony_ci attn_tlv), 43062306a36Sopenharmony_ci 43162306a36Sopenharmony_ci SOC_DOUBLE_R_TLV("VSP-IP Volume", 43262306a36Sopenharmony_ci CS42L73_VSPAIPAA, CS42L73_VSPBIPBA, 0, 0x3F, 1, 43362306a36Sopenharmony_ci attn_tlv), 43462306a36Sopenharmony_ci SOC_DOUBLE_R_TLV("VSP-XSP Volume", 43562306a36Sopenharmony_ci CS42L73_VSPAXSPAA, CS42L73_VSPBXSPBA, 0, 0x3F, 1, 43662306a36Sopenharmony_ci attn_tlv), 43762306a36Sopenharmony_ci SOC_DOUBLE_R_TLV("VSP-ASP Volume", 43862306a36Sopenharmony_ci CS42L73_VSPAASPAA, CS42L73_VSPBASPBA, 0, 0x3F, 1, 43962306a36Sopenharmony_ci attn_tlv), 44062306a36Sopenharmony_ci SOC_DOUBLE_R_TLV("VSP-VSP Volume", 44162306a36Sopenharmony_ci CS42L73_VSPAVSPMA, CS42L73_VSPBVSPMA, 0, 0x3F, 1, 44262306a36Sopenharmony_ci attn_tlv), 44362306a36Sopenharmony_ci 44462306a36Sopenharmony_ci SOC_DOUBLE_R_TLV("HL-IP Volume", 44562306a36Sopenharmony_ci CS42L73_HLAIPAA, CS42L73_HLBIPBA, 0, 0x3F, 1, 44662306a36Sopenharmony_ci attn_tlv), 44762306a36Sopenharmony_ci SOC_DOUBLE_R_TLV("HL-XSP Volume", 44862306a36Sopenharmony_ci CS42L73_HLAXSPAA, CS42L73_HLBXSPBA, 0, 0x3F, 1, 44962306a36Sopenharmony_ci attn_tlv), 45062306a36Sopenharmony_ci SOC_DOUBLE_R_TLV("HL-ASP Volume", 45162306a36Sopenharmony_ci CS42L73_HLAASPAA, CS42L73_HLBASPBA, 0, 0x3F, 1, 45262306a36Sopenharmony_ci attn_tlv), 45362306a36Sopenharmony_ci SOC_DOUBLE_R_TLV("HL-VSP Volume", 45462306a36Sopenharmony_ci CS42L73_HLAVSPMA, CS42L73_HLBVSPMA, 0, 0x3F, 1, 45562306a36Sopenharmony_ci attn_tlv), 45662306a36Sopenharmony_ci 45762306a36Sopenharmony_ci SOC_SINGLE_TLV("SPK-IP Mono Volume", 45862306a36Sopenharmony_ci CS42L73_SPKMIPMA, 0, 0x3F, 1, attn_tlv), 45962306a36Sopenharmony_ci SOC_SINGLE_TLV("SPK-XSP Mono Volume", 46062306a36Sopenharmony_ci CS42L73_SPKMXSPA, 0, 0x3F, 1, attn_tlv), 46162306a36Sopenharmony_ci SOC_SINGLE_TLV("SPK-ASP Mono Volume", 46262306a36Sopenharmony_ci CS42L73_SPKMASPA, 0, 0x3F, 1, attn_tlv), 46362306a36Sopenharmony_ci SOC_SINGLE_TLV("SPK-VSP Mono Volume", 46462306a36Sopenharmony_ci CS42L73_SPKMVSPMA, 0, 0x3F, 1, attn_tlv), 46562306a36Sopenharmony_ci 46662306a36Sopenharmony_ci SOC_SINGLE_TLV("ESL-IP Mono Volume", 46762306a36Sopenharmony_ci CS42L73_ESLMIPMA, 0, 0x3F, 1, attn_tlv), 46862306a36Sopenharmony_ci SOC_SINGLE_TLV("ESL-XSP Mono Volume", 46962306a36Sopenharmony_ci CS42L73_ESLMXSPA, 0, 0x3F, 1, attn_tlv), 47062306a36Sopenharmony_ci SOC_SINGLE_TLV("ESL-ASP Mono Volume", 47162306a36Sopenharmony_ci CS42L73_ESLMASPA, 0, 0x3F, 1, attn_tlv), 47262306a36Sopenharmony_ci SOC_SINGLE_TLV("ESL-VSP Mono Volume", 47362306a36Sopenharmony_ci CS42L73_ESLMVSPMA, 0, 0x3F, 1, attn_tlv), 47462306a36Sopenharmony_ci 47562306a36Sopenharmony_ci SOC_ENUM("IP Digital Swap/Mono Select", ip_swap_enum), 47662306a36Sopenharmony_ci 47762306a36Sopenharmony_ci SOC_ENUM("VSPOUT Mono/Stereo Select", vsp_output_mux_enum), 47862306a36Sopenharmony_ci SOC_ENUM("XSPOUT Mono/Stereo Select", xsp_output_mux_enum), 47962306a36Sopenharmony_ci}; 48062306a36Sopenharmony_ci 48162306a36Sopenharmony_cistatic int cs42l73_spklo_spk_amp_event(struct snd_soc_dapm_widget *w, 48262306a36Sopenharmony_ci struct snd_kcontrol *kcontrol, int event) 48362306a36Sopenharmony_ci{ 48462306a36Sopenharmony_ci struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm); 48562306a36Sopenharmony_ci struct cs42l73_private *priv = snd_soc_component_get_drvdata(component); 48662306a36Sopenharmony_ci switch (event) { 48762306a36Sopenharmony_ci case SND_SOC_DAPM_POST_PMD: 48862306a36Sopenharmony_ci /* 150 ms delay between setting PDN and MCLKDIS */ 48962306a36Sopenharmony_ci priv->shutdwn_delay = 150; 49062306a36Sopenharmony_ci break; 49162306a36Sopenharmony_ci default: 49262306a36Sopenharmony_ci pr_err("Invalid event = 0x%x\n", event); 49362306a36Sopenharmony_ci } 49462306a36Sopenharmony_ci return 0; 49562306a36Sopenharmony_ci} 49662306a36Sopenharmony_ci 49762306a36Sopenharmony_cistatic int cs42l73_ear_amp_event(struct snd_soc_dapm_widget *w, 49862306a36Sopenharmony_ci struct snd_kcontrol *kcontrol, int event) 49962306a36Sopenharmony_ci{ 50062306a36Sopenharmony_ci struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm); 50162306a36Sopenharmony_ci struct cs42l73_private *priv = snd_soc_component_get_drvdata(component); 50262306a36Sopenharmony_ci switch (event) { 50362306a36Sopenharmony_ci case SND_SOC_DAPM_POST_PMD: 50462306a36Sopenharmony_ci /* 50 ms delay between setting PDN and MCLKDIS */ 50562306a36Sopenharmony_ci if (priv->shutdwn_delay < 50) 50662306a36Sopenharmony_ci priv->shutdwn_delay = 50; 50762306a36Sopenharmony_ci break; 50862306a36Sopenharmony_ci default: 50962306a36Sopenharmony_ci pr_err("Invalid event = 0x%x\n", event); 51062306a36Sopenharmony_ci } 51162306a36Sopenharmony_ci return 0; 51262306a36Sopenharmony_ci} 51362306a36Sopenharmony_ci 51462306a36Sopenharmony_ci 51562306a36Sopenharmony_cistatic int cs42l73_hp_amp_event(struct snd_soc_dapm_widget *w, 51662306a36Sopenharmony_ci struct snd_kcontrol *kcontrol, int event) 51762306a36Sopenharmony_ci{ 51862306a36Sopenharmony_ci struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm); 51962306a36Sopenharmony_ci struct cs42l73_private *priv = snd_soc_component_get_drvdata(component); 52062306a36Sopenharmony_ci switch (event) { 52162306a36Sopenharmony_ci case SND_SOC_DAPM_POST_PMD: 52262306a36Sopenharmony_ci /* 30 ms delay between setting PDN and MCLKDIS */ 52362306a36Sopenharmony_ci if (priv->shutdwn_delay < 30) 52462306a36Sopenharmony_ci priv->shutdwn_delay = 30; 52562306a36Sopenharmony_ci break; 52662306a36Sopenharmony_ci default: 52762306a36Sopenharmony_ci pr_err("Invalid event = 0x%x\n", event); 52862306a36Sopenharmony_ci } 52962306a36Sopenharmony_ci return 0; 53062306a36Sopenharmony_ci} 53162306a36Sopenharmony_ci 53262306a36Sopenharmony_cistatic const struct snd_soc_dapm_widget cs42l73_dapm_widgets[] = { 53362306a36Sopenharmony_ci SND_SOC_DAPM_INPUT("DMICA"), 53462306a36Sopenharmony_ci SND_SOC_DAPM_INPUT("DMICB"), 53562306a36Sopenharmony_ci SND_SOC_DAPM_INPUT("LINEINA"), 53662306a36Sopenharmony_ci SND_SOC_DAPM_INPUT("LINEINB"), 53762306a36Sopenharmony_ci SND_SOC_DAPM_INPUT("MIC1"), 53862306a36Sopenharmony_ci SND_SOC_DAPM_SUPPLY("MIC1 Bias", CS42L73_PWRCTL2, 6, 1, NULL, 0), 53962306a36Sopenharmony_ci SND_SOC_DAPM_INPUT("MIC2"), 54062306a36Sopenharmony_ci SND_SOC_DAPM_SUPPLY("MIC2 Bias", CS42L73_PWRCTL2, 7, 1, NULL, 0), 54162306a36Sopenharmony_ci 54262306a36Sopenharmony_ci SND_SOC_DAPM_AIF_OUT("XSPOUTL", NULL, 0, 54362306a36Sopenharmony_ci CS42L73_PWRCTL2, 1, 1), 54462306a36Sopenharmony_ci SND_SOC_DAPM_AIF_OUT("XSPOUTR", NULL, 0, 54562306a36Sopenharmony_ci CS42L73_PWRCTL2, 1, 1), 54662306a36Sopenharmony_ci SND_SOC_DAPM_AIF_OUT("ASPOUTL", NULL, 0, 54762306a36Sopenharmony_ci CS42L73_PWRCTL2, 3, 1), 54862306a36Sopenharmony_ci SND_SOC_DAPM_AIF_OUT("ASPOUTR", NULL, 0, 54962306a36Sopenharmony_ci CS42L73_PWRCTL2, 3, 1), 55062306a36Sopenharmony_ci SND_SOC_DAPM_AIF_OUT("VSPINOUT", NULL, 0, 55162306a36Sopenharmony_ci CS42L73_PWRCTL2, 4, 1), 55262306a36Sopenharmony_ci 55362306a36Sopenharmony_ci SND_SOC_DAPM_PGA("PGA Left", SND_SOC_NOPM, 0, 0, NULL, 0), 55462306a36Sopenharmony_ci SND_SOC_DAPM_PGA("PGA Right", SND_SOC_NOPM, 0, 0, NULL, 0), 55562306a36Sopenharmony_ci 55662306a36Sopenharmony_ci SND_SOC_DAPM_MUX("PGA Left Mux", SND_SOC_NOPM, 0, 0, &pgaa_mux), 55762306a36Sopenharmony_ci SND_SOC_DAPM_MUX("PGA Right Mux", SND_SOC_NOPM, 0, 0, &pgab_mux), 55862306a36Sopenharmony_ci 55962306a36Sopenharmony_ci SND_SOC_DAPM_ADC("ADC Left", NULL, CS42L73_PWRCTL1, 7, 1), 56062306a36Sopenharmony_ci SND_SOC_DAPM_ADC("ADC Right", NULL, CS42L73_PWRCTL1, 5, 1), 56162306a36Sopenharmony_ci SND_SOC_DAPM_ADC("DMIC Left", NULL, CS42L73_PWRCTL1, 6, 1), 56262306a36Sopenharmony_ci SND_SOC_DAPM_ADC("DMIC Right", NULL, CS42L73_PWRCTL1, 4, 1), 56362306a36Sopenharmony_ci 56462306a36Sopenharmony_ci SND_SOC_DAPM_MIXER_NAMED_CTL("Input Left Capture", SND_SOC_NOPM, 56562306a36Sopenharmony_ci 0, 0, input_left_mixer, 56662306a36Sopenharmony_ci ARRAY_SIZE(input_left_mixer)), 56762306a36Sopenharmony_ci 56862306a36Sopenharmony_ci SND_SOC_DAPM_MIXER_NAMED_CTL("Input Right Capture", SND_SOC_NOPM, 56962306a36Sopenharmony_ci 0, 0, input_right_mixer, 57062306a36Sopenharmony_ci ARRAY_SIZE(input_right_mixer)), 57162306a36Sopenharmony_ci 57262306a36Sopenharmony_ci SND_SOC_DAPM_MIXER("ASPL Output Mixer", SND_SOC_NOPM, 0, 0, NULL, 0), 57362306a36Sopenharmony_ci SND_SOC_DAPM_MIXER("ASPR Output Mixer", SND_SOC_NOPM, 0, 0, NULL, 0), 57462306a36Sopenharmony_ci SND_SOC_DAPM_MIXER("XSPL Output Mixer", SND_SOC_NOPM, 0, 0, NULL, 0), 57562306a36Sopenharmony_ci SND_SOC_DAPM_MIXER("XSPR Output Mixer", SND_SOC_NOPM, 0, 0, NULL, 0), 57662306a36Sopenharmony_ci SND_SOC_DAPM_MIXER("VSP Output Mixer", SND_SOC_NOPM, 0, 0, NULL, 0), 57762306a36Sopenharmony_ci 57862306a36Sopenharmony_ci SND_SOC_DAPM_AIF_IN("XSPINL", NULL, 0, 57962306a36Sopenharmony_ci CS42L73_PWRCTL2, 0, 1), 58062306a36Sopenharmony_ci SND_SOC_DAPM_AIF_IN("XSPINR", NULL, 0, 58162306a36Sopenharmony_ci CS42L73_PWRCTL2, 0, 1), 58262306a36Sopenharmony_ci SND_SOC_DAPM_AIF_IN("XSPINM", NULL, 0, 58362306a36Sopenharmony_ci CS42L73_PWRCTL2, 0, 1), 58462306a36Sopenharmony_ci 58562306a36Sopenharmony_ci SND_SOC_DAPM_AIF_IN("ASPINL", NULL, 0, 58662306a36Sopenharmony_ci CS42L73_PWRCTL2, 2, 1), 58762306a36Sopenharmony_ci SND_SOC_DAPM_AIF_IN("ASPINR", NULL, 0, 58862306a36Sopenharmony_ci CS42L73_PWRCTL2, 2, 1), 58962306a36Sopenharmony_ci SND_SOC_DAPM_AIF_IN("ASPINM", NULL, 0, 59062306a36Sopenharmony_ci CS42L73_PWRCTL2, 2, 1), 59162306a36Sopenharmony_ci 59262306a36Sopenharmony_ci SND_SOC_DAPM_AIF_IN("VSPINOUT", NULL, 0, 59362306a36Sopenharmony_ci CS42L73_PWRCTL2, 4, 1), 59462306a36Sopenharmony_ci 59562306a36Sopenharmony_ci SND_SOC_DAPM_MIXER("HL Left Mixer", SND_SOC_NOPM, 0, 0, NULL, 0), 59662306a36Sopenharmony_ci SND_SOC_DAPM_MIXER("HL Right Mixer", SND_SOC_NOPM, 0, 0, NULL, 0), 59762306a36Sopenharmony_ci SND_SOC_DAPM_MIXER("SPK Mixer", SND_SOC_NOPM, 0, 0, NULL, 0), 59862306a36Sopenharmony_ci SND_SOC_DAPM_MIXER("ESL Mixer", SND_SOC_NOPM, 0, 0, NULL, 0), 59962306a36Sopenharmony_ci 60062306a36Sopenharmony_ci SND_SOC_DAPM_MUX("ESL-XSP Mux", SND_SOC_NOPM, 60162306a36Sopenharmony_ci 0, 0, &esl_xsp_mixer), 60262306a36Sopenharmony_ci 60362306a36Sopenharmony_ci SND_SOC_DAPM_MUX("ESL-ASP Mux", SND_SOC_NOPM, 60462306a36Sopenharmony_ci 0, 0, &esl_asp_mixer), 60562306a36Sopenharmony_ci 60662306a36Sopenharmony_ci SND_SOC_DAPM_MUX("SPK-ASP Mux", SND_SOC_NOPM, 60762306a36Sopenharmony_ci 0, 0, &spk_asp_mixer), 60862306a36Sopenharmony_ci 60962306a36Sopenharmony_ci SND_SOC_DAPM_MUX("SPK-XSP Mux", SND_SOC_NOPM, 61062306a36Sopenharmony_ci 0, 0, &spk_xsp_mixer), 61162306a36Sopenharmony_ci 61262306a36Sopenharmony_ci SND_SOC_DAPM_PGA("HL Left DAC", SND_SOC_NOPM, 0, 0, NULL, 0), 61362306a36Sopenharmony_ci SND_SOC_DAPM_PGA("HL Right DAC", SND_SOC_NOPM, 0, 0, NULL, 0), 61462306a36Sopenharmony_ci SND_SOC_DAPM_PGA("SPK DAC", SND_SOC_NOPM, 0, 0, NULL, 0), 61562306a36Sopenharmony_ci SND_SOC_DAPM_PGA("ESL DAC", SND_SOC_NOPM, 0, 0, NULL, 0), 61662306a36Sopenharmony_ci 61762306a36Sopenharmony_ci SND_SOC_DAPM_SWITCH_E("HP Amp", CS42L73_PWRCTL3, 0, 1, 61862306a36Sopenharmony_ci &hp_amp_ctl, cs42l73_hp_amp_event, 61962306a36Sopenharmony_ci SND_SOC_DAPM_POST_PMD), 62062306a36Sopenharmony_ci SND_SOC_DAPM_SWITCH("LO Amp", CS42L73_PWRCTL3, 1, 1, 62162306a36Sopenharmony_ci &lo_amp_ctl), 62262306a36Sopenharmony_ci SND_SOC_DAPM_SWITCH_E("SPK Amp", CS42L73_PWRCTL3, 2, 1, 62362306a36Sopenharmony_ci &spk_amp_ctl, cs42l73_spklo_spk_amp_event, 62462306a36Sopenharmony_ci SND_SOC_DAPM_POST_PMD), 62562306a36Sopenharmony_ci SND_SOC_DAPM_SWITCH_E("EAR Amp", CS42L73_PWRCTL3, 3, 1, 62662306a36Sopenharmony_ci &ear_amp_ctl, cs42l73_ear_amp_event, 62762306a36Sopenharmony_ci SND_SOC_DAPM_POST_PMD), 62862306a36Sopenharmony_ci SND_SOC_DAPM_SWITCH_E("SPKLO Amp", CS42L73_PWRCTL3, 4, 1, 62962306a36Sopenharmony_ci &spklo_amp_ctl, cs42l73_spklo_spk_amp_event, 63062306a36Sopenharmony_ci SND_SOC_DAPM_POST_PMD), 63162306a36Sopenharmony_ci 63262306a36Sopenharmony_ci SND_SOC_DAPM_OUTPUT("HPOUTA"), 63362306a36Sopenharmony_ci SND_SOC_DAPM_OUTPUT("HPOUTB"), 63462306a36Sopenharmony_ci SND_SOC_DAPM_OUTPUT("LINEOUTA"), 63562306a36Sopenharmony_ci SND_SOC_DAPM_OUTPUT("LINEOUTB"), 63662306a36Sopenharmony_ci SND_SOC_DAPM_OUTPUT("EAROUT"), 63762306a36Sopenharmony_ci SND_SOC_DAPM_OUTPUT("SPKOUT"), 63862306a36Sopenharmony_ci SND_SOC_DAPM_OUTPUT("SPKLINEOUT"), 63962306a36Sopenharmony_ci}; 64062306a36Sopenharmony_ci 64162306a36Sopenharmony_cistatic const struct snd_soc_dapm_route cs42l73_audio_map[] = { 64262306a36Sopenharmony_ci 64362306a36Sopenharmony_ci /* SPKLO EARSPK Paths */ 64462306a36Sopenharmony_ci {"EAROUT", NULL, "EAR Amp"}, 64562306a36Sopenharmony_ci {"SPKLINEOUT", NULL, "SPKLO Amp"}, 64662306a36Sopenharmony_ci 64762306a36Sopenharmony_ci {"EAR Amp", "Switch", "ESL DAC"}, 64862306a36Sopenharmony_ci {"SPKLO Amp", "Switch", "ESL DAC"}, 64962306a36Sopenharmony_ci 65062306a36Sopenharmony_ci {"ESL DAC", "ESL-ASP Mono Volume", "ESL Mixer"}, 65162306a36Sopenharmony_ci {"ESL DAC", "ESL-XSP Mono Volume", "ESL Mixer"}, 65262306a36Sopenharmony_ci {"ESL DAC", "ESL-VSP Mono Volume", "VSPINOUT"}, 65362306a36Sopenharmony_ci /* Loopback */ 65462306a36Sopenharmony_ci {"ESL DAC", "ESL-IP Mono Volume", "Input Left Capture"}, 65562306a36Sopenharmony_ci {"ESL DAC", "ESL-IP Mono Volume", "Input Right Capture"}, 65662306a36Sopenharmony_ci 65762306a36Sopenharmony_ci {"ESL Mixer", NULL, "ESL-ASP Mux"}, 65862306a36Sopenharmony_ci {"ESL Mixer", NULL, "ESL-XSP Mux"}, 65962306a36Sopenharmony_ci 66062306a36Sopenharmony_ci {"ESL-ASP Mux", "Left", "ASPINL"}, 66162306a36Sopenharmony_ci {"ESL-ASP Mux", "Right", "ASPINR"}, 66262306a36Sopenharmony_ci {"ESL-ASP Mux", "Mono Mix", "ASPINM"}, 66362306a36Sopenharmony_ci 66462306a36Sopenharmony_ci {"ESL-XSP Mux", "Left", "XSPINL"}, 66562306a36Sopenharmony_ci {"ESL-XSP Mux", "Right", "XSPINR"}, 66662306a36Sopenharmony_ci {"ESL-XSP Mux", "Mono Mix", "XSPINM"}, 66762306a36Sopenharmony_ci 66862306a36Sopenharmony_ci /* Speakerphone Paths */ 66962306a36Sopenharmony_ci {"SPKOUT", NULL, "SPK Amp"}, 67062306a36Sopenharmony_ci {"SPK Amp", "Switch", "SPK DAC"}, 67162306a36Sopenharmony_ci 67262306a36Sopenharmony_ci {"SPK DAC", "SPK-ASP Mono Volume", "SPK Mixer"}, 67362306a36Sopenharmony_ci {"SPK DAC", "SPK-XSP Mono Volume", "SPK Mixer"}, 67462306a36Sopenharmony_ci {"SPK DAC", "SPK-VSP Mono Volume", "VSPINOUT"}, 67562306a36Sopenharmony_ci /* Loopback */ 67662306a36Sopenharmony_ci {"SPK DAC", "SPK-IP Mono Volume", "Input Left Capture"}, 67762306a36Sopenharmony_ci {"SPK DAC", "SPK-IP Mono Volume", "Input Right Capture"}, 67862306a36Sopenharmony_ci 67962306a36Sopenharmony_ci {"SPK Mixer", NULL, "SPK-ASP Mux"}, 68062306a36Sopenharmony_ci {"SPK Mixer", NULL, "SPK-XSP Mux"}, 68162306a36Sopenharmony_ci 68262306a36Sopenharmony_ci {"SPK-ASP Mux", "Left", "ASPINL"}, 68362306a36Sopenharmony_ci {"SPK-ASP Mux", "Mono Mix", "ASPINM"}, 68462306a36Sopenharmony_ci {"SPK-ASP Mux", "Right", "ASPINR"}, 68562306a36Sopenharmony_ci 68662306a36Sopenharmony_ci {"SPK-XSP Mux", "Left", "XSPINL"}, 68762306a36Sopenharmony_ci {"SPK-XSP Mux", "Mono Mix", "XSPINM"}, 68862306a36Sopenharmony_ci {"SPK-XSP Mux", "Right", "XSPINR"}, 68962306a36Sopenharmony_ci 69062306a36Sopenharmony_ci /* HP LineOUT Paths */ 69162306a36Sopenharmony_ci {"HPOUTA", NULL, "HP Amp"}, 69262306a36Sopenharmony_ci {"HPOUTB", NULL, "HP Amp"}, 69362306a36Sopenharmony_ci {"LINEOUTA", NULL, "LO Amp"}, 69462306a36Sopenharmony_ci {"LINEOUTB", NULL, "LO Amp"}, 69562306a36Sopenharmony_ci 69662306a36Sopenharmony_ci {"HP Amp", "Switch", "HL Left DAC"}, 69762306a36Sopenharmony_ci {"HP Amp", "Switch", "HL Right DAC"}, 69862306a36Sopenharmony_ci {"LO Amp", "Switch", "HL Left DAC"}, 69962306a36Sopenharmony_ci {"LO Amp", "Switch", "HL Right DAC"}, 70062306a36Sopenharmony_ci 70162306a36Sopenharmony_ci {"HL Left DAC", "HL-XSP Volume", "HL Left Mixer"}, 70262306a36Sopenharmony_ci {"HL Right DAC", "HL-XSP Volume", "HL Right Mixer"}, 70362306a36Sopenharmony_ci {"HL Left DAC", "HL-ASP Volume", "HL Left Mixer"}, 70462306a36Sopenharmony_ci {"HL Right DAC", "HL-ASP Volume", "HL Right Mixer"}, 70562306a36Sopenharmony_ci {"HL Left DAC", "HL-VSP Volume", "HL Left Mixer"}, 70662306a36Sopenharmony_ci {"HL Right DAC", "HL-VSP Volume", "HL Right Mixer"}, 70762306a36Sopenharmony_ci /* Loopback */ 70862306a36Sopenharmony_ci {"HL Left DAC", "HL-IP Volume", "HL Left Mixer"}, 70962306a36Sopenharmony_ci {"HL Right DAC", "HL-IP Volume", "HL Right Mixer"}, 71062306a36Sopenharmony_ci {"HL Left Mixer", NULL, "Input Left Capture"}, 71162306a36Sopenharmony_ci {"HL Right Mixer", NULL, "Input Right Capture"}, 71262306a36Sopenharmony_ci 71362306a36Sopenharmony_ci {"HL Left Mixer", NULL, "ASPINL"}, 71462306a36Sopenharmony_ci {"HL Right Mixer", NULL, "ASPINR"}, 71562306a36Sopenharmony_ci {"HL Left Mixer", NULL, "XSPINL"}, 71662306a36Sopenharmony_ci {"HL Right Mixer", NULL, "XSPINR"}, 71762306a36Sopenharmony_ci {"HL Left Mixer", NULL, "VSPINOUT"}, 71862306a36Sopenharmony_ci {"HL Right Mixer", NULL, "VSPINOUT"}, 71962306a36Sopenharmony_ci 72062306a36Sopenharmony_ci {"ASPINL", NULL, "ASP Playback"}, 72162306a36Sopenharmony_ci {"ASPINM", NULL, "ASP Playback"}, 72262306a36Sopenharmony_ci {"ASPINR", NULL, "ASP Playback"}, 72362306a36Sopenharmony_ci {"XSPINL", NULL, "XSP Playback"}, 72462306a36Sopenharmony_ci {"XSPINM", NULL, "XSP Playback"}, 72562306a36Sopenharmony_ci {"XSPINR", NULL, "XSP Playback"}, 72662306a36Sopenharmony_ci {"VSPINOUT", NULL, "VSP Playback"}, 72762306a36Sopenharmony_ci 72862306a36Sopenharmony_ci /* Capture Paths */ 72962306a36Sopenharmony_ci {"MIC1", NULL, "MIC1 Bias"}, 73062306a36Sopenharmony_ci {"PGA Left Mux", "Mic 1", "MIC1"}, 73162306a36Sopenharmony_ci {"MIC2", NULL, "MIC2 Bias"}, 73262306a36Sopenharmony_ci {"PGA Right Mux", "Mic 2", "MIC2"}, 73362306a36Sopenharmony_ci 73462306a36Sopenharmony_ci {"PGA Left Mux", "Line A", "LINEINA"}, 73562306a36Sopenharmony_ci {"PGA Right Mux", "Line B", "LINEINB"}, 73662306a36Sopenharmony_ci 73762306a36Sopenharmony_ci {"PGA Left", NULL, "PGA Left Mux"}, 73862306a36Sopenharmony_ci {"PGA Right", NULL, "PGA Right Mux"}, 73962306a36Sopenharmony_ci 74062306a36Sopenharmony_ci {"ADC Left", NULL, "PGA Left"}, 74162306a36Sopenharmony_ci {"ADC Right", NULL, "PGA Right"}, 74262306a36Sopenharmony_ci {"DMIC Left", NULL, "DMICA"}, 74362306a36Sopenharmony_ci {"DMIC Right", NULL, "DMICB"}, 74462306a36Sopenharmony_ci 74562306a36Sopenharmony_ci {"Input Left Capture", "ADC Left Input", "ADC Left"}, 74662306a36Sopenharmony_ci {"Input Right Capture", "ADC Right Input", "ADC Right"}, 74762306a36Sopenharmony_ci {"Input Left Capture", "DMIC Left Input", "DMIC Left"}, 74862306a36Sopenharmony_ci {"Input Right Capture", "DMIC Right Input", "DMIC Right"}, 74962306a36Sopenharmony_ci 75062306a36Sopenharmony_ci /* Audio Capture */ 75162306a36Sopenharmony_ci {"ASPL Output Mixer", NULL, "Input Left Capture"}, 75262306a36Sopenharmony_ci {"ASPR Output Mixer", NULL, "Input Right Capture"}, 75362306a36Sopenharmony_ci 75462306a36Sopenharmony_ci {"ASPOUTL", "ASP-IP Volume", "ASPL Output Mixer"}, 75562306a36Sopenharmony_ci {"ASPOUTR", "ASP-IP Volume", "ASPR Output Mixer"}, 75662306a36Sopenharmony_ci 75762306a36Sopenharmony_ci /* Auxillary Capture */ 75862306a36Sopenharmony_ci {"XSPL Output Mixer", NULL, "Input Left Capture"}, 75962306a36Sopenharmony_ci {"XSPR Output Mixer", NULL, "Input Right Capture"}, 76062306a36Sopenharmony_ci 76162306a36Sopenharmony_ci {"XSPOUTL", "XSP-IP Volume", "XSPL Output Mixer"}, 76262306a36Sopenharmony_ci {"XSPOUTR", "XSP-IP Volume", "XSPR Output Mixer"}, 76362306a36Sopenharmony_ci 76462306a36Sopenharmony_ci {"XSPOUTL", NULL, "XSPL Output Mixer"}, 76562306a36Sopenharmony_ci {"XSPOUTR", NULL, "XSPR Output Mixer"}, 76662306a36Sopenharmony_ci 76762306a36Sopenharmony_ci /* Voice Capture */ 76862306a36Sopenharmony_ci {"VSP Output Mixer", NULL, "Input Left Capture"}, 76962306a36Sopenharmony_ci {"VSP Output Mixer", NULL, "Input Right Capture"}, 77062306a36Sopenharmony_ci 77162306a36Sopenharmony_ci {"VSPINOUT", "VSP-IP Volume", "VSP Output Mixer"}, 77262306a36Sopenharmony_ci 77362306a36Sopenharmony_ci {"VSPINOUT", NULL, "VSP Output Mixer"}, 77462306a36Sopenharmony_ci 77562306a36Sopenharmony_ci {"ASP Capture", NULL, "ASPOUTL"}, 77662306a36Sopenharmony_ci {"ASP Capture", NULL, "ASPOUTR"}, 77762306a36Sopenharmony_ci {"XSP Capture", NULL, "XSPOUTL"}, 77862306a36Sopenharmony_ci {"XSP Capture", NULL, "XSPOUTR"}, 77962306a36Sopenharmony_ci {"VSP Capture", NULL, "VSPINOUT"}, 78062306a36Sopenharmony_ci}; 78162306a36Sopenharmony_ci 78262306a36Sopenharmony_cistruct cs42l73_mclk_div { 78362306a36Sopenharmony_ci u32 mclk; 78462306a36Sopenharmony_ci u32 srate; 78562306a36Sopenharmony_ci u8 mmcc; 78662306a36Sopenharmony_ci}; 78762306a36Sopenharmony_ci 78862306a36Sopenharmony_cistatic const struct cs42l73_mclk_div cs42l73_mclk_coeffs[] = { 78962306a36Sopenharmony_ci /* MCLK, Sample Rate, xMMCC[5:0] */ 79062306a36Sopenharmony_ci {5644800, 11025, 0x30}, 79162306a36Sopenharmony_ci {5644800, 22050, 0x20}, 79262306a36Sopenharmony_ci {5644800, 44100, 0x10}, 79362306a36Sopenharmony_ci 79462306a36Sopenharmony_ci {6000000, 8000, 0x39}, 79562306a36Sopenharmony_ci {6000000, 11025, 0x33}, 79662306a36Sopenharmony_ci {6000000, 12000, 0x31}, 79762306a36Sopenharmony_ci {6000000, 16000, 0x29}, 79862306a36Sopenharmony_ci {6000000, 22050, 0x23}, 79962306a36Sopenharmony_ci {6000000, 24000, 0x21}, 80062306a36Sopenharmony_ci {6000000, 32000, 0x19}, 80162306a36Sopenharmony_ci {6000000, 44100, 0x13}, 80262306a36Sopenharmony_ci {6000000, 48000, 0x11}, 80362306a36Sopenharmony_ci 80462306a36Sopenharmony_ci {6144000, 8000, 0x38}, 80562306a36Sopenharmony_ci {6144000, 12000, 0x30}, 80662306a36Sopenharmony_ci {6144000, 16000, 0x28}, 80762306a36Sopenharmony_ci {6144000, 24000, 0x20}, 80862306a36Sopenharmony_ci {6144000, 32000, 0x18}, 80962306a36Sopenharmony_ci {6144000, 48000, 0x10}, 81062306a36Sopenharmony_ci 81162306a36Sopenharmony_ci {6500000, 8000, 0x3C}, 81262306a36Sopenharmony_ci {6500000, 11025, 0x35}, 81362306a36Sopenharmony_ci {6500000, 12000, 0x34}, 81462306a36Sopenharmony_ci {6500000, 16000, 0x2C}, 81562306a36Sopenharmony_ci {6500000, 22050, 0x25}, 81662306a36Sopenharmony_ci {6500000, 24000, 0x24}, 81762306a36Sopenharmony_ci {6500000, 32000, 0x1C}, 81862306a36Sopenharmony_ci {6500000, 44100, 0x15}, 81962306a36Sopenharmony_ci {6500000, 48000, 0x14}, 82062306a36Sopenharmony_ci 82162306a36Sopenharmony_ci {6400000, 8000, 0x3E}, 82262306a36Sopenharmony_ci {6400000, 11025, 0x37}, 82362306a36Sopenharmony_ci {6400000, 12000, 0x36}, 82462306a36Sopenharmony_ci {6400000, 16000, 0x2E}, 82562306a36Sopenharmony_ci {6400000, 22050, 0x27}, 82662306a36Sopenharmony_ci {6400000, 24000, 0x26}, 82762306a36Sopenharmony_ci {6400000, 32000, 0x1E}, 82862306a36Sopenharmony_ci {6400000, 44100, 0x17}, 82962306a36Sopenharmony_ci {6400000, 48000, 0x16}, 83062306a36Sopenharmony_ci}; 83162306a36Sopenharmony_ci 83262306a36Sopenharmony_cistruct cs42l73_mclkx_div { 83362306a36Sopenharmony_ci u32 mclkx; 83462306a36Sopenharmony_ci u8 ratio; 83562306a36Sopenharmony_ci u8 mclkdiv; 83662306a36Sopenharmony_ci}; 83762306a36Sopenharmony_ci 83862306a36Sopenharmony_cistatic const struct cs42l73_mclkx_div cs42l73_mclkx_coeffs[] = { 83962306a36Sopenharmony_ci {5644800, 1, 0}, /* 5644800 */ 84062306a36Sopenharmony_ci {6000000, 1, 0}, /* 6000000 */ 84162306a36Sopenharmony_ci {6144000, 1, 0}, /* 6144000 */ 84262306a36Sopenharmony_ci {11289600, 2, 2}, /* 5644800 */ 84362306a36Sopenharmony_ci {12288000, 2, 2}, /* 6144000 */ 84462306a36Sopenharmony_ci {12000000, 2, 2}, /* 6000000 */ 84562306a36Sopenharmony_ci {13000000, 2, 2}, /* 6500000 */ 84662306a36Sopenharmony_ci {19200000, 3, 3}, /* 6400000 */ 84762306a36Sopenharmony_ci {24000000, 4, 4}, /* 6000000 */ 84862306a36Sopenharmony_ci {26000000, 4, 4}, /* 6500000 */ 84962306a36Sopenharmony_ci {38400000, 6, 5} /* 6400000 */ 85062306a36Sopenharmony_ci}; 85162306a36Sopenharmony_ci 85262306a36Sopenharmony_cistatic int cs42l73_get_mclkx_coeff(int mclkx) 85362306a36Sopenharmony_ci{ 85462306a36Sopenharmony_ci int i; 85562306a36Sopenharmony_ci 85662306a36Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(cs42l73_mclkx_coeffs); i++) { 85762306a36Sopenharmony_ci if (cs42l73_mclkx_coeffs[i].mclkx == mclkx) 85862306a36Sopenharmony_ci return i; 85962306a36Sopenharmony_ci } 86062306a36Sopenharmony_ci return -EINVAL; 86162306a36Sopenharmony_ci} 86262306a36Sopenharmony_ci 86362306a36Sopenharmony_cistatic int cs42l73_get_mclk_coeff(int mclk, int srate) 86462306a36Sopenharmony_ci{ 86562306a36Sopenharmony_ci int i; 86662306a36Sopenharmony_ci 86762306a36Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(cs42l73_mclk_coeffs); i++) { 86862306a36Sopenharmony_ci if (cs42l73_mclk_coeffs[i].mclk == mclk && 86962306a36Sopenharmony_ci cs42l73_mclk_coeffs[i].srate == srate) 87062306a36Sopenharmony_ci return i; 87162306a36Sopenharmony_ci } 87262306a36Sopenharmony_ci return -EINVAL; 87362306a36Sopenharmony_ci 87462306a36Sopenharmony_ci} 87562306a36Sopenharmony_ci 87662306a36Sopenharmony_cistatic int cs42l73_set_mclk(struct snd_soc_dai *dai, unsigned int freq) 87762306a36Sopenharmony_ci{ 87862306a36Sopenharmony_ci struct snd_soc_component *component = dai->component; 87962306a36Sopenharmony_ci struct cs42l73_private *priv = snd_soc_component_get_drvdata(component); 88062306a36Sopenharmony_ci 88162306a36Sopenharmony_ci int mclkx_coeff; 88262306a36Sopenharmony_ci u32 mclk = 0; 88362306a36Sopenharmony_ci u8 dmmcc = 0; 88462306a36Sopenharmony_ci 88562306a36Sopenharmony_ci /* MCLKX -> MCLK */ 88662306a36Sopenharmony_ci mclkx_coeff = cs42l73_get_mclkx_coeff(freq); 88762306a36Sopenharmony_ci if (mclkx_coeff < 0) 88862306a36Sopenharmony_ci return mclkx_coeff; 88962306a36Sopenharmony_ci 89062306a36Sopenharmony_ci mclk = cs42l73_mclkx_coeffs[mclkx_coeff].mclkx / 89162306a36Sopenharmony_ci cs42l73_mclkx_coeffs[mclkx_coeff].ratio; 89262306a36Sopenharmony_ci 89362306a36Sopenharmony_ci dev_dbg(component->dev, "MCLK%u %u <-> internal MCLK %u\n", 89462306a36Sopenharmony_ci priv->mclksel + 1, cs42l73_mclkx_coeffs[mclkx_coeff].mclkx, 89562306a36Sopenharmony_ci mclk); 89662306a36Sopenharmony_ci 89762306a36Sopenharmony_ci dmmcc = (priv->mclksel << 4) | 89862306a36Sopenharmony_ci (cs42l73_mclkx_coeffs[mclkx_coeff].mclkdiv << 1); 89962306a36Sopenharmony_ci 90062306a36Sopenharmony_ci snd_soc_component_write(component, CS42L73_DMMCC, dmmcc); 90162306a36Sopenharmony_ci 90262306a36Sopenharmony_ci priv->sysclk = mclkx_coeff; 90362306a36Sopenharmony_ci priv->mclk = mclk; 90462306a36Sopenharmony_ci 90562306a36Sopenharmony_ci return 0; 90662306a36Sopenharmony_ci} 90762306a36Sopenharmony_ci 90862306a36Sopenharmony_cistatic int cs42l73_set_sysclk(struct snd_soc_dai *dai, 90962306a36Sopenharmony_ci int clk_id, unsigned int freq, int dir) 91062306a36Sopenharmony_ci{ 91162306a36Sopenharmony_ci struct snd_soc_component *component = dai->component; 91262306a36Sopenharmony_ci struct cs42l73_private *priv = snd_soc_component_get_drvdata(component); 91362306a36Sopenharmony_ci 91462306a36Sopenharmony_ci switch (clk_id) { 91562306a36Sopenharmony_ci case CS42L73_CLKID_MCLK1: 91662306a36Sopenharmony_ci break; 91762306a36Sopenharmony_ci case CS42L73_CLKID_MCLK2: 91862306a36Sopenharmony_ci break; 91962306a36Sopenharmony_ci default: 92062306a36Sopenharmony_ci return -EINVAL; 92162306a36Sopenharmony_ci } 92262306a36Sopenharmony_ci 92362306a36Sopenharmony_ci if ((cs42l73_set_mclk(dai, freq)) < 0) { 92462306a36Sopenharmony_ci dev_err(component->dev, "Unable to set MCLK for dai %s\n", 92562306a36Sopenharmony_ci dai->name); 92662306a36Sopenharmony_ci return -EINVAL; 92762306a36Sopenharmony_ci } 92862306a36Sopenharmony_ci 92962306a36Sopenharmony_ci priv->mclksel = clk_id; 93062306a36Sopenharmony_ci 93162306a36Sopenharmony_ci return 0; 93262306a36Sopenharmony_ci} 93362306a36Sopenharmony_ci 93462306a36Sopenharmony_cistatic int cs42l73_set_dai_fmt(struct snd_soc_dai *codec_dai, unsigned int fmt) 93562306a36Sopenharmony_ci{ 93662306a36Sopenharmony_ci struct snd_soc_component *component = codec_dai->component; 93762306a36Sopenharmony_ci struct cs42l73_private *priv = snd_soc_component_get_drvdata(component); 93862306a36Sopenharmony_ci u8 id = codec_dai->id; 93962306a36Sopenharmony_ci unsigned int inv, format; 94062306a36Sopenharmony_ci u8 spc, mmcc; 94162306a36Sopenharmony_ci 94262306a36Sopenharmony_ci spc = snd_soc_component_read(component, CS42L73_SPC(id)); 94362306a36Sopenharmony_ci mmcc = snd_soc_component_read(component, CS42L73_MMCC(id)); 94462306a36Sopenharmony_ci 94562306a36Sopenharmony_ci switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) { 94662306a36Sopenharmony_ci case SND_SOC_DAIFMT_CBM_CFM: 94762306a36Sopenharmony_ci mmcc |= CS42L73_MS_MASTER; 94862306a36Sopenharmony_ci break; 94962306a36Sopenharmony_ci 95062306a36Sopenharmony_ci case SND_SOC_DAIFMT_CBS_CFS: 95162306a36Sopenharmony_ci mmcc &= ~CS42L73_MS_MASTER; 95262306a36Sopenharmony_ci break; 95362306a36Sopenharmony_ci 95462306a36Sopenharmony_ci default: 95562306a36Sopenharmony_ci return -EINVAL; 95662306a36Sopenharmony_ci } 95762306a36Sopenharmony_ci 95862306a36Sopenharmony_ci format = (fmt & SND_SOC_DAIFMT_FORMAT_MASK); 95962306a36Sopenharmony_ci inv = (fmt & SND_SOC_DAIFMT_INV_MASK); 96062306a36Sopenharmony_ci 96162306a36Sopenharmony_ci switch (format) { 96262306a36Sopenharmony_ci case SND_SOC_DAIFMT_I2S: 96362306a36Sopenharmony_ci spc &= ~CS42L73_SPDIF_PCM; 96462306a36Sopenharmony_ci break; 96562306a36Sopenharmony_ci case SND_SOC_DAIFMT_DSP_A: 96662306a36Sopenharmony_ci case SND_SOC_DAIFMT_DSP_B: 96762306a36Sopenharmony_ci if (mmcc & CS42L73_MS_MASTER) { 96862306a36Sopenharmony_ci dev_err(component->dev, 96962306a36Sopenharmony_ci "PCM format in slave mode only\n"); 97062306a36Sopenharmony_ci return -EINVAL; 97162306a36Sopenharmony_ci } 97262306a36Sopenharmony_ci if (id == CS42L73_ASP) { 97362306a36Sopenharmony_ci dev_err(component->dev, 97462306a36Sopenharmony_ci "PCM format is not supported on ASP port\n"); 97562306a36Sopenharmony_ci return -EINVAL; 97662306a36Sopenharmony_ci } 97762306a36Sopenharmony_ci spc |= CS42L73_SPDIF_PCM; 97862306a36Sopenharmony_ci break; 97962306a36Sopenharmony_ci default: 98062306a36Sopenharmony_ci return -EINVAL; 98162306a36Sopenharmony_ci } 98262306a36Sopenharmony_ci 98362306a36Sopenharmony_ci if (spc & CS42L73_SPDIF_PCM) { 98462306a36Sopenharmony_ci /* Clear PCM mode, clear PCM_BIT_ORDER bit for MSB->LSB */ 98562306a36Sopenharmony_ci spc &= ~(CS42L73_PCM_MODE_MASK | CS42L73_PCM_BIT_ORDER); 98662306a36Sopenharmony_ci switch (format) { 98762306a36Sopenharmony_ci case SND_SOC_DAIFMT_DSP_B: 98862306a36Sopenharmony_ci if (inv == SND_SOC_DAIFMT_IB_IF) 98962306a36Sopenharmony_ci spc |= CS42L73_PCM_MODE0; 99062306a36Sopenharmony_ci if (inv == SND_SOC_DAIFMT_IB_NF) 99162306a36Sopenharmony_ci spc |= CS42L73_PCM_MODE1; 99262306a36Sopenharmony_ci break; 99362306a36Sopenharmony_ci case SND_SOC_DAIFMT_DSP_A: 99462306a36Sopenharmony_ci if (inv == SND_SOC_DAIFMT_IB_IF) 99562306a36Sopenharmony_ci spc |= CS42L73_PCM_MODE1; 99662306a36Sopenharmony_ci break; 99762306a36Sopenharmony_ci default: 99862306a36Sopenharmony_ci return -EINVAL; 99962306a36Sopenharmony_ci } 100062306a36Sopenharmony_ci } 100162306a36Sopenharmony_ci 100262306a36Sopenharmony_ci priv->config[id].spc = spc; 100362306a36Sopenharmony_ci priv->config[id].mmcc = mmcc; 100462306a36Sopenharmony_ci 100562306a36Sopenharmony_ci return 0; 100662306a36Sopenharmony_ci} 100762306a36Sopenharmony_ci 100862306a36Sopenharmony_cistatic const unsigned int cs42l73_asrc_rates[] = { 100962306a36Sopenharmony_ci 8000, 11025, 12000, 16000, 22050, 101062306a36Sopenharmony_ci 24000, 32000, 44100, 48000 101162306a36Sopenharmony_ci}; 101262306a36Sopenharmony_ci 101362306a36Sopenharmony_cistatic unsigned int cs42l73_get_xspfs_coeff(u32 rate) 101462306a36Sopenharmony_ci{ 101562306a36Sopenharmony_ci int i; 101662306a36Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(cs42l73_asrc_rates); i++) { 101762306a36Sopenharmony_ci if (cs42l73_asrc_rates[i] == rate) 101862306a36Sopenharmony_ci return i + 1; 101962306a36Sopenharmony_ci } 102062306a36Sopenharmony_ci return 0; /* 0 = Don't know */ 102162306a36Sopenharmony_ci} 102262306a36Sopenharmony_ci 102362306a36Sopenharmony_cistatic void cs42l73_update_asrc(struct snd_soc_component *component, int id, int srate) 102462306a36Sopenharmony_ci{ 102562306a36Sopenharmony_ci u8 spfs = 0; 102662306a36Sopenharmony_ci 102762306a36Sopenharmony_ci if (srate > 0) 102862306a36Sopenharmony_ci spfs = cs42l73_get_xspfs_coeff(srate); 102962306a36Sopenharmony_ci 103062306a36Sopenharmony_ci switch (id) { 103162306a36Sopenharmony_ci case CS42L73_XSP: 103262306a36Sopenharmony_ci snd_soc_component_update_bits(component, CS42L73_VXSPFS, 0x0f, spfs); 103362306a36Sopenharmony_ci break; 103462306a36Sopenharmony_ci case CS42L73_ASP: 103562306a36Sopenharmony_ci snd_soc_component_update_bits(component, CS42L73_ASPC, 0x3c, spfs << 2); 103662306a36Sopenharmony_ci break; 103762306a36Sopenharmony_ci case CS42L73_VSP: 103862306a36Sopenharmony_ci snd_soc_component_update_bits(component, CS42L73_VXSPFS, 0xf0, spfs << 4); 103962306a36Sopenharmony_ci break; 104062306a36Sopenharmony_ci default: 104162306a36Sopenharmony_ci break; 104262306a36Sopenharmony_ci } 104362306a36Sopenharmony_ci} 104462306a36Sopenharmony_ci 104562306a36Sopenharmony_cistatic int cs42l73_pcm_hw_params(struct snd_pcm_substream *substream, 104662306a36Sopenharmony_ci struct snd_pcm_hw_params *params, 104762306a36Sopenharmony_ci struct snd_soc_dai *dai) 104862306a36Sopenharmony_ci{ 104962306a36Sopenharmony_ci struct snd_soc_component *component = dai->component; 105062306a36Sopenharmony_ci struct cs42l73_private *priv = snd_soc_component_get_drvdata(component); 105162306a36Sopenharmony_ci int id = dai->id; 105262306a36Sopenharmony_ci int mclk_coeff; 105362306a36Sopenharmony_ci int srate = params_rate(params); 105462306a36Sopenharmony_ci 105562306a36Sopenharmony_ci if (priv->config[id].mmcc & CS42L73_MS_MASTER) { 105662306a36Sopenharmony_ci /* CS42L73 Master */ 105762306a36Sopenharmony_ci /* MCLK -> srate */ 105862306a36Sopenharmony_ci mclk_coeff = 105962306a36Sopenharmony_ci cs42l73_get_mclk_coeff(priv->mclk, srate); 106062306a36Sopenharmony_ci 106162306a36Sopenharmony_ci if (mclk_coeff < 0) 106262306a36Sopenharmony_ci return -EINVAL; 106362306a36Sopenharmony_ci 106462306a36Sopenharmony_ci dev_dbg(component->dev, 106562306a36Sopenharmony_ci "DAI[%d]: MCLK %u, srate %u, MMCC[5:0] = %x\n", 106662306a36Sopenharmony_ci id, priv->mclk, srate, 106762306a36Sopenharmony_ci cs42l73_mclk_coeffs[mclk_coeff].mmcc); 106862306a36Sopenharmony_ci 106962306a36Sopenharmony_ci priv->config[id].mmcc &= 0xC0; 107062306a36Sopenharmony_ci priv->config[id].mmcc |= cs42l73_mclk_coeffs[mclk_coeff].mmcc; 107162306a36Sopenharmony_ci priv->config[id].spc &= 0xFC; 107262306a36Sopenharmony_ci /* Use SCLK=64*Fs if internal MCLK >= 6.4MHz */ 107362306a36Sopenharmony_ci if (priv->mclk >= 6400000) 107462306a36Sopenharmony_ci priv->config[id].spc |= CS42L73_MCK_SCLK_64FS; 107562306a36Sopenharmony_ci else 107662306a36Sopenharmony_ci priv->config[id].spc |= CS42L73_MCK_SCLK_MCLK; 107762306a36Sopenharmony_ci } else { 107862306a36Sopenharmony_ci /* CS42L73 Slave */ 107962306a36Sopenharmony_ci priv->config[id].spc &= 0xFC; 108062306a36Sopenharmony_ci priv->config[id].spc |= CS42L73_MCK_SCLK_64FS; 108162306a36Sopenharmony_ci } 108262306a36Sopenharmony_ci /* Update ASRCs */ 108362306a36Sopenharmony_ci priv->config[id].srate = srate; 108462306a36Sopenharmony_ci 108562306a36Sopenharmony_ci snd_soc_component_write(component, CS42L73_SPC(id), priv->config[id].spc); 108662306a36Sopenharmony_ci snd_soc_component_write(component, CS42L73_MMCC(id), priv->config[id].mmcc); 108762306a36Sopenharmony_ci 108862306a36Sopenharmony_ci cs42l73_update_asrc(component, id, srate); 108962306a36Sopenharmony_ci 109062306a36Sopenharmony_ci return 0; 109162306a36Sopenharmony_ci} 109262306a36Sopenharmony_ci 109362306a36Sopenharmony_cistatic int cs42l73_set_bias_level(struct snd_soc_component *component, 109462306a36Sopenharmony_ci enum snd_soc_bias_level level) 109562306a36Sopenharmony_ci{ 109662306a36Sopenharmony_ci struct cs42l73_private *cs42l73 = snd_soc_component_get_drvdata(component); 109762306a36Sopenharmony_ci 109862306a36Sopenharmony_ci switch (level) { 109962306a36Sopenharmony_ci case SND_SOC_BIAS_ON: 110062306a36Sopenharmony_ci snd_soc_component_update_bits(component, CS42L73_DMMCC, CS42L73_MCLKDIS, 0); 110162306a36Sopenharmony_ci snd_soc_component_update_bits(component, CS42L73_PWRCTL1, CS42L73_PDN, 0); 110262306a36Sopenharmony_ci break; 110362306a36Sopenharmony_ci 110462306a36Sopenharmony_ci case SND_SOC_BIAS_PREPARE: 110562306a36Sopenharmony_ci break; 110662306a36Sopenharmony_ci 110762306a36Sopenharmony_ci case SND_SOC_BIAS_STANDBY: 110862306a36Sopenharmony_ci if (snd_soc_component_get_bias_level(component) == SND_SOC_BIAS_OFF) { 110962306a36Sopenharmony_ci regcache_cache_only(cs42l73->regmap, false); 111062306a36Sopenharmony_ci regcache_sync(cs42l73->regmap); 111162306a36Sopenharmony_ci } 111262306a36Sopenharmony_ci snd_soc_component_update_bits(component, CS42L73_PWRCTL1, CS42L73_PDN, 1); 111362306a36Sopenharmony_ci break; 111462306a36Sopenharmony_ci 111562306a36Sopenharmony_ci case SND_SOC_BIAS_OFF: 111662306a36Sopenharmony_ci snd_soc_component_update_bits(component, CS42L73_PWRCTL1, CS42L73_PDN, 1); 111762306a36Sopenharmony_ci if (cs42l73->shutdwn_delay > 0) { 111862306a36Sopenharmony_ci mdelay(cs42l73->shutdwn_delay); 111962306a36Sopenharmony_ci cs42l73->shutdwn_delay = 0; 112062306a36Sopenharmony_ci } else { 112162306a36Sopenharmony_ci mdelay(15); /* Min amount of time requred to power 112262306a36Sopenharmony_ci * down. 112362306a36Sopenharmony_ci */ 112462306a36Sopenharmony_ci } 112562306a36Sopenharmony_ci snd_soc_component_update_bits(component, CS42L73_DMMCC, CS42L73_MCLKDIS, 1); 112662306a36Sopenharmony_ci break; 112762306a36Sopenharmony_ci } 112862306a36Sopenharmony_ci return 0; 112962306a36Sopenharmony_ci} 113062306a36Sopenharmony_ci 113162306a36Sopenharmony_cistatic int cs42l73_set_tristate(struct snd_soc_dai *dai, int tristate) 113262306a36Sopenharmony_ci{ 113362306a36Sopenharmony_ci struct snd_soc_component *component = dai->component; 113462306a36Sopenharmony_ci int id = dai->id; 113562306a36Sopenharmony_ci 113662306a36Sopenharmony_ci return snd_soc_component_update_bits(component, CS42L73_SPC(id), CS42L73_SP_3ST, 113762306a36Sopenharmony_ci tristate << 7); 113862306a36Sopenharmony_ci} 113962306a36Sopenharmony_ci 114062306a36Sopenharmony_cistatic const struct snd_pcm_hw_constraint_list constraints_12_24 = { 114162306a36Sopenharmony_ci .count = ARRAY_SIZE(cs42l73_asrc_rates), 114262306a36Sopenharmony_ci .list = cs42l73_asrc_rates, 114362306a36Sopenharmony_ci}; 114462306a36Sopenharmony_ci 114562306a36Sopenharmony_cistatic int cs42l73_pcm_startup(struct snd_pcm_substream *substream, 114662306a36Sopenharmony_ci struct snd_soc_dai *dai) 114762306a36Sopenharmony_ci{ 114862306a36Sopenharmony_ci snd_pcm_hw_constraint_list(substream->runtime, 0, 114962306a36Sopenharmony_ci SNDRV_PCM_HW_PARAM_RATE, 115062306a36Sopenharmony_ci &constraints_12_24); 115162306a36Sopenharmony_ci return 0; 115262306a36Sopenharmony_ci} 115362306a36Sopenharmony_ci 115462306a36Sopenharmony_ci 115562306a36Sopenharmony_ci#define CS42L73_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE |\ 115662306a36Sopenharmony_ci SNDRV_PCM_FMTBIT_S24_LE) 115762306a36Sopenharmony_ci 115862306a36Sopenharmony_cistatic const struct snd_soc_dai_ops cs42l73_ops = { 115962306a36Sopenharmony_ci .startup = cs42l73_pcm_startup, 116062306a36Sopenharmony_ci .hw_params = cs42l73_pcm_hw_params, 116162306a36Sopenharmony_ci .set_fmt = cs42l73_set_dai_fmt, 116262306a36Sopenharmony_ci .set_sysclk = cs42l73_set_sysclk, 116362306a36Sopenharmony_ci .set_tristate = cs42l73_set_tristate, 116462306a36Sopenharmony_ci}; 116562306a36Sopenharmony_ci 116662306a36Sopenharmony_cistatic struct snd_soc_dai_driver cs42l73_dai[] = { 116762306a36Sopenharmony_ci { 116862306a36Sopenharmony_ci .name = "cs42l73-xsp", 116962306a36Sopenharmony_ci .id = CS42L73_XSP, 117062306a36Sopenharmony_ci .playback = { 117162306a36Sopenharmony_ci .stream_name = "XSP Playback", 117262306a36Sopenharmony_ci .channels_min = 1, 117362306a36Sopenharmony_ci .channels_max = 2, 117462306a36Sopenharmony_ci .rates = SNDRV_PCM_RATE_KNOT, 117562306a36Sopenharmony_ci .formats = CS42L73_FORMATS, 117662306a36Sopenharmony_ci }, 117762306a36Sopenharmony_ci .capture = { 117862306a36Sopenharmony_ci .stream_name = "XSP Capture", 117962306a36Sopenharmony_ci .channels_min = 1, 118062306a36Sopenharmony_ci .channels_max = 2, 118162306a36Sopenharmony_ci .rates = SNDRV_PCM_RATE_KNOT, 118262306a36Sopenharmony_ci .formats = CS42L73_FORMATS, 118362306a36Sopenharmony_ci }, 118462306a36Sopenharmony_ci .ops = &cs42l73_ops, 118562306a36Sopenharmony_ci .symmetric_rate = 1, 118662306a36Sopenharmony_ci }, 118762306a36Sopenharmony_ci { 118862306a36Sopenharmony_ci .name = "cs42l73-asp", 118962306a36Sopenharmony_ci .id = CS42L73_ASP, 119062306a36Sopenharmony_ci .playback = { 119162306a36Sopenharmony_ci .stream_name = "ASP Playback", 119262306a36Sopenharmony_ci .channels_min = 2, 119362306a36Sopenharmony_ci .channels_max = 2, 119462306a36Sopenharmony_ci .rates = SNDRV_PCM_RATE_KNOT, 119562306a36Sopenharmony_ci .formats = CS42L73_FORMATS, 119662306a36Sopenharmony_ci }, 119762306a36Sopenharmony_ci .capture = { 119862306a36Sopenharmony_ci .stream_name = "ASP Capture", 119962306a36Sopenharmony_ci .channels_min = 2, 120062306a36Sopenharmony_ci .channels_max = 2, 120162306a36Sopenharmony_ci .rates = SNDRV_PCM_RATE_KNOT, 120262306a36Sopenharmony_ci .formats = CS42L73_FORMATS, 120362306a36Sopenharmony_ci }, 120462306a36Sopenharmony_ci .ops = &cs42l73_ops, 120562306a36Sopenharmony_ci .symmetric_rate = 1, 120662306a36Sopenharmony_ci }, 120762306a36Sopenharmony_ci { 120862306a36Sopenharmony_ci .name = "cs42l73-vsp", 120962306a36Sopenharmony_ci .id = CS42L73_VSP, 121062306a36Sopenharmony_ci .playback = { 121162306a36Sopenharmony_ci .stream_name = "VSP Playback", 121262306a36Sopenharmony_ci .channels_min = 1, 121362306a36Sopenharmony_ci .channels_max = 2, 121462306a36Sopenharmony_ci .rates = SNDRV_PCM_RATE_KNOT, 121562306a36Sopenharmony_ci .formats = CS42L73_FORMATS, 121662306a36Sopenharmony_ci }, 121762306a36Sopenharmony_ci .capture = { 121862306a36Sopenharmony_ci .stream_name = "VSP Capture", 121962306a36Sopenharmony_ci .channels_min = 1, 122062306a36Sopenharmony_ci .channels_max = 2, 122162306a36Sopenharmony_ci .rates = SNDRV_PCM_RATE_KNOT, 122262306a36Sopenharmony_ci .formats = CS42L73_FORMATS, 122362306a36Sopenharmony_ci }, 122462306a36Sopenharmony_ci .ops = &cs42l73_ops, 122562306a36Sopenharmony_ci .symmetric_rate = 1, 122662306a36Sopenharmony_ci } 122762306a36Sopenharmony_ci}; 122862306a36Sopenharmony_ci 122962306a36Sopenharmony_cistatic int cs42l73_probe(struct snd_soc_component *component) 123062306a36Sopenharmony_ci{ 123162306a36Sopenharmony_ci struct cs42l73_private *cs42l73 = snd_soc_component_get_drvdata(component); 123262306a36Sopenharmony_ci 123362306a36Sopenharmony_ci /* Set Charge Pump Frequency */ 123462306a36Sopenharmony_ci if (cs42l73->pdata.chgfreq) 123562306a36Sopenharmony_ci snd_soc_component_update_bits(component, CS42L73_CPFCHC, 123662306a36Sopenharmony_ci CS42L73_CHARGEPUMP_MASK, 123762306a36Sopenharmony_ci cs42l73->pdata.chgfreq << 4); 123862306a36Sopenharmony_ci 123962306a36Sopenharmony_ci /* MCLK1 as master clk */ 124062306a36Sopenharmony_ci cs42l73->mclksel = CS42L73_CLKID_MCLK1; 124162306a36Sopenharmony_ci cs42l73->mclk = 0; 124262306a36Sopenharmony_ci 124362306a36Sopenharmony_ci return 0; 124462306a36Sopenharmony_ci} 124562306a36Sopenharmony_ci 124662306a36Sopenharmony_cistatic const struct snd_soc_component_driver soc_component_dev_cs42l73 = { 124762306a36Sopenharmony_ci .probe = cs42l73_probe, 124862306a36Sopenharmony_ci .set_bias_level = cs42l73_set_bias_level, 124962306a36Sopenharmony_ci .controls = cs42l73_snd_controls, 125062306a36Sopenharmony_ci .num_controls = ARRAY_SIZE(cs42l73_snd_controls), 125162306a36Sopenharmony_ci .dapm_widgets = cs42l73_dapm_widgets, 125262306a36Sopenharmony_ci .num_dapm_widgets = ARRAY_SIZE(cs42l73_dapm_widgets), 125362306a36Sopenharmony_ci .dapm_routes = cs42l73_audio_map, 125462306a36Sopenharmony_ci .num_dapm_routes = ARRAY_SIZE(cs42l73_audio_map), 125562306a36Sopenharmony_ci .suspend_bias_off = 1, 125662306a36Sopenharmony_ci .idle_bias_on = 1, 125762306a36Sopenharmony_ci .use_pmdown_time = 1, 125862306a36Sopenharmony_ci .endianness = 1, 125962306a36Sopenharmony_ci}; 126062306a36Sopenharmony_ci 126162306a36Sopenharmony_cistatic const struct regmap_config cs42l73_regmap = { 126262306a36Sopenharmony_ci .reg_bits = 8, 126362306a36Sopenharmony_ci .val_bits = 8, 126462306a36Sopenharmony_ci 126562306a36Sopenharmony_ci .max_register = CS42L73_MAX_REGISTER, 126662306a36Sopenharmony_ci .reg_defaults = cs42l73_reg_defaults, 126762306a36Sopenharmony_ci .num_reg_defaults = ARRAY_SIZE(cs42l73_reg_defaults), 126862306a36Sopenharmony_ci .volatile_reg = cs42l73_volatile_register, 126962306a36Sopenharmony_ci .readable_reg = cs42l73_readable_register, 127062306a36Sopenharmony_ci .cache_type = REGCACHE_MAPLE, 127162306a36Sopenharmony_ci 127262306a36Sopenharmony_ci .use_single_read = true, 127362306a36Sopenharmony_ci .use_single_write = true, 127462306a36Sopenharmony_ci}; 127562306a36Sopenharmony_ci 127662306a36Sopenharmony_cistatic int cs42l73_i2c_probe(struct i2c_client *i2c_client) 127762306a36Sopenharmony_ci{ 127862306a36Sopenharmony_ci struct cs42l73_private *cs42l73; 127962306a36Sopenharmony_ci struct cs42l73_platform_data *pdata = dev_get_platdata(&i2c_client->dev); 128062306a36Sopenharmony_ci int ret, devid; 128162306a36Sopenharmony_ci unsigned int reg; 128262306a36Sopenharmony_ci u32 val32; 128362306a36Sopenharmony_ci 128462306a36Sopenharmony_ci cs42l73 = devm_kzalloc(&i2c_client->dev, sizeof(*cs42l73), GFP_KERNEL); 128562306a36Sopenharmony_ci if (!cs42l73) 128662306a36Sopenharmony_ci return -ENOMEM; 128762306a36Sopenharmony_ci 128862306a36Sopenharmony_ci cs42l73->regmap = devm_regmap_init_i2c(i2c_client, &cs42l73_regmap); 128962306a36Sopenharmony_ci if (IS_ERR(cs42l73->regmap)) { 129062306a36Sopenharmony_ci ret = PTR_ERR(cs42l73->regmap); 129162306a36Sopenharmony_ci dev_err(&i2c_client->dev, "regmap_init() failed: %d\n", ret); 129262306a36Sopenharmony_ci return ret; 129362306a36Sopenharmony_ci } 129462306a36Sopenharmony_ci 129562306a36Sopenharmony_ci if (pdata) { 129662306a36Sopenharmony_ci cs42l73->pdata = *pdata; 129762306a36Sopenharmony_ci } else { 129862306a36Sopenharmony_ci pdata = devm_kzalloc(&i2c_client->dev, sizeof(*pdata), 129962306a36Sopenharmony_ci GFP_KERNEL); 130062306a36Sopenharmony_ci if (!pdata) 130162306a36Sopenharmony_ci return -ENOMEM; 130262306a36Sopenharmony_ci 130362306a36Sopenharmony_ci if (i2c_client->dev.of_node) { 130462306a36Sopenharmony_ci if (of_property_read_u32(i2c_client->dev.of_node, 130562306a36Sopenharmony_ci "chgfreq", &val32) >= 0) 130662306a36Sopenharmony_ci pdata->chgfreq = val32; 130762306a36Sopenharmony_ci } 130862306a36Sopenharmony_ci pdata->reset_gpio = of_get_named_gpio(i2c_client->dev.of_node, 130962306a36Sopenharmony_ci "reset-gpio", 0); 131062306a36Sopenharmony_ci cs42l73->pdata = *pdata; 131162306a36Sopenharmony_ci } 131262306a36Sopenharmony_ci 131362306a36Sopenharmony_ci i2c_set_clientdata(i2c_client, cs42l73); 131462306a36Sopenharmony_ci 131562306a36Sopenharmony_ci if (cs42l73->pdata.reset_gpio) { 131662306a36Sopenharmony_ci ret = devm_gpio_request_one(&i2c_client->dev, 131762306a36Sopenharmony_ci cs42l73->pdata.reset_gpio, 131862306a36Sopenharmony_ci GPIOF_OUT_INIT_HIGH, 131962306a36Sopenharmony_ci "CS42L73 /RST"); 132062306a36Sopenharmony_ci if (ret < 0) { 132162306a36Sopenharmony_ci dev_err(&i2c_client->dev, "Failed to request /RST %d: %d\n", 132262306a36Sopenharmony_ci cs42l73->pdata.reset_gpio, ret); 132362306a36Sopenharmony_ci return ret; 132462306a36Sopenharmony_ci } 132562306a36Sopenharmony_ci gpio_set_value_cansleep(cs42l73->pdata.reset_gpio, 0); 132662306a36Sopenharmony_ci gpio_set_value_cansleep(cs42l73->pdata.reset_gpio, 1); 132762306a36Sopenharmony_ci } 132862306a36Sopenharmony_ci 132962306a36Sopenharmony_ci /* initialize codec */ 133062306a36Sopenharmony_ci devid = cirrus_read_device_id(cs42l73->regmap, CS42L73_DEVID_AB); 133162306a36Sopenharmony_ci if (devid < 0) { 133262306a36Sopenharmony_ci ret = devid; 133362306a36Sopenharmony_ci dev_err(&i2c_client->dev, "Failed to read device ID: %d\n", ret); 133462306a36Sopenharmony_ci goto err_reset; 133562306a36Sopenharmony_ci } 133662306a36Sopenharmony_ci 133762306a36Sopenharmony_ci if (devid != CS42L73_DEVID) { 133862306a36Sopenharmony_ci ret = -ENODEV; 133962306a36Sopenharmony_ci dev_err(&i2c_client->dev, 134062306a36Sopenharmony_ci "CS42L73 Device ID (%X). Expected %X\n", 134162306a36Sopenharmony_ci devid, CS42L73_DEVID); 134262306a36Sopenharmony_ci goto err_reset; 134362306a36Sopenharmony_ci } 134462306a36Sopenharmony_ci 134562306a36Sopenharmony_ci ret = regmap_read(cs42l73->regmap, CS42L73_REVID, ®); 134662306a36Sopenharmony_ci if (ret < 0) { 134762306a36Sopenharmony_ci dev_err(&i2c_client->dev, "Get Revision ID failed\n"); 134862306a36Sopenharmony_ci goto err_reset; 134962306a36Sopenharmony_ci } 135062306a36Sopenharmony_ci 135162306a36Sopenharmony_ci dev_info(&i2c_client->dev, 135262306a36Sopenharmony_ci "Cirrus Logic CS42L73, Revision: %02X\n", reg & 0xFF); 135362306a36Sopenharmony_ci 135462306a36Sopenharmony_ci ret = devm_snd_soc_register_component(&i2c_client->dev, 135562306a36Sopenharmony_ci &soc_component_dev_cs42l73, cs42l73_dai, 135662306a36Sopenharmony_ci ARRAY_SIZE(cs42l73_dai)); 135762306a36Sopenharmony_ci if (ret < 0) 135862306a36Sopenharmony_ci goto err_reset; 135962306a36Sopenharmony_ci 136062306a36Sopenharmony_ci return 0; 136162306a36Sopenharmony_ci 136262306a36Sopenharmony_cierr_reset: 136362306a36Sopenharmony_ci gpio_set_value_cansleep(cs42l73->pdata.reset_gpio, 0); 136462306a36Sopenharmony_ci 136562306a36Sopenharmony_ci return ret; 136662306a36Sopenharmony_ci} 136762306a36Sopenharmony_ci 136862306a36Sopenharmony_cistatic const struct of_device_id cs42l73_of_match[] = { 136962306a36Sopenharmony_ci { .compatible = "cirrus,cs42l73", }, 137062306a36Sopenharmony_ci {}, 137162306a36Sopenharmony_ci}; 137262306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, cs42l73_of_match); 137362306a36Sopenharmony_ci 137462306a36Sopenharmony_cistatic const struct i2c_device_id cs42l73_id[] = { 137562306a36Sopenharmony_ci {"cs42l73", 0}, 137662306a36Sopenharmony_ci {} 137762306a36Sopenharmony_ci}; 137862306a36Sopenharmony_ci 137962306a36Sopenharmony_ciMODULE_DEVICE_TABLE(i2c, cs42l73_id); 138062306a36Sopenharmony_ci 138162306a36Sopenharmony_cistatic struct i2c_driver cs42l73_i2c_driver = { 138262306a36Sopenharmony_ci .driver = { 138362306a36Sopenharmony_ci .name = "cs42l73", 138462306a36Sopenharmony_ci .of_match_table = cs42l73_of_match, 138562306a36Sopenharmony_ci }, 138662306a36Sopenharmony_ci .id_table = cs42l73_id, 138762306a36Sopenharmony_ci .probe = cs42l73_i2c_probe, 138862306a36Sopenharmony_ci 138962306a36Sopenharmony_ci}; 139062306a36Sopenharmony_ci 139162306a36Sopenharmony_cimodule_i2c_driver(cs42l73_i2c_driver); 139262306a36Sopenharmony_ci 139362306a36Sopenharmony_ciMODULE_DESCRIPTION("ASoC CS42L73 driver"); 139462306a36Sopenharmony_ciMODULE_AUTHOR("Georgi Vlaev, Nucleus Systems Ltd, <joe@nucleusys.com>"); 139562306a36Sopenharmony_ciMODULE_AUTHOR("Brian Austin, Cirrus Logic Inc, <brian.austin@cirrus.com>"); 139662306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 1397