Merge branches 'release' and 'menlo' into release
Conflicts: drivers/acpi/video.c Signed-off-by: Len Brown <len.brown@intel.com>
This commit is contained in:
commit
26b6f22366
|
@ -0,0 +1,246 @@
|
||||||
|
Generic Thermal Sysfs driver How To
|
||||||
|
=========================
|
||||||
|
|
||||||
|
Written by Sujith Thomas <sujith.thomas@intel.com>, Zhang Rui <rui.zhang@intel.com>
|
||||||
|
|
||||||
|
Updated: 2 January 2008
|
||||||
|
|
||||||
|
Copyright (c) 2008 Intel Corporation
|
||||||
|
|
||||||
|
|
||||||
|
0. Introduction
|
||||||
|
|
||||||
|
The generic thermal sysfs provides a set of interfaces for thermal zone devices (sensors)
|
||||||
|
and thermal cooling devices (fan, processor...) to register with the thermal management
|
||||||
|
solution and to be a part of it.
|
||||||
|
|
||||||
|
This how-to focusses on enabling new thermal zone and cooling devices to participate
|
||||||
|
in thermal management.
|
||||||
|
This solution is platform independent and any type of thermal zone devices and
|
||||||
|
cooling devices should be able to make use of the infrastructure.
|
||||||
|
|
||||||
|
The main task of the thermal sysfs driver is to expose thermal zone attributes as well
|
||||||
|
as cooling device attributes to the user space.
|
||||||
|
An intelligent thermal management application can make decisions based on inputs
|
||||||
|
from thermal zone attributes (the current temperature and trip point temperature)
|
||||||
|
and throttle appropriate devices.
|
||||||
|
|
||||||
|
[0-*] denotes any positive number starting from 0
|
||||||
|
[1-*] denotes any positive number starting from 1
|
||||||
|
|
||||||
|
1. thermal sysfs driver interface functions
|
||||||
|
|
||||||
|
1.1 thermal zone device interface
|
||||||
|
1.1.1 struct thermal_zone_device *thermal_zone_device_register(char *name, int trips,
|
||||||
|
void *devdata, struct thermal_zone_device_ops *ops)
|
||||||
|
|
||||||
|
This interface function adds a new thermal zone device (sensor) to
|
||||||
|
/sys/class/thermal folder as thermal_zone[0-*].
|
||||||
|
It tries to bind all the thermal cooling devices registered at the same time.
|
||||||
|
|
||||||
|
name: the thermal zone name.
|
||||||
|
trips: the total number of trip points this thermal zone supports.
|
||||||
|
devdata: device private data
|
||||||
|
ops: thermal zone device callbacks.
|
||||||
|
.bind: bind the thermal zone device with a thermal cooling device.
|
||||||
|
.unbind: unbing the thermal zone device with a thermal cooling device.
|
||||||
|
.get_temp: get the current temperature of the thermal zone.
|
||||||
|
.get_mode: get the current mode (user/kernel) of the thermal zone.
|
||||||
|
"kernel" means thermal management is done in kernel.
|
||||||
|
"user" will prevent kernel thermal driver actions upon trip points
|
||||||
|
so that user applications can take charge of thermal management.
|
||||||
|
.set_mode: set the mode (user/kernel) of the thermal zone.
|
||||||
|
.get_trip_type: get the type of certain trip point.
|
||||||
|
.get_trip_temp: get the temperature above which the certain trip point
|
||||||
|
will be fired.
|
||||||
|
|
||||||
|
1.1.2 void thermal_zone_device_unregister(struct thermal_zone_device *tz)
|
||||||
|
|
||||||
|
This interface function removes the thermal zone device.
|
||||||
|
It deletes the corresponding entry form /sys/class/thermal folder and unbind all
|
||||||
|
the thermal cooling devices it uses.
|
||||||
|
|
||||||
|
1.2 thermal cooling device interface
|
||||||
|
1.2.1 struct thermal_cooling_device *thermal_cooling_device_register(char *name,
|
||||||
|
void *devdata, struct thermal_cooling_device_ops *)
|
||||||
|
|
||||||
|
This interface function adds a new thermal cooling device (fan/processor/...) to
|
||||||
|
/sys/class/thermal/ folder as cooling_device[0-*].
|
||||||
|
It tries to bind itself to all the thermal zone devices register at the same time.
|
||||||
|
name: the cooling device name.
|
||||||
|
devdata: device private data.
|
||||||
|
ops: thermal cooling devices callbacks.
|
||||||
|
.get_max_state: get the Maximum throttle state of the cooling device.
|
||||||
|
.get_cur_state: get the Current throttle state of the cooling device.
|
||||||
|
.set_cur_state: set the Current throttle state of the cooling device.
|
||||||
|
|
||||||
|
1.2.2 void thermal_cooling_device_unregister(struct thermal_cooling_device *cdev)
|
||||||
|
|
||||||
|
This interface function remove the thermal cooling device.
|
||||||
|
It deletes the corresponding entry form /sys/class/thermal folder and unbind
|
||||||
|
itself from all the thermal zone devices using it.
|
||||||
|
|
||||||
|
1.3 interface for binding a thermal zone device with a thermal cooling device
|
||||||
|
1.3.1 int thermal_zone_bind_cooling_device(struct thermal_zone_device *tz,
|
||||||
|
int trip, struct thermal_cooling_device *cdev);
|
||||||
|
|
||||||
|
This interface function bind a thermal cooling device to the certain trip point
|
||||||
|
of a thermal zone device.
|
||||||
|
This function is usually called in the thermal zone device .bind callback.
|
||||||
|
tz: the thermal zone device
|
||||||
|
cdev: thermal cooling device
|
||||||
|
trip: indicates which trip point the cooling devices is associated with
|
||||||
|
in this thermal zone.
|
||||||
|
|
||||||
|
1.3.2 int thermal_zone_unbind_cooling_device(struct thermal_zone_device *tz,
|
||||||
|
int trip, struct thermal_cooling_device *cdev);
|
||||||
|
|
||||||
|
This interface function unbind a thermal cooling device from the certain trip point
|
||||||
|
of a thermal zone device.
|
||||||
|
This function is usually called in the thermal zone device .unbind callback.
|
||||||
|
tz: the thermal zone device
|
||||||
|
cdev: thermal cooling device
|
||||||
|
trip: indicates which trip point the cooling devices is associated with
|
||||||
|
in this thermal zone.
|
||||||
|
|
||||||
|
2. sysfs attributes structure
|
||||||
|
|
||||||
|
RO read only value
|
||||||
|
RW read/write value
|
||||||
|
|
||||||
|
All thermal sysfs attributes will be represented under /sys/class/thermal
|
||||||
|
/sys/class/thermal/
|
||||||
|
|
||||||
|
Thermal zone device sys I/F, created once it's registered:
|
||||||
|
|thermal_zone[0-*]:
|
||||||
|
|-----type: Type of the thermal zone
|
||||||
|
|-----temp: Current temperature
|
||||||
|
|-----mode: Working mode of the thermal zone
|
||||||
|
|-----trip_point_[0-*]_temp: Trip point temperature
|
||||||
|
|-----trip_point_[0-*]_type: Trip point type
|
||||||
|
|
||||||
|
Thermal cooling device sys I/F, created once it's registered:
|
||||||
|
|cooling_device[0-*]:
|
||||||
|
|-----type : Type of the cooling device(processor/fan/...)
|
||||||
|
|-----max_state: Maximum cooling state of the cooling device
|
||||||
|
|-----cur_state: Current cooling state of the cooling device
|
||||||
|
|
||||||
|
|
||||||
|
These two dynamic attributes are created/removed in pairs.
|
||||||
|
They represent the relationship between a thermal zone and its associated cooling device.
|
||||||
|
They are created/removed for each
|
||||||
|
thermal_zone_bind_cooling_device/thermal_zone_unbind_cooling_device successful exection.
|
||||||
|
|
||||||
|
|thermal_zone[0-*]
|
||||||
|
|-----cdev[0-*]: The [0-*]th cooling device in the current thermal zone
|
||||||
|
|-----cdev[0-*]_trip_point: Trip point that cdev[0-*] is associated with
|
||||||
|
|
||||||
|
|
||||||
|
***************************
|
||||||
|
* Thermal zone attributes *
|
||||||
|
***************************
|
||||||
|
|
||||||
|
type Strings which represent the thermal zone type.
|
||||||
|
This is given by thermal zone driver as part of registration.
|
||||||
|
Eg: "ACPI thermal zone" indicates it's a ACPI thermal device
|
||||||
|
RO
|
||||||
|
Optional
|
||||||
|
|
||||||
|
temp Current temperature as reported by thermal zone (sensor)
|
||||||
|
Unit: degree celsius
|
||||||
|
RO
|
||||||
|
Required
|
||||||
|
|
||||||
|
mode One of the predifned values in [kernel, user]
|
||||||
|
This file gives information about the algorithm
|
||||||
|
that is currently managing the thermal zone.
|
||||||
|
It can be either default kernel based algorithm
|
||||||
|
or user space application.
|
||||||
|
RW
|
||||||
|
Optional
|
||||||
|
kernel = Thermal management in kernel thermal zone driver.
|
||||||
|
user = Preventing kernel thermal zone driver actions upon
|
||||||
|
trip points so that user application can take full
|
||||||
|
charge of the thermal management.
|
||||||
|
|
||||||
|
trip_point_[0-*]_temp The temperature above which trip point will be fired
|
||||||
|
Unit: degree celsius
|
||||||
|
RO
|
||||||
|
Optional
|
||||||
|
|
||||||
|
trip_point_[0-*]_type Strings which indicate the type of the trip point
|
||||||
|
Eg. it can be one of critical, hot, passive,
|
||||||
|
active[0-*] for ACPI thermal zone.
|
||||||
|
RO
|
||||||
|
Optional
|
||||||
|
|
||||||
|
cdev[0-*] Sysfs link to the thermal cooling device node where the sys I/F
|
||||||
|
for cooling device throttling control represents.
|
||||||
|
RO
|
||||||
|
Optional
|
||||||
|
|
||||||
|
cdev[0-*]_trip_point The trip point with which cdev[0-*] is assocated in this thermal zone
|
||||||
|
-1 means the cooling device is not associated with any trip point.
|
||||||
|
RO
|
||||||
|
Optional
|
||||||
|
|
||||||
|
******************************
|
||||||
|
* Cooling device attributes *
|
||||||
|
******************************
|
||||||
|
|
||||||
|
type String which represents the type of device
|
||||||
|
eg: For generic ACPI: this should be "Fan",
|
||||||
|
"Processor" or "LCD"
|
||||||
|
eg. For memory controller device on intel_menlow platform:
|
||||||
|
this should be "Memory controller"
|
||||||
|
RO
|
||||||
|
Optional
|
||||||
|
|
||||||
|
max_state The maximum permissible cooling state of this cooling device.
|
||||||
|
RO
|
||||||
|
Required
|
||||||
|
|
||||||
|
cur_state The current cooling state of this cooling device.
|
||||||
|
the value can any integer numbers between 0 and max_state,
|
||||||
|
cur_state == 0 means no cooling
|
||||||
|
cur_state == max_state means the maximum cooling.
|
||||||
|
RW
|
||||||
|
Required
|
||||||
|
|
||||||
|
3. A simple implementation
|
||||||
|
|
||||||
|
ACPI thermal zone may support multiple trip points like critical/hot/passive/active.
|
||||||
|
If an ACPI thermal zone supports critical, passive, active[0] and active[1] at the same time,
|
||||||
|
it may register itself as a thermale_zone_device (thermal_zone1) with 4 trip points in all.
|
||||||
|
It has one processor and one fan, which are both registered as thermal_cooling_device.
|
||||||
|
If the processor is listed in _PSL method, and the fan is listed in _AL0 method,
|
||||||
|
the sys I/F structure will be built like this:
|
||||||
|
|
||||||
|
/sys/class/thermal:
|
||||||
|
|
||||||
|
|thermal_zone1:
|
||||||
|
|-----type: ACPI thermal zone
|
||||||
|
|-----temp: 37
|
||||||
|
|-----mode: kernel
|
||||||
|
|-----trip_point_0_temp: 100
|
||||||
|
|-----trip_point_0_type: critical
|
||||||
|
|-----trip_point_1_temp: 80
|
||||||
|
|-----trip_point_1_type: passive
|
||||||
|
|-----trip_point_2_temp: 70
|
||||||
|
|-----trip_point_2_type: active[0]
|
||||||
|
|-----trip_point_3_temp: 60
|
||||||
|
|-----trip_point_3_type: active[1]
|
||||||
|
|-----cdev0: --->/sys/class/thermal/cooling_device0
|
||||||
|
|-----cdev0_trip_point: 1 /* cdev0 can be used for passive */
|
||||||
|
|-----cdev1: --->/sys/class/thermal/cooling_device3
|
||||||
|
|-----cdev1_trip_point: 2 /* cdev1 can be used for active[0]*/
|
||||||
|
|
||||||
|
|cooling_device0:
|
||||||
|
|-----type: Processor
|
||||||
|
|-----max_state: 8
|
||||||
|
|-----cur_state: 0
|
||||||
|
|
||||||
|
|cooling_device3:
|
||||||
|
|-----type: Fan
|
||||||
|
|-----max_state: 2
|
||||||
|
|-----cur_state: 0
|
|
@ -60,6 +60,8 @@ source "drivers/power/Kconfig"
|
||||||
|
|
||||||
source "drivers/hwmon/Kconfig"
|
source "drivers/hwmon/Kconfig"
|
||||||
|
|
||||||
|
source "drivers/thermal/Kconfig"
|
||||||
|
|
||||||
source "drivers/watchdog/Kconfig"
|
source "drivers/watchdog/Kconfig"
|
||||||
|
|
||||||
source "drivers/ssb/Kconfig"
|
source "drivers/ssb/Kconfig"
|
||||||
|
|
|
@ -65,6 +65,7 @@ obj-y += i2c/
|
||||||
obj-$(CONFIG_W1) += w1/
|
obj-$(CONFIG_W1) += w1/
|
||||||
obj-$(CONFIG_POWER_SUPPLY) += power/
|
obj-$(CONFIG_POWER_SUPPLY) += power/
|
||||||
obj-$(CONFIG_HWMON) += hwmon/
|
obj-$(CONFIG_HWMON) += hwmon/
|
||||||
|
obj-$(CONFIG_THERMAL) += thermal/
|
||||||
obj-$(CONFIG_WATCHDOG) += watchdog/
|
obj-$(CONFIG_WATCHDOG) += watchdog/
|
||||||
obj-$(CONFIG_PHONE) += telephony/
|
obj-$(CONFIG_PHONE) += telephony/
|
||||||
obj-$(CONFIG_MD) += md/
|
obj-$(CONFIG_MD) += md/
|
||||||
|
|
|
@ -186,6 +186,7 @@ config ACPI_HOTPLUG_CPU
|
||||||
config ACPI_THERMAL
|
config ACPI_THERMAL
|
||||||
tristate "Thermal Zone"
|
tristate "Thermal Zone"
|
||||||
depends on ACPI_PROCESSOR
|
depends on ACPI_PROCESSOR
|
||||||
|
select THERMAL
|
||||||
default y
|
default y
|
||||||
help
|
help
|
||||||
This driver adds support for ACPI thermal zones. Most mobile and
|
This driver adds support for ACPI thermal zones. Most mobile and
|
||||||
|
|
|
@ -122,6 +122,31 @@ int acpi_bus_get_status(struct acpi_device *device)
|
||||||
|
|
||||||
EXPORT_SYMBOL(acpi_bus_get_status);
|
EXPORT_SYMBOL(acpi_bus_get_status);
|
||||||
|
|
||||||
|
void acpi_bus_private_data_handler(acpi_handle handle,
|
||||||
|
u32 function, void *context)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(acpi_bus_private_data_handler);
|
||||||
|
|
||||||
|
int acpi_bus_get_private_data(acpi_handle handle, void **data)
|
||||||
|
{
|
||||||
|
acpi_status status = AE_OK;
|
||||||
|
|
||||||
|
if (!*data)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
status = acpi_get_data(handle, acpi_bus_private_data_handler, data);
|
||||||
|
if (ACPI_FAILURE(status) || !*data) {
|
||||||
|
ACPI_DEBUG_PRINT((ACPI_DB_INFO, "No context for object [%p]\n",
|
||||||
|
handle));
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(acpi_bus_get_private_data);
|
||||||
|
|
||||||
/* --------------------------------------------------------------------------
|
/* --------------------------------------------------------------------------
|
||||||
Power Management
|
Power Management
|
||||||
-------------------------------------------------------------------------- */
|
-------------------------------------------------------------------------- */
|
||||||
|
|
|
@ -30,7 +30,7 @@
|
||||||
#include <linux/proc_fs.h>
|
#include <linux/proc_fs.h>
|
||||||
#include <linux/seq_file.h>
|
#include <linux/seq_file.h>
|
||||||
#include <asm/uaccess.h>
|
#include <asm/uaccess.h>
|
||||||
|
#include <linux/thermal.h>
|
||||||
#include <acpi/acpi_bus.h>
|
#include <acpi/acpi_bus.h>
|
||||||
#include <acpi/acpi_drivers.h>
|
#include <acpi/acpi_drivers.h>
|
||||||
|
|
||||||
|
@ -68,9 +68,55 @@ static struct acpi_driver acpi_fan_driver = {
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/* thermal cooling device callbacks */
|
||||||
|
static int fan_get_max_state(struct thermal_cooling_device *cdev, char *buf)
|
||||||
|
{
|
||||||
|
/* ACPI fan device only support two states: ON/OFF */
|
||||||
|
return sprintf(buf, "1\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int fan_get_cur_state(struct thermal_cooling_device *cdev, char *buf)
|
||||||
|
{
|
||||||
|
struct acpi_device *device = cdev->devdata;
|
||||||
|
int state;
|
||||||
|
int result;
|
||||||
|
|
||||||
|
if (!device)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
result = acpi_bus_get_power(device->handle, &state);
|
||||||
|
if (result)
|
||||||
|
return result;
|
||||||
|
|
||||||
|
return sprintf(buf, "%s\n", state == ACPI_STATE_D3 ? "0" :
|
||||||
|
(state == ACPI_STATE_D0 ? "1" : "unknown"));
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
fan_set_cur_state(struct thermal_cooling_device *cdev, unsigned int state)
|
||||||
|
{
|
||||||
|
struct acpi_device *device = cdev->devdata;
|
||||||
|
int result;
|
||||||
|
|
||||||
|
if (!device || (state != 0 && state != 1))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
result = acpi_bus_set_power(device->handle,
|
||||||
|
state ? ACPI_STATE_D0 : ACPI_STATE_D3);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct thermal_cooling_device_ops fan_cooling_ops = {
|
||||||
|
.get_max_state = fan_get_max_state,
|
||||||
|
.get_cur_state = fan_get_cur_state,
|
||||||
|
.set_cur_state = fan_set_cur_state,
|
||||||
|
};
|
||||||
|
|
||||||
/* --------------------------------------------------------------------------
|
/* --------------------------------------------------------------------------
|
||||||
FS Interface (/proc)
|
FS Interface (/proc)
|
||||||
-------------------------------------------------------------------------- */
|
-------------------------------------------------------------------------- */
|
||||||
|
#ifdef CONFIG_ACPI_PROCFS
|
||||||
|
|
||||||
static struct proc_dir_entry *acpi_fan_dir;
|
static struct proc_dir_entry *acpi_fan_dir;
|
||||||
|
|
||||||
|
@ -171,7 +217,17 @@ static int acpi_fan_remove_fs(struct acpi_device *device)
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
#else
|
||||||
|
static int acpi_fan_add_fs(struct acpi_device *device)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int acpi_fan_remove_fs(struct acpi_device *device)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
/* --------------------------------------------------------------------------
|
/* --------------------------------------------------------------------------
|
||||||
Driver Interface
|
Driver Interface
|
||||||
-------------------------------------------------------------------------- */
|
-------------------------------------------------------------------------- */
|
||||||
|
@ -179,9 +235,8 @@ static int acpi_fan_remove_fs(struct acpi_device *device)
|
||||||
static int acpi_fan_add(struct acpi_device *device)
|
static int acpi_fan_add(struct acpi_device *device)
|
||||||
{
|
{
|
||||||
int result = 0;
|
int result = 0;
|
||||||
struct acpi_fan *fan = NULL;
|
|
||||||
int state = 0;
|
int state = 0;
|
||||||
|
struct thermal_cooling_device *cdev;
|
||||||
|
|
||||||
if (!device)
|
if (!device)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
@ -199,6 +254,25 @@ static int acpi_fan_add(struct acpi_device *device)
|
||||||
acpi_bus_set_power(device->handle, state);
|
acpi_bus_set_power(device->handle, state);
|
||||||
device->flags.force_power_state = 0;
|
device->flags.force_power_state = 0;
|
||||||
|
|
||||||
|
cdev = thermal_cooling_device_register("Fan", device,
|
||||||
|
&fan_cooling_ops);
|
||||||
|
if (cdev)
|
||||||
|
printk(KERN_INFO PREFIX
|
||||||
|
"%s is registered as cooling_device%d\n",
|
||||||
|
device->dev.bus_id, cdev->id);
|
||||||
|
else
|
||||||
|
goto end;
|
||||||
|
acpi_driver_data(device) = cdev;
|
||||||
|
result = sysfs_create_link(&device->dev.kobj, &cdev->device.kobj,
|
||||||
|
"thermal_cooling");
|
||||||
|
if (result)
|
||||||
|
return result;
|
||||||
|
|
||||||
|
result = sysfs_create_link(&cdev->device.kobj, &device->dev.kobj,
|
||||||
|
"device");
|
||||||
|
if (result)
|
||||||
|
return result;
|
||||||
|
|
||||||
result = acpi_fan_add_fs(device);
|
result = acpi_fan_add_fs(device);
|
||||||
if (result)
|
if (result)
|
||||||
goto end;
|
goto end;
|
||||||
|
@ -208,18 +282,20 @@ static int acpi_fan_add(struct acpi_device *device)
|
||||||
!device->power.state ? "on" : "off");
|
!device->power.state ? "on" : "off");
|
||||||
|
|
||||||
end:
|
end:
|
||||||
if (result)
|
|
||||||
kfree(fan);
|
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int acpi_fan_remove(struct acpi_device *device, int type)
|
static int acpi_fan_remove(struct acpi_device *device, int type)
|
||||||
{
|
{
|
||||||
if (!device || !acpi_driver_data(device))
|
struct thermal_cooling_device *cdev = acpi_driver_data(device);
|
||||||
|
|
||||||
|
if (!device || !cdev)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
acpi_fan_remove_fs(device);
|
acpi_fan_remove_fs(device);
|
||||||
|
sysfs_remove_link(&device->dev.kobj, "thermal_cooling");
|
||||||
|
sysfs_remove_link(&cdev->device.kobj, "device");
|
||||||
|
thermal_cooling_device_unregister(cdev);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -261,10 +337,12 @@ static int __init acpi_fan_init(void)
|
||||||
int result = 0;
|
int result = 0;
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef CONFIG_ACPI_PROCFS
|
||||||
acpi_fan_dir = proc_mkdir(ACPI_FAN_CLASS, acpi_root_dir);
|
acpi_fan_dir = proc_mkdir(ACPI_FAN_CLASS, acpi_root_dir);
|
||||||
if (!acpi_fan_dir)
|
if (!acpi_fan_dir)
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
acpi_fan_dir->owner = THIS_MODULE;
|
acpi_fan_dir->owner = THIS_MODULE;
|
||||||
|
#endif
|
||||||
|
|
||||||
result = acpi_bus_register_driver(&acpi_fan_driver);
|
result = acpi_bus_register_driver(&acpi_fan_driver);
|
||||||
if (result < 0) {
|
if (result < 0) {
|
||||||
|
|
|
@ -668,6 +668,24 @@ static int __cpuinit acpi_processor_start(struct acpi_device *device)
|
||||||
|
|
||||||
acpi_processor_power_init(pr, device);
|
acpi_processor_power_init(pr, device);
|
||||||
|
|
||||||
|
pr->cdev = thermal_cooling_device_register("Processor", device,
|
||||||
|
&processor_cooling_ops);
|
||||||
|
if (pr->cdev)
|
||||||
|
printk(KERN_INFO PREFIX
|
||||||
|
"%s is registered as cooling_device%d\n",
|
||||||
|
device->dev.bus_id, pr->cdev->id);
|
||||||
|
else
|
||||||
|
goto end;
|
||||||
|
|
||||||
|
result = sysfs_create_link(&device->dev.kobj, &pr->cdev->device.kobj,
|
||||||
|
"thermal_cooling");
|
||||||
|
if (result)
|
||||||
|
return result;
|
||||||
|
result = sysfs_create_link(&pr->cdev->device.kobj, &device->dev.kobj,
|
||||||
|
"device");
|
||||||
|
if (result)
|
||||||
|
return result;
|
||||||
|
|
||||||
if (pr->flags.throttling) {
|
if (pr->flags.throttling) {
|
||||||
printk(KERN_INFO PREFIX "%s [%s] (supports",
|
printk(KERN_INFO PREFIX "%s [%s] (supports",
|
||||||
acpi_device_name(device), acpi_device_bid(device));
|
acpi_device_name(device), acpi_device_bid(device));
|
||||||
|
@ -791,6 +809,11 @@ static int acpi_processor_remove(struct acpi_device *device, int type)
|
||||||
|
|
||||||
acpi_processor_remove_fs(device);
|
acpi_processor_remove_fs(device);
|
||||||
|
|
||||||
|
sysfs_remove_link(&device->dev.kobj, "thermal_cooling");
|
||||||
|
sysfs_remove_link(&pr->cdev->device.kobj, "device");
|
||||||
|
thermal_cooling_device_unregister(pr->cdev);
|
||||||
|
pr->cdev = NULL;
|
||||||
|
|
||||||
processors[pr->id] = NULL;
|
processors[pr->id] = NULL;
|
||||||
|
|
||||||
kfree(pr);
|
kfree(pr);
|
||||||
|
|
|
@ -32,6 +32,7 @@
|
||||||
#include <linux/cpufreq.h>
|
#include <linux/cpufreq.h>
|
||||||
#include <linux/proc_fs.h>
|
#include <linux/proc_fs.h>
|
||||||
#include <linux/seq_file.h>
|
#include <linux/seq_file.h>
|
||||||
|
#include <linux/sysdev.h>
|
||||||
|
|
||||||
#include <asm/uaccess.h>
|
#include <asm/uaccess.h>
|
||||||
|
|
||||||
|
@ -93,6 +94,9 @@ static int acpi_processor_apply_limit(struct acpi_processor *pr)
|
||||||
* _any_ cpufreq driver and not only the acpi-cpufreq driver.
|
* _any_ cpufreq driver and not only the acpi-cpufreq driver.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define CPUFREQ_THERMAL_MIN_STEP 0
|
||||||
|
#define CPUFREQ_THERMAL_MAX_STEP 3
|
||||||
|
|
||||||
static unsigned int cpufreq_thermal_reduction_pctg[NR_CPUS];
|
static unsigned int cpufreq_thermal_reduction_pctg[NR_CPUS];
|
||||||
static unsigned int acpi_thermal_cpufreq_is_init = 0;
|
static unsigned int acpi_thermal_cpufreq_is_init = 0;
|
||||||
|
|
||||||
|
@ -109,8 +113,9 @@ static int acpi_thermal_cpufreq_increase(unsigned int cpu)
|
||||||
if (!cpu_has_cpufreq(cpu))
|
if (!cpu_has_cpufreq(cpu))
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
|
|
||||||
if (cpufreq_thermal_reduction_pctg[cpu] < 60) {
|
if (cpufreq_thermal_reduction_pctg[cpu] <
|
||||||
cpufreq_thermal_reduction_pctg[cpu] += 20;
|
CPUFREQ_THERMAL_MAX_STEP) {
|
||||||
|
cpufreq_thermal_reduction_pctg[cpu]++;
|
||||||
cpufreq_update_policy(cpu);
|
cpufreq_update_policy(cpu);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -123,8 +128,9 @@ static int acpi_thermal_cpufreq_decrease(unsigned int cpu)
|
||||||
if (!cpu_has_cpufreq(cpu))
|
if (!cpu_has_cpufreq(cpu))
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
|
|
||||||
if (cpufreq_thermal_reduction_pctg[cpu] > 20)
|
if (cpufreq_thermal_reduction_pctg[cpu] >
|
||||||
cpufreq_thermal_reduction_pctg[cpu] -= 20;
|
(CPUFREQ_THERMAL_MIN_STEP + 1))
|
||||||
|
cpufreq_thermal_reduction_pctg[cpu]--;
|
||||||
else
|
else
|
||||||
cpufreq_thermal_reduction_pctg[cpu] = 0;
|
cpufreq_thermal_reduction_pctg[cpu] = 0;
|
||||||
cpufreq_update_policy(cpu);
|
cpufreq_update_policy(cpu);
|
||||||
|
@ -143,7 +149,7 @@ static int acpi_thermal_cpufreq_notifier(struct notifier_block *nb,
|
||||||
|
|
||||||
max_freq =
|
max_freq =
|
||||||
(policy->cpuinfo.max_freq *
|
(policy->cpuinfo.max_freq *
|
||||||
(100 - cpufreq_thermal_reduction_pctg[policy->cpu])) / 100;
|
(100 - cpufreq_thermal_reduction_pctg[policy->cpu] * 20)) / 100;
|
||||||
|
|
||||||
cpufreq_verify_within_limits(policy, 0, max_freq);
|
cpufreq_verify_within_limits(policy, 0, max_freq);
|
||||||
|
|
||||||
|
@ -155,6 +161,32 @@ static struct notifier_block acpi_thermal_cpufreq_notifier_block = {
|
||||||
.notifier_call = acpi_thermal_cpufreq_notifier,
|
.notifier_call = acpi_thermal_cpufreq_notifier,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static int cpufreq_get_max_state(unsigned int cpu)
|
||||||
|
{
|
||||||
|
if (!cpu_has_cpufreq(cpu))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return CPUFREQ_THERMAL_MAX_STEP;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int cpufreq_get_cur_state(unsigned int cpu)
|
||||||
|
{
|
||||||
|
if (!cpu_has_cpufreq(cpu))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return cpufreq_thermal_reduction_pctg[cpu];
|
||||||
|
}
|
||||||
|
|
||||||
|
static int cpufreq_set_cur_state(unsigned int cpu, int state)
|
||||||
|
{
|
||||||
|
if (!cpu_has_cpufreq(cpu))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
cpufreq_thermal_reduction_pctg[cpu] = state;
|
||||||
|
cpufreq_update_policy(cpu);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
void acpi_thermal_cpufreq_init(void)
|
void acpi_thermal_cpufreq_init(void)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
@ -179,6 +211,20 @@ void acpi_thermal_cpufreq_exit(void)
|
||||||
}
|
}
|
||||||
|
|
||||||
#else /* ! CONFIG_CPU_FREQ */
|
#else /* ! CONFIG_CPU_FREQ */
|
||||||
|
static int cpufreq_get_max_state(unsigned int cpu)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int cpufreq_get_cur_state(unsigned int cpu)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int cpufreq_set_cur_state(unsigned int cpu, int state)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static int acpi_thermal_cpufreq_increase(unsigned int cpu)
|
static int acpi_thermal_cpufreq_increase(unsigned int cpu)
|
||||||
{
|
{
|
||||||
|
@ -310,6 +356,84 @@ int acpi_processor_get_limit_info(struct acpi_processor *pr)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* thermal coolign device callbacks */
|
||||||
|
static int acpi_processor_max_state(struct acpi_processor *pr)
|
||||||
|
{
|
||||||
|
int max_state = 0;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* There exists four states according to
|
||||||
|
* cpufreq_thermal_reduction_ptg. 0, 1, 2, 3
|
||||||
|
*/
|
||||||
|
max_state += cpufreq_get_max_state(pr->id);
|
||||||
|
if (pr->flags.throttling)
|
||||||
|
max_state += (pr->throttling.state_count -1);
|
||||||
|
|
||||||
|
return max_state;
|
||||||
|
}
|
||||||
|
static int
|
||||||
|
processor_get_max_state(struct thermal_cooling_device *cdev, char *buf)
|
||||||
|
{
|
||||||
|
struct acpi_device *device = cdev->devdata;
|
||||||
|
struct acpi_processor *pr = acpi_driver_data(device);
|
||||||
|
|
||||||
|
if (!device || !pr)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
return sprintf(buf, "%d\n", acpi_processor_max_state(pr));
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
processor_get_cur_state(struct thermal_cooling_device *cdev, char *buf)
|
||||||
|
{
|
||||||
|
struct acpi_device *device = cdev->devdata;
|
||||||
|
struct acpi_processor *pr = acpi_driver_data(device);
|
||||||
|
int cur_state;
|
||||||
|
|
||||||
|
if (!device || !pr)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
cur_state = cpufreq_get_cur_state(pr->id);
|
||||||
|
if (pr->flags.throttling)
|
||||||
|
cur_state += pr->throttling.state;
|
||||||
|
|
||||||
|
return sprintf(buf, "%d\n", cur_state);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
processor_set_cur_state(struct thermal_cooling_device *cdev, unsigned int state)
|
||||||
|
{
|
||||||
|
struct acpi_device *device = cdev->devdata;
|
||||||
|
struct acpi_processor *pr = acpi_driver_data(device);
|
||||||
|
int result = 0;
|
||||||
|
int max_pstate;
|
||||||
|
|
||||||
|
if (!device || !pr)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
max_pstate = cpufreq_get_max_state(pr->id);
|
||||||
|
|
||||||
|
if (state > acpi_processor_max_state(pr))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (state <= max_pstate) {
|
||||||
|
if (pr->flags.throttling && pr->throttling.state)
|
||||||
|
result = acpi_processor_set_throttling(pr, 0);
|
||||||
|
cpufreq_set_cur_state(pr->id, state);
|
||||||
|
} else {
|
||||||
|
cpufreq_set_cur_state(pr->id, max_pstate);
|
||||||
|
result = acpi_processor_set_throttling(pr,
|
||||||
|
state - max_pstate);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct thermal_cooling_device_ops processor_cooling_ops = {
|
||||||
|
.get_max_state = processor_get_max_state,
|
||||||
|
.get_cur_state = processor_get_cur_state,
|
||||||
|
.set_cur_state = processor_set_cur_state,
|
||||||
|
};
|
||||||
|
|
||||||
/* /proc interface */
|
/* /proc interface */
|
||||||
|
|
||||||
static int acpi_processor_limit_seq_show(struct seq_file *seq, void *offset)
|
static int acpi_processor_limit_seq_show(struct seq_file *seq, void *offset)
|
||||||
|
|
|
@ -43,7 +43,7 @@
|
||||||
#include <linux/seq_file.h>
|
#include <linux/seq_file.h>
|
||||||
#include <linux/reboot.h>
|
#include <linux/reboot.h>
|
||||||
#include <asm/uaccess.h>
|
#include <asm/uaccess.h>
|
||||||
|
#include <linux/thermal.h>
|
||||||
#include <acpi/acpi_bus.h>
|
#include <acpi/acpi_bus.h>
|
||||||
#include <acpi/acpi_drivers.h>
|
#include <acpi/acpi_drivers.h>
|
||||||
|
|
||||||
|
@ -65,9 +65,6 @@
|
||||||
#define ACPI_THERMAL_MAX_ACTIVE 10
|
#define ACPI_THERMAL_MAX_ACTIVE 10
|
||||||
#define ACPI_THERMAL_MAX_LIMIT_STR_LEN 65
|
#define ACPI_THERMAL_MAX_LIMIT_STR_LEN 65
|
||||||
|
|
||||||
#define KELVIN_TO_CELSIUS(t) (long)(((long)t-2732>=0) ? ((long)t-2732+5)/10 : ((long)t-2732-5)/10)
|
|
||||||
#define CELSIUS_TO_KELVIN(t) ((t+273)*10)
|
|
||||||
|
|
||||||
#define _COMPONENT ACPI_THERMAL_COMPONENT
|
#define _COMPONENT ACPI_THERMAL_COMPONENT
|
||||||
ACPI_MODULE_NAME("thermal");
|
ACPI_MODULE_NAME("thermal");
|
||||||
|
|
||||||
|
@ -195,6 +192,8 @@ struct acpi_thermal {
|
||||||
struct acpi_thermal_trips trips;
|
struct acpi_thermal_trips trips;
|
||||||
struct acpi_handle_list devices;
|
struct acpi_handle_list devices;
|
||||||
struct timer_list timer;
|
struct timer_list timer;
|
||||||
|
struct thermal_zone_device *thermal_zone;
|
||||||
|
int tz_enabled;
|
||||||
struct mutex lock;
|
struct mutex lock;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -321,22 +320,49 @@ static int acpi_thermal_set_cooling_mode(struct acpi_thermal *tz, int mode)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int acpi_thermal_get_trip_points(struct acpi_thermal *tz)
|
#define ACPI_TRIPS_CRITICAL 0x01
|
||||||
|
#define ACPI_TRIPS_HOT 0x02
|
||||||
|
#define ACPI_TRIPS_PASSIVE 0x04
|
||||||
|
#define ACPI_TRIPS_ACTIVE 0x08
|
||||||
|
#define ACPI_TRIPS_DEVICES 0x10
|
||||||
|
|
||||||
|
#define ACPI_TRIPS_REFRESH_THRESHOLDS (ACPI_TRIPS_PASSIVE | ACPI_TRIPS_ACTIVE)
|
||||||
|
#define ACPI_TRIPS_REFRESH_DEVICES ACPI_TRIPS_DEVICES
|
||||||
|
|
||||||
|
#define ACPI_TRIPS_INIT (ACPI_TRIPS_CRITICAL | ACPI_TRIPS_HOT | \
|
||||||
|
ACPI_TRIPS_PASSIVE | ACPI_TRIPS_ACTIVE | \
|
||||||
|
ACPI_TRIPS_DEVICES)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This exception is thrown out in two cases:
|
||||||
|
* 1.An invalid trip point becomes invalid or a valid trip point becomes invalid
|
||||||
|
* when re-evaluating the AML code.
|
||||||
|
* 2.TODO: Devices listed in _PSL, _ALx, _TZD may change.
|
||||||
|
* We need to re-bind the cooling devices of a thermal zone when this occurs.
|
||||||
|
*/
|
||||||
|
#define ACPI_THERMAL_TRIPS_EXCEPTION(flags, str) \
|
||||||
|
do { \
|
||||||
|
if (flags != ACPI_TRIPS_INIT) \
|
||||||
|
ACPI_EXCEPTION((AE_INFO, AE_ERROR, \
|
||||||
|
"ACPI thermal trip point %s changed\n" \
|
||||||
|
"Please send acpidump to linux-acpi@vger.kernel.org\n", str)); \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
|
static int acpi_thermal_trips_update(struct acpi_thermal *tz, int flag)
|
||||||
{
|
{
|
||||||
acpi_status status = AE_OK;
|
acpi_status status = AE_OK;
|
||||||
int i = 0;
|
struct acpi_handle_list devices;
|
||||||
|
int valid = 0;
|
||||||
|
int i;
|
||||||
if (!tz)
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
/* Critical Shutdown (required) */
|
/* Critical Shutdown (required) */
|
||||||
|
if (flag & ACPI_TRIPS_CRITICAL) {
|
||||||
status = acpi_evaluate_integer(tz->device->handle, "_CRT", NULL,
|
status = acpi_evaluate_integer(tz->device->handle,
|
||||||
&tz->trips.critical.temperature);
|
"_CRT", NULL, &tz->trips.critical.temperature);
|
||||||
if (ACPI_FAILURE(status)) {
|
if (ACPI_FAILURE(status)) {
|
||||||
tz->trips.critical.flags.valid = 0;
|
tz->trips.critical.flags.valid = 0;
|
||||||
ACPI_EXCEPTION((AE_INFO, status, "No critical threshold"));
|
ACPI_EXCEPTION((AE_INFO, status,
|
||||||
|
"No critical threshold"));
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
} else {
|
} else {
|
||||||
tz->trips.critical.flags.valid = 1;
|
tz->trips.critical.flags.valid = 1;
|
||||||
|
@ -344,13 +370,11 @@ static int acpi_thermal_get_trip_points(struct acpi_thermal *tz)
|
||||||
"Found critical threshold [%lu]\n",
|
"Found critical threshold [%lu]\n",
|
||||||
tz->trips.critical.temperature));
|
tz->trips.critical.temperature));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (tz->trips.critical.flags.valid == 1) {
|
if (tz->trips.critical.flags.valid == 1) {
|
||||||
if (crt == -1) {
|
if (crt == -1) {
|
||||||
tz->trips.critical.flags.valid = 0;
|
tz->trips.critical.flags.valid = 0;
|
||||||
} else if (crt > 0) {
|
} else if (crt > 0) {
|
||||||
unsigned long crt_k = CELSIUS_TO_KELVIN(crt);
|
unsigned long crt_k = CELSIUS_TO_KELVIN(crt);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Allow override to lower critical threshold
|
* Allow override to lower critical threshold
|
||||||
*/
|
*/
|
||||||
|
@ -358,23 +382,27 @@ static int acpi_thermal_get_trip_points(struct acpi_thermal *tz)
|
||||||
tz->trips.critical.temperature = crt_k;
|
tz->trips.critical.temperature = crt_k;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Critical Sleep (optional) */
|
|
||||||
|
|
||||||
status =
|
|
||||||
acpi_evaluate_integer(tz->device->handle, "_HOT", NULL,
|
|
||||||
&tz->trips.hot.temperature);
|
|
||||||
if (ACPI_FAILURE(status)) {
|
|
||||||
tz->trips.hot.flags.valid = 0;
|
|
||||||
ACPI_DEBUG_PRINT((ACPI_DB_INFO, "No hot threshold\n"));
|
|
||||||
} else {
|
|
||||||
tz->trips.hot.flags.valid = 1;
|
|
||||||
ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Found hot threshold [%lu]\n",
|
|
||||||
tz->trips.hot.temperature));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Passive: Processors (optional) */
|
/* Critical Sleep (optional) */
|
||||||
|
if (flag & ACPI_TRIPS_HOT) {
|
||||||
|
status = acpi_evaluate_integer(tz->device->handle,
|
||||||
|
"_HOT", NULL, &tz->trips.hot.temperature);
|
||||||
|
if (ACPI_FAILURE(status)) {
|
||||||
|
tz->trips.hot.flags.valid = 0;
|
||||||
|
ACPI_DEBUG_PRINT((ACPI_DB_INFO,
|
||||||
|
"No hot threshold\n"));
|
||||||
|
} else {
|
||||||
|
tz->trips.hot.flags.valid = 1;
|
||||||
|
ACPI_DEBUG_PRINT((ACPI_DB_INFO,
|
||||||
|
"Found hot threshold [%lu]\n",
|
||||||
|
tz->trips.critical.temperature));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Passive (optional) */
|
||||||
|
if (flag & ACPI_TRIPS_PASSIVE) {
|
||||||
|
valid = tz->trips.passive.flags.valid;
|
||||||
if (psv == -1) {
|
if (psv == -1) {
|
||||||
status = AE_SUPPORT;
|
status = AE_SUPPORT;
|
||||||
} else if (psv > 0) {
|
} else if (psv > 0) {
|
||||||
|
@ -385,65 +413,71 @@ static int acpi_thermal_get_trip_points(struct acpi_thermal *tz)
|
||||||
"_PSV", NULL, &tz->trips.passive.temperature);
|
"_PSV", NULL, &tz->trips.passive.temperature);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ACPI_FAILURE(status)) {
|
if (ACPI_FAILURE(status))
|
||||||
tz->trips.passive.flags.valid = 0;
|
tz->trips.passive.flags.valid = 0;
|
||||||
ACPI_DEBUG_PRINT((ACPI_DB_INFO, "No passive threshold\n"));
|
else {
|
||||||
} else {
|
tz->trips.passive.flags.valid = 1;
|
||||||
|
if (flag == ACPI_TRIPS_INIT) {
|
||||||
|
status = acpi_evaluate_integer(
|
||||||
|
tz->device->handle, "_TC1",
|
||||||
|
NULL, &tz->trips.passive.tc1);
|
||||||
|
if (ACPI_FAILURE(status))
|
||||||
|
tz->trips.passive.flags.valid = 0;
|
||||||
|
status = acpi_evaluate_integer(
|
||||||
|
tz->device->handle, "_TC2",
|
||||||
|
NULL, &tz->trips.passive.tc2);
|
||||||
|
if (ACPI_FAILURE(status))
|
||||||
|
tz->trips.passive.flags.valid = 0;
|
||||||
|
status = acpi_evaluate_integer(
|
||||||
|
tz->device->handle, "_TSP",
|
||||||
|
NULL, &tz->trips.passive.tsp);
|
||||||
|
if (ACPI_FAILURE(status))
|
||||||
|
tz->trips.passive.flags.valid = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if ((flag & ACPI_TRIPS_DEVICES) && tz->trips.passive.flags.valid) {
|
||||||
|
memset(&devices, 0, sizeof(struct acpi_handle_list));
|
||||||
|
status = acpi_evaluate_reference(tz->device->handle, "_PSL",
|
||||||
|
NULL, &devices);
|
||||||
|
if (ACPI_FAILURE(status))
|
||||||
|
tz->trips.passive.flags.valid = 0;
|
||||||
|
else
|
||||||
tz->trips.passive.flags.valid = 1;
|
tz->trips.passive.flags.valid = 1;
|
||||||
|
|
||||||
status =
|
if (memcmp(&tz->trips.passive.devices, &devices,
|
||||||
acpi_evaluate_integer(tz->device->handle, "_TC1", NULL,
|
sizeof(struct acpi_handle_list))) {
|
||||||
&tz->trips.passive.tc1);
|
memcpy(&tz->trips.passive.devices, &devices,
|
||||||
if (ACPI_FAILURE(status))
|
sizeof(struct acpi_handle_list));
|
||||||
tz->trips.passive.flags.valid = 0;
|
ACPI_THERMAL_TRIPS_EXCEPTION(flag, "device");
|
||||||
|
}
|
||||||
status =
|
}
|
||||||
acpi_evaluate_integer(tz->device->handle, "_TC2", NULL,
|
if ((flag & ACPI_TRIPS_PASSIVE) || (flag & ACPI_TRIPS_DEVICES)) {
|
||||||
&tz->trips.passive.tc2);
|
if (valid != tz->trips.passive.flags.valid)
|
||||||
if (ACPI_FAILURE(status))
|
ACPI_THERMAL_TRIPS_EXCEPTION(flag, "state");
|
||||||
tz->trips.passive.flags.valid = 0;
|
|
||||||
|
|
||||||
status =
|
|
||||||
acpi_evaluate_integer(tz->device->handle, "_TSP", NULL,
|
|
||||||
&tz->trips.passive.tsp);
|
|
||||||
if (ACPI_FAILURE(status))
|
|
||||||
tz->trips.passive.flags.valid = 0;
|
|
||||||
|
|
||||||
status =
|
|
||||||
acpi_evaluate_reference(tz->device->handle, "_PSL", NULL,
|
|
||||||
&tz->trips.passive.devices);
|
|
||||||
if (ACPI_FAILURE(status))
|
|
||||||
tz->trips.passive.flags.valid = 0;
|
|
||||||
|
|
||||||
if (!tz->trips.passive.flags.valid)
|
|
||||||
printk(KERN_WARNING PREFIX "Invalid passive threshold\n");
|
|
||||||
else
|
|
||||||
ACPI_DEBUG_PRINT((ACPI_DB_INFO,
|
|
||||||
"Found passive threshold [%lu]\n",
|
|
||||||
tz->trips.passive.temperature));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Active: Fans, etc. (optional) */
|
/* Active (optional) */
|
||||||
|
|
||||||
for (i = 0; i < ACPI_THERMAL_MAX_ACTIVE; i++) {
|
for (i = 0; i < ACPI_THERMAL_MAX_ACTIVE; i++) {
|
||||||
|
|
||||||
char name[5] = { '_', 'A', 'C', ('0' + i), '\0' };
|
char name[5] = { '_', 'A', 'C', ('0' + i), '\0' };
|
||||||
|
valid = tz->trips.active[i].flags.valid;
|
||||||
|
|
||||||
if (act == -1)
|
if (act == -1)
|
||||||
break; /* disable all active trip points */
|
break; /* disable all active trip points */
|
||||||
|
|
||||||
|
if (flag & ACPI_TRIPS_ACTIVE) {
|
||||||
status = acpi_evaluate_integer(tz->device->handle,
|
status = acpi_evaluate_integer(tz->device->handle,
|
||||||
name, NULL, &tz->trips.active[i].temperature);
|
name, NULL, &tz->trips.active[i].temperature);
|
||||||
|
|
||||||
if (ACPI_FAILURE(status)) {
|
if (ACPI_FAILURE(status)) {
|
||||||
if (i == 0) /* no active trip points */
|
tz->trips.active[i].flags.valid = 0;
|
||||||
|
if (i == 0)
|
||||||
break;
|
break;
|
||||||
if (act <= 0) /* no override requested */
|
if (act <= 0)
|
||||||
break;
|
break;
|
||||||
if (i == 1) { /* 1 trip point */
|
if (i == 1)
|
||||||
tz->trips.active[0].temperature =
|
tz->trips.active[0].temperature =
|
||||||
CELSIUS_TO_KELVIN(act);
|
CELSIUS_TO_KELVIN(act);
|
||||||
} else { /* multiple trips */
|
else
|
||||||
/*
|
/*
|
||||||
* Don't allow override higher than
|
* Don't allow override higher than
|
||||||
* the next higher trip point
|
* the next higher trip point
|
||||||
|
@ -453,41 +487,54 @@ static int acpi_thermal_get_trip_points(struct acpi_thermal *tz)
|
||||||
CELSIUS_TO_KELVIN(act) ?
|
CELSIUS_TO_KELVIN(act) ?
|
||||||
tz->trips.active[i - 2].temperature :
|
tz->trips.active[i - 2].temperature :
|
||||||
CELSIUS_TO_KELVIN(act));
|
CELSIUS_TO_KELVIN(act));
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
} else
|
||||||
|
tz->trips.active[i].flags.valid = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
name[2] = 'L';
|
name[2] = 'L';
|
||||||
status =
|
if ((flag & ACPI_TRIPS_DEVICES) && tz->trips.active[i].flags.valid ) {
|
||||||
acpi_evaluate_reference(tz->device->handle, name, NULL,
|
memset(&devices, 0, sizeof(struct acpi_handle_list));
|
||||||
&tz->trips.active[i].devices);
|
status = acpi_evaluate_reference(tz->device->handle,
|
||||||
if (ACPI_SUCCESS(status)) {
|
name, NULL, &devices);
|
||||||
|
if (ACPI_FAILURE(status))
|
||||||
|
tz->trips.active[i].flags.valid = 0;
|
||||||
|
else
|
||||||
tz->trips.active[i].flags.valid = 1;
|
tz->trips.active[i].flags.valid = 1;
|
||||||
ACPI_DEBUG_PRINT((ACPI_DB_INFO,
|
|
||||||
"Found active threshold [%d]:[%lu]\n",
|
if (memcmp(&tz->trips.active[i].devices, &devices,
|
||||||
i, tz->trips.active[i].temperature));
|
sizeof(struct acpi_handle_list))) {
|
||||||
} else
|
memcpy(&tz->trips.active[i].devices, &devices,
|
||||||
ACPI_EXCEPTION((AE_INFO, status,
|
sizeof(struct acpi_handle_list));
|
||||||
"Invalid active threshold [%d]", i));
|
ACPI_THERMAL_TRIPS_EXCEPTION(flag, "device");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if ((flag & ACPI_TRIPS_ACTIVE) || (flag & ACPI_TRIPS_DEVICES))
|
||||||
|
if (valid != tz->trips.active[i].flags.valid)
|
||||||
|
ACPI_THERMAL_TRIPS_EXCEPTION(flag, "state");
|
||||||
|
|
||||||
|
if (!tz->trips.active[i].flags.valid)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (flag & ACPI_TRIPS_DEVICES) {
|
||||||
|
memset(&devices, 0, sizeof(struct acpi_handle_list));
|
||||||
|
status = acpi_evaluate_reference(tz->device->handle, "_TZD",
|
||||||
|
NULL, &devices);
|
||||||
|
if (memcmp(&tz->devices, &devices,
|
||||||
|
sizeof(struct acpi_handle_list))) {
|
||||||
|
memcpy(&tz->devices, &devices,
|
||||||
|
sizeof(struct acpi_handle_list));
|
||||||
|
ACPI_THERMAL_TRIPS_EXCEPTION(flag, "device");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int acpi_thermal_get_devices(struct acpi_thermal *tz)
|
static int acpi_thermal_get_trip_points(struct acpi_thermal *tz)
|
||||||
{
|
{
|
||||||
acpi_status status = AE_OK;
|
return acpi_thermal_trips_update(tz, ACPI_TRIPS_INIT);
|
||||||
|
|
||||||
|
|
||||||
if (!tz)
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
status =
|
|
||||||
acpi_evaluate_reference(tz->device->handle, "_TZD", NULL, &tz->devices);
|
|
||||||
if (ACPI_FAILURE(status))
|
|
||||||
return -ENODEV;
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int acpi_thermal_critical(struct acpi_thermal *tz)
|
static int acpi_thermal_critical(struct acpi_thermal *tz)
|
||||||
|
@ -735,6 +782,9 @@ static void acpi_thermal_check(void *data)
|
||||||
if (result)
|
if (result)
|
||||||
goto unlock;
|
goto unlock;
|
||||||
|
|
||||||
|
if (!tz->tz_enabled)
|
||||||
|
goto unlock;
|
||||||
|
|
||||||
memset(&tz->state, 0, sizeof(tz->state));
|
memset(&tz->state, 0, sizeof(tz->state));
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -828,6 +878,290 @@ static void acpi_thermal_check(void *data)
|
||||||
mutex_unlock(&tz->lock);
|
mutex_unlock(&tz->lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* sys I/F for generic thermal sysfs support */
|
||||||
|
static int thermal_get_temp(struct thermal_zone_device *thermal, char *buf)
|
||||||
|
{
|
||||||
|
struct acpi_thermal *tz = thermal->devdata;
|
||||||
|
|
||||||
|
if (!tz)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
return sprintf(buf, "%ld\n", KELVIN_TO_CELSIUS(tz->temperature));
|
||||||
|
}
|
||||||
|
|
||||||
|
static const char enabled[] = "kernel";
|
||||||
|
static const char disabled[] = "user";
|
||||||
|
static int thermal_get_mode(struct thermal_zone_device *thermal,
|
||||||
|
char *buf)
|
||||||
|
{
|
||||||
|
struct acpi_thermal *tz = thermal->devdata;
|
||||||
|
|
||||||
|
if (!tz)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
return sprintf(buf, "%s\n", tz->tz_enabled ?
|
||||||
|
enabled : disabled);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int thermal_set_mode(struct thermal_zone_device *thermal,
|
||||||
|
const char *buf)
|
||||||
|
{
|
||||||
|
struct acpi_thermal *tz = thermal->devdata;
|
||||||
|
int enable;
|
||||||
|
|
||||||
|
if (!tz)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* enable/disable thermal management from ACPI thermal driver
|
||||||
|
*/
|
||||||
|
if (!strncmp(buf, enabled, sizeof enabled - 1))
|
||||||
|
enable = 1;
|
||||||
|
else if (!strncmp(buf, disabled, sizeof disabled - 1))
|
||||||
|
enable = 0;
|
||||||
|
else
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (enable != tz->tz_enabled) {
|
||||||
|
tz->tz_enabled = enable;
|
||||||
|
ACPI_DEBUG_PRINT((ACPI_DB_INFO,
|
||||||
|
"%s ACPI thermal control\n",
|
||||||
|
tz->tz_enabled ? enabled : disabled));
|
||||||
|
acpi_thermal_check(tz);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int thermal_get_trip_type(struct thermal_zone_device *thermal,
|
||||||
|
int trip, char *buf)
|
||||||
|
{
|
||||||
|
struct acpi_thermal *tz = thermal->devdata;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (!tz || trip < 0)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (tz->trips.critical.flags.valid) {
|
||||||
|
if (!trip)
|
||||||
|
return sprintf(buf, "critical\n");
|
||||||
|
trip--;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tz->trips.hot.flags.valid) {
|
||||||
|
if (!trip)
|
||||||
|
return sprintf(buf, "hot\n");
|
||||||
|
trip--;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tz->trips.passive.flags.valid) {
|
||||||
|
if (!trip)
|
||||||
|
return sprintf(buf, "passive\n");
|
||||||
|
trip--;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 0; i < ACPI_THERMAL_MAX_ACTIVE &&
|
||||||
|
tz->trips.active[i].flags.valid; i++) {
|
||||||
|
if (!trip)
|
||||||
|
return sprintf(buf, "active%d\n", i);
|
||||||
|
trip--;
|
||||||
|
}
|
||||||
|
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int thermal_get_trip_temp(struct thermal_zone_device *thermal,
|
||||||
|
int trip, char *buf)
|
||||||
|
{
|
||||||
|
struct acpi_thermal *tz = thermal->devdata;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (!tz || trip < 0)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (tz->trips.critical.flags.valid) {
|
||||||
|
if (!trip)
|
||||||
|
return sprintf(buf, "%ld\n", KELVIN_TO_CELSIUS(
|
||||||
|
tz->trips.critical.temperature));
|
||||||
|
trip--;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tz->trips.hot.flags.valid) {
|
||||||
|
if (!trip)
|
||||||
|
return sprintf(buf, "%ld\n", KELVIN_TO_CELSIUS(
|
||||||
|
tz->trips.hot.temperature));
|
||||||
|
trip--;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tz->trips.passive.flags.valid) {
|
||||||
|
if (!trip)
|
||||||
|
return sprintf(buf, "%ld\n", KELVIN_TO_CELSIUS(
|
||||||
|
tz->trips.passive.temperature));
|
||||||
|
trip--;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 0; i < ACPI_THERMAL_MAX_ACTIVE &&
|
||||||
|
tz->trips.active[i].flags.valid; i++) {
|
||||||
|
if (!trip)
|
||||||
|
return sprintf(buf, "%ld\n", KELVIN_TO_CELSIUS(
|
||||||
|
tz->trips.active[i].temperature));
|
||||||
|
trip--;
|
||||||
|
}
|
||||||
|
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
typedef int (*cb)(struct thermal_zone_device *, int,
|
||||||
|
struct thermal_cooling_device *);
|
||||||
|
static int acpi_thermal_cooling_device_cb(struct thermal_zone_device *thermal,
|
||||||
|
struct thermal_cooling_device *cdev,
|
||||||
|
cb action)
|
||||||
|
{
|
||||||
|
struct acpi_device *device = cdev->devdata;
|
||||||
|
struct acpi_thermal *tz = thermal->devdata;
|
||||||
|
struct acpi_device *dev;
|
||||||
|
acpi_status status;
|
||||||
|
acpi_handle handle;
|
||||||
|
int i;
|
||||||
|
int j;
|
||||||
|
int trip = -1;
|
||||||
|
int result = 0;
|
||||||
|
|
||||||
|
if (tz->trips.critical.flags.valid)
|
||||||
|
trip++;
|
||||||
|
|
||||||
|
if (tz->trips.hot.flags.valid)
|
||||||
|
trip++;
|
||||||
|
|
||||||
|
if (tz->trips.passive.flags.valid) {
|
||||||
|
trip++;
|
||||||
|
for (i = 0; i < tz->trips.passive.devices.count;
|
||||||
|
i++) {
|
||||||
|
handle = tz->trips.passive.devices.handles[i];
|
||||||
|
status = acpi_bus_get_device(handle, &dev);
|
||||||
|
if (ACPI_SUCCESS(status) && (dev == device)) {
|
||||||
|
result = action(thermal, trip, cdev);
|
||||||
|
if (result)
|
||||||
|
goto failed;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 0; i < ACPI_THERMAL_MAX_ACTIVE; i++) {
|
||||||
|
if (!tz->trips.active[i].flags.valid)
|
||||||
|
break;
|
||||||
|
trip++;
|
||||||
|
for (j = 0;
|
||||||
|
j < tz->trips.active[i].devices.count;
|
||||||
|
j++) {
|
||||||
|
handle = tz->trips.active[i].devices.handles[j];
|
||||||
|
status = acpi_bus_get_device(handle, &dev);
|
||||||
|
if (ACPI_SUCCESS(status) && (dev == device)) {
|
||||||
|
result = action(thermal, trip, cdev);
|
||||||
|
if (result)
|
||||||
|
goto failed;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 0; i < tz->devices.count; i++) {
|
||||||
|
handle = tz->devices.handles[i];
|
||||||
|
status = acpi_bus_get_device(handle, &dev);
|
||||||
|
if (ACPI_SUCCESS(status) && (dev == device)) {
|
||||||
|
result = action(thermal, -1, cdev);
|
||||||
|
if (result)
|
||||||
|
goto failed;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
failed:
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
acpi_thermal_bind_cooling_device(struct thermal_zone_device *thermal,
|
||||||
|
struct thermal_cooling_device *cdev)
|
||||||
|
{
|
||||||
|
return acpi_thermal_cooling_device_cb(thermal, cdev,
|
||||||
|
thermal_zone_bind_cooling_device);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
acpi_thermal_unbind_cooling_device(struct thermal_zone_device *thermal,
|
||||||
|
struct thermal_cooling_device *cdev)
|
||||||
|
{
|
||||||
|
return acpi_thermal_cooling_device_cb(thermal, cdev,
|
||||||
|
thermal_zone_unbind_cooling_device);
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct thermal_zone_device_ops acpi_thermal_zone_ops = {
|
||||||
|
.bind = acpi_thermal_bind_cooling_device,
|
||||||
|
.unbind = acpi_thermal_unbind_cooling_device,
|
||||||
|
.get_temp = thermal_get_temp,
|
||||||
|
.get_mode = thermal_get_mode,
|
||||||
|
.set_mode = thermal_set_mode,
|
||||||
|
.get_trip_type = thermal_get_trip_type,
|
||||||
|
.get_trip_temp = thermal_get_trip_temp,
|
||||||
|
};
|
||||||
|
|
||||||
|
static int acpi_thermal_register_thermal_zone(struct acpi_thermal *tz)
|
||||||
|
{
|
||||||
|
int trips = 0;
|
||||||
|
int result;
|
||||||
|
acpi_status status;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (tz->trips.critical.flags.valid)
|
||||||
|
trips++;
|
||||||
|
|
||||||
|
if (tz->trips.hot.flags.valid)
|
||||||
|
trips++;
|
||||||
|
|
||||||
|
if (tz->trips.passive.flags.valid)
|
||||||
|
trips++;
|
||||||
|
|
||||||
|
for (i = 0; i < ACPI_THERMAL_MAX_ACTIVE &&
|
||||||
|
tz->trips.active[i].flags.valid; i++, trips++);
|
||||||
|
tz->thermal_zone = thermal_zone_device_register("ACPI thermal zone",
|
||||||
|
trips, tz, &acpi_thermal_zone_ops);
|
||||||
|
if (!tz->thermal_zone)
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
result = sysfs_create_link(&tz->device->dev.kobj,
|
||||||
|
&tz->thermal_zone->device.kobj, "thermal_zone");
|
||||||
|
if (result)
|
||||||
|
return result;
|
||||||
|
|
||||||
|
result = sysfs_create_link(&tz->thermal_zone->device.kobj,
|
||||||
|
&tz->device->dev.kobj, "device");
|
||||||
|
if (result)
|
||||||
|
return result;
|
||||||
|
|
||||||
|
status = acpi_attach_data(tz->device->handle,
|
||||||
|
acpi_bus_private_data_handler,
|
||||||
|
tz->thermal_zone);
|
||||||
|
if (ACPI_FAILURE(status)) {
|
||||||
|
ACPI_DEBUG_PRINT((ACPI_DB_ERROR,
|
||||||
|
"Error attaching device data\n"));
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
|
||||||
|
tz->tz_enabled = 1;
|
||||||
|
|
||||||
|
printk(KERN_INFO PREFIX "%s is registered as thermal_zone%d\n",
|
||||||
|
tz->device->dev.bus_id, tz->thermal_zone->id);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void acpi_thermal_unregister_thermal_zone(struct acpi_thermal *tz)
|
||||||
|
{
|
||||||
|
sysfs_remove_link(&tz->device->dev.kobj, "thermal_zone");
|
||||||
|
sysfs_remove_link(&tz->thermal_zone->device.kobj, "device");
|
||||||
|
thermal_zone_device_unregister(tz->thermal_zone);
|
||||||
|
tz->thermal_zone = NULL;
|
||||||
|
acpi_detach_data(tz->device->handle, acpi_bus_private_data_handler);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* --------------------------------------------------------------------------
|
/* --------------------------------------------------------------------------
|
||||||
FS Interface (/proc)
|
FS Interface (/proc)
|
||||||
-------------------------------------------------------------------------- */
|
-------------------------------------------------------------------------- */
|
||||||
|
@ -1184,15 +1518,15 @@ static void acpi_thermal_notify(acpi_handle handle, u32 event, void *data)
|
||||||
acpi_thermal_check(tz);
|
acpi_thermal_check(tz);
|
||||||
break;
|
break;
|
||||||
case ACPI_THERMAL_NOTIFY_THRESHOLDS:
|
case ACPI_THERMAL_NOTIFY_THRESHOLDS:
|
||||||
acpi_thermal_get_trip_points(tz);
|
acpi_thermal_trips_update(tz, ACPI_TRIPS_REFRESH_THRESHOLDS);
|
||||||
acpi_thermal_check(tz);
|
acpi_thermal_check(tz);
|
||||||
acpi_bus_generate_proc_event(device, event, 0);
|
acpi_bus_generate_proc_event(device, event, 0);
|
||||||
acpi_bus_generate_netlink_event(device->pnp.device_class,
|
acpi_bus_generate_netlink_event(device->pnp.device_class,
|
||||||
device->dev.bus_id, event, 0);
|
device->dev.bus_id, event, 0);
|
||||||
break;
|
break;
|
||||||
case ACPI_THERMAL_NOTIFY_DEVICES:
|
case ACPI_THERMAL_NOTIFY_DEVICES:
|
||||||
if (tz->flags.devices)
|
acpi_thermal_trips_update(tz, ACPI_TRIPS_REFRESH_DEVICES);
|
||||||
acpi_thermal_get_devices(tz);
|
acpi_thermal_check(tz);
|
||||||
acpi_bus_generate_proc_event(device, event, 0);
|
acpi_bus_generate_proc_event(device, event, 0);
|
||||||
acpi_bus_generate_netlink_event(device->pnp.device_class,
|
acpi_bus_generate_netlink_event(device->pnp.device_class,
|
||||||
device->dev.bus_id, event, 0);
|
device->dev.bus_id, event, 0);
|
||||||
|
@ -1235,11 +1569,6 @@ static int acpi_thermal_get_info(struct acpi_thermal *tz)
|
||||||
else
|
else
|
||||||
acpi_thermal_get_polling_frequency(tz);
|
acpi_thermal_get_polling_frequency(tz);
|
||||||
|
|
||||||
/* Get devices in this thermal zone [_TZD] (optional) */
|
|
||||||
result = acpi_thermal_get_devices(tz);
|
|
||||||
if (!result)
|
|
||||||
tz->flags.devices = 1;
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1263,13 +1592,19 @@ static int acpi_thermal_add(struct acpi_device *device)
|
||||||
strcpy(acpi_device_class(device), ACPI_THERMAL_CLASS);
|
strcpy(acpi_device_class(device), ACPI_THERMAL_CLASS);
|
||||||
acpi_driver_data(device) = tz;
|
acpi_driver_data(device) = tz;
|
||||||
mutex_init(&tz->lock);
|
mutex_init(&tz->lock);
|
||||||
|
|
||||||
|
|
||||||
result = acpi_thermal_get_info(tz);
|
result = acpi_thermal_get_info(tz);
|
||||||
if (result)
|
if (result)
|
||||||
goto end;
|
goto free_memory;
|
||||||
|
|
||||||
|
result = acpi_thermal_register_thermal_zone(tz);
|
||||||
|
if (result)
|
||||||
|
goto free_memory;
|
||||||
|
|
||||||
result = acpi_thermal_add_fs(device);
|
result = acpi_thermal_add_fs(device);
|
||||||
if (result)
|
if (result)
|
||||||
goto end;
|
goto unregister_thermal_zone;
|
||||||
|
|
||||||
init_timer(&tz->timer);
|
init_timer(&tz->timer);
|
||||||
|
|
||||||
|
@ -1280,19 +1615,21 @@ static int acpi_thermal_add(struct acpi_device *device)
|
||||||
acpi_thermal_notify, tz);
|
acpi_thermal_notify, tz);
|
||||||
if (ACPI_FAILURE(status)) {
|
if (ACPI_FAILURE(status)) {
|
||||||
result = -ENODEV;
|
result = -ENODEV;
|
||||||
goto end;
|
goto remove_fs;
|
||||||
}
|
}
|
||||||
|
|
||||||
printk(KERN_INFO PREFIX "%s [%s] (%ld C)\n",
|
printk(KERN_INFO PREFIX "%s [%s] (%ld C)\n",
|
||||||
acpi_device_name(device), acpi_device_bid(device),
|
acpi_device_name(device), acpi_device_bid(device),
|
||||||
KELVIN_TO_CELSIUS(tz->temperature));
|
KELVIN_TO_CELSIUS(tz->temperature));
|
||||||
|
goto end;
|
||||||
|
|
||||||
end:
|
remove_fs:
|
||||||
if (result) {
|
|
||||||
acpi_thermal_remove_fs(device);
|
acpi_thermal_remove_fs(device);
|
||||||
|
unregister_thermal_zone:
|
||||||
|
thermal_zone_device_unregister(tz->thermal_zone);
|
||||||
|
free_memory:
|
||||||
kfree(tz);
|
kfree(tz);
|
||||||
}
|
end:
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1332,6 +1669,7 @@ static int acpi_thermal_remove(struct acpi_device *device, int type)
|
||||||
}
|
}
|
||||||
|
|
||||||
acpi_thermal_remove_fs(device);
|
acpi_thermal_remove_fs(device);
|
||||||
|
acpi_thermal_unregister_thermal_zone(tz);
|
||||||
mutex_destroy(&tz->lock);
|
mutex_destroy(&tz->lock);
|
||||||
kfree(tz);
|
kfree(tz);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
|
@ -34,6 +34,7 @@
|
||||||
#include <linux/seq_file.h>
|
#include <linux/seq_file.h>
|
||||||
#include <linux/input.h>
|
#include <linux/input.h>
|
||||||
#include <linux/backlight.h>
|
#include <linux/backlight.h>
|
||||||
|
#include <linux/thermal.h>
|
||||||
#include <linux/video_output.h>
|
#include <linux/video_output.h>
|
||||||
#include <asm/uaccess.h>
|
#include <asm/uaccess.h>
|
||||||
|
|
||||||
|
@ -179,6 +180,7 @@ struct acpi_video_device {
|
||||||
struct acpi_device *dev;
|
struct acpi_device *dev;
|
||||||
struct acpi_video_device_brightness *brightness;
|
struct acpi_video_device_brightness *brightness;
|
||||||
struct backlight_device *backlight;
|
struct backlight_device *backlight;
|
||||||
|
struct thermal_cooling_device *cdev;
|
||||||
struct output_device *output_dev;
|
struct output_device *output_dev;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -342,6 +344,54 @@ static struct output_properties acpi_output_properties = {
|
||||||
.set_state = acpi_video_output_set,
|
.set_state = acpi_video_output_set,
|
||||||
.get_status = acpi_video_output_get,
|
.get_status = acpi_video_output_get,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/* thermal cooling device callbacks */
|
||||||
|
static int video_get_max_state(struct thermal_cooling_device *cdev, char *buf)
|
||||||
|
{
|
||||||
|
struct acpi_device *device = cdev->devdata;
|
||||||
|
struct acpi_video_device *video = acpi_driver_data(device);
|
||||||
|
|
||||||
|
return sprintf(buf, "%d\n", video->brightness->count - 3);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int video_get_cur_state(struct thermal_cooling_device *cdev, char *buf)
|
||||||
|
{
|
||||||
|
struct acpi_device *device = cdev->devdata;
|
||||||
|
struct acpi_video_device *video = acpi_driver_data(device);
|
||||||
|
unsigned long level;
|
||||||
|
int state;
|
||||||
|
|
||||||
|
acpi_video_device_lcd_get_level_current(video, &level);
|
||||||
|
for (state = 2; state < video->brightness->count; state++)
|
||||||
|
if (level == video->brightness->levels[state])
|
||||||
|
return sprintf(buf, "%d\n",
|
||||||
|
video->brightness->count - state - 1);
|
||||||
|
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
video_set_cur_state(struct thermal_cooling_device *cdev, unsigned int state)
|
||||||
|
{
|
||||||
|
struct acpi_device *device = cdev->devdata;
|
||||||
|
struct acpi_video_device *video = acpi_driver_data(device);
|
||||||
|
int level;
|
||||||
|
|
||||||
|
if ( state >= video->brightness->count - 2)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
state = video->brightness->count - state;
|
||||||
|
level = video->brightness->levels[state -1];
|
||||||
|
return acpi_video_device_lcd_set_level(video, level);
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct thermal_cooling_device_ops video_cooling_ops = {
|
||||||
|
.get_max_state = video_get_max_state,
|
||||||
|
.get_cur_state = video_get_cur_state,
|
||||||
|
.set_cur_state = video_set_cur_state,
|
||||||
|
};
|
||||||
|
|
||||||
/* --------------------------------------------------------------------------
|
/* --------------------------------------------------------------------------
|
||||||
Video Management
|
Video Management
|
||||||
-------------------------------------------------------------------------- */
|
-------------------------------------------------------------------------- */
|
||||||
|
@ -660,6 +710,7 @@ static void acpi_video_device_find_cap(struct acpi_video_device *device)
|
||||||
kfree(obj);
|
kfree(obj);
|
||||||
|
|
||||||
if (device->cap._BCL && device->cap._BCM && device->cap._BQC && max_level > 0){
|
if (device->cap._BCL && device->cap._BCM && device->cap._BQC && max_level > 0){
|
||||||
|
int result;
|
||||||
static int count = 0;
|
static int count = 0;
|
||||||
char *name;
|
char *name;
|
||||||
name = kzalloc(MAX_NAME_LEN, GFP_KERNEL);
|
name = kzalloc(MAX_NAME_LEN, GFP_KERNEL);
|
||||||
|
@ -672,8 +723,25 @@ static void acpi_video_device_find_cap(struct acpi_video_device *device)
|
||||||
device->backlight->props.max_brightness = device->brightness->count-3;
|
device->backlight->props.max_brightness = device->brightness->count-3;
|
||||||
device->backlight->props.brightness = acpi_video_get_brightness(device->backlight);
|
device->backlight->props.brightness = acpi_video_get_brightness(device->backlight);
|
||||||
backlight_update_status(device->backlight);
|
backlight_update_status(device->backlight);
|
||||||
|
|
||||||
kfree(name);
|
kfree(name);
|
||||||
|
|
||||||
|
device->cdev = thermal_cooling_device_register("LCD",
|
||||||
|
device->dev, &video_cooling_ops);
|
||||||
|
if (device->cdev) {
|
||||||
|
printk(KERN_INFO PREFIX
|
||||||
|
"%s is registered as cooling_device%d\n",
|
||||||
|
device->dev->dev.bus_id, device->cdev->id);
|
||||||
|
result = sysfs_create_link(&device->dev->dev.kobj,
|
||||||
|
&device->cdev->device.kobj,
|
||||||
|
"thermal_cooling");
|
||||||
|
if (result)
|
||||||
|
printk(KERN_ERR PREFIX "Create sysfs link\n");
|
||||||
|
result = sysfs_create_link(&device->cdev->device.kobj,
|
||||||
|
&device->dev->dev.kobj,
|
||||||
|
"device");
|
||||||
|
if (result)
|
||||||
|
printk(KERN_ERR PREFIX "Create sysfs link\n");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (device->cap._DCS && device->cap._DSS){
|
if (device->cap._DCS && device->cap._DSS){
|
||||||
static int count = 0;
|
static int count = 0;
|
||||||
|
@ -1764,6 +1832,14 @@ static int acpi_video_bus_put_one_device(struct acpi_video_device *device)
|
||||||
ACPI_DEVICE_NOTIFY,
|
ACPI_DEVICE_NOTIFY,
|
||||||
acpi_video_device_notify);
|
acpi_video_device_notify);
|
||||||
backlight_device_unregister(device->backlight);
|
backlight_device_unregister(device->backlight);
|
||||||
|
if (device->cdev) {
|
||||||
|
sysfs_remove_link(&device->dev->dev.kobj,
|
||||||
|
"thermal_cooling");
|
||||||
|
sysfs_remove_link(&device->cdev->device.kobj,
|
||||||
|
"device");
|
||||||
|
thermal_cooling_device_unregister(device->cdev);
|
||||||
|
device->cdev = NULL;
|
||||||
|
}
|
||||||
video_output_unregister(device->output_dev);
|
video_output_unregister(device->output_dev);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
|
@ -251,4 +251,13 @@ config ATMEL_SSC
|
||||||
|
|
||||||
If unsure, say N.
|
If unsure, say N.
|
||||||
|
|
||||||
|
config INTEL_MENLOW
|
||||||
|
tristate "Thermal Management driver for Intel menlow platform"
|
||||||
|
depends on ACPI_THERMAL
|
||||||
|
---help---
|
||||||
|
ACPI thermal management enhancement driver on
|
||||||
|
Intel Menlow platform.
|
||||||
|
|
||||||
|
If unsure, say N.
|
||||||
|
|
||||||
endif # MISC_DEVICES
|
endif # MISC_DEVICES
|
||||||
|
|
|
@ -17,3 +17,4 @@ obj-$(CONFIG_SONY_LAPTOP) += sony-laptop.o
|
||||||
obj-$(CONFIG_THINKPAD_ACPI) += thinkpad_acpi.o
|
obj-$(CONFIG_THINKPAD_ACPI) += thinkpad_acpi.o
|
||||||
obj-$(CONFIG_FUJITSU_LAPTOP) += fujitsu-laptop.o
|
obj-$(CONFIG_FUJITSU_LAPTOP) += fujitsu-laptop.o
|
||||||
obj-$(CONFIG_EEPROM_93CX6) += eeprom_93cx6.o
|
obj-$(CONFIG_EEPROM_93CX6) += eeprom_93cx6.o
|
||||||
|
obj-$(CONFIG_INTEL_MENLOW) += intel_menlow.o
|
||||||
|
|
|
@ -0,0 +1,526 @@
|
||||||
|
/*
|
||||||
|
* intel_menlow.c - Intel menlow Driver for thermal management extension
|
||||||
|
*
|
||||||
|
* Copyright (C) 2008 Intel Corp
|
||||||
|
* Copyright (C) 2008 Sujith Thomas <sujith.thomas@intel.com>
|
||||||
|
* Copyright (C) 2008 Zhang Rui <rui.zhang@intel.com>
|
||||||
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; version 2 of the License.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
|
||||||
|
*
|
||||||
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <linux/kernel.h>
|
||||||
|
#include <linux/module.h>
|
||||||
|
#include <linux/init.h>
|
||||||
|
#include <linux/types.h>
|
||||||
|
#include <linux/pci.h>
|
||||||
|
#include <linux/pm.h>
|
||||||
|
|
||||||
|
#include <linux/thermal.h>
|
||||||
|
#include <acpi/acpi_bus.h>
|
||||||
|
#include <acpi/acpi_drivers.h>
|
||||||
|
|
||||||
|
MODULE_AUTHOR("Thomas Sujith");
|
||||||
|
MODULE_AUTHOR("Zhang Rui");
|
||||||
|
MODULE_DESCRIPTION("Intel Menlow platform specific driver");
|
||||||
|
MODULE_LICENSE("GPL");
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 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
|
||||||
|
|
||||||
|
static int memory_get_int_max_bandwidth(struct thermal_cooling_device *cdev,
|
||||||
|
unsigned long *max_state)
|
||||||
|
{
|
||||||
|
struct acpi_device *device = cdev->devdata;
|
||||||
|
acpi_handle handle = device->handle;
|
||||||
|
unsigned long value;
|
||||||
|
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;
|
||||||
|
|
||||||
|
*max_state = value - 1;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int memory_get_max_bandwidth(struct thermal_cooling_device *cdev,
|
||||||
|
char *buf)
|
||||||
|
{
|
||||||
|
unsigned long value;
|
||||||
|
if (memory_get_int_max_bandwidth(cdev, &value))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
return sprintf(buf, "%ld\n", value);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int memory_get_cur_bandwidth(struct thermal_cooling_device *cdev,
|
||||||
|
char *buf)
|
||||||
|
{
|
||||||
|
struct acpi_device *device = cdev->devdata;
|
||||||
|
acpi_handle handle = device->handle;
|
||||||
|
unsigned long value;
|
||||||
|
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,
|
||||||
|
&arg_list, &value);
|
||||||
|
if (ACPI_FAILURE(status))
|
||||||
|
return -EFAULT;
|
||||||
|
|
||||||
|
return sprintf(buf, "%ld\n", value);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int memory_set_cur_bandwidth(struct thermal_cooling_device *cdev,
|
||||||
|
unsigned int state)
|
||||||
|
{
|
||||||
|
struct acpi_device *device = cdev->devdata;
|
||||||
|
acpi_handle handle = device->handle;
|
||||||
|
struct acpi_object_list arg_list;
|
||||||
|
union acpi_object arg;
|
||||||
|
acpi_status status;
|
||||||
|
int temp;
|
||||||
|
unsigned long max_state;
|
||||||
|
|
||||||
|
if (memory_get_int_max_bandwidth(cdev, &max_state))
|
||||||
|
return -EFAULT;
|
||||||
|
|
||||||
|
if (max_state < 0 || state > max_state)
|
||||||
|
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,
|
||||||
|
(unsigned long *)&temp);
|
||||||
|
|
||||||
|
printk(KERN_INFO
|
||||||
|
"Bandwidth value was %d: status is %d\n", state, status);
|
||||||
|
if (ACPI_FAILURE(status))
|
||||||
|
return -EFAULT;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct thermal_cooling_device_ops memory_cooling_ops = {
|
||||||
|
.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;
|
||||||
|
acpi_status status = AE_OK;
|
||||||
|
acpi_handle dummy;
|
||||||
|
struct thermal_cooling_device *cdev;
|
||||||
|
|
||||||
|
if (!device)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
status = acpi_get_handle(device->handle, MEMORY_GET_BANDWIDTH, &dummy);
|
||||||
|
if (ACPI_FAILURE(status))
|
||||||
|
goto end;
|
||||||
|
|
||||||
|
status = acpi_get_handle(device->handle, MEMORY_SET_BANDWIDTH, &dummy);
|
||||||
|
if (ACPI_FAILURE(status))
|
||||||
|
goto end;
|
||||||
|
|
||||||
|
cdev = thermal_cooling_device_register("Memory controller", device,
|
||||||
|
&memory_cooling_ops);
|
||||||
|
acpi_driver_data(device) = cdev;
|
||||||
|
if (!cdev)
|
||||||
|
result = -ENODEV;
|
||||||
|
else {
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
end:
|
||||||
|
return result;
|
||||||
|
|
||||||
|
unregister:
|
||||||
|
thermal_cooling_device_unregister(cdev);
|
||||||
|
return result;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
static int intel_menlow_memory_remove(struct acpi_device *device, int type)
|
||||||
|
{
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
const static struct acpi_device_id intel_menlow_memory_ids[] = {
|
||||||
|
{"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
|
||||||
|
*/
|
||||||
|
static int sensor_get_auxtrip(acpi_handle handle, int index, int *value)
|
||||||
|
{
|
||||||
|
acpi_status status;
|
||||||
|
|
||||||
|
if ((index != 0 && index != 1) || !value)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
status = acpi_evaluate_integer(handle, index ? GET_AUX1 : GET_AUX0,
|
||||||
|
NULL, (unsigned long *)value);
|
||||||
|
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
|
||||||
|
};
|
||||||
|
int temp;
|
||||||
|
|
||||||
|
if (index != 0 && index != 1)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
status = acpi_evaluate_integer(handle, index ? GET_AUX0 : GET_AUX1,
|
||||||
|
NULL, (unsigned long *)&temp);
|
||||||
|
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,
|
||||||
|
&args, (unsigned long *)&temp);
|
||||||
|
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)
|
||||||
|
|
||||||
|
static ssize_t aux0_show(struct device *dev,
|
||||||
|
struct device_attribute *dev_attr, char *buf)
|
||||||
|
{
|
||||||
|
struct intel_menlow_attribute *attr = to_intel_menlow_attr(dev_attr);
|
||||||
|
int value;
|
||||||
|
int result;
|
||||||
|
|
||||||
|
result = sensor_get_auxtrip(attr->handle, 0, &value);
|
||||||
|
|
||||||
|
return result ? result : sprintf(buf, "%lu", KELVIN_TO_CELSIUS(value));
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t aux1_show(struct device *dev,
|
||||||
|
struct device_attribute *dev_attr, char *buf)
|
||||||
|
{
|
||||||
|
struct intel_menlow_attribute *attr = to_intel_menlow_attr(dev_attr);
|
||||||
|
int value;
|
||||||
|
int result;
|
||||||
|
|
||||||
|
result = sensor_get_auxtrip(attr->handle, 1, &value);
|
||||||
|
|
||||||
|
return result ? result : sprintf(buf, "%lu", KELVIN_TO_CELSIUS(value));
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t aux0_store(struct device *dev,
|
||||||
|
struct device_attribute *dev_attr,
|
||||||
|
const char *buf, size_t count)
|
||||||
|
{
|
||||||
|
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;
|
||||||
|
|
||||||
|
result = sensor_set_auxtrip(attr->handle, 0, CELSIUS_TO_KELVIN(value));
|
||||||
|
return result ? result : count;
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t aux1_store(struct device *dev,
|
||||||
|
struct device_attribute *dev_attr,
|
||||||
|
const char *buf, size_t count)
|
||||||
|
{
|
||||||
|
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;
|
||||||
|
|
||||||
|
result = sensor_set_auxtrip(attr->handle, 1, CELSIUS_TO_KELVIN(value));
|
||||||
|
return result ? result : count;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 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;
|
||||||
|
unsigned long bios_enabled;
|
||||||
|
|
||||||
|
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");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int intel_menlow_add_one_attribute(char *name, int mode, void *show,
|
||||||
|
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;
|
||||||
|
|
||||||
|
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);
|
||||||
|
if (result)
|
||||||
|
return result;
|
||||||
|
|
||||||
|
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))
|
||||||
|
goto not_found;
|
||||||
|
|
||||||
|
status = acpi_get_handle(handle, SET_AUX0, &dummy);
|
||||||
|
if (ACPI_FAILURE(status))
|
||||||
|
goto not_found;
|
||||||
|
|
||||||
|
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))
|
||||||
|
goto not_found;
|
||||||
|
|
||||||
|
status = acpi_get_handle(handle, SET_AUX1, &dummy);
|
||||||
|
if (ACPI_FAILURE(status))
|
||||||
|
goto not_found;
|
||||||
|
|
||||||
|
result = intel_menlow_add_one_attribute("aux1", 0644,
|
||||||
|
aux1_show, aux1_store,
|
||||||
|
&thermal->device, handle);
|
||||||
|
if (result)
|
||||||
|
return AE_ERROR;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 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);
|
||||||
|
if (result)
|
||||||
|
return AE_ERROR;
|
||||||
|
|
||||||
|
not_found:
|
||||||
|
if (status == AE_NOT_FOUND)
|
||||||
|
return AE_OK;
|
||||||
|
else
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
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;
|
||||||
|
unsigned long enable;
|
||||||
|
|
||||||
|
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,
|
||||||
|
intel_menlow_register_sensor, NULL, NULL);
|
||||||
|
if (ACPI_FAILURE(status))
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
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);
|
|
@ -0,0 +1,15 @@
|
||||||
|
#
|
||||||
|
# Generic thermal sysfs drivers configuration
|
||||||
|
#
|
||||||
|
|
||||||
|
menuconfig THERMAL
|
||||||
|
bool "Generic Thermal sysfs driver"
|
||||||
|
default y
|
||||||
|
help
|
||||||
|
Generic Thermal Sysfs driver offers a generic mechanism for
|
||||||
|
thermal management. Usually it's made up of one or more thermal
|
||||||
|
zone and cooling device.
|
||||||
|
each thermal zone contains its own temperature, trip points,
|
||||||
|
cooling devices.
|
||||||
|
All platforms with ACPI thermal support can use this driver.
|
||||||
|
If you want this support, you should say Y here
|
|
@ -0,0 +1,5 @@
|
||||||
|
#
|
||||||
|
# Makefile for sensor chip drivers.
|
||||||
|
#
|
||||||
|
|
||||||
|
obj-$(CONFIG_THERMAL) += thermal.o
|
|
@ -0,0 +1,714 @@
|
||||||
|
/*
|
||||||
|
* thermal.c - Generic Thermal Management Sysfs support.
|
||||||
|
*
|
||||||
|
* Copyright (C) 2008 Intel Corp
|
||||||
|
* Copyright (C) 2008 Zhang Rui <rui.zhang@intel.com>
|
||||||
|
* Copyright (C) 2008 Sujith Thomas <sujith.thomas@intel.com>
|
||||||
|
*
|
||||||
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; version 2 of the License.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
|
||||||
|
*
|
||||||
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <linux/module.h>
|
||||||
|
#include <linux/device.h>
|
||||||
|
#include <linux/err.h>
|
||||||
|
#include <linux/kdev_t.h>
|
||||||
|
#include <linux/idr.h>
|
||||||
|
#include <linux/thermal.h>
|
||||||
|
#include <linux/spinlock.h>
|
||||||
|
|
||||||
|
MODULE_AUTHOR("Zhang Rui")
|
||||||
|
MODULE_DESCRIPTION("Generic thermal management sysfs support");
|
||||||
|
MODULE_LICENSE("GPL");
|
||||||
|
|
||||||
|
#define PREFIX "Thermal: "
|
||||||
|
|
||||||
|
struct thermal_cooling_device_instance {
|
||||||
|
int id;
|
||||||
|
char name[THERMAL_NAME_LENGTH];
|
||||||
|
struct thermal_zone_device *tz;
|
||||||
|
struct thermal_cooling_device *cdev;
|
||||||
|
int trip;
|
||||||
|
char attr_name[THERMAL_NAME_LENGTH];
|
||||||
|
struct device_attribute attr;
|
||||||
|
struct list_head node;
|
||||||
|
};
|
||||||
|
|
||||||
|
static DEFINE_IDR(thermal_tz_idr);
|
||||||
|
static DEFINE_IDR(thermal_cdev_idr);
|
||||||
|
static DEFINE_MUTEX(thermal_idr_lock);
|
||||||
|
|
||||||
|
static LIST_HEAD(thermal_tz_list);
|
||||||
|
static LIST_HEAD(thermal_cdev_list);
|
||||||
|
static DEFINE_MUTEX(thermal_list_lock);
|
||||||
|
|
||||||
|
static int get_idr(struct idr *idr, struct mutex *lock, int *id)
|
||||||
|
{
|
||||||
|
int err;
|
||||||
|
|
||||||
|
again:
|
||||||
|
if (unlikely(idr_pre_get(idr, GFP_KERNEL) == 0))
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
if (lock)
|
||||||
|
mutex_lock(lock);
|
||||||
|
err = idr_get_new(idr, NULL, id);
|
||||||
|
if (lock)
|
||||||
|
mutex_unlock(lock);
|
||||||
|
if (unlikely(err == -EAGAIN))
|
||||||
|
goto again;
|
||||||
|
else if (unlikely(err))
|
||||||
|
return err;
|
||||||
|
|
||||||
|
*id = *id & MAX_ID_MASK;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void release_idr(struct idr *idr, struct mutex *lock, int id)
|
||||||
|
{
|
||||||
|
if (lock)
|
||||||
|
mutex_lock(lock);
|
||||||
|
idr_remove(idr, id);
|
||||||
|
if (lock)
|
||||||
|
mutex_unlock(lock);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* sys I/F for thermal zone */
|
||||||
|
|
||||||
|
#define to_thermal_zone(_dev) \
|
||||||
|
container_of(_dev, struct thermal_zone_device, device)
|
||||||
|
|
||||||
|
static ssize_t
|
||||||
|
type_show(struct device *dev, struct device_attribute *attr, char *buf)
|
||||||
|
{
|
||||||
|
struct thermal_zone_device *tz = to_thermal_zone(dev);
|
||||||
|
|
||||||
|
return sprintf(buf, "%s\n", tz->type);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t
|
||||||
|
temp_show(struct device *dev, struct device_attribute *attr, char *buf)
|
||||||
|
{
|
||||||
|
struct thermal_zone_device *tz = to_thermal_zone(dev);
|
||||||
|
|
||||||
|
if (!tz->ops->get_temp)
|
||||||
|
return -EPERM;
|
||||||
|
|
||||||
|
return tz->ops->get_temp(tz, buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t
|
||||||
|
mode_show(struct device *dev, struct device_attribute *attr, char *buf)
|
||||||
|
{
|
||||||
|
struct thermal_zone_device *tz = to_thermal_zone(dev);
|
||||||
|
|
||||||
|
if (!tz->ops->get_mode)
|
||||||
|
return -EPERM;
|
||||||
|
|
||||||
|
return tz->ops->get_mode(tz, buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t
|
||||||
|
mode_store(struct device *dev, struct device_attribute *attr,
|
||||||
|
const char *buf, size_t count)
|
||||||
|
{
|
||||||
|
struct thermal_zone_device *tz = to_thermal_zone(dev);
|
||||||
|
int result;
|
||||||
|
|
||||||
|
if (!tz->ops->set_mode)
|
||||||
|
return -EPERM;
|
||||||
|
|
||||||
|
result = tz->ops->set_mode(tz, buf);
|
||||||
|
if (result)
|
||||||
|
return result;
|
||||||
|
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t
|
||||||
|
trip_point_type_show(struct device *dev, struct device_attribute *attr,
|
||||||
|
char *buf)
|
||||||
|
{
|
||||||
|
struct thermal_zone_device *tz = to_thermal_zone(dev);
|
||||||
|
int trip;
|
||||||
|
|
||||||
|
if (!tz->ops->get_trip_type)
|
||||||
|
return -EPERM;
|
||||||
|
|
||||||
|
if (!sscanf(attr->attr.name, "trip_point_%d_type", &trip))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
return tz->ops->get_trip_type(tz, trip, buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t
|
||||||
|
trip_point_temp_show(struct device *dev, struct device_attribute *attr,
|
||||||
|
char *buf)
|
||||||
|
{
|
||||||
|
struct thermal_zone_device *tz = to_thermal_zone(dev);
|
||||||
|
int trip;
|
||||||
|
|
||||||
|
if (!tz->ops->get_trip_temp)
|
||||||
|
return -EPERM;
|
||||||
|
|
||||||
|
if (!sscanf(attr->attr.name, "trip_point_%d_temp", &trip))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
return tz->ops->get_trip_temp(tz, trip, buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
static DEVICE_ATTR(type, 0444, type_show, NULL);
|
||||||
|
static DEVICE_ATTR(temp, 0444, temp_show, NULL);
|
||||||
|
static DEVICE_ATTR(mode, 0644, mode_show, mode_store);
|
||||||
|
|
||||||
|
static struct device_attribute trip_point_attrs[] = {
|
||||||
|
__ATTR(trip_point_0_type, 0444, trip_point_type_show, NULL),
|
||||||
|
__ATTR(trip_point_0_temp, 0444, trip_point_temp_show, NULL),
|
||||||
|
__ATTR(trip_point_1_type, 0444, trip_point_type_show, NULL),
|
||||||
|
__ATTR(trip_point_1_temp, 0444, trip_point_temp_show, NULL),
|
||||||
|
__ATTR(trip_point_2_type, 0444, trip_point_type_show, NULL),
|
||||||
|
__ATTR(trip_point_2_temp, 0444, trip_point_temp_show, NULL),
|
||||||
|
__ATTR(trip_point_3_type, 0444, trip_point_type_show, NULL),
|
||||||
|
__ATTR(trip_point_3_temp, 0444, trip_point_temp_show, NULL),
|
||||||
|
__ATTR(trip_point_4_type, 0444, trip_point_type_show, NULL),
|
||||||
|
__ATTR(trip_point_4_temp, 0444, trip_point_temp_show, NULL),
|
||||||
|
__ATTR(trip_point_5_type, 0444, trip_point_type_show, NULL),
|
||||||
|
__ATTR(trip_point_5_temp, 0444, trip_point_temp_show, NULL),
|
||||||
|
__ATTR(trip_point_6_type, 0444, trip_point_type_show, NULL),
|
||||||
|
__ATTR(trip_point_6_temp, 0444, trip_point_temp_show, NULL),
|
||||||
|
__ATTR(trip_point_7_type, 0444, trip_point_type_show, NULL),
|
||||||
|
__ATTR(trip_point_7_temp, 0444, trip_point_temp_show, NULL),
|
||||||
|
__ATTR(trip_point_8_type, 0444, trip_point_type_show, NULL),
|
||||||
|
__ATTR(trip_point_8_temp, 0444, trip_point_temp_show, NULL),
|
||||||
|
__ATTR(trip_point_9_type, 0444, trip_point_type_show, NULL),
|
||||||
|
__ATTR(trip_point_9_temp, 0444, trip_point_temp_show, NULL),
|
||||||
|
};
|
||||||
|
|
||||||
|
#define TRIP_POINT_ATTR_ADD(_dev, _index, result) \
|
||||||
|
do { \
|
||||||
|
result = device_create_file(_dev, \
|
||||||
|
&trip_point_attrs[_index * 2]); \
|
||||||
|
if (result) \
|
||||||
|
break; \
|
||||||
|
result = device_create_file(_dev, \
|
||||||
|
&trip_point_attrs[_index * 2 + 1]); \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
|
#define TRIP_POINT_ATTR_REMOVE(_dev, _index) \
|
||||||
|
do { \
|
||||||
|
device_remove_file(_dev, &trip_point_attrs[_index * 2]); \
|
||||||
|
device_remove_file(_dev, &trip_point_attrs[_index * 2 + 1]); \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
|
/* sys I/F for cooling device */
|
||||||
|
#define to_cooling_device(_dev) \
|
||||||
|
container_of(_dev, struct thermal_cooling_device, device)
|
||||||
|
|
||||||
|
static ssize_t
|
||||||
|
thermal_cooling_device_type_show(struct device *dev,
|
||||||
|
struct device_attribute *attr, char *buf)
|
||||||
|
{
|
||||||
|
struct thermal_cooling_device *cdev = to_cooling_device(dev);
|
||||||
|
|
||||||
|
return sprintf(buf, "%s\n", cdev->type);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t
|
||||||
|
thermal_cooling_device_max_state_show(struct device *dev,
|
||||||
|
struct device_attribute *attr, char *buf)
|
||||||
|
{
|
||||||
|
struct thermal_cooling_device *cdev = to_cooling_device(dev);
|
||||||
|
|
||||||
|
return cdev->ops->get_max_state(cdev, buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t
|
||||||
|
thermal_cooling_device_cur_state_show(struct device *dev,
|
||||||
|
struct device_attribute *attr, char *buf)
|
||||||
|
{
|
||||||
|
struct thermal_cooling_device *cdev = to_cooling_device(dev);
|
||||||
|
|
||||||
|
return cdev->ops->get_cur_state(cdev, buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t
|
||||||
|
thermal_cooling_device_cur_state_store(struct device *dev,
|
||||||
|
struct device_attribute *attr,
|
||||||
|
const char *buf, size_t count)
|
||||||
|
{
|
||||||
|
struct thermal_cooling_device *cdev = to_cooling_device(dev);
|
||||||
|
int state;
|
||||||
|
int result;
|
||||||
|
|
||||||
|
if (!sscanf(buf, "%d\n", &state))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (state < 0)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
result = cdev->ops->set_cur_state(cdev, state);
|
||||||
|
if (result)
|
||||||
|
return result;
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct device_attribute dev_attr_cdev_type =
|
||||||
|
__ATTR(type, 0444, thermal_cooling_device_type_show, NULL);
|
||||||
|
static DEVICE_ATTR(max_state, 0444,
|
||||||
|
thermal_cooling_device_max_state_show, NULL);
|
||||||
|
static DEVICE_ATTR(cur_state, 0644,
|
||||||
|
thermal_cooling_device_cur_state_show,
|
||||||
|
thermal_cooling_device_cur_state_store);
|
||||||
|
|
||||||
|
static ssize_t
|
||||||
|
thermal_cooling_device_trip_point_show(struct device *dev,
|
||||||
|
struct device_attribute *attr, char *buf)
|
||||||
|
{
|
||||||
|
struct thermal_cooling_device_instance *instance;
|
||||||
|
|
||||||
|
instance =
|
||||||
|
container_of(attr, struct thermal_cooling_device_instance, attr);
|
||||||
|
|
||||||
|
if (instance->trip == THERMAL_TRIPS_NONE)
|
||||||
|
return sprintf(buf, "-1\n");
|
||||||
|
else
|
||||||
|
return sprintf(buf, "%d\n", instance->trip);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Device management */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* thermal_zone_bind_cooling_device - bind a cooling device to a thermal zone
|
||||||
|
* this function is usually called in the thermal zone device .bind callback.
|
||||||
|
* @tz: thermal zone device
|
||||||
|
* @trip: indicates which trip point the cooling devices is
|
||||||
|
* associated with in this thermal zone.
|
||||||
|
* @cdev: thermal cooling device
|
||||||
|
*/
|
||||||
|
int thermal_zone_bind_cooling_device(struct thermal_zone_device *tz,
|
||||||
|
int trip,
|
||||||
|
struct thermal_cooling_device *cdev)
|
||||||
|
{
|
||||||
|
struct thermal_cooling_device_instance *dev;
|
||||||
|
struct thermal_cooling_device_instance *pos;
|
||||||
|
int result;
|
||||||
|
|
||||||
|
if (trip >= tz->trips ||
|
||||||
|
(trip < 0 && trip != THERMAL_TRIPS_NONE))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (!tz || !cdev)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
dev =
|
||||||
|
kzalloc(sizeof(struct thermal_cooling_device_instance), GFP_KERNEL);
|
||||||
|
if (!dev)
|
||||||
|
return -ENOMEM;
|
||||||
|
dev->tz = tz;
|
||||||
|
dev->cdev = cdev;
|
||||||
|
dev->trip = trip;
|
||||||
|
result = get_idr(&tz->idr, &tz->lock, &dev->id);
|
||||||
|
if (result)
|
||||||
|
goto free_mem;
|
||||||
|
|
||||||
|
sprintf(dev->name, "cdev%d", dev->id);
|
||||||
|
result =
|
||||||
|
sysfs_create_link(&tz->device.kobj, &cdev->device.kobj, dev->name);
|
||||||
|
if (result)
|
||||||
|
goto release_idr;
|
||||||
|
|
||||||
|
sprintf(dev->attr_name, "cdev%d_trip_point", dev->id);
|
||||||
|
dev->attr.attr.name = dev->attr_name;
|
||||||
|
dev->attr.attr.mode = 0444;
|
||||||
|
dev->attr.show = thermal_cooling_device_trip_point_show;
|
||||||
|
result = device_create_file(&tz->device, &dev->attr);
|
||||||
|
if (result)
|
||||||
|
goto remove_symbol_link;
|
||||||
|
|
||||||
|
mutex_lock(&tz->lock);
|
||||||
|
list_for_each_entry(pos, &tz->cooling_devices, node)
|
||||||
|
if (pos->tz == tz && pos->trip == trip && pos->cdev == cdev) {
|
||||||
|
result = -EEXIST;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (!result)
|
||||||
|
list_add_tail(&dev->node, &tz->cooling_devices);
|
||||||
|
mutex_unlock(&tz->lock);
|
||||||
|
|
||||||
|
if (!result)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
device_remove_file(&tz->device, &dev->attr);
|
||||||
|
remove_symbol_link:
|
||||||
|
sysfs_remove_link(&tz->device.kobj, dev->name);
|
||||||
|
release_idr:
|
||||||
|
release_idr(&tz->idr, &tz->lock, dev->id);
|
||||||
|
free_mem:
|
||||||
|
kfree(dev);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(thermal_zone_bind_cooling_device);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* thermal_zone_unbind_cooling_device - unbind a cooling device from a thermal zone
|
||||||
|
* this function is usually called in the thermal zone device .unbind callback.
|
||||||
|
* @tz: thermal zone device
|
||||||
|
* @trip: indicates which trip point the cooling devices is
|
||||||
|
* associated with in this thermal zone.
|
||||||
|
* @cdev: thermal cooling device
|
||||||
|
*/
|
||||||
|
int thermal_zone_unbind_cooling_device(struct thermal_zone_device *tz,
|
||||||
|
int trip,
|
||||||
|
struct thermal_cooling_device *cdev)
|
||||||
|
{
|
||||||
|
struct thermal_cooling_device_instance *pos, *next;
|
||||||
|
|
||||||
|
mutex_lock(&tz->lock);
|
||||||
|
list_for_each_entry_safe(pos, next, &tz->cooling_devices, node) {
|
||||||
|
if (pos->tz == tz && pos->trip == trip
|
||||||
|
&& pos->cdev == cdev) {
|
||||||
|
list_del(&pos->node);
|
||||||
|
mutex_unlock(&tz->lock);
|
||||||
|
goto unbind;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
mutex_unlock(&tz->lock);
|
||||||
|
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
unbind:
|
||||||
|
device_remove_file(&tz->device, &pos->attr);
|
||||||
|
sysfs_remove_link(&tz->device.kobj, pos->name);
|
||||||
|
release_idr(&tz->idr, &tz->lock, pos->id);
|
||||||
|
kfree(pos);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(thermal_zone_unbind_cooling_device);
|
||||||
|
|
||||||
|
static void thermal_release(struct device *dev)
|
||||||
|
{
|
||||||
|
struct thermal_zone_device *tz;
|
||||||
|
struct thermal_cooling_device *cdev;
|
||||||
|
|
||||||
|
if (!strncmp(dev->bus_id, "thermal_zone", sizeof "thermal_zone" - 1)) {
|
||||||
|
tz = to_thermal_zone(dev);
|
||||||
|
kfree(tz);
|
||||||
|
} else {
|
||||||
|
cdev = to_cooling_device(dev);
|
||||||
|
kfree(cdev);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct class thermal_class = {
|
||||||
|
.name = "thermal",
|
||||||
|
.dev_release = thermal_release,
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* thermal_cooling_device_register - register a new thermal cooling device
|
||||||
|
* @type: the thermal cooling device type.
|
||||||
|
* @devdata: device private data.
|
||||||
|
* @ops: standard thermal cooling devices callbacks.
|
||||||
|
*/
|
||||||
|
struct thermal_cooling_device *thermal_cooling_device_register(char *type,
|
||||||
|
void *devdata, struct thermal_cooling_device_ops *ops)
|
||||||
|
{
|
||||||
|
struct thermal_cooling_device *cdev;
|
||||||
|
struct thermal_zone_device *pos;
|
||||||
|
int result;
|
||||||
|
|
||||||
|
if (strlen(type) >= THERMAL_NAME_LENGTH)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
if (!ops || !ops->get_max_state || !ops->get_cur_state ||
|
||||||
|
!ops->set_cur_state)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
cdev = kzalloc(sizeof(struct thermal_cooling_device), GFP_KERNEL);
|
||||||
|
if (!cdev)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
result = get_idr(&thermal_cdev_idr, &thermal_idr_lock, &cdev->id);
|
||||||
|
if (result) {
|
||||||
|
kfree(cdev);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
strcpy(cdev->type, type);
|
||||||
|
cdev->ops = ops;
|
||||||
|
cdev->device.class = &thermal_class;
|
||||||
|
cdev->devdata = devdata;
|
||||||
|
sprintf(cdev->device.bus_id, "cooling_device%d", cdev->id);
|
||||||
|
result = device_register(&cdev->device);
|
||||||
|
if (result) {
|
||||||
|
release_idr(&thermal_cdev_idr, &thermal_idr_lock, cdev->id);
|
||||||
|
kfree(cdev);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* sys I/F */
|
||||||
|
if (type) {
|
||||||
|
result = device_create_file(&cdev->device,
|
||||||
|
&dev_attr_cdev_type);
|
||||||
|
if (result)
|
||||||
|
goto unregister;
|
||||||
|
}
|
||||||
|
|
||||||
|
result = device_create_file(&cdev->device, &dev_attr_max_state);
|
||||||
|
if (result)
|
||||||
|
goto unregister;
|
||||||
|
|
||||||
|
result = device_create_file(&cdev->device, &dev_attr_cur_state);
|
||||||
|
if (result)
|
||||||
|
goto unregister;
|
||||||
|
|
||||||
|
mutex_lock(&thermal_list_lock);
|
||||||
|
list_add(&cdev->node, &thermal_cdev_list);
|
||||||
|
list_for_each_entry(pos, &thermal_tz_list, node) {
|
||||||
|
if (!pos->ops->bind)
|
||||||
|
continue;
|
||||||
|
result = pos->ops->bind(pos, cdev);
|
||||||
|
if (result)
|
||||||
|
break;
|
||||||
|
|
||||||
|
}
|
||||||
|
mutex_unlock(&thermal_list_lock);
|
||||||
|
|
||||||
|
if (!result)
|
||||||
|
return cdev;
|
||||||
|
|
||||||
|
unregister:
|
||||||
|
release_idr(&thermal_cdev_idr, &thermal_idr_lock, cdev->id);
|
||||||
|
device_unregister(&cdev->device);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(thermal_cooling_device_register);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* thermal_cooling_device_unregister - removes the registered thermal cooling device
|
||||||
|
*
|
||||||
|
* @cdev: the thermal cooling device to remove.
|
||||||
|
*
|
||||||
|
* thermal_cooling_device_unregister() must be called when the device is no
|
||||||
|
* longer needed.
|
||||||
|
*/
|
||||||
|
void thermal_cooling_device_unregister(struct
|
||||||
|
thermal_cooling_device
|
||||||
|
*cdev)
|
||||||
|
{
|
||||||
|
struct thermal_zone_device *tz;
|
||||||
|
struct thermal_cooling_device *pos = NULL;
|
||||||
|
|
||||||
|
if (!cdev)
|
||||||
|
return;
|
||||||
|
|
||||||
|
mutex_lock(&thermal_list_lock);
|
||||||
|
list_for_each_entry(pos, &thermal_cdev_list, node)
|
||||||
|
if (pos == cdev)
|
||||||
|
break;
|
||||||
|
if (pos != cdev) {
|
||||||
|
/* thermal cooling device not found */
|
||||||
|
mutex_unlock(&thermal_list_lock);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
list_del(&cdev->node);
|
||||||
|
list_for_each_entry(tz, &thermal_tz_list, node) {
|
||||||
|
if (!tz->ops->unbind)
|
||||||
|
continue;
|
||||||
|
tz->ops->unbind(tz, cdev);
|
||||||
|
}
|
||||||
|
mutex_unlock(&thermal_list_lock);
|
||||||
|
if (cdev->type[0])
|
||||||
|
device_remove_file(&cdev->device,
|
||||||
|
&dev_attr_cdev_type);
|
||||||
|
device_remove_file(&cdev->device, &dev_attr_max_state);
|
||||||
|
device_remove_file(&cdev->device, &dev_attr_cur_state);
|
||||||
|
|
||||||
|
release_idr(&thermal_cdev_idr, &thermal_idr_lock, cdev->id);
|
||||||
|
device_unregister(&cdev->device);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(thermal_cooling_device_unregister);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* thermal_zone_device_register - register a new thermal zone device
|
||||||
|
* @type: the thermal zone device type
|
||||||
|
* @trips: the number of trip points the thermal zone support
|
||||||
|
* @devdata: private device data
|
||||||
|
* @ops: standard thermal zone device callbacks
|
||||||
|
*
|
||||||
|
* thermal_zone_device_unregister() must be called when the device is no
|
||||||
|
* longer needed.
|
||||||
|
*/
|
||||||
|
struct thermal_zone_device *thermal_zone_device_register(char *type,
|
||||||
|
int trips, void *devdata,
|
||||||
|
struct thermal_zone_device_ops *ops)
|
||||||
|
{
|
||||||
|
struct thermal_zone_device *tz;
|
||||||
|
struct thermal_cooling_device *pos;
|
||||||
|
int result;
|
||||||
|
int count;
|
||||||
|
|
||||||
|
if (strlen(type) >= THERMAL_NAME_LENGTH)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
if (trips > THERMAL_MAX_TRIPS || trips < 0)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
if (!ops || !ops->get_temp)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
tz = kzalloc(sizeof(struct thermal_zone_device), GFP_KERNEL);
|
||||||
|
if (!tz)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
INIT_LIST_HEAD(&tz->cooling_devices);
|
||||||
|
idr_init(&tz->idr);
|
||||||
|
mutex_init(&tz->lock);
|
||||||
|
result = get_idr(&thermal_tz_idr, &thermal_idr_lock, &tz->id);
|
||||||
|
if (result) {
|
||||||
|
kfree(tz);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
strcpy(tz->type, type);
|
||||||
|
tz->ops = ops;
|
||||||
|
tz->device.class = &thermal_class;
|
||||||
|
tz->devdata = devdata;
|
||||||
|
tz->trips = trips;
|
||||||
|
sprintf(tz->device.bus_id, "thermal_zone%d", tz->id);
|
||||||
|
result = device_register(&tz->device);
|
||||||
|
if (result) {
|
||||||
|
release_idr(&thermal_tz_idr, &thermal_idr_lock, tz->id);
|
||||||
|
kfree(tz);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* sys I/F */
|
||||||
|
if (type) {
|
||||||
|
result = device_create_file(&tz->device, &dev_attr_type);
|
||||||
|
if (result)
|
||||||
|
goto unregister;
|
||||||
|
}
|
||||||
|
|
||||||
|
result = device_create_file(&tz->device, &dev_attr_temp);
|
||||||
|
if (result)
|
||||||
|
goto unregister;
|
||||||
|
|
||||||
|
if (ops->get_mode) {
|
||||||
|
result = device_create_file(&tz->device, &dev_attr_mode);
|
||||||
|
if (result)
|
||||||
|
goto unregister;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (count = 0; count < trips; count++) {
|
||||||
|
TRIP_POINT_ATTR_ADD(&tz->device, count, result);
|
||||||
|
if (result)
|
||||||
|
goto unregister;
|
||||||
|
}
|
||||||
|
|
||||||
|
mutex_lock(&thermal_list_lock);
|
||||||
|
list_add_tail(&tz->node, &thermal_tz_list);
|
||||||
|
if (ops->bind)
|
||||||
|
list_for_each_entry(pos, &thermal_cdev_list, node) {
|
||||||
|
result = ops->bind(tz, pos);
|
||||||
|
if (result)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
mutex_unlock(&thermal_list_lock);
|
||||||
|
|
||||||
|
if (!result)
|
||||||
|
return tz;
|
||||||
|
|
||||||
|
unregister:
|
||||||
|
release_idr(&thermal_tz_idr, &thermal_idr_lock, tz->id);
|
||||||
|
device_unregister(&tz->device);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(thermal_zone_device_register);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* thermal_device_unregister - removes the registered thermal zone device
|
||||||
|
*
|
||||||
|
* @tz: the thermal zone device to remove
|
||||||
|
*/
|
||||||
|
void thermal_zone_device_unregister(struct thermal_zone_device *tz)
|
||||||
|
{
|
||||||
|
struct thermal_cooling_device *cdev;
|
||||||
|
struct thermal_zone_device *pos = NULL;
|
||||||
|
int count;
|
||||||
|
|
||||||
|
if (!tz)
|
||||||
|
return;
|
||||||
|
|
||||||
|
mutex_lock(&thermal_list_lock);
|
||||||
|
list_for_each_entry(pos, &thermal_tz_list, node)
|
||||||
|
if (pos == tz)
|
||||||
|
break;
|
||||||
|
if (pos != tz) {
|
||||||
|
/* thermal zone device not found */
|
||||||
|
mutex_unlock(&thermal_list_lock);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
list_del(&tz->node);
|
||||||
|
if (tz->ops->unbind)
|
||||||
|
list_for_each_entry(cdev, &thermal_cdev_list, node)
|
||||||
|
tz->ops->unbind(tz, cdev);
|
||||||
|
mutex_unlock(&thermal_list_lock);
|
||||||
|
|
||||||
|
if (tz->type[0])
|
||||||
|
device_remove_file(&tz->device, &dev_attr_type);
|
||||||
|
device_remove_file(&tz->device, &dev_attr_temp);
|
||||||
|
if (tz->ops->get_mode)
|
||||||
|
device_remove_file(&tz->device, &dev_attr_mode);
|
||||||
|
|
||||||
|
for (count = 0; count < tz->trips; count++)
|
||||||
|
TRIP_POINT_ATTR_REMOVE(&tz->device, count);
|
||||||
|
|
||||||
|
release_idr(&thermal_tz_idr, &thermal_idr_lock, tz->id);
|
||||||
|
idr_destroy(&tz->idr);
|
||||||
|
mutex_destroy(&tz->lock);
|
||||||
|
device_unregister(&tz->device);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(thermal_zone_device_unregister);
|
||||||
|
|
||||||
|
static int __init thermal_init(void)
|
||||||
|
{
|
||||||
|
int result = 0;
|
||||||
|
|
||||||
|
result = class_register(&thermal_class);
|
||||||
|
if (result) {
|
||||||
|
idr_destroy(&thermal_tz_idr);
|
||||||
|
idr_destroy(&thermal_cdev_idr);
|
||||||
|
mutex_destroy(&thermal_idr_lock);
|
||||||
|
mutex_destroy(&thermal_list_lock);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void __exit thermal_exit(void)
|
||||||
|
{
|
||||||
|
class_unregister(&thermal_class);
|
||||||
|
idr_destroy(&thermal_tz_idr);
|
||||||
|
idr_destroy(&thermal_cdev_idr);
|
||||||
|
mutex_destroy(&thermal_idr_lock);
|
||||||
|
mutex_destroy(&thermal_list_lock);
|
||||||
|
}
|
||||||
|
|
||||||
|
subsys_initcall(thermal_init);
|
||||||
|
module_exit(thermal_exit);
|
|
@ -321,6 +321,8 @@ struct acpi_bus_event {
|
||||||
|
|
||||||
extern struct kobject *acpi_kobj;
|
extern struct kobject *acpi_kobj;
|
||||||
extern int acpi_bus_generate_netlink_event(const char*, const char*, u8, int);
|
extern int acpi_bus_generate_netlink_event(const char*, const char*, u8, int);
|
||||||
|
void acpi_bus_private_data_handler(acpi_handle, u32, void *);
|
||||||
|
int acpi_bus_get_private_data(acpi_handle, void **);
|
||||||
/*
|
/*
|
||||||
* External Functions
|
* External Functions
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
#include <linux/kernel.h>
|
#include <linux/kernel.h>
|
||||||
#include <linux/cpu.h>
|
#include <linux/cpu.h>
|
||||||
#include <linux/cpuidle.h>
|
#include <linux/cpuidle.h>
|
||||||
|
#include <linux/thermal.h>
|
||||||
#include <asm/acpi.h>
|
#include <asm/acpi.h>
|
||||||
|
|
||||||
#define ACPI_PROCESSOR_BUSY_METRIC 10
|
#define ACPI_PROCESSOR_BUSY_METRIC 10
|
||||||
|
@ -219,7 +219,7 @@ struct acpi_processor {
|
||||||
struct acpi_processor_performance *performance;
|
struct acpi_processor_performance *performance;
|
||||||
struct acpi_processor_throttling throttling;
|
struct acpi_processor_throttling throttling;
|
||||||
struct acpi_processor_limit limit;
|
struct acpi_processor_limit limit;
|
||||||
|
struct thermal_cooling_device *cdev;
|
||||||
/* the _PDC objects for this processor, if any */
|
/* the _PDC objects for this processor, if any */
|
||||||
struct acpi_object_list *pdc;
|
struct acpi_object_list *pdc;
|
||||||
};
|
};
|
||||||
|
@ -331,7 +331,7 @@ extern struct cpuidle_driver acpi_idle_driver;
|
||||||
/* in processor_thermal.c */
|
/* in processor_thermal.c */
|
||||||
int acpi_processor_get_limit_info(struct acpi_processor *pr);
|
int acpi_processor_get_limit_info(struct acpi_processor *pr);
|
||||||
extern struct file_operations acpi_processor_limit_fops;
|
extern struct file_operations acpi_processor_limit_fops;
|
||||||
|
extern struct thermal_cooling_device_ops processor_cooling_ops;
|
||||||
#ifdef CONFIG_CPU_FREQ
|
#ifdef CONFIG_CPU_FREQ
|
||||||
void acpi_thermal_cpufreq_init(void);
|
void acpi_thermal_cpufreq_init(void);
|
||||||
void acpi_thermal_cpufreq_exit(void);
|
void acpi_thermal_cpufreq_exit(void);
|
||||||
|
|
|
@ -0,0 +1,94 @@
|
||||||
|
/*
|
||||||
|
* thermal.h ($Revision: 0 $)
|
||||||
|
*
|
||||||
|
* Copyright (C) 2008 Intel Corp
|
||||||
|
* Copyright (C) 2008 Zhang Rui <rui.zhang@intel.com>
|
||||||
|
* Copyright (C) 2008 Sujith Thomas <sujith.thomas@intel.com>
|
||||||
|
*
|
||||||
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; version 2 of the License.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
|
||||||
|
*
|
||||||
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __THERMAL_H__
|
||||||
|
#define __THERMAL_H__
|
||||||
|
|
||||||
|
#include <linux/idr.h>
|
||||||
|
#include <linux/device.h>
|
||||||
|
|
||||||
|
struct thermal_zone_device;
|
||||||
|
struct thermal_cooling_device;
|
||||||
|
|
||||||
|
struct thermal_zone_device_ops {
|
||||||
|
int (*bind) (struct thermal_zone_device *,
|
||||||
|
struct thermal_cooling_device *);
|
||||||
|
int (*unbind) (struct thermal_zone_device *,
|
||||||
|
struct thermal_cooling_device *);
|
||||||
|
int (*get_temp) (struct thermal_zone_device *, char *);
|
||||||
|
int (*get_mode) (struct thermal_zone_device *, char *);
|
||||||
|
int (*set_mode) (struct thermal_zone_device *, const char *);
|
||||||
|
int (*get_trip_type) (struct thermal_zone_device *, int, char *);
|
||||||
|
int (*get_trip_temp) (struct thermal_zone_device *, int, char *);
|
||||||
|
};
|
||||||
|
|
||||||
|
struct thermal_cooling_device_ops {
|
||||||
|
int (*get_max_state) (struct thermal_cooling_device *, char *);
|
||||||
|
int (*get_cur_state) (struct thermal_cooling_device *, char *);
|
||||||
|
int (*set_cur_state) (struct thermal_cooling_device *, unsigned int);
|
||||||
|
};
|
||||||
|
|
||||||
|
#define THERMAL_TRIPS_NONE -1
|
||||||
|
#define THERMAL_MAX_TRIPS 10
|
||||||
|
#define THERMAL_NAME_LENGTH 20
|
||||||
|
struct thermal_cooling_device {
|
||||||
|
int id;
|
||||||
|
char type[THERMAL_NAME_LENGTH];
|
||||||
|
struct device device;
|
||||||
|
void *devdata;
|
||||||
|
struct thermal_cooling_device_ops *ops;
|
||||||
|
struct list_head node;
|
||||||
|
};
|
||||||
|
|
||||||
|
#define KELVIN_TO_CELSIUS(t) (long)(((long)t-2732 >= 0) ? \
|
||||||
|
((long)t-2732+5)/10 : ((long)t-2732-5)/10)
|
||||||
|
#define CELSIUS_TO_KELVIN(t) ((t)*10+2732)
|
||||||
|
|
||||||
|
struct thermal_zone_device {
|
||||||
|
int id;
|
||||||
|
char type[THERMAL_NAME_LENGTH];
|
||||||
|
struct device device;
|
||||||
|
void *devdata;
|
||||||
|
int trips;
|
||||||
|
struct thermal_zone_device_ops *ops;
|
||||||
|
struct list_head cooling_devices;
|
||||||
|
struct idr idr;
|
||||||
|
struct mutex lock; /* protect cooling devices list */
|
||||||
|
struct list_head node;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct thermal_zone_device *thermal_zone_device_register(char *, int, void *,
|
||||||
|
struct thermal_zone_device_ops *);
|
||||||
|
void thermal_zone_device_unregister(struct thermal_zone_device *);
|
||||||
|
|
||||||
|
int thermal_zone_bind_cooling_device(struct thermal_zone_device *, int,
|
||||||
|
struct thermal_cooling_device *);
|
||||||
|
int thermal_zone_unbind_cooling_device(struct thermal_zone_device *, int,
|
||||||
|
struct thermal_cooling_device *);
|
||||||
|
|
||||||
|
struct thermal_cooling_device *thermal_cooling_device_register(char *, void *,
|
||||||
|
struct thermal_cooling_device_ops *);
|
||||||
|
void thermal_cooling_device_unregister(struct thermal_cooling_device *);
|
||||||
|
|
||||||
|
#endif /* __THERMAL_H__ */
|
Loading…
Reference in New Issue