2019-05-29 17:17:58 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2015-02-05 23:53:40 +03:00
/* Copyright (c) 2010-2011,2013-2015 The Linux Foundation. All rights reserved.
*
* max98357a . c - - MAX98357A ALSA SoC Codec driver
*/
2015-12-11 22:29:06 +03:00
# include <linux/acpi.h>
2020-02-12 08:55:16 +03:00
# include <linux/delay.h>
2015-02-17 11:53:11 +03:00
# include <linux/device.h>
# include <linux/err.h>
2015-02-05 23:53:40 +03:00
# include <linux/gpio.h>
2015-02-12 01:08:59 +03:00
# include <linux/gpio/consumer.h>
2015-02-17 11:53:11 +03:00
# include <linux/kernel.h>
# include <linux/mod_devicetable.h>
# include <linux/module.h>
# include <linux/of.h>
# include <linux/platform_device.h>
# include <sound/pcm.h>
2015-02-05 23:53:40 +03:00
# include <sound/soc.h>
2015-02-17 11:53:11 +03:00
# include <sound/soc-dai.h>
# include <sound/soc-dapm.h>
2015-02-05 23:53:40 +03:00
2019-06-19 13:18:33 +03:00
struct max98357a_priv {
struct gpio_desc * sdmode ;
unsigned int sdmode_delay ;
2020-07-21 14:42:32 +03:00
int sdmode_switch ;
2019-06-19 13:18:33 +03:00
} ;
2020-07-21 14:42:32 +03:00
static int max98357a_daiops_trigger ( struct snd_pcm_substream * substream ,
int cmd , struct snd_soc_dai * dai )
2015-02-05 23:53:40 +03:00
{
2020-07-21 14:42:32 +03:00
struct snd_soc_component * component = dai - > component ;
2020-02-12 08:55:16 +03:00
struct max98357a_priv * max98357a =
snd_soc_component_get_drvdata ( component ) ;
2015-02-05 23:53:40 +03:00
2019-06-19 13:18:33 +03:00
if ( ! max98357a - > sdmode )
2015-07-12 18:14:19 +03:00
return 0 ;
2020-07-21 14:42:32 +03:00
switch ( cmd ) {
case SNDRV_PCM_TRIGGER_START :
case SNDRV_PCM_TRIGGER_RESUME :
case SNDRV_PCM_TRIGGER_PAUSE_RELEASE :
mdelay ( max98357a - > sdmode_delay ) ;
if ( max98357a - > sdmode_switch ) {
gpiod_set_value ( max98357a - > sdmode , 1 ) ;
dev_dbg ( component - > dev , " set sdmode to 1 " ) ;
}
break ;
case SNDRV_PCM_TRIGGER_STOP :
case SNDRV_PCM_TRIGGER_SUSPEND :
case SNDRV_PCM_TRIGGER_PAUSE_PUSH :
2019-06-19 13:18:33 +03:00
gpiod_set_value ( max98357a - > sdmode , 0 ) ;
2020-02-12 08:55:16 +03:00
dev_dbg ( component - > dev , " set sdmode to 0 " ) ;
2020-07-21 14:42:32 +03:00
break ;
2015-02-05 23:53:40 +03:00
}
return 0 ;
}
2020-07-21 14:42:32 +03:00
static int max98357a_sdmode_event ( struct snd_soc_dapm_widget * w ,
struct snd_kcontrol * kcontrol , int event )
{
struct snd_soc_component * component =
snd_soc_dapm_to_component ( w - > dapm ) ;
struct max98357a_priv * max98357a =
snd_soc_component_get_drvdata ( component ) ;
if ( event & SND_SOC_DAPM_POST_PMU )
max98357a - > sdmode_switch = 1 ;
else if ( event & SND_SOC_DAPM_POST_PMD )
max98357a - > sdmode_switch = 0 ;
return 0 ;
}
2015-02-05 23:53:40 +03:00
static const struct snd_soc_dapm_widget max98357a_dapm_widgets [ ] = {
SND_SOC_DAPM_OUTPUT ( " Speaker " ) ,
2020-02-12 08:55:16 +03:00
SND_SOC_DAPM_OUT_DRV_E ( " SD_MODE " , SND_SOC_NOPM , 0 , 0 , NULL , 0 ,
max98357a_sdmode_event ,
2020-07-21 14:42:32 +03:00
SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD ) ,
2015-02-05 23:53:40 +03:00
} ;
static const struct snd_soc_dapm_route max98357a_dapm_routes [ ] = {
2020-02-12 08:55:16 +03:00
{ " SD_MODE " , NULL , " HiFi Playback " } ,
{ " Speaker " , NULL , " SD_MODE " } ,
2015-02-05 23:53:40 +03:00
} ;
2018-01-29 07:09:20 +03:00
static const struct snd_soc_component_driver max98357a_component_driver = {
. dapm_widgets = max98357a_dapm_widgets ,
. num_dapm_widgets = ARRAY_SIZE ( max98357a_dapm_widgets ) ,
. dapm_routes = max98357a_dapm_routes ,
. num_dapm_routes = ARRAY_SIZE ( max98357a_dapm_routes ) ,
. idle_bias_on = 1 ,
. use_pmdown_time = 1 ,
. endianness = 1 ,
2015-02-05 23:53:40 +03:00
} ;
2020-07-21 14:42:32 +03:00
static const struct snd_soc_dai_ops max98357a_dai_ops = {
. trigger = max98357a_daiops_trigger ,
} ;
2015-02-05 23:53:40 +03:00
static struct snd_soc_dai_driver max98357a_dai_driver = {
2015-02-25 09:39:04 +03:00
. name = " HiFi " ,
2015-02-05 23:53:40 +03:00
. playback = {
2015-02-25 09:39:04 +03:00
. stream_name = " HiFi Playback " ,
2015-02-05 23:53:40 +03:00
. formats = SNDRV_PCM_FMTBIT_S16 |
SNDRV_PCM_FMTBIT_S24 |
SNDRV_PCM_FMTBIT_S32 ,
. rates = SNDRV_PCM_RATE_8000 |
SNDRV_PCM_RATE_16000 |
2019-04-04 14:50:15 +03:00
SNDRV_PCM_RATE_32000 |
SNDRV_PCM_RATE_44100 |
2015-02-05 23:53:40 +03:00
SNDRV_PCM_RATE_48000 |
2019-04-04 14:50:15 +03:00
SNDRV_PCM_RATE_88200 |
2015-02-05 23:53:40 +03:00
SNDRV_PCM_RATE_96000 ,
. rate_min = 8000 ,
. rate_max = 96000 ,
. channels_min = 1 ,
. channels_max = 2 ,
} ,
2020-07-21 14:42:32 +03:00
. ops = & max98357a_dai_ops ,
2015-02-05 23:53:40 +03:00
} ;
static int max98357a_platform_probe ( struct platform_device * pdev )
{
2019-06-19 13:18:33 +03:00
struct max98357a_priv * max98357a ;
int ret ;
2019-05-09 06:04:54 +03:00
2019-06-19 13:18:33 +03:00
max98357a = devm_kzalloc ( & pdev - > dev , sizeof ( * max98357a ) , GFP_KERNEL ) ;
if ( ! max98357a )
return - ENOMEM ;
max98357a - > sdmode = devm_gpiod_get_optional ( & pdev - > dev ,
2019-05-09 06:04:54 +03:00
" sdmode " , GPIOD_OUT_LOW ) ;
2019-06-19 13:18:33 +03:00
if ( IS_ERR ( max98357a - > sdmode ) )
return PTR_ERR ( max98357a - > sdmode ) ;
ret = device_property_read_u32 ( & pdev - > dev , " sdmode-delay " ,
& max98357a - > sdmode_delay ) ;
if ( ret ) {
max98357a - > sdmode_delay = 0 ;
dev_dbg ( & pdev - > dev ,
2019-07-08 17:19:01 +03:00
" no optional property 'sdmode-delay' found, "
" default: no delay \n " ) ;
2019-06-19 13:18:33 +03:00
}
dev_set_drvdata ( & pdev - > dev , max98357a ) ;
2018-01-29 07:09:20 +03:00
return devm_snd_soc_register_component ( & pdev - > dev ,
& max98357a_component_driver ,
2015-02-05 23:53:40 +03:00
& max98357a_dai_driver , 1 ) ;
}
# ifdef CONFIG_OF
static const struct of_device_id max98357a_device_id [ ] = {
2015-02-17 11:53:12 +03:00
{ . compatible = " maxim,max98357a " } ,
2020-06-05 06:49:30 +03:00
{ . compatible = " maxim,max98360a " } ,
2015-02-05 23:53:40 +03:00
{ }
} ;
2015-02-09 09:36:47 +03:00
MODULE_DEVICE_TABLE ( of , max98357a_device_id ) ;
2015-02-05 23:53:40 +03:00
# endif
2015-12-11 22:29:06 +03:00
# ifdef CONFIG_ACPI
static const struct acpi_device_id max98357a_acpi_match [ ] = {
{ " MX98357A " , 0 } ,
2020-03-13 18:55:26 +03:00
{ " MX98360A " , 0 } ,
2015-12-11 22:29:06 +03:00
{ } ,
} ;
MODULE_DEVICE_TABLE ( acpi , max98357a_acpi_match ) ;
# endif
2015-02-05 23:53:40 +03:00
static struct platform_driver max98357a_platform_driver = {
. driver = {
2015-02-17 11:53:12 +03:00
. name = " max98357a " ,
2015-02-05 23:53:40 +03:00
. of_match_table = of_match_ptr ( max98357a_device_id ) ,
2015-12-11 22:29:06 +03:00
. acpi_match_table = ACPI_PTR ( max98357a_acpi_match ) ,
2015-02-05 23:53:40 +03:00
} ,
. probe = max98357a_platform_probe ,
} ;
module_platform_driver ( max98357a_platform_driver ) ;
MODULE_DESCRIPTION ( " Maxim MAX98357A Codec Driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;