summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/dev/acpi/acpi.c5
-rw-r--r--sys/dev/acpi/acpireg.h3
-rw-r--r--sys/dev/acpi/atk0110.c382
-rw-r--r--sys/dev/acpi/files.acpi7
4 files changed, 393 insertions, 4 deletions
diff --git a/sys/dev/acpi/acpi.c b/sys/dev/acpi/acpi.c
index 86eb1722072..45d49b9ff66 100644
--- a/sys/dev/acpi/acpi.c
+++ b/sys/dev/acpi/acpi.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: acpi.c,v 1.140 2009/06/03 07:13:48 pirofti Exp $ */
+/* $OpenBSD: acpi.c,v 1.141 2009/07/23 01:38:16 cnst Exp $ */
/*
* Copyright (c) 2005 Thorsten Lockert <tholo@sigmasoft.com>
* Copyright (c) 2005 Jordan Hargrave <jordan@openbsd.org>
@@ -2244,7 +2244,8 @@ acpi_foundhid(struct aml_node *node, void *arg)
else if (!strcmp(dev, ACPI_DEV_THINKPAD)) {
aaa.aaa_name = "acpithinkpad";
acpi_thinkpad_enabled = 1;
- }
+ } else if (!strcmp(dev, ACPI_DEV_ASUSAIBOOSTER))
+ aaa.aaa_name = "aibs";
if (aaa.aaa_name)
config_found(self, &aaa, acpi_print);
diff --git a/sys/dev/acpi/acpireg.h b/sys/dev/acpi/acpireg.h
index 3da2888e4fe..9550a684a41 100644
--- a/sys/dev/acpi/acpireg.h
+++ b/sys/dev/acpi/acpireg.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: acpireg.h,v 1.17 2009/04/11 08:22:48 kettenis Exp $ */
+/* $OpenBSD: acpireg.h,v 1.18 2009/07/23 01:38:16 cnst Exp $ */
/*
* Copyright (c) 2005 Thorsten Lockert <tholo@sigmasoft.com>
* Copyright (c) 2005 Marco Peereboom <marco@openbsd.org>
@@ -494,5 +494,6 @@ struct acpi_facs {
#define ACPI_DEV_FFB "FIXEDBUTTON" /* Fixed Feature Button */
#define ACPI_DEV_ASUS "ASUS010" /* ASUS Hotkeys */
#define ACPI_DEV_THINKPAD "IBM0068" /* ThinkPad support */
+#define ACPI_DEV_ASUSAIBOOSTER "ATK0110" /* ASUSTeK AI Booster */
#endif /* !_DEV_ACPI_ACPIREG_H_ */
diff --git a/sys/dev/acpi/atk0110.c b/sys/dev/acpi/atk0110.c
new file mode 100644
index 00000000000..a5b6f66dde8
--- /dev/null
+++ b/sys/dev/acpi/atk0110.c
@@ -0,0 +1,382 @@
+/* $OpenBSD: atk0110.c,v 1.1 2009/07/23 01:38:16 cnst Exp $ */
+
+/*
+ * Copyright (c) 2009 Constantine A. Murenin <cnst+openbsd@bugmail.mojo.ru>
+ *
+ * Permission to use, copy, modify, and distribute this software for any
+ * purpose with or without fee is hereby granted, provided that the above
+ * copyright notice and this permission notice appear in all copies.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ */
+
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/device.h>
+#include <sys/malloc.h>
+#include <sys/sensors.h>
+
+#include <dev/acpi/acpireg.h>
+#include <dev/acpi/acpivar.h>
+#include <dev/acpi/acpidev.h>
+#include <dev/acpi/amltypes.h>
+#include <dev/acpi/dsdt.h>
+
+/*
+ * ASUSTeK AI Booster (ACPI ATK0110).
+ *
+ * The driver was inspired by Takanori Watanabe's acpi_aiboost driver.
+ * http://cvsweb.freebsd.org/src/sys/dev/acpi_support/acpi_aiboost.c
+ *
+ * Special thanks goes to Sam Fourman Jr. for providing access to several
+ * ASUS boxes where the driver could be tested.
+ *
+ * -- cnst.su.
+ */
+
+#define AIBS_MORE_SENSORS
+#define AIBS_VERBOSE
+
+struct aibs_sensor {
+ struct ksensor s;
+ int64_t i;
+ int64_t l;
+ int64_t h;
+};
+
+struct aibs_softc {
+ struct device sc_dev;
+
+ struct acpi_softc *sc_acpi;
+ struct aml_node *sc_devnode;
+
+ struct aibs_sensor *sc_asens_volt;
+ struct aibs_sensor *sc_asens_temp;
+ struct aibs_sensor *sc_asens_fan;
+
+ struct ksensordev sc_sensordev;
+};
+
+
+int aibs_match(struct device *, void *, void *);
+void aibs_attach(struct device *, struct device *, void *);
+void aibs_refresh(void *);
+
+void aibs_attach_sif(struct aibs_softc *, enum sensor_type);
+void aibs_refresh_r(struct aibs_softc *, enum sensor_type);
+
+
+struct cfattach aibs_ca = {
+ sizeof(struct aibs_softc), aibs_match, aibs_attach
+};
+
+struct cfdriver aibs_cd = {
+ NULL, "aibs", DV_DULL
+};
+
+static const char* aibs_hids[] = {
+ ACPI_DEV_ASUSAIBOOSTER,
+ NULL
+};
+
+int
+aibs_match(struct device *parent, void *match, void *aux)
+{
+ struct acpi_attach_args *aa = aux;
+ struct cfdata *cf = match;
+
+ return acpi_matchhids(aa, aibs_hids, cf->cf_driver->cd_name);
+}
+
+void
+aibs_attach(struct device *parent, struct device *self, void *aux)
+{
+ struct aibs_softc *sc = (struct aibs_softc *)self;
+ struct acpi_attach_args *aa = aux;
+
+ sc->sc_acpi = (struct acpi_softc *)parent;
+ sc->sc_devnode = aa->aaa_node;
+
+ printf("\n");
+
+ strlcpy(sc->sc_sensordev.xname, sc->sc_dev.dv_xname,
+ sizeof(sc->sc_sensordev.xname));
+
+ aibs_attach_sif(sc, SENSOR_TEMP);
+ aibs_attach_sif(sc, SENSOR_FANRPM);
+ aibs_attach_sif(sc, SENSOR_VOLTS_DC);
+
+ if (sc->sc_sensordev.sensors_count == 0) {
+ printf("%s: no sensors found\n", DEVNAME(sc));
+ return;
+ }
+
+ if (sensor_task_register(sc, aibs_refresh, 5) == NULL) {
+ printf("%s: unable to register update task\n", DEVNAME(sc));
+ return;
+ }
+
+ sensordev_install(&sc->sc_sensordev);
+}
+
+void
+aibs_attach_sif(struct aibs_softc *sc, enum sensor_type st)
+{
+ struct aml_value res;
+ struct aml_value **v;
+ int i, n;
+ char *name = "?SIF";
+ struct aibs_sensor *as;
+
+ switch (st) {
+ case SENSOR_TEMP:
+ name[0] = 'T';
+ break;
+ case SENSOR_FANRPM:
+ name[0] = 'F';
+ break;
+ case SENSOR_VOLTS_DC:
+ name[0] = 'V';
+ break;
+ default:
+ return;
+ }
+
+ if (aml_evalname(sc->sc_acpi, sc->sc_devnode, name, 0, NULL, &res)) {
+ printf("%s: %s not found\n", DEVNAME(sc), name);
+ aml_freevalue(&res);
+ return;
+ }
+
+ if (res.type != AML_OBJTYPE_PACKAGE) {
+ printf("%s: %s: not a package\n", DEVNAME(sc), name);
+ aml_freevalue(&res);
+ return;
+ }
+
+ v = res.v_package;
+ if (v[0]->type != AML_OBJTYPE_INTEGER) {
+ printf("%s: %s[0]: invalid type\n", DEVNAME(sc), name);
+ aml_freevalue(&res);
+ return;
+ }
+
+ n = v[0]->v_integer;
+ if (res.length - 1 < n) {
+ printf("%s: %s: invalid package\n", DEVNAME(sc), name);
+ aml_freevalue(&res);
+ return;
+ } else if (res.length - 1 > n) {
+ printf("%s: %s: misformed package: %i/%i",
+ DEVNAME(sc), name, n, res.length - 1);
+#ifdef AIBS_MORE_SENSORS
+ n = res.length - 1;
+#endif
+ printf(", assume %i\n", n);
+ }
+ if (n < 1) {
+ printf("%s: %s: no members in the package\n",
+ DEVNAME(sc), name);
+ aml_freevalue(&res);
+ return;
+ }
+
+ as = malloc(sizeof(*as) * n, M_DEVBUF, M_NOWAIT | M_ZERO);
+ if (as == NULL) {
+ printf("%s: %s: malloc fail\n", DEVNAME(sc), name);
+ aml_freevalue(&res);
+ return;
+ }
+
+ switch (st) {
+ case SENSOR_TEMP:
+ sc->sc_asens_temp = as;
+ break;
+ case SENSOR_FANRPM:
+ sc->sc_asens_fan = as;
+ break;
+ case SENSOR_VOLTS_DC:
+ sc->sc_asens_volt = as;
+ break;
+ default:
+ /* NOTREACHED */
+ return;
+ }
+
+ for (i = 0, v++; i < n; i++, v++) {
+ struct aml_value ri;
+
+ if(v[0]->type != AML_OBJTYPE_STRING) {
+ printf("%s: %s: %i: not a string: %i type\n",
+ DEVNAME(sc), name, i, v[0]->type);
+ continue;
+ }
+ if (aml_evalname(sc->sc_acpi, sc->sc_devnode, v[0]->v_string,
+ 0, NULL, &ri)) {
+ printf("%s: %s: %i: %s not found\n",
+ DEVNAME(sc), name, i, v[0]->v_string);
+ aml_freevalue(&ri);
+ continue;
+ }
+ if (ri.type != AML_OBJTYPE_PACKAGE) {
+ printf("%s: %s: %i: %s: not a package\n",
+ DEVNAME(sc), name, i, v[0]->v_string);
+ aml_freevalue(&ri);
+ continue;
+ }
+ if (ri.length != 5 ||
+ ri.v_package[0]->type != AML_OBJTYPE_INTEGER ||
+ ri.v_package[1]->type != AML_OBJTYPE_STRING ||
+ ri.v_package[2]->type != AML_OBJTYPE_INTEGER ||
+ ri.v_package[3]->type != AML_OBJTYPE_INTEGER ||
+ ri.v_package[4]->type != AML_OBJTYPE_INTEGER) {
+ printf("%s: %s: %i: %s: invalid package\n",
+ DEVNAME(sc), name, i, v[0]->v_string);
+ aml_freevalue(&ri);
+ continue;
+ }
+ as[i].i = ri.v_package[0]->v_integer;
+ strlcpy(as[i].s.desc, ri.v_package[1]->v_string,
+ sizeof(as[i].s.desc));
+ as[i].l = ri.v_package[2]->v_integer;
+ as[i].h = ri.v_package[3]->v_integer;
+ as[i].s.type = st;
+#ifdef AIBS_VERBOSE
+ printf("%s: %s %2i: %4s: "
+ "0x%08llx %20s %5lli / %5lli 0x%llx\n",
+ DEVNAME(sc), name, i, v[0]->v_string,
+ as[i].i, as[i].s.desc, as[i].l, as[i].h,
+ ri.v_package[4]->v_integer);
+#endif
+ sensor_attach(&sc->sc_sensordev, &as[i].s);
+ aml_freevalue(&ri);
+ }
+
+ aml_freevalue(&res);
+ return;
+}
+
+void
+aibs_refresh(void *arg)
+{
+ struct aibs_softc *sc = arg;
+
+ aibs_refresh_r(sc, SENSOR_TEMP);
+ aibs_refresh_r(sc, SENSOR_FANRPM);
+ aibs_refresh_r(sc, SENSOR_VOLTS_DC);
+}
+
+void
+aibs_refresh_r(struct aibs_softc *sc, enum sensor_type st)
+{
+ struct aml_node *node;
+ int i, n = sc->sc_sensordev.maxnumt[st];
+ char *name;
+ struct aibs_sensor *as;
+
+ switch (st) {
+ case SENSOR_TEMP:
+ name = "RTMP";
+ as = sc->sc_asens_temp;
+ break;
+ case SENSOR_FANRPM:
+ name = "RFAN";
+ as = sc->sc_asens_fan;
+ break;
+ case SENSOR_VOLTS_DC:
+ name = "RVLT";
+ as = sc->sc_asens_volt;
+ break;
+ default:
+ return;
+ }
+
+ if (as == NULL)
+ return;
+
+ node = aml_searchname(sc->sc_devnode, name);
+ if (node == NULL || node->value == NULL ||
+ node->value->type != AML_OBJTYPE_METHOD) {
+ dprintf("%s: %s: method node not found\n",
+ DEVNAME(sc), name);
+ for (i = 0; i < n; i++)
+ as[i].s.flags |= SENSOR_FINVALID;
+ return;
+ }
+
+ for (i = 0; i < n; i++) {
+ struct aml_value req, res;
+ struct ksensor *s;
+ int64_t v, l, h;
+
+ req.type = AML_OBJTYPE_INTEGER;
+ req.v_integer = as[i].i;
+ if (aml_evalnode(sc->sc_acpi, node, 1, &req, &res)) {
+ dprintf("%s: %s: %i: evaluation failed\n",
+ DEVNAME(sc), name, i);
+ aml_freevalue(&res);
+ s->flags |= SENSOR_FINVALID;
+ continue;
+ }
+ if (res.type != AML_OBJTYPE_INTEGER) {
+ dprintf("%s: %s: %i: not an integer: type %i\n",
+ DEVNAME(sc), name, i, res.type);
+ aml_freevalue(&res);
+ s->flags |= SENSOR_FINVALID;
+ continue;
+ }
+
+ v = res.v_integer;
+ s = &as[i].s;
+ l = as[i].l;
+ h = as[i].h;
+
+ switch (st) {
+ case SENSOR_TEMP:
+ s->value = v * 100 * 1000 + 273150000;
+ if (v == 0) {
+ s->status = SENSOR_S_UNKNOWN;
+ s->flags |= SENSOR_FINVALID;
+ } else {
+ if (v > h)
+ s->status = SENSOR_S_CRIT;
+ else if (v > l)
+ s->status = SENSOR_S_WARN;
+ else
+ s->status = SENSOR_S_OK;
+ s->flags &= ~SENSOR_FINVALID;
+ }
+ break;
+ case SENSOR_FANRPM:
+ s->value = v;
+ /* some boards have strange limits for fans */
+ if ((l != 0 && l < v && v < h) ||
+ (l == 0 && v > h))
+ s->status = SENSOR_S_OK;
+ else
+ s->status = SENSOR_S_WARN;
+ s->flags &= ~SENSOR_FINVALID;
+ break;
+ case SENSOR_VOLTS_DC:
+ s->value = v * 1000;
+ if (l < v && v < h)
+ s->status = SENSOR_S_OK;
+ else
+ s->status = SENSOR_S_WARN;
+ s->flags &= ~SENSOR_FINVALID;
+ break;
+ default:
+ /* NOTREACHED */
+ break;
+ }
+ aml_freevalue(&res);
+ }
+
+ return;
+}
diff --git a/sys/dev/acpi/files.acpi b/sys/dev/acpi/files.acpi
index be89febd85f..7a1c444bbc5 100644
--- a/sys/dev/acpi/files.acpi
+++ b/sys/dev/acpi/files.acpi
@@ -1,4 +1,4 @@
-# $OpenBSD: files.acpi,v 1.22 2009/06/03 07:13:48 pirofti Exp $
+# $OpenBSD: files.acpi,v 1.23 2009/07/23 01:38:16 cnst Exp $
#
# Config file and device description for machine-independent ACPI code.
# Included by ports that need it.
@@ -91,3 +91,8 @@ file dev/acpi/acpivout.c acpivout
device acpipwrres
attach acpipwrres at acpi
file dev/acpi/acpipwrres.c acpipwrres
+
+# ASUSTeK AI Booster ATK0110
+device aibs
+attach aibs at acpi
+file dev/acpi/atk0110.c aibs