diff options
| -rw-r--r-- | drivers/acpi/thermal.c | 41 |
1 files changed, 34 insertions, 7 deletions
diff --git a/drivers/acpi/thermal.c b/drivers/acpi/thermal.c index e8c143caf0fd..ddbb7c8f994d 100644 --- a/drivers/acpi/thermal.c +++ b/drivers/acpi/thermal.c | |||
| @@ -192,6 +192,7 @@ struct acpi_thermal { | |||
| 192 | struct acpi_handle_list devices; | 192 | struct acpi_handle_list devices; |
| 193 | struct thermal_zone_device *thermal_zone; | 193 | struct thermal_zone_device *thermal_zone; |
| 194 | int tz_enabled; | 194 | int tz_enabled; |
| 195 | int kelvin_offset; | ||
| 195 | struct mutex lock; | 196 | struct mutex lock; |
| 196 | }; | 197 | }; |
| 197 | 198 | ||
| @@ -581,7 +582,7 @@ static void acpi_thermal_check(void *data) | |||
| 581 | } | 582 | } |
| 582 | 583 | ||
| 583 | /* sys I/F for generic thermal sysfs support */ | 584 | /* sys I/F for generic thermal sysfs support */ |
| 584 | #define KELVIN_TO_MILLICELSIUS(t) (t * 100 - 273200) | 585 | #define KELVIN_TO_MILLICELSIUS(t, off) (((t) - (off)) * 100) |
| 585 | 586 | ||
| 586 | static int thermal_get_temp(struct thermal_zone_device *thermal, | 587 | static int thermal_get_temp(struct thermal_zone_device *thermal, |
| 587 | unsigned long *temp) | 588 | unsigned long *temp) |
| @@ -596,7 +597,7 @@ static int thermal_get_temp(struct thermal_zone_device *thermal, | |||
| 596 | if (result) | 597 | if (result) |
| 597 | return result; | 598 | return result; |
| 598 | 599 | ||
| 599 | *temp = KELVIN_TO_MILLICELSIUS(tz->temperature); | 600 | *temp = KELVIN_TO_MILLICELSIUS(tz->temperature, tz->kelvin_offset); |
| 600 | return 0; | 601 | return 0; |
| 601 | } | 602 | } |
| 602 | 603 | ||
| @@ -702,7 +703,8 @@ static int thermal_get_trip_temp(struct thermal_zone_device *thermal, | |||
| 702 | if (tz->trips.critical.flags.valid) { | 703 | if (tz->trips.critical.flags.valid) { |
| 703 | if (!trip) { | 704 | if (!trip) { |
| 704 | *temp = KELVIN_TO_MILLICELSIUS( | 705 | *temp = KELVIN_TO_MILLICELSIUS( |
| 705 | tz->trips.critical.temperature); | 706 | tz->trips.critical.temperature, |
| 707 | tz->kelvin_offset); | ||
| 706 | return 0; | 708 | return 0; |
| 707 | } | 709 | } |
| 708 | trip--; | 710 | trip--; |
| @@ -711,7 +713,8 @@ static int thermal_get_trip_temp(struct thermal_zone_device *thermal, | |||
| 711 | if (tz->trips.hot.flags.valid) { | 713 | if (tz->trips.hot.flags.valid) { |
| 712 | if (!trip) { | 714 | if (!trip) { |
| 713 | *temp = KELVIN_TO_MILLICELSIUS( | 715 | *temp = KELVIN_TO_MILLICELSIUS( |
| 714 | tz->trips.hot.temperature); | 716 | tz->trips.hot.temperature, |
| 717 | tz->kelvin_offset); | ||
| 715 | return 0; | 718 | return 0; |
| 716 | } | 719 | } |
| 717 | trip--; | 720 | trip--; |
| @@ -720,7 +723,8 @@ static int thermal_get_trip_temp(struct thermal_zone_device *thermal, | |||
| 720 | if (tz->trips.passive.flags.valid) { | 723 | if (tz->trips.passive.flags.valid) { |
| 721 | if (!trip) { | 724 | if (!trip) { |
| 722 | *temp = KELVIN_TO_MILLICELSIUS( | 725 | *temp = KELVIN_TO_MILLICELSIUS( |
| 723 | tz->trips.passive.temperature); | 726 | tz->trips.passive.temperature, |
| 727 | tz->kelvin_offset); | ||
| 724 | return 0; | 728 | return 0; |
| 725 | } | 729 | } |
| 726 | trip--; | 730 | trip--; |
| @@ -730,7 +734,8 @@ static int thermal_get_trip_temp(struct thermal_zone_device *thermal, | |||
| 730 | tz->trips.active[i].flags.valid; i++) { | 734 | tz->trips.active[i].flags.valid; i++) { |
| 731 | if (!trip) { | 735 | if (!trip) { |
| 732 | *temp = KELVIN_TO_MILLICELSIUS( | 736 | *temp = KELVIN_TO_MILLICELSIUS( |
| 733 | tz->trips.active[i].temperature); | 737 | tz->trips.active[i].temperature, |
| 738 | tz->kelvin_offset); | ||
| 734 | return 0; | 739 | return 0; |
| 735 | } | 740 | } |
| 736 | trip--; | 741 | trip--; |
| @@ -745,7 +750,8 @@ static int thermal_get_crit_temp(struct thermal_zone_device *thermal, | |||
| 745 | 750 | ||
| 746 | if (tz->trips.critical.flags.valid) { | 751 | if (tz->trips.critical.flags.valid) { |
| 747 | *temperature = KELVIN_TO_MILLICELSIUS( | 752 | *temperature = KELVIN_TO_MILLICELSIUS( |
| 748 | tz->trips.critical.temperature); | 753 | tz->trips.critical.temperature, |
| 754 | tz->kelvin_offset); | ||
| 749 | return 0; | 755 | return 0; |
| 750 | } else | 756 | } else |
| 751 | return -EINVAL; | 757 | return -EINVAL; |
| @@ -1334,6 +1340,25 @@ static int acpi_thermal_get_info(struct acpi_thermal *tz) | |||
| 1334 | return 0; | 1340 | return 0; |
| 1335 | } | 1341 | } |
| 1336 | 1342 | ||
| 1343 | /* | ||
| 1344 | * The exact offset between Kelvin and degree Celsius is 273.15. However ACPI | ||
| 1345 | * handles temperature values with a single decimal place. As a consequence, | ||
| 1346 | * some implementations use an offset of 273.1 and others use an offset of | ||
| 1347 | * 273.2. Try to find out which one is being used, to present the most | ||
| 1348 | * accurate and visually appealing number. | ||
| 1349 | * | ||
| 1350 | * The heuristic below should work for all ACPI thermal zones which have a | ||
| 1351 | * critical trip point with a value being a multiple of 0.5 degree Celsius. | ||
| 1352 | */ | ||
| 1353 | static void acpi_thermal_guess_offset(struct acpi_thermal *tz) | ||
| 1354 | { | ||
| 1355 | if (tz->trips.critical.flags.valid && | ||
| 1356 | (tz->trips.critical.temperature % 5) == 1) | ||
| 1357 | tz->kelvin_offset = 2731; | ||
| 1358 | else | ||
| 1359 | tz->kelvin_offset = 2732; | ||
| 1360 | } | ||
| 1361 | |||
| 1337 | static int acpi_thermal_add(struct acpi_device *device) | 1362 | static int acpi_thermal_add(struct acpi_device *device) |
| 1338 | { | 1363 | { |
| 1339 | int result = 0; | 1364 | int result = 0; |
| @@ -1360,6 +1385,8 @@ static int acpi_thermal_add(struct acpi_device *device) | |||
| 1360 | if (result) | 1385 | if (result) |
| 1361 | goto free_memory; | 1386 | goto free_memory; |
| 1362 | 1387 | ||
| 1388 | acpi_thermal_guess_offset(tz); | ||
| 1389 | |||
| 1363 | result = acpi_thermal_register_thermal_zone(tz); | 1390 | result = acpi_thermal_register_thermal_zone(tz); |
| 1364 | if (result) | 1391 | if (result) |
| 1365 | goto free_memory; | 1392 | goto free_memory; |
