2019-06-04 11:11:33 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2012-06-19 19:36:18 +04:00
/*
* arizona - spi . c - - Arizona SPI bus interface
*
* Copyright 2012 Wolfson Microelectronics plc
*
* Author : Mark Brown < broonie @ opensource . wolfsonmicro . com >
*/
2021-01-21 00:49:55 +03:00
# include <linux/acpi.h>
2012-06-19 19:36:18 +04:00
# include <linux/err.h>
2021-01-21 00:49:55 +03:00
# include <linux/gpio/consumer.h>
# include <linux/gpio/machine.h>
2012-06-19 19:36:18 +04:00
# include <linux/module.h>
# include <linux/pm_runtime.h>
# include <linux/regmap.h>
# include <linux/regulator/consumer.h>
# include <linux/slab.h>
# include <linux/spi/spi.h>
2013-10-16 12:56:56 +04:00
# include <linux/of.h>
2021-01-21 00:49:55 +03:00
# include <uapi/linux/input-event-codes.h>
2012-06-19 19:36:18 +04:00
# include <linux/mfd/arizona/core.h>
# include "arizona.h"
2021-01-21 00:49:55 +03:00
# ifdef CONFIG_ACPI
2021-02-10 10:56:26 +03:00
static const struct acpi_gpio_params reset_gpios = { 1 , 0 , false } ;
static const struct acpi_gpio_params ldoena_gpios = { 2 , 0 , false } ;
2021-01-21 00:49:55 +03:00
static const struct acpi_gpio_mapping arizona_acpi_gpios [ ] = {
{ " reset-gpios " , & reset_gpios , 1 , } ,
{ " wlf,ldoena-gpios " , & ldoena_gpios , 1 } ,
{ }
} ;
/*
* The ACPI resources for the device only describe external GPIO - s . They do
* not provide mappings for the GPIO - s coming from the Arizona codec itself .
*/
static const struct gpiod_lookup arizona_soc_gpios [ ] = {
{ " arizona " , 2 , " wlf,spkvdd-ena " , 0 , GPIO_ACTIVE_HIGH } ,
{ " arizona " , 4 , " wlf,micd-pol " , 0 , GPIO_ACTIVE_LOW } ,
} ;
static void arizona_spi_acpi_remove_lookup ( void * lookup )
{
gpiod_remove_lookup_table ( lookup ) ;
}
2022-03-07 20:38:43 +03:00
/* For ACPI tables from boards which ship with Windows as factory OS */
static int arizona_spi_acpi_windows_probe ( struct arizona * arizona )
2021-01-21 00:49:55 +03:00
{
struct gpiod_lookup_table * lookup ;
acpi_status status ;
int ret ;
/* Add mappings for the 2 ACPI declared GPIOs used for reset and ldo-ena */
devm_acpi_dev_add_driver_gpios ( arizona - > dev , arizona_acpi_gpios ) ;
/* Add lookups for the SoCs own GPIOs used for micdet-polarity and spkVDD-enable */
lookup = devm_kzalloc ( arizona - > dev ,
struct_size ( lookup , table , ARRAY_SIZE ( arizona_soc_gpios ) + 1 ) ,
GFP_KERNEL ) ;
if ( ! lookup )
return - ENOMEM ;
lookup - > dev_id = dev_name ( arizona - > dev ) ;
memcpy ( lookup - > table , arizona_soc_gpios , sizeof ( arizona_soc_gpios ) ) ;
gpiod_add_lookup_table ( lookup ) ;
ret = devm_add_action_or_reset ( arizona - > dev , arizona_spi_acpi_remove_lookup , lookup ) ;
if ( ret )
return ret ;
/* Enable 32KHz clock from SoC to codec for jack-detect */
status = acpi_evaluate_object ( ACPI_HANDLE ( arizona - > dev ) , " CLKE " , NULL , NULL ) ;
if ( ACPI_FAILURE ( status ) )
dev_warn ( arizona - > dev , " Failed to enable 32KHz clk ACPI error %d \n " , status ) ;
2022-03-07 20:38:43 +03:00
return 0 ;
}
2022-03-07 20:38:44 +03:00
/* For ACPI tables from boards which ship with Android as factory OS */
static int arizona_spi_acpi_android_probe ( struct arizona * arizona )
{
int ret ;
/*
* Get the reset GPIO , treating - ENOENT as - EPROBE_DEFER to wait for
* the x86 - android - tablets module to register the board specific GPIO
* lookup table .
*/
arizona - > pdata . reset = devm_gpiod_get ( arizona - > dev , " reset " , GPIOD_OUT_LOW ) ;
if ( IS_ERR ( arizona - > pdata . reset ) ) {
ret = PTR_ERR ( arizona - > pdata . reset ) ;
if ( ret = = - ENOENT ) {
dev_info_once ( arizona - > dev ,
" Deferring probe till GPIO lookup is registered \n " ) ;
ret = - EPROBE_DEFER ;
}
return dev_err_probe ( arizona - > dev , ret , " getting reset GPIO \n " ) ;
}
return 0 ;
}
2022-03-07 20:38:43 +03:00
/*
* The AOSP 3.5 mm Headset : Accessory Specification gives the following values :
* Function A Play / Pause : 0 ohm
* Function D Voice assistant : 135 ohm
* Function B Volume Up 240 ohm
* Function C Volume Down 470 ohm
* Minimum Mic DC resistance 1000 ohm
* Minimum Ear speaker impedance 16 ohm
* Note the first max value below must be less then the min . speaker impedance ,
* to allow CTIA / OMTP detection to work . The other max values are the closest
* value from extcon - arizona . c : arizona_micd_levels halfway 2 button resistances .
*/
static const struct arizona_micd_range arizona_micd_aosp_ranges [ ] = {
{ . max = 11 , . key = KEY_PLAYPAUSE } ,
{ . max = 186 , . key = KEY_VOICECOMMAND } ,
{ . max = 348 , . key = KEY_VOLUMEUP } ,
{ . max = 752 , . key = KEY_VOLUMEDOWN } ,
} ;
static int arizona_spi_acpi_probe ( struct arizona * arizona )
{
2022-03-07 20:38:44 +03:00
struct acpi_device * adev = ACPI_COMPANION ( arizona - > dev ) ;
2022-03-07 20:38:43 +03:00
int ret ;
2022-03-07 20:38:44 +03:00
if ( acpi_dev_hid_uid_match ( adev , " 10WM5102 " , NULL ) )
ret = arizona_spi_acpi_android_probe ( arizona ) ;
else
ret = arizona_spi_acpi_windows_probe ( arizona ) ;
2022-03-07 20:38:43 +03:00
if ( ret )
return ret ;
2021-01-21 00:49:55 +03:00
/*
* Some DSDTs wrongly declare the IRQ trigger - type as IRQF_TRIGGER_FALLING
* The IRQ line will stay low when a new IRQ event happens between reading
* the IRQ status flags and acknowledging them . When the IRQ line stays
* low like this the IRQ will never trigger again when its type is set
* to IRQF_TRIGGER_FALLING . Correct the IRQ trigger - type to fix this .
*
* Note theoretically it is possible that some boards are not capable
* of handling active low level interrupts . In that case setting the
* flag to IRQF_TRIGGER_FALLING would not be a bug ( and we would need
* to work around this ) but so far all known usages of IRQF_TRIGGER_FALLING
* are a bug in the board ' s DSDT .
*/
arizona - > pdata . irq_flags = IRQF_TRIGGER_LOW ;
/* Wait 200 ms after jack insertion */
arizona - > pdata . micd_detect_debounce = 200 ;
/* Use standard AOSP values for headset-button mappings */
arizona - > pdata . micd_ranges = arizona_micd_aosp_ranges ;
arizona - > pdata . num_micd_ranges = ARRAY_SIZE ( arizona_micd_aosp_ranges ) ;
return 0 ;
}
static const struct acpi_device_id arizona_acpi_match [ ] = {
{
. id = " WM510204 " ,
. driver_data = WM5102 ,
} ,
{
. id = " WM510205 " ,
. driver_data = WM5102 ,
} ,
2022-03-07 20:38:44 +03:00
{
. id = " 10WM5102 " ,
. driver_data = WM5102 ,
} ,
2021-01-21 00:49:55 +03:00
{ }
} ;
MODULE_DEVICE_TABLE ( acpi , arizona_acpi_match ) ;
# else
static int arizona_spi_acpi_probe ( struct arizona * arizona )
{
return - ENODEV ;
}
# endif
2012-11-19 22:23:04 +04:00
static int arizona_spi_probe ( struct spi_device * spi )
2012-06-19 19:36:18 +04:00
{
const struct spi_device_id * id = spi_get_device_id ( spi ) ;
2021-01-21 00:49:54 +03:00
const void * match_data ;
2012-06-19 19:36:18 +04:00
struct arizona * arizona ;
2015-10-02 15:29:14 +03:00
const struct regmap_config * regmap_config = NULL ;
2021-01-21 00:49:54 +03:00
unsigned long type = 0 ;
2014-07-02 17:28:46 +04:00
int ret ;
2012-06-19 19:36:18 +04:00
2021-01-21 00:49:54 +03:00
match_data = device_get_match_data ( & spi - > dev ) ;
if ( match_data )
type = ( unsigned long ) match_data ;
else if ( id )
2013-03-25 04:11:27 +04:00
type = id - > driver_data ;
switch ( type ) {
2012-06-19 19:36:18 +04:00
case WM5102 :
2015-10-02 15:29:14 +03:00
if ( IS_ENABLED ( CONFIG_MFD_WM5102 ) )
regmap_config = & wm5102_spi_regmap ;
2012-06-19 19:36:18 +04:00
break ;
2012-07-10 15:37:58 +04:00
case WM5110 :
2015-01-17 18:21:22 +03:00
case WM8280 :
2015-10-02 15:29:14 +03:00
if ( IS_ENABLED ( CONFIG_MFD_WM5110 ) )
regmap_config = & wm5110_spi_regmap ;
2012-07-10 15:37:58 +04:00
break ;
2015-11-03 18:08:32 +03:00
case WM1831 :
case CS47L24 :
if ( IS_ENABLED ( CONFIG_MFD_CS47L24 ) )
regmap_config = & cs47l24_spi_regmap ;
break ;
2012-06-19 19:36:18 +04:00
default :
2015-10-02 15:29:15 +03:00
dev_err ( & spi - > dev , " Unknown device type %ld \n " , type ) ;
2012-06-19 19:36:18 +04:00
return - EINVAL ;
}
2015-10-02 15:29:14 +03:00
if ( ! regmap_config ) {
dev_err ( & spi - > dev ,
" No kernel support for device type %ld \n " , type ) ;
2012-06-19 19:36:18 +04:00
return - EINVAL ;
}
arizona = devm_kzalloc ( & spi - > dev , sizeof ( * arizona ) , GFP_KERNEL ) ;
if ( arizona = = NULL )
return - ENOMEM ;
arizona - > regmap = devm_regmap_init_spi ( spi , regmap_config ) ;
if ( IS_ERR ( arizona - > regmap ) ) {
ret = PTR_ERR ( arizona - > regmap ) ;
dev_err ( & spi - > dev , " Failed to allocate register map: %d \n " ,
ret ) ;
return ret ;
}
2015-10-02 15:29:15 +03:00
arizona - > type = type ;
2012-06-19 19:36:18 +04:00
arizona - > dev = & spi - > dev ;
arizona - > irq = spi - > irq ;
2021-01-21 00:49:55 +03:00
if ( has_acpi_companion ( & spi - > dev ) ) {
ret = arizona_spi_acpi_probe ( arizona ) ;
if ( ret )
return ret ;
}
2012-06-19 19:36:18 +04:00
return arizona_dev_init ( arizona ) ;
}
2022-01-23 20:52:01 +03:00
static void arizona_spi_remove ( struct spi_device * spi )
2012-06-19 19:36:18 +04:00
{
2013-04-06 10:44:30 +04:00
struct arizona * arizona = spi_get_drvdata ( spi ) ;
2014-10-15 12:38:47 +04:00
2012-06-19 19:36:18 +04:00
arizona_dev_exit ( arizona ) ;
}
static const struct spi_device_id arizona_spi_ids [ ] = {
{ " wm5102 " , WM5102 } ,
2012-07-10 15:37:58 +04:00
{ " wm5110 " , WM5110 } ,
2015-01-17 18:21:22 +03:00
{ " wm8280 " , WM8280 } ,
2015-11-03 18:08:32 +03:00
{ " wm1831 " , WM1831 } ,
{ " cs47l24 " , CS47L24 } ,
2012-06-19 19:36:18 +04:00
{ } ,
} ;
MODULE_DEVICE_TABLE ( spi , arizona_spi_ids ) ;
2021-09-28 19:30:35 +03:00
# ifdef CONFIG_OF
2022-01-04 19:54:35 +03:00
static const struct of_device_id arizona_spi_of_match [ ] = {
2021-09-28 19:30:35 +03:00
{ . compatible = " wlf,wm5102 " , . data = ( void * ) WM5102 } ,
{ . compatible = " wlf,wm5110 " , . data = ( void * ) WM5110 } ,
{ . compatible = " wlf,wm8280 " , . data = ( void * ) WM8280 } ,
{ . compatible = " wlf,wm1831 " , . data = ( void * ) WM1831 } ,
{ . compatible = " cirrus,cs47l24 " , . data = ( void * ) CS47L24 } ,
{ } ,
} ;
# endif
2012-06-19 19:36:18 +04:00
static struct spi_driver arizona_spi_driver = {
. driver = {
. name = " arizona " ,
2022-10-23 12:48:43 +03:00
. pm = pm_ptr ( & arizona_pm_ops ) ,
2021-09-28 19:30:35 +03:00
. of_match_table = of_match_ptr ( arizona_spi_of_match ) ,
2021-01-21 00:49:55 +03:00
. acpi_match_table = ACPI_PTR ( arizona_acpi_match ) ,
2012-06-19 19:36:18 +04:00
} ,
. probe = arizona_spi_probe ,
2012-11-19 22:20:24 +04:00
. remove = arizona_spi_remove ,
2012-06-19 19:36:18 +04:00
. id_table = arizona_spi_ids ,
} ;
module_spi_driver ( arizona_spi_driver ) ;
2021-01-21 00:49:53 +03:00
MODULE_SOFTDEP ( " pre: arizona_ldo1 " ) ;
2012-06-19 19:36:18 +04:00
MODULE_DESCRIPTION ( " Arizona SPI bus interface " ) ;
MODULE_AUTHOR ( " Mark Brown <broonie@opensource.wolfsonmicro.com> " ) ;
MODULE_LICENSE ( " GPL " ) ;