2023-03-21 10:26:44 +08:00
// SPDX-License-Identifier: GPL-2.0
/*
* Copyright ( C ) 2021 Emil Renner Berthing < kernel @ esmil . dk >
* Copyright ( C ) 2021 Samin Guo < samin . guo @ starfivetech . com >
*/
# include <linux/bits.h>
# include <linux/clk.h>
# include <linux/delay.h>
# include <linux/hwmon.h>
# include <linux/io.h>
# include <linux/module.h>
# include <linux/mutex.h>
# include <linux/of.h>
# include <linux/platform_device.h>
# include <linux/reset.h>
/*
* TempSensor reset . The RSTN can be de - asserted once the analog core has
* powered up . Trst ( min 100 ns )
* 0 : reset 1 : de - assert
*/
# define SFCTEMP_RSTN BIT(0)
/*
* TempSensor analog core power down . The analog core will be powered up
* Tpu ( min 50u s ) after PD is de - asserted . RSTN should be held low until the
* analog core is powered up .
* 0 : power up 1 : power down
*/
# define SFCTEMP_PD BIT(1)
/*
* TempSensor start conversion enable .
* 0 : disable 1 : enable
*/
# define SFCTEMP_RUN BIT(2)
/*
* TempSensor conversion value output .
* Temp ( C ) = DOUT * Y / 4094 - K
*/
# define SFCTEMP_DOUT_POS 16
# define SFCTEMP_DOUT_MSK GENMASK(27, 16)
/* DOUT to Celcius conversion constants */
# define SFCTEMP_Y1000 237500L
# define SFCTEMP_Z 4094L
# define SFCTEMP_K1000 81100L
struct sfctemp {
/* serialize access to hardware register and enabled below */
struct mutex lock ;
void __iomem * regs ;
struct clk * clk_sense ;
struct clk * clk_bus ;
struct reset_control * rst_sense ;
struct reset_control * rst_bus ;
bool enabled ;
} ;
static void sfctemp_power_up ( struct sfctemp * sfctemp )
{
/* make sure we're powered down first */
writel ( SFCTEMP_PD , sfctemp - > regs ) ;
udelay ( 1 ) ;
writel ( 0 , sfctemp - > regs ) ;
/* wait t_pu(50us) + t_rst(100ns) */
usleep_range ( 60 , 200 ) ;
/* de-assert reset */
writel ( SFCTEMP_RSTN , sfctemp - > regs ) ;
udelay ( 1 ) ; /* wait t_su(500ps) */
}
static void sfctemp_power_down ( struct sfctemp * sfctemp )
{
writel ( SFCTEMP_PD , sfctemp - > regs ) ;
}
static void sfctemp_run ( struct sfctemp * sfctemp )
{
writel ( SFCTEMP_RSTN | SFCTEMP_RUN , sfctemp - > regs ) ;
udelay ( 1 ) ;
}
static void sfctemp_stop ( struct sfctemp * sfctemp )
{
writel ( SFCTEMP_RSTN , sfctemp - > regs ) ;
}
static int sfctemp_enable ( struct sfctemp * sfctemp )
{
int ret = 0 ;
mutex_lock ( & sfctemp - > lock ) ;
if ( sfctemp - > enabled )
goto done ;
ret = clk_prepare_enable ( sfctemp - > clk_bus ) ;
if ( ret )
goto err ;
ret = reset_control_deassert ( sfctemp - > rst_bus ) ;
if ( ret )
goto err_disable_bus ;
ret = clk_prepare_enable ( sfctemp - > clk_sense ) ;
if ( ret )
goto err_assert_bus ;
ret = reset_control_deassert ( sfctemp - > rst_sense ) ;
if ( ret )
goto err_disable_sense ;
sfctemp_power_up ( sfctemp ) ;
sfctemp_run ( sfctemp ) ;
sfctemp - > enabled = true ;
done :
mutex_unlock ( & sfctemp - > lock ) ;
return ret ;
err_disable_sense :
clk_disable_unprepare ( sfctemp - > clk_sense ) ;
err_assert_bus :
reset_control_assert ( sfctemp - > rst_bus ) ;
err_disable_bus :
clk_disable_unprepare ( sfctemp - > clk_bus ) ;
err :
mutex_unlock ( & sfctemp - > lock ) ;
return ret ;
}
static int sfctemp_disable ( struct sfctemp * sfctemp )
{
mutex_lock ( & sfctemp - > lock ) ;
if ( ! sfctemp - > enabled )
goto done ;
sfctemp_stop ( sfctemp ) ;
sfctemp_power_down ( sfctemp ) ;
reset_control_assert ( sfctemp - > rst_sense ) ;
clk_disable_unprepare ( sfctemp - > clk_sense ) ;
reset_control_assert ( sfctemp - > rst_bus ) ;
clk_disable_unprepare ( sfctemp - > clk_bus ) ;
sfctemp - > enabled = false ;
done :
mutex_unlock ( & sfctemp - > lock ) ;
return 0 ;
}
static void sfctemp_disable_action ( void * data )
{
sfctemp_disable ( data ) ;
}
static int sfctemp_convert ( struct sfctemp * sfctemp , long * val )
{
int ret ;
mutex_lock ( & sfctemp - > lock ) ;
if ( ! sfctemp - > enabled ) {
ret = - ENODATA ;
goto out ;
}
/* calculate temperature in milli Celcius */
* val = ( long ) ( ( readl ( sfctemp - > regs ) & SFCTEMP_DOUT_MSK ) > > SFCTEMP_DOUT_POS )
* SFCTEMP_Y1000 / SFCTEMP_Z - SFCTEMP_K1000 ;
ret = 0 ;
out :
mutex_unlock ( & sfctemp - > lock ) ;
return ret ;
}
static umode_t sfctemp_is_visible ( const void * data , enum hwmon_sensor_types type ,
u32 attr , int channel )
{
switch ( type ) {
case hwmon_temp :
switch ( attr ) {
case hwmon_temp_enable :
return 0644 ;
case hwmon_temp_input :
return 0444 ;
default :
return 0 ;
}
default :
return 0 ;
}
}
static int sfctemp_read ( struct device * dev , enum hwmon_sensor_types type ,
u32 attr , int channel , long * val )
{
struct sfctemp * sfctemp = dev_get_drvdata ( dev ) ;
switch ( type ) {
case hwmon_temp :
switch ( attr ) {
case hwmon_temp_enable :
* val = sfctemp - > enabled ;
return 0 ;
case hwmon_temp_input :
return sfctemp_convert ( sfctemp , val ) ;
default :
return - EINVAL ;
}
default :
return - EINVAL ;
}
}
static int sfctemp_write ( struct device * dev , enum hwmon_sensor_types type ,
u32 attr , int channel , long val )
{
struct sfctemp * sfctemp = dev_get_drvdata ( dev ) ;
switch ( type ) {
case hwmon_temp :
switch ( attr ) {
case hwmon_temp_enable :
if ( val = = 0 )
return sfctemp_disable ( sfctemp ) ;
if ( val = = 1 )
return sfctemp_enable ( sfctemp ) ;
return - EINVAL ;
default :
return - EINVAL ;
}
default :
return - EINVAL ;
}
}
static const struct hwmon_channel_info * sfctemp_info [ ] = {
HWMON_CHANNEL_INFO ( chip , HWMON_C_REGISTER_TZ ) ,
HWMON_CHANNEL_INFO ( temp , HWMON_T_ENABLE | HWMON_T_INPUT ) ,
NULL
} ;
static const struct hwmon_ops sfctemp_hwmon_ops = {
. is_visible = sfctemp_is_visible ,
. read = sfctemp_read ,
. write = sfctemp_write ,
} ;
static const struct hwmon_chip_info sfctemp_chip_info = {
. ops = & sfctemp_hwmon_ops ,
. info = sfctemp_info ,
} ;
static int sfctemp_probe ( struct platform_device * pdev )
{
struct device * dev = & pdev - > dev ;
struct device * hwmon_dev ;
struct sfctemp * sfctemp ;
int ret ;
sfctemp = devm_kzalloc ( dev , sizeof ( * sfctemp ) , GFP_KERNEL ) ;
if ( ! sfctemp )
return - ENOMEM ;
dev_set_drvdata ( dev , sfctemp ) ;
mutex_init ( & sfctemp - > lock ) ;
sfctemp - > regs = devm_platform_ioremap_resource ( pdev , 0 ) ;
if ( IS_ERR ( sfctemp - > regs ) )
return PTR_ERR ( sfctemp - > regs ) ;
sfctemp - > clk_sense = devm_clk_get ( dev , " sense " ) ;
if ( IS_ERR ( sfctemp - > clk_sense ) )
return dev_err_probe ( dev , PTR_ERR ( sfctemp - > clk_sense ) ,
" error getting sense clock \n " ) ;
sfctemp - > clk_bus = devm_clk_get ( dev , " bus " ) ;
if ( IS_ERR ( sfctemp - > clk_bus ) )
return dev_err_probe ( dev , PTR_ERR ( sfctemp - > clk_bus ) ,
" error getting bus clock \n " ) ;
sfctemp - > rst_sense = devm_reset_control_get_exclusive ( dev , " sense " ) ;
if ( IS_ERR ( sfctemp - > rst_sense ) )
return dev_err_probe ( dev , PTR_ERR ( sfctemp - > rst_sense ) ,
" error getting sense reset \n " ) ;
sfctemp - > rst_bus = devm_reset_control_get_exclusive ( dev , " bus " ) ;
if ( IS_ERR ( sfctemp - > rst_bus ) )
return dev_err_probe ( dev , PTR_ERR ( sfctemp - > rst_bus ) ,
" error getting busreset \n " ) ;
ret = reset_control_assert ( sfctemp - > rst_sense ) ;
if ( ret )
return dev_err_probe ( dev , ret , " error asserting sense reset \n " ) ;
ret = reset_control_assert ( sfctemp - > rst_bus ) ;
if ( ret )
return dev_err_probe ( dev , ret , " error asserting bus reset \n " ) ;
ret = devm_add_action ( dev , sfctemp_disable_action , sfctemp ) ;
if ( ret )
return ret ;
ret = sfctemp_enable ( sfctemp ) ;
if ( ret )
2023-04-15 18:23:37 +02:00
return dev_err_probe ( dev , ret , " error enabling temperature sensor \n " ) ;
2023-03-21 10:26:44 +08:00
hwmon_dev = devm_hwmon_device_register_with_info ( dev , " sfctemp " , sfctemp ,
& sfctemp_chip_info , NULL ) ;
return PTR_ERR_OR_ZERO ( hwmon_dev ) ;
}
static const struct of_device_id sfctemp_of_match [ ] = {
{ . compatible = " starfive,jh7100-temp " } ,
{ . compatible = " starfive,jh7110-temp " } ,
{ /* sentinel */ }
} ;
MODULE_DEVICE_TABLE ( of , sfctemp_of_match ) ;
static struct platform_driver sfctemp_driver = {
. probe = sfctemp_probe ,
. driver = {
. name = " sfctemp " ,
. of_match_table = sfctemp_of_match ,
} ,
} ;
module_platform_driver ( sfctemp_driver ) ;
MODULE_AUTHOR ( " Emil Renner Berthing " ) ;
MODULE_DESCRIPTION ( " StarFive JH71x0 temperature sensor driver " ) ;
MODULE_LICENSE ( " GPL " ) ;