2019-06-26 14:33:36 +01:00
// SPDX-License-Identifier: GPL-2.0-only
2018-05-21 10:59:56 +01:00
/*
* Core MFD support for Cirrus Logic Madera codecs
*
* Copyright ( C ) 2015 - 2018 Cirrus Logic
*/
# include <linux/device.h>
# include <linux/delay.h>
# include <linux/err.h>
2022-09-25 22:44:18 -07:00
# include <linux/gpio/consumer.h>
2018-05-21 10:59:56 +01:00
# include <linux/mfd/core.h>
# include <linux/module.h>
2018-11-12 15:28:37 +00:00
# include <linux/mutex.h>
2018-05-21 10:59:56 +01:00
# include <linux/notifier.h>
# include <linux/of.h>
# include <linux/platform_device.h>
# include <linux/pm_runtime.h>
# include <linux/regmap.h>
# include <linux/regulator/consumer.h>
# include <linux/regulator/machine.h>
# include <linux/regulator/of_regulator.h>
# include <linux/mfd/madera/core.h>
# include <linux/mfd/madera/registers.h>
# include "madera.h"
2019-05-30 15:39:52 +01:00
# define CS47L15_SILICON_ID 0x6370
2018-05-21 10:59:56 +01:00
# define CS47L35_SILICON_ID 0x6360
# define CS47L85_SILICON_ID 0x6338
# define CS47L90_SILICON_ID 0x6364
2019-05-30 15:39:53 +01:00
# define CS47L92_SILICON_ID 0x6371
2018-05-21 10:59:56 +01:00
# define MADERA_32KZ_MCLK2 1
2020-01-14 15:10:48 +00:00
# define MADERA_RESET_MIN_US 2000
# define MADERA_RESET_MAX_US 3000
2020-10-27 09:41:32 +00:00
# define ERRATA_DCVDD_MIN_US 10000
# define ERRATA_DCVDD_MAX_US 15000
2018-05-21 10:59:56 +01:00
static const char * const madera_core_supplies [ ] = {
" AVDD " ,
" DBVDD1 " ,
} ;
static const struct mfd_cell madera_ldo1_devs [ ] = {
2020-07-23 11:54:59 +01:00
{
. name = " madera-ldo1 " ,
. level = MFD_DEP_LEVEL_HIGH ,
} ,
2018-05-21 10:59:56 +01:00
} ;
2019-05-30 15:39:52 +01:00
static const char * const cs47l15_supplies [ ] = {
" MICVDD " ,
" CPVDD1 " ,
" SPKVDD " ,
} ;
static const struct mfd_cell cs47l15_devs [ ] = {
{ . name = " madera-pinctrl " , } ,
2020-06-15 14:53:23 +01:00
{ . name = " madera-irq " , } ,
{ . name = " madera-gpio " , } ,
2019-05-30 15:39:52 +01:00
{
. name = " madera-extcon " ,
. parent_supplies = cs47l15_supplies ,
. num_parent_supplies = 1 , /* We only need MICVDD */
} ,
{
. name = " cs47l15-codec " ,
. parent_supplies = cs47l15_supplies ,
. num_parent_supplies = ARRAY_SIZE ( cs47l15_supplies ) ,
} ,
} ;
2018-05-21 10:59:56 +01:00
static const char * const cs47l35_supplies [ ] = {
" MICVDD " ,
" DBVDD2 " ,
" CPVDD1 " ,
" CPVDD2 " ,
" SPKVDD " ,
} ;
static const struct mfd_cell cs47l35_devs [ ] = {
{ . name = " madera-pinctrl " , } ,
{ . name = " madera-irq " , } ,
{ . name = " madera-micsupp " , } ,
{ . name = " madera-gpio " , } ,
2019-05-20 10:06:28 +01:00
{
. name = " madera-extcon " ,
. parent_supplies = cs47l35_supplies ,
. num_parent_supplies = 1 , /* We only need MICVDD */
} ,
2018-05-21 10:59:56 +01:00
{
. name = " cs47l35-codec " ,
. parent_supplies = cs47l35_supplies ,
. num_parent_supplies = ARRAY_SIZE ( cs47l35_supplies ) ,
} ,
} ;
static const char * const cs47l85_supplies [ ] = {
" MICVDD " ,
" DBVDD2 " ,
" DBVDD3 " ,
" DBVDD4 " ,
" CPVDD1 " ,
" CPVDD2 " ,
" SPKVDDL " ,
" SPKVDDR " ,
} ;
static const struct mfd_cell cs47l85_devs [ ] = {
{ . name = " madera-pinctrl " , } ,
{ . name = " madera-irq " , } ,
2020-06-15 14:53:23 +01:00
{ . name = " madera-micsupp " , } ,
2018-05-21 10:59:56 +01:00
{ . name = " madera-gpio " , } ,
2019-05-20 10:06:28 +01:00
{
. name = " madera-extcon " ,
. parent_supplies = cs47l85_supplies ,
. num_parent_supplies = 1 , /* We only need MICVDD */
} ,
2018-05-21 10:59:56 +01:00
{
. name = " cs47l85-codec " ,
. parent_supplies = cs47l85_supplies ,
. num_parent_supplies = ARRAY_SIZE ( cs47l85_supplies ) ,
} ,
} ;
static const char * const cs47l90_supplies [ ] = {
" MICVDD " ,
" DBVDD2 " ,
" DBVDD3 " ,
" DBVDD4 " ,
" CPVDD1 " ,
" CPVDD2 " ,
} ;
static const struct mfd_cell cs47l90_devs [ ] = {
{ . name = " madera-pinctrl " , } ,
{ . name = " madera-irq " , } ,
{ . name = " madera-micsupp " , } ,
{ . name = " madera-gpio " , } ,
2019-05-20 10:06:28 +01:00
{
. name = " madera-extcon " ,
. parent_supplies = cs47l90_supplies ,
. num_parent_supplies = 1 , /* We only need MICVDD */
} ,
2018-05-21 10:59:56 +01:00
{
. name = " cs47l90-codec " ,
. parent_supplies = cs47l90_supplies ,
. num_parent_supplies = ARRAY_SIZE ( cs47l90_supplies ) ,
} ,
} ;
2019-05-30 15:39:53 +01:00
static const char * const cs47l92_supplies [ ] = {
" MICVDD " ,
" CPVDD1 " ,
" CPVDD2 " ,
} ;
static const struct mfd_cell cs47l92_devs [ ] = {
2020-06-15 14:53:23 +01:00
{ . name = " madera-pinctrl " , } ,
2019-05-30 15:39:53 +01:00
{ . name = " madera-irq " , } ,
{ . name = " madera-micsupp " , } ,
2020-06-15 14:53:23 +01:00
{ . name = " madera-gpio " , } ,
2019-05-30 15:39:53 +01:00
{
. name = " madera-extcon " ,
. parent_supplies = cs47l92_supplies ,
. num_parent_supplies = 1 , /* We only need MICVDD */
} ,
{
. name = " cs47l92-codec " ,
. parent_supplies = cs47l92_supplies ,
. num_parent_supplies = ARRAY_SIZE ( cs47l92_supplies ) ,
} ,
} ;
2018-05-21 10:59:56 +01:00
/* Used by madera-i2c and madera-spi drivers */
const char * madera_name_from_type ( enum madera_type type )
{
switch ( type ) {
2019-05-30 15:39:52 +01:00
case CS47L15 :
return " CS47L15 " ;
2018-05-21 10:59:56 +01:00
case CS47L35 :
return " CS47L35 " ;
case CS47L85 :
return " CS47L85 " ;
case CS47L90 :
return " CS47L90 " ;
case CS47L91 :
return " CS47L91 " ;
2019-05-30 15:39:53 +01:00
case CS42L92 :
return " CS42L92 " ;
case CS47L92 :
return " CS47L92 " ;
case CS47L93 :
return " CS47L93 " ;
2018-05-21 10:59:56 +01:00
case WM1840 :
return " WM1840 " ;
default :
return " Unknown " ;
}
}
EXPORT_SYMBOL_GPL ( madera_name_from_type ) ;
2018-08-27 14:51:15 +01:00
# define MADERA_BOOT_POLL_INTERVAL_USEC 5000
# define MADERA_BOOT_POLL_TIMEOUT_USEC 25000
2018-05-21 10:59:56 +01:00
2020-01-06 10:28:34 +00:00
static int madera_wait_for_boot_noack ( struct madera * madera )
2018-05-21 10:59:56 +01:00
{
2018-08-27 14:51:15 +01:00
ktime_t timeout ;
2019-05-20 10:06:27 +01:00
unsigned int val = 0 ;
2018-08-27 14:51:15 +01:00
int ret = 0 ;
2018-05-21 10:59:56 +01:00
/*
* We can ' t use an interrupt as we need to runtime resume to do so ,
* so we poll the status bit . This won ' t race with the interrupt
* handler because it will be blocked on runtime resume .
2018-08-27 14:51:15 +01:00
* The chip could NAK a read request while it is booting so ignore
* errors from regmap_read .
2018-05-21 10:59:56 +01:00
*/
2018-08-27 14:51:15 +01:00
timeout = ktime_add_us ( ktime_get ( ) , MADERA_BOOT_POLL_TIMEOUT_USEC ) ;
regmap_read ( madera - > regmap , MADERA_IRQ1_RAW_STATUS_1 , & val ) ;
while ( ! ( val & MADERA_BOOT_DONE_STS1 ) & &
! ktime_after ( ktime_get ( ) , timeout ) ) {
usleep_range ( MADERA_BOOT_POLL_INTERVAL_USEC / 2 ,
MADERA_BOOT_POLL_INTERVAL_USEC ) ;
regmap_read ( madera - > regmap , MADERA_IRQ1_RAW_STATUS_1 , & val ) ;
2018-12-14 14:39:14 +00:00
}
2018-08-27 14:51:15 +01:00
if ( ! ( val & MADERA_BOOT_DONE_STS1 ) ) {
dev_err ( madera - > dev , " Polling BOOT_DONE_STS timed out \n " ) ;
ret = - ETIMEDOUT ;
}
2018-05-21 10:59:56 +01:00
2020-01-06 10:28:34 +00:00
return ret ;
}
static int madera_wait_for_boot ( struct madera * madera )
{
int ret = madera_wait_for_boot_noack ( madera ) ;
2018-05-21 10:59:56 +01:00
/*
* BOOT_DONE defaults to unmasked on boot so we must ack it .
2018-08-27 14:51:15 +01:00
* Do this even after a timeout to avoid interrupt storms .
2018-05-21 10:59:56 +01:00
*/
regmap_write ( madera - > regmap , MADERA_IRQ1_STATUS_1 ,
MADERA_BOOT_DONE_EINT1 ) ;
pm_runtime_mark_last_busy ( madera - > dev ) ;
return ret ;
}
static int madera_soft_reset ( struct madera * madera )
{
int ret ;
ret = regmap_write ( madera - > regmap , MADERA_SOFTWARE_RESET , 0 ) ;
if ( ret ! = 0 ) {
dev_err ( madera - > dev , " Failed to soft reset device: %d \n " , ret ) ;
return ret ;
}
/* Allow time for internal clocks to startup after reset */
2020-01-14 15:10:48 +00:00
usleep_range ( MADERA_RESET_MIN_US , MADERA_RESET_MAX_US ) ;
2018-05-21 10:59:56 +01:00
return 0 ;
}
static void madera_enable_hard_reset ( struct madera * madera )
{
/*
* There are many existing out - of - tree users of these codecs that we
* can ' t break so preserve the expected behaviour of setting the line
* low to assert reset .
*/
gpiod_set_raw_value_cansleep ( madera - > pdata . reset , 0 ) ;
}
static void madera_disable_hard_reset ( struct madera * madera )
{
gpiod_set_raw_value_cansleep ( madera - > pdata . reset , 1 ) ;
2020-01-14 15:10:48 +00:00
usleep_range ( MADERA_RESET_MIN_US , MADERA_RESET_MAX_US ) ;
2018-05-21 10:59:56 +01:00
}
static int __maybe_unused madera_runtime_resume ( struct device * dev )
{
struct madera * madera = dev_get_drvdata ( dev ) ;
int ret ;
dev_dbg ( dev , " Leaving sleep mode \n " ) ;
2020-10-27 09:41:32 +00:00
if ( ! madera - > reset_errata )
madera_enable_hard_reset ( madera ) ;
2020-10-27 09:41:31 +00:00
2018-05-21 10:59:56 +01:00
ret = regulator_enable ( madera - > dcvdd ) ;
if ( ret ) {
dev_err ( dev , " Failed to enable DCVDD: %d \n " , ret ) ;
return ret ;
}
regcache_cache_only ( madera - > regmap , false ) ;
regcache_cache_only ( madera - > regmap_32bit , false ) ;
2020-10-27 09:41:32 +00:00
if ( madera - > reset_errata )
usleep_range ( ERRATA_DCVDD_MIN_US , ERRATA_DCVDD_MAX_US ) ;
else
madera_disable_hard_reset ( madera ) ;
2020-10-27 09:41:31 +00:00
2020-10-27 09:41:32 +00:00
if ( ! madera - > pdata . reset | | madera - > reset_errata ) {
2020-10-27 09:41:31 +00:00
ret = madera_wait_for_boot ( madera ) ;
if ( ret )
goto err ;
ret = madera_soft_reset ( madera ) ;
if ( ret ) {
dev_err ( dev , " Failed to reset: %d \n " , ret ) ;
goto err ;
}
}
2020-01-14 15:10:48 +00:00
2018-05-21 10:59:56 +01:00
ret = madera_wait_for_boot ( madera ) ;
if ( ret )
goto err ;
ret = regcache_sync ( madera - > regmap ) ;
if ( ret ) {
dev_err ( dev , " Failed to restore 16-bit register cache \n " ) ;
goto err ;
}
ret = regcache_sync ( madera - > regmap_32bit ) ;
if ( ret ) {
dev_err ( dev , " Failed to restore 32-bit register cache \n " ) ;
goto err ;
}
return 0 ;
err :
regcache_cache_only ( madera - > regmap_32bit , true ) ;
regcache_cache_only ( madera - > regmap , true ) ;
regulator_disable ( madera - > dcvdd ) ;
return ret ;
}
static int __maybe_unused madera_runtime_suspend ( struct device * dev )
{
struct madera * madera = dev_get_drvdata ( dev ) ;
dev_dbg ( madera - > dev , " Entering sleep mode \n " ) ;
regcache_cache_only ( madera - > regmap , true ) ;
regcache_mark_dirty ( madera - > regmap ) ;
regcache_cache_only ( madera - > regmap_32bit , true ) ;
regcache_mark_dirty ( madera - > regmap_32bit ) ;
regulator_disable ( madera - > dcvdd ) ;
return 0 ;
}
const struct dev_pm_ops madera_pm_ops = {
SET_RUNTIME_PM_OPS ( madera_runtime_suspend ,
madera_runtime_resume ,
NULL )
} ;
EXPORT_SYMBOL_GPL ( madera_pm_ops ) ;
const struct of_device_id madera_of_match [ ] = {
2019-05-30 15:39:52 +01:00
{ . compatible = " cirrus,cs47l15 " , . data = ( void * ) CS47L15 } ,
2018-05-21 10:59:56 +01:00
{ . compatible = " cirrus,cs47l35 " , . data = ( void * ) CS47L35 } ,
{ . compatible = " cirrus,cs47l85 " , . data = ( void * ) CS47L85 } ,
{ . compatible = " cirrus,cs47l90 " , . data = ( void * ) CS47L90 } ,
{ . compatible = " cirrus,cs47l91 " , . data = ( void * ) CS47L91 } ,
2019-05-30 15:39:53 +01:00
{ . compatible = " cirrus,cs42l92 " , . data = ( void * ) CS42L92 } ,
{ . compatible = " cirrus,cs47l92 " , . data = ( void * ) CS47L92 } ,
{ . compatible = " cirrus,cs47l93 " , . data = ( void * ) CS47L93 } ,
2018-05-21 10:59:56 +01:00
{ . compatible = " cirrus,wm1840 " , . data = ( void * ) WM1840 } ,
{ }
} ;
mfd: madera: Add missing of table registration
MODULE_DEVICE_TABLE(of, <of_match_table>) should be called to complete DT
OF mathing mechanism and register it.
Before this patch:
modinfo ./drivers/mfd/madera.ko | grep alias
After this patch:
modinfo ./drivers/mfd/madera.ko | grep alias
alias: of:N*T*Ccirrus,wm1840C*
alias: of:N*T*Ccirrus,wm1840
alias: of:N*T*Ccirrus,cs47l91C*
alias: of:N*T*Ccirrus,cs47l91
alias: of:N*T*Ccirrus,cs47l90C*
alias: of:N*T*Ccirrus,cs47l90
alias: of:N*T*Ccirrus,cs47l85C*
alias: of:N*T*Ccirrus,cs47l85
alias: of:N*T*Ccirrus,cs47l35C*
alias: of:N*T*Ccirrus,cs47l35
Reported-by: Javier Martinez Canillas <javier@dowhile0.org>
Signed-off-by: Daniel Gomez <dagmcr@gmail.com>
Signed-off-by: Lee Jones <lee.jones@linaro.org>
2019-05-11 12:03:58 +02:00
MODULE_DEVICE_TABLE ( of , madera_of_match ) ;
2018-05-21 10:59:56 +01:00
EXPORT_SYMBOL_GPL ( madera_of_match ) ;
static int madera_get_reset_gpio ( struct madera * madera )
{
struct gpio_desc * reset ;
if ( madera - > pdata . reset )
return 0 ;
reset = devm_gpiod_get_optional ( madera - > dev , " reset " , GPIOD_OUT_LOW ) ;
2020-08-26 16:49:33 +02:00
if ( IS_ERR ( reset ) )
return dev_err_probe ( madera - > dev , PTR_ERR ( reset ) ,
" Failed to request /RESET " ) ;
2018-05-21 10:59:56 +01:00
/*
* A hard reset is needed for full reset of the chip . We allow running
* without hard reset only because it can be useful for early
* prototyping and some debugging , but we need to warn it ' s not ideal .
*/
if ( ! reset )
dev_warn ( madera - > dev ,
" Running without reset GPIO is not recommended \n " ) ;
madera - > pdata . reset = reset ;
return 0 ;
}
static void madera_set_micbias_info ( struct madera * madera )
{
/*
* num_childbias is an array because future codecs can have different
* childbiases for each micbias . Unspecified values default to 0.
*/
switch ( madera - > type ) {
2019-05-30 15:39:52 +01:00
case CS47L15 :
madera - > num_micbias = 1 ;
madera - > num_childbias [ 0 ] = 3 ;
return ;
2018-05-21 10:59:56 +01:00
case CS47L35 :
madera - > num_micbias = 2 ;
madera - > num_childbias [ 0 ] = 2 ;
madera - > num_childbias [ 1 ] = 2 ;
return ;
case CS47L85 :
case WM1840 :
madera - > num_micbias = 4 ;
/* no child biases */
return ;
case CS47L90 :
case CS47L91 :
madera - > num_micbias = 2 ;
madera - > num_childbias [ 0 ] = 4 ;
madera - > num_childbias [ 1 ] = 4 ;
return ;
2019-05-30 15:39:53 +01:00
case CS42L92 :
case CS47L92 :
case CS47L93 :
madera - > num_micbias = 2 ;
madera - > num_childbias [ 0 ] = 4 ;
madera - > num_childbias [ 1 ] = 2 ;
return ;
2018-05-21 10:59:56 +01:00
default :
return ;
}
}
int madera_dev_init ( struct madera * madera )
{
struct device * dev = madera - > dev ;
unsigned int hwid ;
int ( * patch_fn ) ( struct madera * ) = NULL ;
const struct mfd_cell * mfd_devs ;
int n_devs = 0 ;
int i , ret ;
dev_set_drvdata ( madera - > dev , madera ) ;
BLOCKING_INIT_NOTIFIER_HEAD ( & madera - > notifier ) ;
2018-11-12 15:28:37 +00:00
mutex_init ( & madera - > dapm_ptr_lock ) ;
2018-05-21 10:59:56 +01:00
madera_set_micbias_info ( madera ) ;
/*
* We need writable hw config info that all children can share .
* Simplest to take one shared copy of pdata struct .
*/
if ( dev_get_platdata ( madera - > dev ) ) {
memcpy ( & madera - > pdata , dev_get_platdata ( madera - > dev ) ,
sizeof ( madera - > pdata ) ) ;
}
2019-10-21 14:58:13 +01:00
madera - > mclk [ MADERA_MCLK1 ] . id = " mclk1 " ;
madera - > mclk [ MADERA_MCLK2 ] . id = " mclk2 " ;
madera - > mclk [ MADERA_MCLK3 ] . id = " mclk3 " ;
ret = devm_clk_bulk_get_optional ( madera - > dev , ARRAY_SIZE ( madera - > mclk ) ,
madera - > mclk ) ;
if ( ret ) {
dev_err ( madera - > dev , " Failed to get clocks: %d \n " , ret ) ;
return ret ;
}
/* Not using devm_clk_get to prevent breakage of existing DTs */
if ( ! madera - > mclk [ MADERA_MCLK2 ] . clk )
dev_warn ( madera - > dev , " Missing MCLK2, requires 32kHz clock \n " ) ;
2018-05-21 10:59:56 +01:00
ret = madera_get_reset_gpio ( madera ) ;
if ( ret )
return ret ;
regcache_cache_only ( madera - > regmap , true ) ;
regcache_cache_only ( madera - > regmap_32bit , true ) ;
for ( i = 0 ; i < ARRAY_SIZE ( madera_core_supplies ) ; i + + )
madera - > core_supplies [ i ] . supply = madera_core_supplies [ i ] ;
madera - > num_core_supplies = ARRAY_SIZE ( madera_core_supplies ) ;
/*
* On some codecs DCVDD could be supplied by the internal LDO1 .
* For those we must add the LDO1 driver before requesting DCVDD
* No devm_ because we need to control shutdown order of children .
*/
switch ( madera - > type ) {
2019-05-30 15:39:52 +01:00
case CS47L15 :
2020-10-27 09:41:32 +00:00
madera - > reset_errata = true ;
break ;
2018-05-21 10:59:56 +01:00
case CS47L35 :
case CS47L90 :
case CS47L91 :
2019-05-30 15:39:53 +01:00
case CS42L92 :
case CS47L92 :
case CS47L93 :
2018-05-21 10:59:56 +01:00
break ;
case CS47L85 :
case WM1840 :
ret = mfd_add_devices ( madera - > dev , PLATFORM_DEVID_NONE ,
madera_ldo1_devs ,
ARRAY_SIZE ( madera_ldo1_devs ) ,
NULL , 0 , NULL ) ;
if ( ret ) {
dev_err ( dev , " Failed to add LDO1 child: %d \n " , ret ) ;
return ret ;
}
break ;
default :
/* No point continuing if the type is unknown */
dev_err ( madera - > dev , " Unknown device type %d \n " , madera - > type ) ;
return - ENODEV ;
}
ret = devm_regulator_bulk_get ( dev , madera - > num_core_supplies ,
madera - > core_supplies ) ;
if ( ret ) {
dev_err ( dev , " Failed to request core supplies: %d \n " , ret ) ;
goto err_devs ;
}
/*
* Don ' t use devres here . If the regulator is one of our children it
* will already have been removed before devres cleanup on this mfd
* driver tries to call put ( ) on it . We need control of shutdown order .
*/
madera - > dcvdd = regulator_get ( madera - > dev , " DCVDD " ) ;
if ( IS_ERR ( madera - > dcvdd ) ) {
ret = PTR_ERR ( madera - > dcvdd ) ;
dev_err ( dev , " Failed to request DCVDD: %d \n " , ret ) ;
goto err_devs ;
}
ret = regulator_bulk_enable ( madera - > num_core_supplies ,
madera - > core_supplies ) ;
if ( ret ) {
dev_err ( dev , " Failed to enable core supplies: %d \n " , ret ) ;
goto err_dcvdd ;
}
2020-10-27 09:41:32 +00:00
if ( madera - > reset_errata )
madera_disable_hard_reset ( madera ) ;
2018-05-21 10:59:56 +01:00
ret = regulator_enable ( madera - > dcvdd ) ;
if ( ret ) {
dev_err ( dev , " Failed to enable DCVDD: %d \n " , ret ) ;
goto err_enable ;
}
2020-10-27 09:41:32 +00:00
if ( madera - > reset_errata )
usleep_range ( ERRATA_DCVDD_MIN_US , ERRATA_DCVDD_MAX_US ) ;
else
madera_disable_hard_reset ( madera ) ;
2018-05-21 10:59:56 +01:00
regcache_cache_only ( madera - > regmap , false ) ;
regcache_cache_only ( madera - > regmap_32bit , false ) ;
2020-01-06 10:28:34 +00:00
ret = madera_wait_for_boot_noack ( madera ) ;
if ( ret ) {
dev_err ( madera - > dev , " Device failed initial boot: %d \n " , ret ) ;
goto err_reset ;
}
2018-05-21 10:59:56 +01:00
/*
* Now we can power up and verify that this is a chip we know about
* before we start doing any writes to its registers .
*/
ret = regmap_read ( madera - > regmap , MADERA_SOFTWARE_RESET , & hwid ) ;
if ( ret ) {
dev_err ( dev , " Failed to read ID register: %d \n " , ret ) ;
goto err_reset ;
}
switch ( hwid ) {
2019-05-30 15:39:52 +01:00
case CS47L15_SILICON_ID :
if ( IS_ENABLED ( CONFIG_MFD_CS47L15 ) ) {
switch ( madera - > type ) {
case CS47L15 :
patch_fn = & cs47l15_patch ;
mfd_devs = cs47l15_devs ;
n_devs = ARRAY_SIZE ( cs47l15_devs ) ;
break ;
default :
break ;
}
}
break ;
2018-05-21 10:59:56 +01:00
case CS47L35_SILICON_ID :
if ( IS_ENABLED ( CONFIG_MFD_CS47L35 ) ) {
switch ( madera - > type ) {
case CS47L35 :
patch_fn = cs47l35_patch ;
mfd_devs = cs47l35_devs ;
n_devs = ARRAY_SIZE ( cs47l35_devs ) ;
break ;
default :
break ;
}
}
break ;
case CS47L85_SILICON_ID :
if ( IS_ENABLED ( CONFIG_MFD_CS47L85 ) ) {
switch ( madera - > type ) {
case CS47L85 :
case WM1840 :
patch_fn = cs47l85_patch ;
mfd_devs = cs47l85_devs ;
n_devs = ARRAY_SIZE ( cs47l85_devs ) ;
break ;
default :
break ;
}
}
break ;
case CS47L90_SILICON_ID :
if ( IS_ENABLED ( CONFIG_MFD_CS47L90 ) ) {
switch ( madera - > type ) {
case CS47L90 :
case CS47L91 :
patch_fn = cs47l90_patch ;
mfd_devs = cs47l90_devs ;
n_devs = ARRAY_SIZE ( cs47l90_devs ) ;
break ;
default :
break ;
}
}
break ;
2019-05-30 15:39:53 +01:00
case CS47L92_SILICON_ID :
if ( IS_ENABLED ( CONFIG_MFD_CS47L92 ) ) {
switch ( madera - > type ) {
case CS42L92 :
case CS47L92 :
case CS47L93 :
patch_fn = cs47l92_patch ;
mfd_devs = cs47l92_devs ;
n_devs = ARRAY_SIZE ( cs47l92_devs ) ;
break ;
default :
break ;
}
}
break ;
2018-05-21 10:59:56 +01:00
default :
dev_err ( madera - > dev , " Unknown device ID: %x \n " , hwid ) ;
ret = - EINVAL ;
goto err_reset ;
}
if ( ! n_devs ) {
dev_err ( madera - > dev , " Device ID 0x%x not a %s \n " , hwid ,
madera - > type_name ) ;
ret = - ENODEV ;
goto err_reset ;
}
/*
* It looks like a device we support . If we don ' t have a hard reset
* we can now attempt a soft reset .
*/
2020-10-27 09:41:32 +00:00
if ( ! madera - > pdata . reset | | madera - > reset_errata ) {
2018-05-21 10:59:56 +01:00
ret = madera_soft_reset ( madera ) ;
if ( ret )
goto err_reset ;
}
ret = madera_wait_for_boot ( madera ) ;
if ( ret ) {
2020-01-06 10:28:34 +00:00
dev_err ( madera - > dev , " Failed to clear boot done: %d \n " , ret ) ;
2018-05-21 10:59:56 +01:00
goto err_reset ;
}
ret = regmap_read ( madera - > regmap , MADERA_HARDWARE_REVISION ,
& madera - > rev ) ;
if ( ret ) {
dev_err ( dev , " Failed to read revision register: %d \n " , ret ) ;
goto err_reset ;
}
madera - > rev & = MADERA_HW_REVISION_MASK ;
dev_info ( dev , " %s silicon revision %d \n " , madera - > type_name ,
madera - > rev ) ;
/* Apply hardware patch */
if ( patch_fn ) {
ret = patch_fn ( madera ) ;
if ( ret ) {
dev_err ( madera - > dev , " Failed to apply patch %d \n " , ret ) ;
goto err_reset ;
}
}
/* Init 32k clock sourced from MCLK2 */
2019-10-21 14:58:13 +01:00
ret = clk_prepare_enable ( madera - > mclk [ MADERA_MCLK2 ] . clk ) ;
if ( ret ) {
dev_err ( madera - > dev , " Failed to enable 32k clock: %d \n " , ret ) ;
goto err_reset ;
}
2018-05-21 10:59:56 +01:00
ret = regmap_update_bits ( madera - > regmap ,
MADERA_CLOCK_32K_1 ,
MADERA_CLK_32K_ENA_MASK | MADERA_CLK_32K_SRC_MASK ,
MADERA_CLK_32K_ENA | MADERA_32KZ_MCLK2 ) ;
if ( ret ) {
dev_err ( madera - > dev , " Failed to init 32k clock: %d \n " , ret ) ;
2019-10-21 14:58:13 +01:00
goto err_clock ;
2018-05-21 10:59:56 +01:00
}
pm_runtime_set_active ( madera - > dev ) ;
pm_runtime_enable ( madera - > dev ) ;
pm_runtime_set_autosuspend_delay ( madera - > dev , 100 ) ;
pm_runtime_use_autosuspend ( madera - > dev ) ;
/* No devm_ because we need to control shutdown order of children */
ret = mfd_add_devices ( madera - > dev , PLATFORM_DEVID_NONE ,
mfd_devs , n_devs ,
NULL , 0 , NULL ) ;
if ( ret ) {
dev_err ( madera - > dev , " Failed to add subdevices: %d \n " , ret ) ;
goto err_pm_runtime ;
}
return 0 ;
err_pm_runtime :
pm_runtime_disable ( madera - > dev ) ;
2019-10-21 14:58:13 +01:00
err_clock :
clk_disable_unprepare ( madera - > mclk [ MADERA_MCLK2 ] . clk ) ;
2018-05-21 10:59:56 +01:00
err_reset :
madera_enable_hard_reset ( madera ) ;
regulator_disable ( madera - > dcvdd ) ;
err_enable :
regulator_bulk_disable ( madera - > num_core_supplies ,
madera - > core_supplies ) ;
err_dcvdd :
regulator_put ( madera - > dcvdd ) ;
err_devs :
mfd_remove_devices ( dev ) ;
return ret ;
}
EXPORT_SYMBOL_GPL ( madera_dev_init ) ;
int madera_dev_exit ( struct madera * madera )
{
/* Prevent any IRQs being serviced while we clean up */
disable_irq ( madera - > irq ) ;
2020-07-23 11:54:59 +01:00
pm_runtime_get_sync ( madera - > dev ) ;
2018-05-21 10:59:56 +01:00
2020-07-23 11:54:59 +01:00
mfd_remove_devices ( madera - > dev ) ;
pm_runtime_disable ( madera - > dev ) ;
2019-10-21 14:58:13 +01:00
2018-05-21 10:59:56 +01:00
regulator_disable ( madera - > dcvdd ) ;
regulator_put ( madera - > dcvdd ) ;
2020-07-23 11:54:59 +01:00
mfd_remove_devices_late ( madera - > dev ) ;
pm_runtime_set_suspended ( madera - > dev ) ;
pm_runtime_put_noidle ( madera - > dev ) ;
clk_disable_unprepare ( madera - > mclk [ MADERA_MCLK2 ] . clk ) ;
2018-05-21 10:59:56 +01:00
madera_enable_hard_reset ( madera ) ;
regulator_bulk_disable ( madera - > num_core_supplies ,
madera - > core_supplies ) ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( madera_dev_exit ) ;
MODULE_DESCRIPTION ( " Madera core MFD driver " ) ;
MODULE_AUTHOR ( " Richard Fitzgerald <rf@opensource.cirrus.com> " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;