hwmon: (coretemp) Fix Atom CPUs support

Fix Atom CPUs support. Intel documents TjMax at 90 degrees C but
some Atoms may have 125 degrees C (this is undocumented speculation).

Signed-off-by: Rudolf Marek <r.marek@assembler.cz>
Cc: Huaxu Wan <huaxu.wan@linux.intel.com>
Cc: Kent Liu <kent.liu@linux.intel.com>
Signed-off-by: Jean Delvare <khali@linux-fr.org>
This commit is contained in:
Rudolf Marek 2009-09-23 22:59:42 +02:00 committed by Jean Delvare
parent 91f17e02a2
commit 708a62bcd5
3 changed files with 23 additions and 14 deletions

View file

@ -4,7 +4,7 @@ Kernel driver coretemp
Supported chips: Supported chips:
* All Intel Core family * All Intel Core family
Prefix: 'coretemp' Prefix: 'coretemp'
CPUID: family 0x6, models 0xe, 0xf, 0x16, 0x17 CPUID: family 0x6, models 0xe, 0xf, 0x16, 0x17, 0x1c (Atom)
Datasheet: Intel 64 and IA-32 Architectures Software Developer's Manual Datasheet: Intel 64 and IA-32 Architectures Software Developer's Manual
Volume 3A: System Programming Guide Volume 3A: System Programming Guide
http://softwarecommunity.intel.com/Wiki/Mobility/720.htm http://softwarecommunity.intel.com/Wiki/Mobility/720.htm

View file

@ -373,12 +373,12 @@ config SENSORS_GL520SM
will be called gl520sm. will be called gl520sm.
config SENSORS_CORETEMP config SENSORS_CORETEMP
tristate "Intel Core (2) Duo/Solo temperature sensor" tristate "Intel Core/Core2/Atom temperature sensor"
depends on X86 && EXPERIMENTAL depends on X86 && EXPERIMENTAL
help help
If you say yes here you get support for the temperature If you say yes here you get support for the temperature
sensor inside your CPU. Supported all are all known variants sensor inside your CPU. Most of the family 6 CPUs
of Intel Core family. are supported. Check documentation/driver for details.
config SENSORS_IBMAEM config SENSORS_IBMAEM
tristate "IBM Active Energy Manager temperature/power sensors and control" tristate "IBM Active Energy Manager temperature/power sensors and control"

View file

@ -157,17 +157,24 @@ static int __devinit adjust_tjmax(struct cpuinfo_x86 *c, u32 id, struct device *
/* The 100C is default for both mobile and non mobile CPUs */ /* The 100C is default for both mobile and non mobile CPUs */
int tjmax = 100000; int tjmax = 100000;
int ismobile = 1; int usemsr_ee = 1;
int err; int err;
u32 eax, edx; u32 eax, edx;
/* Early chips have no MSR for TjMax */ /* Early chips have no MSR for TjMax */
if ((c->x86_model == 0xf) && (c->x86_mask < 4)) { if ((c->x86_model == 0xf) && (c->x86_mask < 4)) {
ismobile = 0; usemsr_ee = 0;
} }
if ((c->x86_model > 0xe) && (ismobile)) { /* Atoms seems to have TjMax at 90C */
if (c->x86_model == 0x1c) {
usemsr_ee = 0;
tjmax = 90000;
}
if ((c->x86_model > 0xe) && (usemsr_ee)) {
/* Now we can detect the mobile CPU using Intel provided table /* Now we can detect the mobile CPU using Intel provided table
http://softwarecommunity.intel.com/Wiki/Mobility/720.htm http://softwarecommunity.intel.com/Wiki/Mobility/720.htm
@ -179,13 +186,13 @@ static int __devinit adjust_tjmax(struct cpuinfo_x86 *c, u32 id, struct device *
dev_warn(dev, dev_warn(dev,
"Unable to access MSR 0x17, assuming desktop" "Unable to access MSR 0x17, assuming desktop"
" CPU\n"); " CPU\n");
ismobile = 0; usemsr_ee = 0;
} else if (!(eax & 0x10000000)) { } else if (!(eax & 0x10000000)) {
ismobile = 0; usemsr_ee = 0;
} }
} }
if (ismobile || c->x86_model == 0x1c) { if (usemsr_ee) {
err = rdmsr_safe_on_cpu(id, 0xee, &eax, &edx); err = rdmsr_safe_on_cpu(id, 0xee, &eax, &edx);
if (err) { if (err) {
@ -195,7 +202,9 @@ static int __devinit adjust_tjmax(struct cpuinfo_x86 *c, u32 id, struct device *
} else if (eax & 0x40000000) { } else if (eax & 0x40000000) {
tjmax = 85000; tjmax = 85000;
} }
} else { /* if we dont use msr EE it means we are desktop CPU (with exeception
of Atom) */
} else if (tjmax == 100000) {
dev_warn(dev, "Using relative temperature scale!\n"); dev_warn(dev, "Using relative temperature scale!\n");
} }
@ -248,9 +257,9 @@ static int __devinit coretemp_probe(struct platform_device *pdev)
platform_set_drvdata(pdev, data); platform_set_drvdata(pdev, data);
/* read the still undocumented IA32_TEMPERATURE_TARGET it exists /* read the still undocumented IA32_TEMPERATURE_TARGET it exists
on older CPUs but not in this register */ on older CPUs but not in this register, Atoms don't have it either */
if (c->x86_model > 0xe) { if ((c->x86_model > 0xe) && (c->x86_model != 0x1c)) {
err = rdmsr_safe_on_cpu(data->id, 0x1a2, &eax, &edx); err = rdmsr_safe_on_cpu(data->id, 0x1a2, &eax, &edx);
if (err) { if (err) {
dev_warn(&pdev->dev, "Unable to read" dev_warn(&pdev->dev, "Unable to read"
@ -413,7 +422,7 @@ static int __init coretemp_init(void)
for_each_online_cpu(i) { for_each_online_cpu(i) {
struct cpuinfo_x86 *c = &cpu_data(i); struct cpuinfo_x86 *c = &cpu_data(i);
/* check if family 6, models 0xe, 0xf, 0x16, 0x17, 0x1A */ /* check if family 6, models 0xe, 0xf, 0x16, 0x17, 0x1A, 0x1c */
if ((c->cpuid_level < 0) || (c->x86 != 0x6) || if ((c->cpuid_level < 0) || (c->x86 != 0x6) ||
!((c->x86_model == 0xe) || (c->x86_model == 0xf) || !((c->x86_model == 0xe) || (c->x86_model == 0xf) ||
(c->x86_model == 0x16) || (c->x86_model == 0x17) || (c->x86_model == 0x16) || (c->x86_model == 0x17) ||