summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMarco Peereboom <marco@cvs.openbsd.org>2008-01-06 18:06:51 +0000
committerMarco Peereboom <marco@cvs.openbsd.org>2008-01-06 18:06:51 +0000
commitc2e51257c1f212071e959a7dfcff600433dac334 (patch)
treef4b57a397e21b9650346b1efade49ff720f776fd
parent379ad048e58ff0bf4a6a8022ce465bf88c8b3499 (diff)
Add a more robust gettempreading that debounces the value (idea from FreeBSD)
Add a macro to convert from K to C Redo notify and sprinkle some debug Tested on D810 T60 & Q2010 and tested by several others.
-rw-r--r--sys/dev/acpi/acpitz.c122
1 files changed, 92 insertions, 30 deletions
diff --git a/sys/dev/acpi/acpitz.c b/sys/dev/acpi/acpitz.c
index 861c508a0bc..10a989d9c3c 100644
--- a/sys/dev/acpi/acpitz.c
+++ b/sys/dev/acpi/acpitz.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: acpitz.c,v 1.20 2007/11/12 21:38:31 deraadt Exp $ */
+/* $OpenBSD: acpitz.c,v 1.21 2008/01/06 18:06:50 marco Exp $ */
/*
* Copyright (c) 2006 Can Erkin Acar <canacar@openbsd.org>
* Copyright (c) 2005 Marco Peereboom <marco@openbsd.org>
@@ -22,6 +22,7 @@
#include <sys/systm.h>
#include <sys/device.h>
#include <sys/malloc.h>
+#include <sys/kernel.h>
#include <machine/bus.h>
@@ -33,7 +34,9 @@
#include <sys/sensors.h>
-#define ACPITZ_MAX_AC 10
+#define KTOC(k) ((k - 2732) / 10)
+#define ACPITZ_MAX_AC (10)
+#define ACPITZ_TMP_RETRY (3)
struct acpitz_softc {
struct device sc_dev;
@@ -69,6 +72,7 @@ struct cfdriver acpitz_cd = {
void acpitz_monitor(struct acpitz_softc *);
void acpitz_refresh(void *);
int acpitz_notify(struct aml_node *, int, void *);
+int acpitz_gettempreading(struct acpitz_softc *, char *);
int acpitz_getreading(struct acpitz_softc *, char *);
int acpitz_setfan(struct acpitz_softc *, int, char *);
int acpitz_setcpu(struct acpitz_softc *, int);
@@ -103,28 +107,31 @@ acpitz_attach(struct device *parent, struct device *self, void *aux)
sc->sc_devnode = aa->aaa_node->child;
sc->sc_lasttmp = -1;
- if ((sc->sc_tmp = acpitz_getreading(sc, "_TMP")) == -1) {
+ if ((sc->sc_tmp = acpitz_gettempreading(sc, "_TMP")) == -1) {
printf(": failed to read _TMP\n");
return;
}
- if ((sc->sc_crt = acpitz_getreading(sc, "_CRT")) == -1) {
+ if ((sc->sc_crt = acpitz_gettempreading(sc, "_CRT")) == -1)
printf(": no critical temperature defined\n");
- sc->sc_crt = 0;
- } else
- printf(": critical temperature %d degC\n",
- (sc->sc_crt - 2732) / 10);
+ else
+ printf(": critical temperature %d degC\n", KTOC(sc->sc_crt));
for (i = 0; i < ACPITZ_MAX_AC; i++) {
snprintf(name, sizeof name, "_AC%d", i);
- sc->sc_ac[i] = acpitz_getreading(sc, name);
+ sc->sc_ac[i] = acpitz_gettempreading(sc, name);
sc->sc_ac_stat[0] = -1;
}
- sc->sc_hot = acpitz_getreading(sc, "_HOT");
+ sc->sc_hot = acpitz_gettempreading(sc, "_HOT");
sc->sc_tc1 = acpitz_getreading(sc, "_TC1");
sc->sc_tc2 = acpitz_getreading(sc, "_TC2");
- sc->sc_psv = acpitz_getreading(sc, "_PSV");
+ sc->sc_psv = acpitz_gettempreading(sc, "_PSV");
+ dnprintf(10, "%s: _HOT: %d _TC1: %d _TC2: %d _PSV: %d _TMP: %d "
+ "_CRT: %d\n", DEVNAME(sc), sc->sc_hot, sc->sc_tc1, sc->sc_tc2,
+ sc->sc_psv, sc->sc_tmp, sc->sc_crt);
+
+ /* get _PSL */
strlcpy(sc->sc_sensdev.xname, DEVNAME(sc),
sizeof(sc->sc_sensdev.xname));
@@ -140,7 +147,7 @@ acpitz_attach(struct device *parent, struct device *self, void *aux)
}
int
-acpitz_setcpu(struct acpitz_softc *sc, int trend)
+acpitz_setcpu(struct acpitz_softc *sc, int perc)
{
struct aml_value res0, *ref;
int x;
@@ -155,7 +162,8 @@ acpitz_setcpu(struct acpitz_softc *sc, int trend)
}
for (x = 0; x < res0.length; x++) {
if (res0.v_package[x]->type != AML_OBJTYPE_OBJREF) {
- printf("%s: _PSL[%d] not a object ref\n", DEVNAME(sc), x);
+ printf("%s: _PSL[%d] not a object ref\n",
+ DEVNAME(sc), x);
continue;
}
ref = res0.v_package[x]->v_objref.ref;
@@ -241,40 +249,53 @@ void
acpitz_refresh(void *arg)
{
struct acpitz_softc *sc = arg;
- int i, trend;
+ int i, perc;
extern int acpi_s5;
dnprintf(30, "%s: %s: refresh\n", DEVNAME(sc),
sc->sc_devnode->parent->name);
- if (-1 == (sc->sc_tmp = acpitz_getreading(sc, "_TMP"))) {
- dnprintf(30, "%s: %s: failed to read temp\n", DEVNAME(sc),
+ /* get _TMP and debounce the value */
+ if (-1 == (sc->sc_tmp = acpitz_gettempreading(sc, "_TMP"))) {
+ printf("%s: %s: failed to read temp\n", DEVNAME(sc),
sc->sc_devnode->parent->name);
- sc->sc_tmp = 0; /* XXX */
+ return;
}
-
+ /* critical trip points */
if (sc->sc_crt != -1 && sc->sc_crt <= sc->sc_tmp) {
- /* Do critical shutdown */
+ /* do critical shutdown */
printf("%s: Critical temperature, shutting down\n",
DEVNAME(sc));
acpi_s5 = 1;
psignal(initproc, SIGUSR1);
}
- if (sc->sc_hot != -1 && sc->sc_hot <= sc->sc_tmp)
+ if (sc->sc_hot != -1 && sc->sc_hot <= sc->sc_tmp) {
printf("%s: _HOT temperature\n", DEVNAME(sc));
+ /* XXX go to S4, until then cool as hard as we can */
+ }
+ /* passive cooling */
if (sc->sc_lasttmp != -1 && sc->sc_tc1 != -1 && sc->sc_tc2 != -1 &&
sc->sc_psv != -1) {
+ dnprintf(30, "%s: passive cooling: lasttmp: %d tc1: %d "
+ "tc2: %d psv: %d\n", DEVNAME(sc), sc->sc_lasttmp,
+ sc->sc_tc1, sc->sc_tc2, sc->sc_psv);
if (sc->sc_psv <= sc->sc_tmp) {
sc->sc_pse = 1;
- trend = sc->sc_tc1 * (sc->sc_tmp - sc->sc_lasttmp) +
+ perc = sc->sc_tc1 * (sc->sc_tmp - sc->sc_lasttmp) +
sc->sc_tc2 * (sc->sc_tmp - sc->sc_psv);
- /* printf("_TZ trend = %d\n", trend); */
+ perc /= 10;
+ if (perc < 0)
+ perc = 0;
+ else if (perc > 100)
+ perc = 100;
+ /* printf("_TZ perc = %d\n", perc); */
} else if (sc->sc_pse)
sc->sc_pse = 0;
}
sc->sc_lasttmp = sc->sc_tmp;
+ /* active cooling */
for (i = 0; i < ACPITZ_MAX_AC; i++) {
if (sc->sc_ac[i] != -1 && sc->sc_ac[i] <= sc->sc_tmp) {
/* turn on fan i */
@@ -296,36 +317,77 @@ acpitz_getreading(struct acpitz_softc *sc, char *name)
int rv = -1;
if (aml_evalname(sc->sc_acpi, sc->sc_devnode, name, 0, NULL, &res)) {
- dnprintf(10, "%s: no %s\n", DEVNAME(sc), name);
+ dnprintf(10, "%s: acpitz_getreading: no %s\n", DEVNAME(sc), name);
+ aml_freevalue(&res);
goto out;
}
rv = aml_val2int(&res);
- out:
+out:
aml_freevalue(&res);
return (rv);
}
int
+acpitz_gettempreading(struct acpitz_softc *sc, char *name)
+{
+ int rv = -1, tmp = -1, i;
+
+ for (i = 0; i < ACPITZ_TMP_RETRY; i++) {
+ tmp = acpitz_getreading(sc, name);
+ if (tmp == -1)
+ goto out;
+ if (KTOC(tmp) > 0) {
+ rv = tmp;
+ break;
+ } else {
+ dnprintf(20, "%s: %d invalid reading on %s, "
+ "debouncing\n", DEVNAME(sc), tmp, name);
+ }
+
+ /* debounce value */
+ if (cold)
+ delay(1000000);
+ else
+ while (tsleep(sc, PWAIT, "tzsleep", hz) !=
+ EWOULDBLOCK);
+ }
+ if (i >= ACPITZ_TMP_RETRY) {
+ printf("%s: %s: failed to read %s\n", DEVNAME(sc),
+ sc->sc_devnode->parent->name, name);
+ goto out;
+ }
+ out:
+ dnprintf(30, "%s: name: %s tmp: %dK => %dC, rv: %d\n", DEVNAME(sc),
+ name, tmp, KTOC(tmp), rv);
+ return (rv);
+}
+
+int
acpitz_notify(struct aml_node *node, int notify_type, void *arg)
{
struct acpitz_softc *sc = arg;
- u_int64_t crt;
+ int crt;
dnprintf(10, "%s notify: %.2x %s\n", DEVNAME(sc), notify_type,
sc->sc_devnode->parent->name);
switch (notify_type) {
- case 0x81: /* Operating Points changed */
- sc->sc_psv = acpitz_getreading(sc, "_PSV");
+ case 0x80: /* hardware notifications */
+ break;
+ case 0x81: /* operating Points changed */
+ sc->sc_psv = acpitz_gettempreading(sc, "_PSV");
crt = sc->sc_crt;
- sc->sc_crt = acpitz_getreading(sc, "_CRT");
+ sc->sc_crt = acpitz_gettempreading(sc, "_CRT");
if (crt != sc->sc_crt)
- printf("%s: critical temperature: %u degC",
- DEVNAME(sc), (sc->sc_crt - 2732) / 10);
+ printf("%s: new critical temperature: %u degC",
+ DEVNAME(sc), KTOC(sc->sc_crt));
+ break;
+ case 0x82: /* re-evaluate thermal device list */
break;
default:
break;
}
+
acpitz_refresh(sc);
return (0);
}