summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--sys/dev/acpica/acpi.c50
-rw-r--r--sys/dev/acpica/acpivar.h3
2 files changed, 53 insertions, 0 deletions
diff --git a/sys/dev/acpica/acpi.c b/sys/dev/acpica/acpi.c
index 4eac94b..2e627da 100644
--- a/sys/dev/acpica/acpi.c
+++ b/sys/dev/acpica/acpi.c
@@ -84,6 +84,9 @@ static struct cdevsw acpi_cdevsw = {
-1
};
+static const char* sleep_state_names[] = {
+ "S0", "S1", "S2", "S3", "S4", "S4B", "S5" };
+
static void acpi_identify(driver_t *driver, device_t parent);
static int acpi_probe(device_t dev);
static int acpi_attach(device_t dev);
@@ -115,6 +118,7 @@ static void acpi_set_debugging(void);
static void acpi_system_eventhandler_sleep(void *arg, int state);
static void acpi_system_eventhandler_wakeup(void *arg, int state);
+static int acpi_sleep_state_sysctl(SYSCTL_HANDLER_ARGS);
static device_method_t acpi_methods[] = {
/* Device interface */
@@ -301,6 +305,25 @@ acpi_attach(device_t dev)
}
/*
+ * Setup our sysctl tree.
+ *
+ * XXX: This doesn't check to make sure that none of these fail.
+ */
+ sysctl_ctx_init(&sc->acpi_sysctl_ctx);
+ sc->acpi_sysctl_tree = SYSCTL_ADD_NODE(&sc->acpi_sysctl_ctx,
+ SYSCTL_STATIC_CHILDREN(_hw), OID_AUTO,
+ device_get_name(dev), CTLFLAG_RD, 0, "");
+ SYSCTL_ADD_PROC(&sc->acpi_sysctl_ctx, SYSCTL_CHILDREN(sc->acpi_sysctl_tree),
+ OID_AUTO, "power_button_state", CTLTYPE_STRING | CTLFLAG_RW,
+ &sc->acpi_power_button_sx, 0, acpi_sleep_state_sysctl, "A", "");
+ SYSCTL_ADD_PROC(&sc->acpi_sysctl_ctx, SYSCTL_CHILDREN(sc->acpi_sysctl_tree),
+ OID_AUTO, "sleep_button_state", CTLTYPE_STRING | CTLFLAG_RW,
+ &sc->acpi_sleep_button_sx, 0, acpi_sleep_state_sysctl, "A", "");
+ SYSCTL_ADD_PROC(&sc->acpi_sysctl_ctx, SYSCTL_CHILDREN(sc->acpi_sysctl_tree),
+ OID_AUTO, "lid_switch_state", CTLTYPE_STRING | CTLFLAG_RW,
+ &sc->acpi_lid_switch_sx, 0, acpi_sleep_state_sysctl, "A", "");
+
+ /*
* Dispatch the default sleep state to devices.
* TBD: should be configured from userland policy manager.
*/
@@ -1194,6 +1217,33 @@ out:
return(error);
}
+static int
+acpi_sleep_state_sysctl(SYSCTL_HANDLER_ARGS)
+{
+ char sleep_state[10];
+ int error;
+ u_int new_state, old_state;
+
+ old_state = *(u_int *)oidp->oid_arg1;
+ if (old_state > ACPI_STATE_S5)
+ strcpy(sleep_state, "unknown");
+ else
+ strncpy(sleep_state, sleep_state_names[old_state],
+ sizeof(sleep_state_names[old_state]));
+ error = sysctl_handle_string(oidp, sleep_state, sizeof(sleep_state), req);
+ if (error == 0 && req->newptr != NULL) {
+ for (new_state = ACPI_STATE_S0; new_state <= ACPI_STATE_S5; new_state++)
+ if (strncmp(sleep_state, sleep_state_names[new_state],
+ sizeof(sleep_state)) == 0)
+ break;
+ if (new_state != old_state && new_state <= ACPI_STATE_S5)
+ *(u_int *)oidp->oid_arg1 = new_state;
+ else
+ error = EINVAL;
+ }
+ return(error);
+}
+
#ifdef ACPI_DEBUG
/*
* Support for parsing debug options from the kernel environment.
diff --git a/sys/dev/acpica/acpivar.h b/sys/dev/acpica/acpivar.h
index 960d175..1b27e83 100644
--- a/sys/dev/acpica/acpivar.h
+++ b/sys/dev/acpica/acpivar.h
@@ -30,6 +30,7 @@
#include "bus_if.h"
#include <sys/eventhandler.h>
+#include <sys/sysctl.h>
extern devclass_t acpi_devclass;
@@ -44,6 +45,8 @@ struct acpi_softc {
int acpi_enabled;
int acpi_sstate;
+ struct sysctl_ctx_list acpi_sysctl_ctx;
+ struct sysctl_oid *acpi_sysctl_tree;
#define ACPI_POWER_BUTTON_DEFAULT_SX ACPI_STATE_S5;
#define ACPI_SLEEP_BUTTON_DEFAULT_SX ACPI_STATE_S1;
#define ACPI_LID_SWITCH_DEFAULT_SX ACPI_STATE_S1;
OpenPOWER on IntegriCloud