summaryrefslogtreecommitdiff
path: root/common
diff options
context:
space:
mode:
Diffstat (limited to 'common')
-rw-r--r--common/build.mk1
-rw-r--r--common/device_event.c129
-rw-r--r--common/host_command.c3
3 files changed, 133 insertions, 0 deletions
diff --git a/common/build.mk b/common/build.mk
index cc190e08d6..d264c6c2e5 100644
--- a/common/build.mk
+++ b/common/build.mk
@@ -45,6 +45,7 @@ common-$(CONFIG_COMMON_RUNTIME)+=hooks.o main.o system.o
common-$(CONFIG_COMMON_TIMER)+=timer.o
common-$(CONFIG_CRC8)+= crc8.o
common-$(CONFIG_CURVE25519)+=curve25519.o
+common-$(CONFIG_DEVICE_EVENT)+=device_event.o
common-$(CONFIG_DEVICE_STATE)+=device_state.o
common-$(CONFIG_DPTF)+=dptf.o
common-$(CONFIG_EXTENSION_COMMAND)+=extension.o
diff --git a/common/device_event.c b/common/device_event.c
new file mode 100644
index 0000000000..4b4e37724c
--- /dev/null
+++ b/common/device_event.c
@@ -0,0 +1,129 @@
+/* Copyright (c) 2017 The Chromium OS Authors. All rights reserved.
+ * Use of this source code is governed by a BSD-style license that can be
+ * found in the LICENSE file.
+ */
+
+/* Device event commands for Chrome EC */
+
+#include "atomic.h"
+#include "common.h"
+#include "console.h"
+#include "host_command.h"
+#include "lpc.h"
+#include "mkbp_event.h"
+#include "util.h"
+
+/* Console output macros */
+#define CPUTS(outstr) cputs(CC_EVENTS, outstr)
+#define CPRINTS(format, args...) cprints(CC_EVENTS, format, ## args)
+
+static uint32_t device_current_events;
+static uint32_t device_enabled_events;
+
+uint32_t device_get_current_events(void)
+{
+ return device_current_events;
+}
+
+static uint32_t device_get_and_clear_events(void)
+{
+ return atomic_read_clear(&device_current_events);
+}
+
+static uint32_t device_get_enabled_events(void)
+{
+ return device_enabled_events;
+}
+
+void device_set_events(uint32_t mask)
+{
+ /* Ignore events that are not enabled */
+ mask &= device_enabled_events;
+
+ if ((device_current_events & mask) != mask)
+ CPRINTS("device event set 0x%08x", mask);
+
+ atomic_or(&device_current_events, mask);
+
+ /* Signal host that a device event is pending */
+ host_set_single_event(EC_HOST_EVENT_DEVICE);
+}
+
+void device_clear_events(uint32_t mask)
+{
+ /* Only print if something's about to change */
+ if (device_current_events & mask)
+ CPRINTS("device event clear 0x%08x", mask);
+
+ atomic_clear(&device_current_events, mask);
+}
+
+static void device_set_enabled_events(uint32_t mask)
+{
+ if ((device_enabled_events & mask) != mask)
+ CPRINTS("device enabled events set 0x%08x", mask);
+
+ device_enabled_events = mask;
+}
+
+/*****************************************************************************/
+/* Console commands */
+
+#ifdef CONFIG_CMD_DEVICE_EVENT
+static int command_device_event(int argc, char **argv)
+{
+ /* Handle sub-commands */
+ if (argc == 3) {
+ char *e;
+ int i = strtoi(argv[2], &e, 0);
+
+ if (*e)
+ return EC_ERROR_PARAM2;
+ else if (!strcasecmp(argv[1], "set"))
+ device_set_events(i);
+ else if (!strcasecmp(argv[1], "clear"))
+ device_clear_events(i);
+ else if (!strcasecmp(argv[1], "enable"))
+ device_set_enabled_events(i);
+ else
+ return EC_ERROR_PARAM1;
+ }
+
+ ccprintf("Enabled Events: 0x%08x\n", device_get_enabled_events());
+ ccprintf("Current Events: 0x%08x\n", device_get_current_events());
+
+ return EC_SUCCESS;
+}
+DECLARE_CONSOLE_COMMAND(deviceevent, command_device_event,
+ "[set | clear | enable] [mask]",
+ "Print / set device event state");
+#endif
+
+/*****************************************************************************/
+/* Host commands */
+
+static int device_event_cmd(struct host_cmd_handler_args *args)
+{
+ const struct ec_params_device_event *p = args->params;
+ struct ec_response_device_event *r = args->response;
+
+ switch (p->param) {
+ case EC_DEVICE_EVENT_PARAM_GET_CURRENT_EVENTS:
+ r->event_mask = device_get_and_clear_events();
+ break;
+ case EC_DEVICE_EVENT_PARAM_GET_ENABLED_EVENTS:
+ r->event_mask = device_get_enabled_events();
+ break;
+ case EC_DEVICE_EVENT_PARAM_SET_ENABLED_EVENTS:
+ device_set_enabled_events(p->event_mask);
+ r->event_mask = device_get_enabled_events();
+ break;
+ default:
+ return EC_RES_INVALID_PARAM;
+ }
+
+ args->response_size = sizeof(*r);
+
+ return EC_RES_SUCCESS;
+}
+DECLARE_HOST_COMMAND(EC_CMD_DEVICE_EVENT, device_event_cmd, EC_VER_MASK(0));
diff --git a/common/host_command.c b/common/host_command.c
index 06d687f51e..825515a66f 100644
--- a/common/host_command.c
+++ b/common/host_command.c
@@ -805,6 +805,9 @@ static int host_command_get_features(struct host_cmd_handler_args *args)
#ifdef HAS_TASK_RWSIG
| EC_FEATURE_MASK_0(EC_FEATURE_RWSIG)
#endif
+#ifdef CONFIG_DEVICE_EVENT
+ | EC_FEATURE_MASK_0(EC_FEATURE_DEVICE_EVENT)
+#endif
;
return EC_RES_SUCCESS;
}