Lines Matching +full:vmon +full:- +full:slot +full:- +full:no
1 // SPDX-License-Identifier: GPL-2.0-only
3 * cs35l34.c -- CS35l34 ALSA SoC audio driver
29 #include <sound/soc-dapm.h>
50 struct gpio_desc *reset_gpio; /* Active-low reset GPIO */
237 struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm); in cs35l34_sdin_event()
243 if (priv->tdm_mode) in cs35l34_sdin_event()
244 regmap_update_bits(priv->regmap, CS35L34_PWRCTL3, in cs35l34_sdin_event()
247 ret = regmap_update_bits(priv->regmap, CS35L34_PWRCTL1, in cs35l34_sdin_event()
250 dev_err(component->dev, "Cannot set Power bits %d\n", ret); in cs35l34_sdin_event()
256 if (priv->tdm_mode) { in cs35l34_sdin_event()
257 regmap_update_bits(priv->regmap, CS35L34_PWRCTL3, in cs35l34_sdin_event()
260 ret = regmap_update_bits(priv->regmap, CS35L34_PWRCTL1, in cs35l34_sdin_event()
272 struct snd_soc_component *component = dai->component; in cs35l34_set_tdm_slot()
275 int slot, slot_num; in cs35l34_set_tdm_slot() local
278 return -EINVAL; in cs35l34_set_tdm_slot()
280 priv->tdm_mode = true; in cs35l34_set_tdm_slot()
281 /* scan rx_mask for aud slot */ in cs35l34_set_tdm_slot()
282 slot = ffs(rx_mask) - 1; in cs35l34_set_tdm_slot()
283 if (slot >= 0) in cs35l34_set_tdm_slot()
285 CS35L34_X_LOC, slot); in cs35l34_set_tdm_slot()
287 /* scan tx_mask: vmon(2 slots); imon (2 slots); vpmon (1 slot) in cs35l34_set_tdm_slot()
288 * vbstmon (1 slot) in cs35l34_set_tdm_slot()
290 slot = ffs(tx_mask) - 1; in cs35l34_set_tdm_slot()
302 while (slot >= 0) { in cs35l34_set_tdm_slot()
306 CS35L34_X_STATE | CS35L34_X_LOC, slot); in cs35l34_set_tdm_slot()
311 CS35L34_X_STATE | CS35L34_X_LOC, slot); in cs35l34_set_tdm_slot()
316 CS35L34_X_STATE | CS35L34_X_LOC, slot); in cs35l34_set_tdm_slot()
322 CS35L34_X_STATE | CS35L34_X_LOC, slot); in cs35l34_set_tdm_slot()
325 /* Enable the relevant tx slot */ in cs35l34_set_tdm_slot()
326 reg = CS35L34_TDM_TX_SLOT_EN_4 - (slot/8); in cs35l34_set_tdm_slot()
327 bit_pos = slot - ((slot / 8) * (8)); in cs35l34_set_tdm_slot()
331 tx_mask &= ~(1 << slot); in cs35l34_set_tdm_slot()
332 slot = ffs(tx_mask) - 1; in cs35l34_set_tdm_slot()
342 struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm); in cs35l34_main_amp_event()
347 regmap_update_bits(priv->regmap, CS35L34_BST_CVTR_V_CTL, in cs35l34_main_amp_event()
348 CS35L34_BST_CVTL_MASK, priv->pdata.boost_vtge); in cs35l34_main_amp_event()
350 regmap_update_bits(priv->regmap, CS35L34_PROTECT_CTL, in cs35l34_main_amp_event()
354 regmap_update_bits(priv->regmap, CS35L34_BST_CVTR_V_CTL, in cs35l34_main_amp_event()
356 regmap_update_bits(priv->regmap, CS35L34_PROTECT_CTL, in cs35l34_main_amp_event()
366 static DECLARE_TLV_DB_SCALE(dig_vol_tlv, -10200, 50, 0);
382 struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm); in cs35l34_mclk_event()
389 ret = regmap_read(priv->regmap, CS35L34_AMP_DIG_VOL_CTL, in cs35l34_mclk_event()
401 ret = regmap_read(priv->regmap, CS35L34_INT_STATUS_2, in cs35l34_mclk_event()
441 SND_SOC_DAPM_ADC("VMON ADC", NULL, CS35L34_PWRCTL2, 7, 1),
466 {"VMON ADC", NULL, "VSENSE"},
468 {"SDOUT", NULL, "VMON ADC"},
519 return -EINVAL; in cs35l34_get_mclk_coeff()
524 struct snd_soc_component *component = codec_dai->component; in cs35l34_set_dai_fmt()
529 regmap_update_bits(priv->regmap, CS35L34_ADSP_CLK_CTL, in cs35l34_set_dai_fmt()
533 regmap_update_bits(priv->regmap, CS35L34_ADSP_CLK_CTL, in cs35l34_set_dai_fmt()
537 return -EINVAL; in cs35l34_set_dai_fmt()
546 struct snd_soc_component *component = dai->component; in cs35l34_pcm_hw_params()
551 int coeff = cs35l34_get_mclk_coeff(priv->mclk_int, srate); in cs35l34_pcm_hw_params()
554 dev_err(component->dev, "ERROR: Invalid mclk %d and/or srate %d\n", in cs35l34_pcm_hw_params()
555 priv->mclk_int, srate); in cs35l34_pcm_hw_params()
559 ret = regmap_update_bits(priv->regmap, CS35L34_ADSP_CLK_CTL, in cs35l34_pcm_hw_params()
562 dev_err(component->dev, "Failed to set clock state %d\n", ret); in cs35l34_pcm_hw_params()
581 snd_pcm_hw_constraint_list(substream->runtime, 0, in cs35l34_pcm_startup()
590 struct snd_soc_component *component = dai->component; in cs35l34_set_tristate()
604 struct snd_soc_component *component = dai->component; in cs35l34_dai_set_sysclk()
611 cs35l34->mclk_int = freq; in cs35l34_dai_set_sysclk()
615 cs35l34->mclk_int = freq; in cs35l34_dai_set_sysclk()
619 cs35l34->mclk_int = freq; in cs35l34_dai_set_sysclk()
623 cs35l34->mclk_int = freq / 2; in cs35l34_dai_set_sysclk()
627 cs35l34->mclk_int = freq / 2; in cs35l34_dai_set_sysclk()
631 cs35l34->mclk_int = freq / 2; in cs35l34_dai_set_sysclk()
634 dev_err(component->dev, "ERROR: Invalid Frequency %d\n", freq); in cs35l34_dai_set_sysclk()
635 cs35l34->mclk_int = 0; in cs35l34_dai_set_sysclk()
636 return -EINVAL; in cs35l34_dai_set_sysclk()
638 regmap_update_bits(cs35l34->regmap, CS35L34_MCLK_CTL, in cs35l34_dai_set_sysclk()
676 struct snd_soc_component *component = cs35l34->component; in cs35l34_boost_inductor()
680 regmap_write(cs35l34->regmap, CS35L34_BST_CONV_COEF_1, 0x24); in cs35l34_boost_inductor()
681 regmap_write(cs35l34->regmap, CS35L34_BST_CONV_COEF_2, 0x24); in cs35l34_boost_inductor()
682 regmap_write(cs35l34->regmap, CS35L34_BST_CONV_SLOPE_COMP, in cs35l34_boost_inductor()
684 regmap_write(cs35l34->regmap, CS35L34_BST_CONV_SW_FREQ, 0); in cs35l34_boost_inductor()
687 regmap_write(cs35l34->regmap, CS35L34_BST_CONV_COEF_1, 0x20); in cs35l34_boost_inductor()
688 regmap_write(cs35l34->regmap, CS35L34_BST_CONV_COEF_2, 0x20); in cs35l34_boost_inductor()
689 regmap_write(cs35l34->regmap, CS35L34_BST_CONV_SLOPE_COMP, in cs35l34_boost_inductor()
691 regmap_write(cs35l34->regmap, CS35L34_BST_CONV_SW_FREQ, 1); in cs35l34_boost_inductor()
694 regmap_write(cs35l34->regmap, CS35L34_BST_CONV_COEF_1, 0x20); in cs35l34_boost_inductor()
695 regmap_write(cs35l34->regmap, CS35L34_BST_CONV_COEF_2, 0x20); in cs35l34_boost_inductor()
696 regmap_write(cs35l34->regmap, CS35L34_BST_CONV_SLOPE_COMP, in cs35l34_boost_inductor()
698 regmap_write(cs35l34->regmap, CS35L34_BST_CONV_SW_FREQ, 2); in cs35l34_boost_inductor()
701 regmap_write(cs35l34->regmap, CS35L34_BST_CONV_COEF_1, 0x19); in cs35l34_boost_inductor()
702 regmap_write(cs35l34->regmap, CS35L34_BST_CONV_COEF_2, 0x25); in cs35l34_boost_inductor()
703 regmap_write(cs35l34->regmap, CS35L34_BST_CONV_SLOPE_COMP, in cs35l34_boost_inductor()
705 regmap_write(cs35l34->regmap, CS35L34_BST_CONV_SW_FREQ, 3); in cs35l34_boost_inductor()
708 dev_err(component->dev, "%s Invalid Inductor Value %d uH\n", in cs35l34_boost_inductor()
710 return -EINVAL; in cs35l34_boost_inductor()
720 pm_runtime_get_sync(component->dev); in cs35l34_probe()
723 regmap_update_bits(cs35l34->regmap, CS35L34_PROTECT_CTL, in cs35l34_probe()
729 regmap_write(cs35l34->regmap, CS35L34_PWRCTL2, 0xFD); in cs35l34_probe()
730 regmap_write(cs35l34->regmap, CS35L34_PWRCTL3, 0x1F); in cs35l34_probe()
733 regmap_update_bits(cs35l34->regmap, CS35L34_PROTECT_CTL, in cs35l34_probe()
737 if (cs35l34->pdata.boost_peak) in cs35l34_probe()
738 regmap_update_bits(cs35l34->regmap, CS35L34_BST_PEAK_I, in cs35l34_probe()
740 cs35l34->pdata.boost_peak); in cs35l34_probe()
742 if (cs35l34->pdata.gain_zc_disable) in cs35l34_probe()
743 regmap_update_bits(cs35l34->regmap, CS35L34_PROTECT_CTL, in cs35l34_probe()
746 regmap_update_bits(cs35l34->regmap, CS35L34_PROTECT_CTL, in cs35l34_probe()
749 if (cs35l34->pdata.aif_half_drv) in cs35l34_probe()
750 regmap_update_bits(cs35l34->regmap, CS35L34_ADSP_CLK_CTL, in cs35l34_probe()
753 if (cs35l34->pdata.digsft_disable) in cs35l34_probe()
754 regmap_update_bits(cs35l34->regmap, CS35L34_AMP_DIG_VOL_CTL, in cs35l34_probe()
757 if (cs35l34->pdata.amp_inv) in cs35l34_probe()
758 regmap_update_bits(cs35l34->regmap, CS35L34_AMP_DIG_VOL_CTL, in cs35l34_probe()
761 if (cs35l34->pdata.boost_ind) in cs35l34_probe()
762 ret = cs35l34_boost_inductor(cs35l34, cs35l34->pdata.boost_ind); in cs35l34_probe()
764 if (cs35l34->pdata.i2s_sdinloc) in cs35l34_probe()
765 regmap_update_bits(cs35l34->regmap, CS35L34_ADSP_I2S_CTL, in cs35l34_probe()
767 cs35l34->pdata.i2s_sdinloc << CS35L34_I2S_LOC_SHIFT); in cs35l34_probe()
769 if (cs35l34->pdata.tdm_rising_edge) in cs35l34_probe()
770 regmap_update_bits(cs35l34->regmap, CS35L34_ADSP_TDM_CTL, in cs35l34_probe()
773 pm_runtime_put_sync(component->dev); in cs35l34_probe()
811 struct device_node *np = i2c_client->dev.of_node; in cs35l34_handle_of_data()
814 if (of_property_read_u32(np, "cirrus,boost-vtge-millivolt", in cs35l34_handle_of_data()
818 dev_err(&i2c_client->dev, in cs35l34_handle_of_data()
820 return -EINVAL; in cs35l34_handle_of_data()
823 pdata->boost_vtge = 0; /* Use VP */ in cs35l34_handle_of_data()
825 pdata->boost_vtge = ((val - 3300)/100) + 1; in cs35l34_handle_of_data()
827 dev_warn(&i2c_client->dev, in cs35l34_handle_of_data()
831 if (of_property_read_u32(np, "cirrus,boost-ind-nanohenry", &val) >= 0) { in cs35l34_handle_of_data()
832 pdata->boost_ind = val; in cs35l34_handle_of_data()
834 dev_err(&i2c_client->dev, "Inductor not specified.\n"); in cs35l34_handle_of_data()
835 return -EINVAL; in cs35l34_handle_of_data()
838 if (of_property_read_u32(np, "cirrus,boost-peak-milliamp", &val) >= 0) { in cs35l34_handle_of_data()
840 dev_err(&i2c_client->dev, in cs35l34_handle_of_data()
842 return -EINVAL; in cs35l34_handle_of_data()
844 pdata->boost_peak = ((val - 1200)/80) + 1; in cs35l34_handle_of_data()
847 pdata->aif_half_drv = of_property_read_bool(np, in cs35l34_handle_of_data()
848 "cirrus,aif-half-drv"); in cs35l34_handle_of_data()
849 pdata->digsft_disable = of_property_read_bool(np, in cs35l34_handle_of_data()
850 "cirrus,digsft-disable"); in cs35l34_handle_of_data()
852 pdata->gain_zc_disable = of_property_read_bool(np, in cs35l34_handle_of_data()
853 "cirrus,gain-zc-disable"); in cs35l34_handle_of_data()
854 pdata->amp_inv = of_property_read_bool(np, "cirrus,amp-inv"); in cs35l34_handle_of_data()
856 if (of_property_read_u32(np, "cirrus,i2s-sdinloc", &val) >= 0) in cs35l34_handle_of_data()
857 pdata->i2s_sdinloc = val; in cs35l34_handle_of_data()
858 if (of_property_read_u32(np, "cirrus,tdm-rising-edge", &val) >= 0) in cs35l34_handle_of_data()
859 pdata->tdm_rising_edge = val; in cs35l34_handle_of_data()
867 struct snd_soc_component *component = cs35l34->component; in cs35l34_irq_thread()
873 regmap_read(cs35l34->regmap, CS35L34_INT_STATUS_4, &sticky4); in cs35l34_irq_thread()
874 regmap_read(cs35l34->regmap, CS35L34_INT_STATUS_3, &sticky3); in cs35l34_irq_thread()
875 regmap_read(cs35l34->regmap, CS35L34_INT_STATUS_2, &sticky2); in cs35l34_irq_thread()
876 regmap_read(cs35l34->regmap, CS35L34_INT_STATUS_1, &sticky1); in cs35l34_irq_thread()
878 regmap_read(cs35l34->regmap, CS35L34_INT_MASK_4, &mask4); in cs35l34_irq_thread()
879 regmap_read(cs35l34->regmap, CS35L34_INT_MASK_3, &mask3); in cs35l34_irq_thread()
880 regmap_read(cs35l34->regmap, CS35L34_INT_MASK_2, &mask2); in cs35l34_irq_thread()
881 regmap_read(cs35l34->regmap, CS35L34_INT_MASK_1, &mask1); in cs35l34_irq_thread()
887 regmap_read(cs35l34->regmap, CS35L34_INT_STATUS_1, ¤t1); in cs35l34_irq_thread()
890 dev_err(component->dev, "Cal error\n"); in cs35l34_irq_thread()
892 /* error is no longer asserted; safe to reset */ in cs35l34_irq_thread()
894 dev_dbg(component->dev, "Cal error release\n"); in cs35l34_irq_thread()
895 regmap_update_bits(cs35l34->regmap, in cs35l34_irq_thread()
898 regmap_update_bits(cs35l34->regmap, in cs35l34_irq_thread()
902 regmap_update_bits(cs35l34->regmap, in cs35l34_irq_thread()
905 /* note: amp will re-calibrate on next resume */ in cs35l34_irq_thread()
910 dev_err(component->dev, "Alive error\n"); in cs35l34_irq_thread()
913 dev_crit(component->dev, "Amp short error\n"); in cs35l34_irq_thread()
915 /* error is no longer asserted; safe to reset */ in cs35l34_irq_thread()
917 dev_dbg(component->dev, in cs35l34_irq_thread()
919 regmap_update_bits(cs35l34->regmap, in cs35l34_irq_thread()
922 regmap_update_bits(cs35l34->regmap, in cs35l34_irq_thread()
926 regmap_update_bits(cs35l34->regmap, in cs35l34_irq_thread()
933 dev_crit(component->dev, "Over temperature warning\n"); in cs35l34_irq_thread()
935 /* error is no longer asserted; safe to reset */ in cs35l34_irq_thread()
937 dev_dbg(component->dev, in cs35l34_irq_thread()
939 regmap_update_bits(cs35l34->regmap, in cs35l34_irq_thread()
942 regmap_update_bits(cs35l34->regmap, in cs35l34_irq_thread()
946 regmap_update_bits(cs35l34->regmap, in cs35l34_irq_thread()
953 dev_crit(component->dev, "Over temperature error\n"); in cs35l34_irq_thread()
955 /* error is no longer asserted; safe to reset */ in cs35l34_irq_thread()
957 dev_dbg(component->dev, in cs35l34_irq_thread()
959 regmap_update_bits(cs35l34->regmap, in cs35l34_irq_thread()
962 regmap_update_bits(cs35l34->regmap, in cs35l34_irq_thread()
966 regmap_update_bits(cs35l34->regmap, in cs35l34_irq_thread()
973 dev_crit(component->dev, "VBST too high error; powering off!\n"); in cs35l34_irq_thread()
974 regmap_update_bits(cs35l34->regmap, CS35L34_PWRCTL2, in cs35l34_irq_thread()
976 regmap_update_bits(cs35l34->regmap, CS35L34_PWRCTL1, in cs35l34_irq_thread()
981 dev_crit(component->dev, "LBST short error; powering off!\n"); in cs35l34_irq_thread()
982 regmap_update_bits(cs35l34->regmap, CS35L34_PWRCTL2, in cs35l34_irq_thread()
984 regmap_update_bits(cs35l34->regmap, CS35L34_PWRCTL1, in cs35l34_irq_thread()
1000 dev_get_platdata(&i2c_client->dev); in cs35l34_i2c_probe()
1005 cs35l34 = devm_kzalloc(&i2c_client->dev, sizeof(*cs35l34), GFP_KERNEL); in cs35l34_i2c_probe()
1007 return -ENOMEM; in cs35l34_i2c_probe()
1010 cs35l34->regmap = devm_regmap_init_i2c(i2c_client, &cs35l34_regmap); in cs35l34_i2c_probe()
1011 if (IS_ERR(cs35l34->regmap)) { in cs35l34_i2c_probe()
1012 ret = PTR_ERR(cs35l34->regmap); in cs35l34_i2c_probe()
1013 dev_err(&i2c_client->dev, "regmap_init() failed: %d\n", ret); in cs35l34_i2c_probe()
1017 cs35l34->num_core_supplies = ARRAY_SIZE(cs35l34_core_supplies); in cs35l34_i2c_probe()
1019 cs35l34->core_supplies[i].supply = cs35l34_core_supplies[i]; in cs35l34_i2c_probe()
1021 ret = devm_regulator_bulk_get(&i2c_client->dev, in cs35l34_i2c_probe()
1022 cs35l34->num_core_supplies, in cs35l34_i2c_probe()
1023 cs35l34->core_supplies); in cs35l34_i2c_probe()
1025 dev_err(&i2c_client->dev, in cs35l34_i2c_probe()
1030 ret = regulator_bulk_enable(cs35l34->num_core_supplies, in cs35l34_i2c_probe()
1031 cs35l34->core_supplies); in cs35l34_i2c_probe()
1033 dev_err(&i2c_client->dev, in cs35l34_i2c_probe()
1039 cs35l34->pdata = *pdata; in cs35l34_i2c_probe()
1041 pdata = devm_kzalloc(&i2c_client->dev, sizeof(*pdata), in cs35l34_i2c_probe()
1044 ret = -ENOMEM; in cs35l34_i2c_probe()
1048 if (i2c_client->dev.of_node) { in cs35l34_i2c_probe()
1054 cs35l34->pdata = *pdata; in cs35l34_i2c_probe()
1057 ret = devm_request_threaded_irq(&i2c_client->dev, i2c_client->irq, NULL, in cs35l34_i2c_probe()
1061 dev_err(&i2c_client->dev, "Failed to request IRQ: %d\n", ret); in cs35l34_i2c_probe()
1063 cs35l34->reset_gpio = devm_gpiod_get_optional(&i2c_client->dev, in cs35l34_i2c_probe()
1064 "reset-gpios", GPIOD_OUT_LOW); in cs35l34_i2c_probe()
1065 if (IS_ERR(cs35l34->reset_gpio)) { in cs35l34_i2c_probe()
1066 ret = PTR_ERR(cs35l34->reset_gpio); in cs35l34_i2c_probe()
1070 gpiod_set_value_cansleep(cs35l34->reset_gpio, 1); in cs35l34_i2c_probe()
1074 devid = cirrus_read_device_id(cs35l34->regmap, CS35L34_DEVID_AB); in cs35l34_i2c_probe()
1077 dev_err(&i2c_client->dev, "Failed to read device ID: %d\n", ret); in cs35l34_i2c_probe()
1082 dev_err(&i2c_client->dev, in cs35l34_i2c_probe()
1085 ret = -ENODEV; in cs35l34_i2c_probe()
1089 ret = regmap_read(cs35l34->regmap, CS35L34_REV_ID, ®); in cs35l34_i2c_probe()
1091 dev_err(&i2c_client->dev, "Get Revision ID failed\n"); in cs35l34_i2c_probe()
1095 dev_info(&i2c_client->dev, in cs35l34_i2c_probe()
1100 regmap_update_bits(cs35l34->regmap, CS35L34_INT_MASK_1, in cs35l34_i2c_probe()
1104 regmap_update_bits(cs35l34->regmap, CS35L34_INT_MASK_3, in cs35l34_i2c_probe()
1107 pm_runtime_set_autosuspend_delay(&i2c_client->dev, 100); in cs35l34_i2c_probe()
1108 pm_runtime_use_autosuspend(&i2c_client->dev); in cs35l34_i2c_probe()
1109 pm_runtime_set_active(&i2c_client->dev); in cs35l34_i2c_probe()
1110 pm_runtime_enable(&i2c_client->dev); in cs35l34_i2c_probe()
1112 ret = devm_snd_soc_register_component(&i2c_client->dev, in cs35l34_i2c_probe()
1115 dev_err(&i2c_client->dev, in cs35l34_i2c_probe()
1123 gpiod_set_value_cansleep(cs35l34->reset_gpio, 0); in cs35l34_i2c_probe()
1125 regulator_bulk_disable(cs35l34->num_core_supplies, in cs35l34_i2c_probe()
1126 cs35l34->core_supplies); in cs35l34_i2c_probe()
1135 gpiod_set_value_cansleep(cs35l34->reset_gpio, 0); in cs35l34_i2c_remove()
1137 pm_runtime_disable(&client->dev); in cs35l34_i2c_remove()
1138 regulator_bulk_disable(cs35l34->num_core_supplies, in cs35l34_i2c_remove()
1139 cs35l34->core_supplies); in cs35l34_i2c_remove()
1147 ret = regulator_bulk_enable(cs35l34->num_core_supplies, in cs35l34_runtime_resume()
1148 cs35l34->core_supplies); in cs35l34_runtime_resume()
1156 regcache_cache_only(cs35l34->regmap, false); in cs35l34_runtime_resume()
1158 gpiod_set_value_cansleep(cs35l34->reset_gpio, 1); in cs35l34_runtime_resume()
1161 ret = regcache_sync(cs35l34->regmap); in cs35l34_runtime_resume()
1168 regcache_cache_only(cs35l34->regmap, true); in cs35l34_runtime_resume()
1169 regulator_bulk_disable(cs35l34->num_core_supplies, in cs35l34_runtime_resume()
1170 cs35l34->core_supplies); in cs35l34_runtime_resume()
1179 regcache_cache_only(cs35l34->regmap, true); in cs35l34_runtime_suspend()
1180 regcache_mark_dirty(cs35l34->regmap); in cs35l34_runtime_suspend()
1182 gpiod_set_value_cansleep(cs35l34->reset_gpio, 0); in cs35l34_runtime_suspend()
1184 regulator_bulk_disable(cs35l34->num_core_supplies, in cs35l34_runtime_suspend()
1185 cs35l34->core_supplies); in cs35l34_runtime_suspend()