18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 28c2ecf20Sopenharmony_ci// 38c2ecf20Sopenharmony_ci// ROHM BD28623MUV class D speaker amplifier codec driver. 48c2ecf20Sopenharmony_ci// 58c2ecf20Sopenharmony_ci// Copyright (c) 2018 Socionext Inc. 68c2ecf20Sopenharmony_ci 78c2ecf20Sopenharmony_ci#include <linux/delay.h> 88c2ecf20Sopenharmony_ci#include <linux/gpio/consumer.h> 98c2ecf20Sopenharmony_ci#include <linux/module.h> 108c2ecf20Sopenharmony_ci#include <linux/of.h> 118c2ecf20Sopenharmony_ci#include <linux/regulator/consumer.h> 128c2ecf20Sopenharmony_ci#include <sound/pcm.h> 138c2ecf20Sopenharmony_ci#include <sound/soc.h> 148c2ecf20Sopenharmony_ci 158c2ecf20Sopenharmony_ci#define BD28623_NUM_SUPPLIES 3 168c2ecf20Sopenharmony_ci 178c2ecf20Sopenharmony_cistatic const char *const bd28623_supply_names[BD28623_NUM_SUPPLIES] = { 188c2ecf20Sopenharmony_ci "VCCA", 198c2ecf20Sopenharmony_ci "VCCP1", 208c2ecf20Sopenharmony_ci "VCCP2", 218c2ecf20Sopenharmony_ci}; 228c2ecf20Sopenharmony_ci 238c2ecf20Sopenharmony_cistruct bd28623_priv { 248c2ecf20Sopenharmony_ci struct device *dev; 258c2ecf20Sopenharmony_ci struct regulator_bulk_data supplies[BD28623_NUM_SUPPLIES]; 268c2ecf20Sopenharmony_ci struct gpio_desc *reset_gpio; 278c2ecf20Sopenharmony_ci struct gpio_desc *mute_gpio; 288c2ecf20Sopenharmony_ci 298c2ecf20Sopenharmony_ci int switch_spk; 308c2ecf20Sopenharmony_ci}; 318c2ecf20Sopenharmony_ci 328c2ecf20Sopenharmony_cistatic const struct snd_soc_dapm_widget bd28623_widgets[] = { 338c2ecf20Sopenharmony_ci SND_SOC_DAPM_DAC("DAC", "Playback", SND_SOC_NOPM, 0, 0), 348c2ecf20Sopenharmony_ci SND_SOC_DAPM_OUTPUT("OUT1P"), 358c2ecf20Sopenharmony_ci SND_SOC_DAPM_OUTPUT("OUT1N"), 368c2ecf20Sopenharmony_ci SND_SOC_DAPM_OUTPUT("OUT2P"), 378c2ecf20Sopenharmony_ci SND_SOC_DAPM_OUTPUT("OUT2N"), 388c2ecf20Sopenharmony_ci}; 398c2ecf20Sopenharmony_ci 408c2ecf20Sopenharmony_cistatic const struct snd_soc_dapm_route bd28623_routes[] = { 418c2ecf20Sopenharmony_ci { "OUT1P", NULL, "DAC" }, 428c2ecf20Sopenharmony_ci { "OUT1N", NULL, "DAC" }, 438c2ecf20Sopenharmony_ci { "OUT2P", NULL, "DAC" }, 448c2ecf20Sopenharmony_ci { "OUT2N", NULL, "DAC" }, 458c2ecf20Sopenharmony_ci}; 468c2ecf20Sopenharmony_ci 478c2ecf20Sopenharmony_cistatic int bd28623_power_on(struct bd28623_priv *bd) 488c2ecf20Sopenharmony_ci{ 498c2ecf20Sopenharmony_ci int ret; 508c2ecf20Sopenharmony_ci 518c2ecf20Sopenharmony_ci ret = regulator_bulk_enable(ARRAY_SIZE(bd->supplies), bd->supplies); 528c2ecf20Sopenharmony_ci if (ret) { 538c2ecf20Sopenharmony_ci dev_err(bd->dev, "Failed to enable supplies: %d\n", ret); 548c2ecf20Sopenharmony_ci return ret; 558c2ecf20Sopenharmony_ci } 568c2ecf20Sopenharmony_ci 578c2ecf20Sopenharmony_ci gpiod_set_value_cansleep(bd->reset_gpio, 0); 588c2ecf20Sopenharmony_ci usleep_range(300000, 400000); 598c2ecf20Sopenharmony_ci 608c2ecf20Sopenharmony_ci return 0; 618c2ecf20Sopenharmony_ci} 628c2ecf20Sopenharmony_ci 638c2ecf20Sopenharmony_cistatic void bd28623_power_off(struct bd28623_priv *bd) 648c2ecf20Sopenharmony_ci{ 658c2ecf20Sopenharmony_ci gpiod_set_value_cansleep(bd->reset_gpio, 1); 668c2ecf20Sopenharmony_ci 678c2ecf20Sopenharmony_ci regulator_bulk_disable(ARRAY_SIZE(bd->supplies), bd->supplies); 688c2ecf20Sopenharmony_ci} 698c2ecf20Sopenharmony_ci 708c2ecf20Sopenharmony_cistatic int bd28623_get_switch_spk(struct snd_kcontrol *kcontrol, 718c2ecf20Sopenharmony_ci struct snd_ctl_elem_value *ucontrol) 728c2ecf20Sopenharmony_ci{ 738c2ecf20Sopenharmony_ci struct snd_soc_component *component = 748c2ecf20Sopenharmony_ci snd_soc_kcontrol_component(kcontrol); 758c2ecf20Sopenharmony_ci struct bd28623_priv *bd = snd_soc_component_get_drvdata(component); 768c2ecf20Sopenharmony_ci 778c2ecf20Sopenharmony_ci ucontrol->value.integer.value[0] = bd->switch_spk; 788c2ecf20Sopenharmony_ci 798c2ecf20Sopenharmony_ci return 0; 808c2ecf20Sopenharmony_ci} 818c2ecf20Sopenharmony_ci 828c2ecf20Sopenharmony_cistatic int bd28623_set_switch_spk(struct snd_kcontrol *kcontrol, 838c2ecf20Sopenharmony_ci struct snd_ctl_elem_value *ucontrol) 848c2ecf20Sopenharmony_ci{ 858c2ecf20Sopenharmony_ci struct snd_soc_component *component = 868c2ecf20Sopenharmony_ci snd_soc_kcontrol_component(kcontrol); 878c2ecf20Sopenharmony_ci struct bd28623_priv *bd = snd_soc_component_get_drvdata(component); 888c2ecf20Sopenharmony_ci 898c2ecf20Sopenharmony_ci if (bd->switch_spk == ucontrol->value.integer.value[0]) 908c2ecf20Sopenharmony_ci return 0; 918c2ecf20Sopenharmony_ci 928c2ecf20Sopenharmony_ci bd->switch_spk = ucontrol->value.integer.value[0]; 938c2ecf20Sopenharmony_ci 948c2ecf20Sopenharmony_ci gpiod_set_value_cansleep(bd->mute_gpio, bd->switch_spk ? 0 : 1); 958c2ecf20Sopenharmony_ci 968c2ecf20Sopenharmony_ci return 0; 978c2ecf20Sopenharmony_ci} 988c2ecf20Sopenharmony_ci 998c2ecf20Sopenharmony_cistatic const struct snd_kcontrol_new bd28623_controls[] = { 1008c2ecf20Sopenharmony_ci SOC_SINGLE_BOOL_EXT("Speaker Switch", 0, 1018c2ecf20Sopenharmony_ci bd28623_get_switch_spk, bd28623_set_switch_spk), 1028c2ecf20Sopenharmony_ci}; 1038c2ecf20Sopenharmony_ci 1048c2ecf20Sopenharmony_cistatic int bd28623_codec_probe(struct snd_soc_component *component) 1058c2ecf20Sopenharmony_ci{ 1068c2ecf20Sopenharmony_ci struct bd28623_priv *bd = snd_soc_component_get_drvdata(component); 1078c2ecf20Sopenharmony_ci int ret; 1088c2ecf20Sopenharmony_ci 1098c2ecf20Sopenharmony_ci bd->switch_spk = 1; 1108c2ecf20Sopenharmony_ci 1118c2ecf20Sopenharmony_ci ret = bd28623_power_on(bd); 1128c2ecf20Sopenharmony_ci if (ret) 1138c2ecf20Sopenharmony_ci return ret; 1148c2ecf20Sopenharmony_ci 1158c2ecf20Sopenharmony_ci gpiod_set_value_cansleep(bd->mute_gpio, bd->switch_spk ? 0 : 1); 1168c2ecf20Sopenharmony_ci 1178c2ecf20Sopenharmony_ci return 0; 1188c2ecf20Sopenharmony_ci} 1198c2ecf20Sopenharmony_ci 1208c2ecf20Sopenharmony_cistatic void bd28623_codec_remove(struct snd_soc_component *component) 1218c2ecf20Sopenharmony_ci{ 1228c2ecf20Sopenharmony_ci struct bd28623_priv *bd = snd_soc_component_get_drvdata(component); 1238c2ecf20Sopenharmony_ci 1248c2ecf20Sopenharmony_ci bd28623_power_off(bd); 1258c2ecf20Sopenharmony_ci} 1268c2ecf20Sopenharmony_ci 1278c2ecf20Sopenharmony_cistatic int bd28623_codec_suspend(struct snd_soc_component *component) 1288c2ecf20Sopenharmony_ci{ 1298c2ecf20Sopenharmony_ci struct bd28623_priv *bd = snd_soc_component_get_drvdata(component); 1308c2ecf20Sopenharmony_ci 1318c2ecf20Sopenharmony_ci bd28623_power_off(bd); 1328c2ecf20Sopenharmony_ci 1338c2ecf20Sopenharmony_ci return 0; 1348c2ecf20Sopenharmony_ci} 1358c2ecf20Sopenharmony_ci 1368c2ecf20Sopenharmony_cistatic int bd28623_codec_resume(struct snd_soc_component *component) 1378c2ecf20Sopenharmony_ci{ 1388c2ecf20Sopenharmony_ci struct bd28623_priv *bd = snd_soc_component_get_drvdata(component); 1398c2ecf20Sopenharmony_ci int ret; 1408c2ecf20Sopenharmony_ci 1418c2ecf20Sopenharmony_ci ret = bd28623_power_on(bd); 1428c2ecf20Sopenharmony_ci if (ret) 1438c2ecf20Sopenharmony_ci return ret; 1448c2ecf20Sopenharmony_ci 1458c2ecf20Sopenharmony_ci gpiod_set_value_cansleep(bd->mute_gpio, bd->switch_spk ? 0 : 1); 1468c2ecf20Sopenharmony_ci 1478c2ecf20Sopenharmony_ci return 0; 1488c2ecf20Sopenharmony_ci} 1498c2ecf20Sopenharmony_ci 1508c2ecf20Sopenharmony_cistatic const struct snd_soc_component_driver soc_codec_bd = { 1518c2ecf20Sopenharmony_ci .probe = bd28623_codec_probe, 1528c2ecf20Sopenharmony_ci .remove = bd28623_codec_remove, 1538c2ecf20Sopenharmony_ci .suspend = bd28623_codec_suspend, 1548c2ecf20Sopenharmony_ci .resume = bd28623_codec_resume, 1558c2ecf20Sopenharmony_ci .dapm_widgets = bd28623_widgets, 1568c2ecf20Sopenharmony_ci .num_dapm_widgets = ARRAY_SIZE(bd28623_widgets), 1578c2ecf20Sopenharmony_ci .dapm_routes = bd28623_routes, 1588c2ecf20Sopenharmony_ci .num_dapm_routes = ARRAY_SIZE(bd28623_routes), 1598c2ecf20Sopenharmony_ci .controls = bd28623_controls, 1608c2ecf20Sopenharmony_ci .num_controls = ARRAY_SIZE(bd28623_controls), 1618c2ecf20Sopenharmony_ci .idle_bias_on = 1, 1628c2ecf20Sopenharmony_ci .use_pmdown_time = 1, 1638c2ecf20Sopenharmony_ci .endianness = 1, 1648c2ecf20Sopenharmony_ci .non_legacy_dai_naming = 1, 1658c2ecf20Sopenharmony_ci}; 1668c2ecf20Sopenharmony_ci 1678c2ecf20Sopenharmony_cistatic struct snd_soc_dai_driver soc_dai_bd = { 1688c2ecf20Sopenharmony_ci .name = "bd28623-speaker", 1698c2ecf20Sopenharmony_ci .playback = { 1708c2ecf20Sopenharmony_ci .stream_name = "Playback", 1718c2ecf20Sopenharmony_ci .formats = SNDRV_PCM_FMTBIT_S32_LE | 1728c2ecf20Sopenharmony_ci SNDRV_PCM_FMTBIT_S24_LE | 1738c2ecf20Sopenharmony_ci SNDRV_PCM_FMTBIT_S16_LE, 1748c2ecf20Sopenharmony_ci .rates = SNDRV_PCM_RATE_48000 | 1758c2ecf20Sopenharmony_ci SNDRV_PCM_RATE_44100 | 1768c2ecf20Sopenharmony_ci SNDRV_PCM_RATE_32000, 1778c2ecf20Sopenharmony_ci .channels_min = 2, 1788c2ecf20Sopenharmony_ci .channels_max = 2, 1798c2ecf20Sopenharmony_ci }, 1808c2ecf20Sopenharmony_ci}; 1818c2ecf20Sopenharmony_ci 1828c2ecf20Sopenharmony_cistatic int bd28623_probe(struct platform_device *pdev) 1838c2ecf20Sopenharmony_ci{ 1848c2ecf20Sopenharmony_ci struct bd28623_priv *bd; 1858c2ecf20Sopenharmony_ci struct device *dev = &pdev->dev; 1868c2ecf20Sopenharmony_ci int i, ret; 1878c2ecf20Sopenharmony_ci 1888c2ecf20Sopenharmony_ci bd = devm_kzalloc(&pdev->dev, sizeof(struct bd28623_priv), GFP_KERNEL); 1898c2ecf20Sopenharmony_ci if (!bd) 1908c2ecf20Sopenharmony_ci return -ENOMEM; 1918c2ecf20Sopenharmony_ci 1928c2ecf20Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(bd->supplies); i++) 1938c2ecf20Sopenharmony_ci bd->supplies[i].supply = bd28623_supply_names[i]; 1948c2ecf20Sopenharmony_ci 1958c2ecf20Sopenharmony_ci ret = devm_regulator_bulk_get(dev, ARRAY_SIZE(bd->supplies), 1968c2ecf20Sopenharmony_ci bd->supplies); 1978c2ecf20Sopenharmony_ci if (ret) { 1988c2ecf20Sopenharmony_ci dev_err(dev, "Failed to get supplies: %d\n", ret); 1998c2ecf20Sopenharmony_ci return ret; 2008c2ecf20Sopenharmony_ci } 2018c2ecf20Sopenharmony_ci 2028c2ecf20Sopenharmony_ci bd->reset_gpio = devm_gpiod_get_optional(dev, "reset", 2038c2ecf20Sopenharmony_ci GPIOD_OUT_HIGH); 2048c2ecf20Sopenharmony_ci if (IS_ERR(bd->reset_gpio)) { 2058c2ecf20Sopenharmony_ci dev_err(dev, "Failed to request reset_gpio: %ld\n", 2068c2ecf20Sopenharmony_ci PTR_ERR(bd->reset_gpio)); 2078c2ecf20Sopenharmony_ci return PTR_ERR(bd->reset_gpio); 2088c2ecf20Sopenharmony_ci } 2098c2ecf20Sopenharmony_ci 2108c2ecf20Sopenharmony_ci bd->mute_gpio = devm_gpiod_get_optional(dev, "mute", 2118c2ecf20Sopenharmony_ci GPIOD_OUT_HIGH); 2128c2ecf20Sopenharmony_ci if (IS_ERR(bd->mute_gpio)) { 2138c2ecf20Sopenharmony_ci dev_err(dev, "Failed to request mute_gpio: %ld\n", 2148c2ecf20Sopenharmony_ci PTR_ERR(bd->mute_gpio)); 2158c2ecf20Sopenharmony_ci return PTR_ERR(bd->mute_gpio); 2168c2ecf20Sopenharmony_ci } 2178c2ecf20Sopenharmony_ci 2188c2ecf20Sopenharmony_ci platform_set_drvdata(pdev, bd); 2198c2ecf20Sopenharmony_ci bd->dev = dev; 2208c2ecf20Sopenharmony_ci 2218c2ecf20Sopenharmony_ci return devm_snd_soc_register_component(dev, &soc_codec_bd, 2228c2ecf20Sopenharmony_ci &soc_dai_bd, 1); 2238c2ecf20Sopenharmony_ci} 2248c2ecf20Sopenharmony_ci 2258c2ecf20Sopenharmony_cistatic const struct of_device_id bd28623_of_match[] = { 2268c2ecf20Sopenharmony_ci { .compatible = "rohm,bd28623", }, 2278c2ecf20Sopenharmony_ci {} 2288c2ecf20Sopenharmony_ci}; 2298c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(of, bd28623_of_match); 2308c2ecf20Sopenharmony_ci 2318c2ecf20Sopenharmony_cistatic struct platform_driver bd28623_codec_driver = { 2328c2ecf20Sopenharmony_ci .driver = { 2338c2ecf20Sopenharmony_ci .name = "bd28623", 2348c2ecf20Sopenharmony_ci .of_match_table = of_match_ptr(bd28623_of_match), 2358c2ecf20Sopenharmony_ci }, 2368c2ecf20Sopenharmony_ci .probe = bd28623_probe, 2378c2ecf20Sopenharmony_ci}; 2388c2ecf20Sopenharmony_cimodule_platform_driver(bd28623_codec_driver); 2398c2ecf20Sopenharmony_ci 2408c2ecf20Sopenharmony_ciMODULE_AUTHOR("Katsuhiro Suzuki <suzuki.katsuhiro@socionext.com>"); 2418c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("ROHM BD28623 speaker amplifier driver"); 2428c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL v2"); 243