2018-09-26 18:27:40 +03:00
// SPDX-License-Identifier: GPL-2.0
2008-01-25 11:45:44 +08:00
/*
2018-09-26 18:27:40 +03:00
* Intel menlow Driver for thermal management extension
2008-01-25 11:45:44 +08:00
*
* Copyright ( C ) 2008 Intel Corp
* Copyright ( C ) 2008 Sujith Thomas < sujith . thomas @ intel . com >
* Copyright ( C ) 2008 Zhang Rui < rui . zhang @ intel . com >
*
* This driver creates the sys I / F for programming the sensors .
* It also implements the driver for intel menlow memory controller ( hardware
* id is INT0002 ) which makes use of the platform specific ACPI methods
* to get / set bandwidth .
*/
2011-03-29 15:21:44 -07:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2018-09-26 18:27:14 +03:00
# include <linux/acpi.h>
2008-01-25 11:45:44 +08:00
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/pci.h>
# include <linux/pm.h>
2018-09-26 18:27:14 +03:00
# include <linux/slab.h>
2008-01-25 11:45:44 +08:00
# include <linux/thermal.h>
2018-09-26 18:27:14 +03:00
# include <linux/types.h>
2008-01-25 11:45:44 +08:00
MODULE_AUTHOR ( " Thomas Sujith " ) ;
MODULE_AUTHOR ( " Zhang Rui " ) ;
MODULE_DESCRIPTION ( " Intel Menlow platform specific driver " ) ;
2018-09-26 18:27:40 +03:00
MODULE_LICENSE ( " GPL v2 " ) ;
2008-01-25 11:45:44 +08:00
/*
* Memory controller device control
*/
# define MEMORY_GET_BANDWIDTH "GTHS"
# define MEMORY_SET_BANDWIDTH "STHS"
# define MEMORY_ARG_CUR_BANDWIDTH 1
# define MEMORY_ARG_MAX_BANDWIDTH 0
2010-05-21 17:10:33 +08:00
static void intel_menlow_unregister_sensor ( void ) ;
2008-11-05 16:15:13 +05:30
/*
* GTHS returning ' n ' would mean that [ 0 , n - 1 ] states are supported
* In that case max_cstate would be n - 1
* GTHS returning ' 0 ' would mean that no bandwidth control states are supported
*/
2008-11-27 17:48:13 +00:00
static int memory_get_max_bandwidth ( struct thermal_cooling_device * cdev ,
unsigned long * max_state )
2008-01-25 11:45:44 +08:00
{
struct acpi_device * device = cdev - > devdata ;
acpi_handle handle = device - > handle ;
2008-10-10 02:22:59 -04:00
unsigned long long value ;
2008-01-25 11:45:44 +08:00
struct acpi_object_list arg_list ;
union acpi_object arg ;
acpi_status status = AE_OK ;
arg_list . count = 1 ;
arg_list . pointer = & arg ;
arg . type = ACPI_TYPE_INTEGER ;
arg . integer . value = MEMORY_ARG_MAX_BANDWIDTH ;
status = acpi_evaluate_integer ( handle , MEMORY_GET_BANDWIDTH ,
& arg_list , & value ) ;
if ( ACPI_FAILURE ( status ) )
return - EFAULT ;
2008-09-11 10:56:00 +08:00
if ( ! value )
return - EINVAL ;
2008-01-25 11:45:44 +08:00
* max_state = value - 1 ;
return 0 ;
}
static int memory_get_cur_bandwidth ( struct thermal_cooling_device * cdev ,
2008-11-27 17:48:13 +00:00
unsigned long * value )
2008-01-25 11:45:44 +08:00
{
struct acpi_device * device = cdev - > devdata ;
acpi_handle handle = device - > handle ;
2008-11-27 17:48:13 +00:00
unsigned long long result ;
2008-01-25 11:45:44 +08:00
struct acpi_object_list arg_list ;
union acpi_object arg ;
acpi_status status = AE_OK ;
arg_list . count = 1 ;
arg_list . pointer = & arg ;
arg . type = ACPI_TYPE_INTEGER ;
arg . integer . value = MEMORY_ARG_CUR_BANDWIDTH ;
status = acpi_evaluate_integer ( handle , MEMORY_GET_BANDWIDTH ,
2008-11-27 17:48:13 +00:00
& arg_list , & result ) ;
2008-01-25 11:45:44 +08:00
if ( ACPI_FAILURE ( status ) )
return - EFAULT ;
2008-11-27 17:48:13 +00:00
* value = result ;
return 0 ;
2008-01-25 11:45:44 +08:00
}
static int memory_set_cur_bandwidth ( struct thermal_cooling_device * cdev ,
2008-11-27 17:48:13 +00:00
unsigned long state )
2008-01-25 11:45:44 +08:00
{
struct acpi_device * device = cdev - > devdata ;
acpi_handle handle = device - > handle ;
struct acpi_object_list arg_list ;
union acpi_object arg ;
acpi_status status ;
2008-10-10 02:22:59 -04:00
unsigned long long temp ;
2008-01-25 11:45:44 +08:00
unsigned long max_state ;
2008-11-27 17:48:13 +00:00
if ( memory_get_max_bandwidth ( cdev , & max_state ) )
2008-01-25 11:45:44 +08:00
return - EFAULT ;
2008-09-11 10:56:00 +08:00
if ( state > max_state )
2008-01-25 11:45:44 +08:00
return - EINVAL ;
arg_list . count = 1 ;
arg_list . pointer = & arg ;
arg . type = ACPI_TYPE_INTEGER ;
arg . integer . value = state ;
status =
acpi_evaluate_integer ( handle , MEMORY_SET_BANDWIDTH , & arg_list ,
2008-10-10 02:22:59 -04:00
& temp ) ;
2008-01-25 11:45:44 +08:00
2011-03-29 15:21:44 -07:00
pr_info ( " Bandwidth value was %ld: status is %d \n " , state , status ) ;
2008-01-25 11:45:44 +08:00
if ( ACPI_FAILURE ( status ) )
return - EFAULT ;
return 0 ;
}
2017-06-21 12:34:17 +05:30
static const struct thermal_cooling_device_ops memory_cooling_ops = {
2008-01-25 11:45:44 +08:00
. get_max_state = memory_get_max_bandwidth ,
. get_cur_state = memory_get_cur_bandwidth ,
. set_cur_state = memory_set_cur_bandwidth ,
} ;
/*
* Memory Device Management
*/
static int intel_menlow_memory_add ( struct acpi_device * device )
{
int result = - ENODEV ;
struct thermal_cooling_device * cdev ;
if ( ! device )
return - EINVAL ;
2013-09-03 08:32:04 +08:00
if ( ! acpi_has_method ( device - > handle , MEMORY_GET_BANDWIDTH ) )
2008-01-25 11:45:44 +08:00
goto end ;
2013-09-03 08:32:04 +08:00
if ( ! acpi_has_method ( device - > handle , MEMORY_SET_BANDWIDTH ) )
2008-01-25 11:45:44 +08:00
goto end ;
cdev = thermal_cooling_device_register ( " Memory controller " , device ,
& memory_cooling_ops ) ;
2008-02-15 01:05:23 -05:00
if ( IS_ERR ( cdev ) ) {
result = PTR_ERR ( cdev ) ;
goto end ;
}
2008-09-22 14:37:34 -07:00
device - > driver_data = cdev ;
2008-04-29 00:59:30 -07:00
result = sysfs_create_link ( & device - > dev . kobj ,
& cdev - > device . kobj , " thermal_cooling " ) ;
if ( result )
goto unregister ;
result = sysfs_create_link ( & cdev - > device . kobj ,
& device - > dev . kobj , " device " ) ;
if ( result ) {
sysfs_remove_link ( & device - > dev . kobj , " thermal_cooling " ) ;
goto unregister ;
2008-01-25 11:45:44 +08:00
}
end :
return result ;
unregister :
thermal_cooling_device_unregister ( cdev ) ;
return result ;
}
2013-01-24 00:24:48 +01:00
static int intel_menlow_memory_remove ( struct acpi_device * device )
2008-01-25 11:45:44 +08:00
{
struct thermal_cooling_device * cdev = acpi_driver_data ( device ) ;
if ( ! device | | ! cdev )
return - EINVAL ;
sysfs_remove_link ( & device - > dev . kobj , " thermal_cooling " ) ;
sysfs_remove_link ( & cdev - > device . kobj , " device " ) ;
thermal_cooling_device_unregister ( cdev ) ;
return 0 ;
}
2008-04-21 22:28:49 +00:00
static const struct acpi_device_id intel_menlow_memory_ids [ ] = {
2008-01-25 11:45:44 +08:00
{ " INT0002 " , 0 } ,
{ " " , 0 } ,
} ;
static struct acpi_driver intel_menlow_memory_driver = {
. name = " intel_menlow_thermal_control " ,
. ids = intel_menlow_memory_ids ,
. ops = {
. add = intel_menlow_memory_add ,
. remove = intel_menlow_memory_remove ,
} ,
} ;
/*
* Sensor control on menlow platform
*/
# define THERMAL_AUX0 0
# define THERMAL_AUX1 1
# define GET_AUX0 "GAX0"
# define GET_AUX1 "GAX1"
# define SET_AUX0 "SAX0"
# define SET_AUX1 "SAX1"
struct intel_menlow_attribute {
struct device_attribute attr ;
struct device * device ;
acpi_handle handle ;
struct list_head node ;
} ;
static LIST_HEAD ( intel_menlow_attr_list ) ;
static DEFINE_MUTEX ( intel_menlow_attr_lock ) ;
/*
* sensor_get_auxtrip - get the current auxtrip value from sensor
* @ name : Thermalzone name
* @ auxtype : AUX0 / AUX1
* @ buf : syfs buffer
*/
2008-10-10 02:22:59 -04:00
static int sensor_get_auxtrip ( acpi_handle handle , int index ,
unsigned long long * value )
2008-01-25 11:45:44 +08:00
{
acpi_status status ;
if ( ( index ! = 0 & & index ! = 1 ) | | ! value )
return - EINVAL ;
status = acpi_evaluate_integer ( handle , index ? GET_AUX1 : GET_AUX0 ,
2008-10-10 02:22:59 -04:00
NULL , value ) ;
2008-01-25 11:45:44 +08:00
if ( ACPI_FAILURE ( status ) )
return - EIO ;
return 0 ;
}
/*
* sensor_set_auxtrip - set the new auxtrip value to sensor
* @ name : Thermalzone name
* @ auxtype : AUX0 / AUX1
* @ buf : syfs buffer
*/
static int sensor_set_auxtrip ( acpi_handle handle , int index , int value )
{
acpi_status status ;
union acpi_object arg = {
ACPI_TYPE_INTEGER
} ;
struct acpi_object_list args = {
1 , & arg
} ;
2008-10-10 02:22:59 -04:00
unsigned long long temp ;
2008-01-25 11:45:44 +08:00
if ( index ! = 0 & & index ! = 1 )
return - EINVAL ;
status = acpi_evaluate_integer ( handle , index ? GET_AUX0 : GET_AUX1 ,
2008-10-10 02:22:59 -04:00
NULL , & temp ) ;
2008-01-25 11:45:44 +08:00
if ( ACPI_FAILURE ( status ) )
return - EIO ;
if ( ( index & & value < temp ) | | ( ! index & & value > temp ) )
return - EINVAL ;
arg . integer . value = value ;
status = acpi_evaluate_integer ( handle , index ? SET_AUX1 : SET_AUX0 ,
2008-10-10 02:22:59 -04:00
& args , & temp ) ;
2008-01-25 11:45:44 +08:00
if ( ACPI_FAILURE ( status ) )
return - EIO ;
/* do we need to check the return value of SAX0/SAX1 ? */
return 0 ;
}
# define to_intel_menlow_attr(_attr) \
container_of ( _attr , struct intel_menlow_attribute , attr )
2016-03-26 22:40:26 +01:00
static ssize_t aux_show ( struct device * dev , struct device_attribute * dev_attr ,
char * buf , int idx )
2008-01-25 11:45:44 +08:00
{
struct intel_menlow_attribute * attr = to_intel_menlow_attr ( dev_attr ) ;
2008-10-10 02:22:59 -04:00
unsigned long long value ;
2008-01-25 11:45:44 +08:00
int result ;
2016-03-26 22:40:26 +01:00
result = sensor_get_auxtrip ( attr - > handle , idx , & value ) ;
2008-01-25 11:45:44 +08:00
2015-10-01 23:45:31 +02:00
return result ? result : sprintf ( buf , " %lu " , DECI_KELVIN_TO_CELSIUS ( value ) ) ;
2008-01-25 11:45:44 +08:00
}
2016-03-26 22:40:26 +01:00
static ssize_t aux0_show ( struct device * dev ,
2008-01-25 11:45:44 +08:00
struct device_attribute * dev_attr , char * buf )
{
2016-03-26 22:40:26 +01:00
return aux_show ( dev , dev_attr , buf , 0 ) ;
}
2008-01-25 11:45:44 +08:00
2016-03-26 22:40:26 +01:00
static ssize_t aux1_show ( struct device * dev ,
struct device_attribute * dev_attr , char * buf )
{
return aux_show ( dev , dev_attr , buf , 1 ) ;
2008-01-25 11:45:44 +08:00
}
2016-03-26 22:40:26 +01:00
static ssize_t aux_store ( struct device * dev , struct device_attribute * dev_attr ,
const char * buf , size_t count , int idx )
2008-01-25 11:45:44 +08:00
{
struct intel_menlow_attribute * attr = to_intel_menlow_attr ( dev_attr ) ;
int value ;
int result ;
/*Sanity check; should be a positive integer */
if ( ! sscanf ( buf , " %d " , & value ) )
return - EINVAL ;
if ( value < 0 )
return - EINVAL ;
2016-03-26 22:40:26 +01:00
result = sensor_set_auxtrip ( attr - > handle , idx ,
CELSIUS_TO_DECI_KELVIN ( value ) ) ;
2008-01-25 11:45:44 +08:00
return result ? result : count ;
}
2016-03-26 22:40:26 +01:00
static ssize_t aux0_store ( struct device * dev ,
2008-01-25 11:45:44 +08:00
struct device_attribute * dev_attr ,
const char * buf , size_t count )
{
2016-03-26 22:40:26 +01:00
return aux_store ( dev , dev_attr , buf , count , 0 ) ;
}
2008-01-25 11:45:44 +08:00
2016-03-26 22:40:26 +01:00
static ssize_t aux1_store ( struct device * dev ,
struct device_attribute * dev_attr ,
const char * buf , size_t count )
{
return aux_store ( dev , dev_attr , buf , count , 1 ) ;
2008-01-25 11:45:44 +08:00
}
/* BIOS can enable/disable the thermal user application in dabney platform */
# define BIOS_ENABLED "\\_TZ.GSTS"
static ssize_t bios_enabled_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
acpi_status status ;
2008-10-10 02:22:59 -04:00
unsigned long long bios_enabled ;
2008-01-25 11:45:44 +08:00
status = acpi_evaluate_integer ( NULL , BIOS_ENABLED , NULL , & bios_enabled ) ;
if ( ACPI_FAILURE ( status ) )
return - ENODEV ;
return sprintf ( buf , " %s \n " , bios_enabled ? " enabled " : " disabled " ) ;
}
2011-07-23 23:10:46 -04:00
static int intel_menlow_add_one_attribute ( char * name , umode_t mode , void * show ,
2008-01-25 11:45:44 +08:00
void * store , struct device * dev ,
acpi_handle handle )
{
struct intel_menlow_attribute * attr ;
int result ;
attr = kzalloc ( sizeof ( struct intel_menlow_attribute ) , GFP_KERNEL ) ;
if ( ! attr )
return - ENOMEM ;
2010-04-06 14:34:52 -07:00
sysfs_attr_init ( & attr - > attr . attr ) ; /* That is consistent naming :D */
2008-01-25 11:45:44 +08:00
attr - > attr . attr . name = name ;
attr - > attr . attr . mode = mode ;
attr - > attr . show = show ;
attr - > attr . store = store ;
attr - > device = dev ;
attr - > handle = handle ;
result = device_create_file ( dev , & attr - > attr ) ;
2010-05-21 17:10:33 +08:00
if ( result ) {
kfree ( attr ) ;
2008-01-25 11:45:44 +08:00
return result ;
2010-05-21 17:10:33 +08:00
}
2008-01-25 11:45:44 +08:00
mutex_lock ( & intel_menlow_attr_lock ) ;
list_add_tail ( & attr - > node , & intel_menlow_attr_list ) ;
mutex_unlock ( & intel_menlow_attr_lock ) ;
return 0 ;
}
static acpi_status intel_menlow_register_sensor ( acpi_handle handle , u32 lvl ,
void * context , void * * rv )
{
acpi_status status ;
acpi_handle dummy ;
struct thermal_zone_device * thermal ;
int result ;
result = acpi_bus_get_private_data ( handle , ( void * * ) & thermal ) ;
if ( result )
return 0 ;
/* _TZ must have the AUX0/1 methods */
status = acpi_get_handle ( handle , GET_AUX0 , & dummy ) ;
if ( ACPI_FAILURE ( status ) )
2010-05-21 17:10:33 +08:00
return ( status = = AE_NOT_FOUND ) ? AE_OK : status ;
2008-01-25 11:45:44 +08:00
status = acpi_get_handle ( handle , SET_AUX0 , & dummy ) ;
if ( ACPI_FAILURE ( status ) )
2010-05-21 17:10:33 +08:00
return ( status = = AE_NOT_FOUND ) ? AE_OK : status ;
2008-01-25 11:45:44 +08:00
result = intel_menlow_add_one_attribute ( " aux0 " , 0644 ,
aux0_show , aux0_store ,
& thermal - > device , handle ) ;
if ( result )
return AE_ERROR ;
status = acpi_get_handle ( handle , GET_AUX1 , & dummy ) ;
if ( ACPI_FAILURE ( status ) )
2010-05-21 17:10:33 +08:00
goto aux1_not_found ;
2008-01-25 11:45:44 +08:00
status = acpi_get_handle ( handle , SET_AUX1 , & dummy ) ;
if ( ACPI_FAILURE ( status ) )
2010-05-21 17:10:33 +08:00
goto aux1_not_found ;
2008-01-25 11:45:44 +08:00
result = intel_menlow_add_one_attribute ( " aux1 " , 0644 ,
aux1_show , aux1_store ,
& thermal - > device , handle ) ;
2010-05-21 17:10:33 +08:00
if ( result ) {
intel_menlow_unregister_sensor ( ) ;
2008-01-25 11:45:44 +08:00
return AE_ERROR ;
2010-05-21 17:10:33 +08:00
}
2008-01-25 11:45:44 +08:00
/*
* create the " dabney_enabled " attribute which means the user app
* should be loaded or not
*/
result = intel_menlow_add_one_attribute ( " bios_enabled " , 0444 ,
bios_enabled_show , NULL ,
& thermal - > device , handle ) ;
2010-05-21 17:10:33 +08:00
if ( result ) {
intel_menlow_unregister_sensor ( ) ;
2008-01-25 11:45:44 +08:00
return AE_ERROR ;
2010-05-21 17:10:33 +08:00
}
2008-01-25 11:45:44 +08:00
2011-07-07 10:05:49 +08:00
return AE_OK ;
2010-05-21 17:10:33 +08:00
aux1_not_found :
2008-01-25 11:45:44 +08:00
if ( status = = AE_NOT_FOUND )
return AE_OK ;
2010-05-21 17:10:33 +08:00
intel_menlow_unregister_sensor ( ) ;
return status ;
2008-01-25 11:45:44 +08:00
}
static void intel_menlow_unregister_sensor ( void )
{
struct intel_menlow_attribute * pos , * next ;
mutex_lock ( & intel_menlow_attr_lock ) ;
list_for_each_entry_safe ( pos , next , & intel_menlow_attr_list , node ) {
list_del ( & pos - > node ) ;
device_remove_file ( pos - > device , & pos - > attr ) ;
kfree ( pos ) ;
}
mutex_unlock ( & intel_menlow_attr_lock ) ;
return ;
}
static int __init intel_menlow_module_init ( void )
{
int result = - ENODEV ;
acpi_status status ;
2008-10-10 02:22:59 -04:00
unsigned long long enable ;
2008-01-25 11:45:44 +08:00
if ( acpi_disabled )
return result ;
/* Looking for the \_TZ.GSTS method */
status = acpi_evaluate_integer ( NULL , BIOS_ENABLED , NULL , & enable ) ;
if ( ACPI_FAILURE ( status ) | | ! enable )
return - ENODEV ;
/* Looking for ACPI device MEM0 with hardware id INT0002 */
result = acpi_bus_register_driver ( & intel_menlow_memory_driver ) ;
if ( result )
return result ;
/* Looking for sensors in each ACPI thermal zone */
status = acpi_walk_namespace ( ACPI_TYPE_THERMAL , ACPI_ROOT_OBJECT ,
ACPI_UINT32_MAX ,
2009-11-13 10:06:08 +08:00
intel_menlow_register_sensor , NULL , NULL , NULL ) ;
2010-05-21 17:10:33 +08:00
if ( ACPI_FAILURE ( status ) ) {
acpi_bus_unregister_driver ( & intel_menlow_memory_driver ) ;
2008-01-25 11:45:44 +08:00
return - ENODEV ;
2010-05-21 17:10:33 +08:00
}
2008-01-25 11:45:44 +08:00
return 0 ;
}
static void __exit intel_menlow_module_exit ( void )
{
acpi_bus_unregister_driver ( & intel_menlow_memory_driver ) ;
intel_menlow_unregister_sensor ( ) ;
}
module_init ( intel_menlow_module_init ) ;
module_exit ( intel_menlow_module_exit ) ;