2019-05-29 07:17:56 -07:00
// SPDX-License-Identifier: GPL-2.0-only
2013-02-06 06:35:24 +00:00
/*
* Kirkwood thermal sensor driver
*
* Copyright ( C ) 2012 Nobuhiro Iwamatsu < iwamatsu @ nigauri . org >
*/
# include <linux/device.h>
# include <linux/err.h>
# include <linux/io.h>
# include <linux/kernel.h>
# include <linux/of.h>
# include <linux/module.h>
# include <linux/platform_device.h>
# include <linux/thermal.h>
# define KIRKWOOD_THERMAL_VALID_OFFSET 9
# define KIRKWOOD_THERMAL_VALID_MASK 0x1
# define KIRKWOOD_THERMAL_TEMP_OFFSET 10
# define KIRKWOOD_THERMAL_TEMP_MASK 0x1FF
/* Kirkwood Thermal Sensor Dev Structure */
struct kirkwood_thermal_priv {
void __iomem * sensor ;
} ;
static int kirkwood_get_temp ( struct thermal_zone_device * thermal ,
2015-07-24 08:12:54 +02:00
int * temp )
2013-02-06 06:35:24 +00:00
{
unsigned long reg ;
struct kirkwood_thermal_priv * priv = thermal - > devdata ;
reg = readl_relaxed ( priv - > sensor ) ;
/* Valid check */
2013-03-21 17:42:07 -03:00
if ( ! ( ( reg > > KIRKWOOD_THERMAL_VALID_OFFSET ) &
KIRKWOOD_THERMAL_VALID_MASK ) ) {
2013-02-06 06:35:24 +00:00
dev_err ( & thermal - > device ,
" Temperature sensor reading not valid \n " ) ;
return - EIO ;
}
/*
2013-03-22 09:23:02 -03:00
* Calculate temperature . According to Marvell internal
* documentation the formula for this is :
* Celsius = ( 322 - reg ) / 1.3625
2013-02-06 06:35:24 +00:00
*/
reg = ( reg > > KIRKWOOD_THERMAL_TEMP_OFFSET ) &
KIRKWOOD_THERMAL_TEMP_MASK ;
2013-03-22 09:23:02 -03:00
* temp = ( ( 3220000000UL - ( 10000000UL * reg ) ) / 13625 ) ;
2013-02-06 06:35:24 +00:00
return 0 ;
}
static struct thermal_zone_device_ops ops = {
. get_temp = kirkwood_get_temp ,
} ;
static const struct of_device_id kirkwood_thermal_id_table [ ] = {
{ . compatible = " marvell,kirkwood-thermal " } ,
{ }
} ;
static int kirkwood_thermal_probe ( struct platform_device * pdev )
{
struct thermal_zone_device * thermal = NULL ;
struct kirkwood_thermal_priv * priv ;
struct resource * res ;
2020-06-29 14:29:22 +02:00
int ret ;
2013-02-06 06:35:24 +00:00
priv = devm_kzalloc ( & pdev - > dev , sizeof ( * priv ) , GFP_KERNEL ) ;
if ( ! priv )
return - ENOMEM ;
2013-05-10 08:17:11 +00:00
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
2013-03-04 06:45:34 +00:00
priv - > sensor = devm_ioremap_resource ( & pdev - > dev , res ) ;
if ( IS_ERR ( priv - > sensor ) )
return PTR_ERR ( priv - > sensor ) ;
2013-02-06 06:35:24 +00:00
thermal = thermal_zone_device_register ( " kirkwood_thermal " , 0 , 0 ,
priv , & ops , NULL , 0 , 0 ) ;
if ( IS_ERR ( thermal ) ) {
dev_err ( & pdev - > dev ,
" Failed to register thermal zone device \n " ) ;
return PTR_ERR ( thermal ) ;
}
2020-06-29 14:29:22 +02:00
ret = thermal_zone_device_enable ( thermal ) ;
if ( ret ) {
thermal_zone_device_unregister ( thermal ) ;
dev_err ( & pdev - > dev , " Failed to enable thermal zone device \n " ) ;
return ret ;
}
2013-02-06 06:35:24 +00:00
platform_set_drvdata ( pdev , thermal ) ;
return 0 ;
}
static int kirkwood_thermal_exit ( struct platform_device * pdev )
{
struct thermal_zone_device * kirkwood_thermal =
platform_get_drvdata ( pdev ) ;
thermal_zone_device_unregister ( kirkwood_thermal ) ;
return 0 ;
}
MODULE_DEVICE_TABLE ( of , kirkwood_thermal_id_table ) ;
static struct platform_driver kirkwood_thermal_driver = {
. probe = kirkwood_thermal_probe ,
. remove = kirkwood_thermal_exit ,
. driver = {
. name = " kirkwood_thermal " ,
2013-05-16 10:28:10 +00:00
. of_match_table = kirkwood_thermal_id_table ,
2013-02-06 06:35:24 +00:00
} ,
} ;
module_platform_driver ( kirkwood_thermal_driver ) ;
MODULE_AUTHOR ( " Nobuhiro Iwamatsu <iwamatsu@nigauri.org> " ) ;
MODULE_DESCRIPTION ( " kirkwood thermal driver " ) ;
MODULE_LICENSE ( " GPL " ) ;