/* Copyright 2022 The ChromiumOS Authors * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include #include #include #include #include #include #include "console.h" #include "host_command.h" #include "mkbp_event.h" #include "mkbp_fifo.h" #include "test/drivers/test_mocks.h" #include "test/drivers/test_state.h" /** * @brief FFF fake that will be registered as a callback to monitor the EC->AP * interrupt pin. Implements `gpio_callback_handler_t`. */ FAKE_VOID_FUNC(interrupt_gpio_monitor, const struct device *, struct gpio_callback *, gpio_port_pins_t); /** * @brief Fixture to hold state while the suite is running. */ struct event_fixture { /** Configuration for the interrupt pin change callback */ struct gpio_callback callback_config; }; static struct event_fixture fixture; ZTEST(mkbp_event, host_command_get_events__empty) { /* Issue a host command to get the next event (from any source) */ uint16_t ret; struct ec_response_get_next_event response; struct host_cmd_handler_args args = BUILD_HOST_COMMAND_RESPONSE(EC_CMD_GET_NEXT_EVENT, 0, response); ret = host_command_process(&args); zassert_equal(EC_RES_UNAVAILABLE, ret, "Expected EC_RES_UNAVAILABLE but got %d", ret); } ZTEST(mkbp_event, host_command_get_events__get_event) { /* Dispatch a fake keyboard event and ensure it gets returned by the * host command. */ int ret; struct ec_response_get_next_event expected_event = { .event_type = EC_MKBP_EVENT_KEY_MATRIX, .data.key_matrix = { /* Arbitrary key matrix data (uint8_t[13]) */ 0x1, 0x2, 0x3, 0x4, 0x5, 0x6, 0x7, 0x8, 0x9, 0xa, 0xb, 0xc, 0xd }, }; /* Add the above event to the MKBP keyboard FIFO and raise the event */ ret = mkbp_fifo_add(expected_event.event_type, (const uint8_t *)&expected_event.data.key_matrix); activate_mkbp_with_events(BIT(expected_event.event_type)); zassert_equal(EC_SUCCESS, ret, "Got %d when adding to FIFO", ret); /* Retrieve this event via host command */ struct ec_response_get_next_event response; struct host_cmd_handler_args args = BUILD_HOST_COMMAND_RESPONSE(EC_CMD_GET_NEXT_EVENT, 0, response); ret = host_command_process(&args); zassert_equal(EC_RES_SUCCESS, ret, "Expected EC_RES_SUCCESS but got %d", ret); /* Compare event data in response */ zassert_equal(expected_event.event_type, response.event_type, "Got event type 0x%02x", response.event_type); zassert_mem_equal(&expected_event.data.key_matrix, &response.data.key_matrix, sizeof(expected_event.data.key_matrix), "Event data payload does not match."); /* Check for two pin change events (initial assertion when the event * was sent, and a de-assertion once we retrieved it through the host * command) */ zassert_equal(2, interrupt_gpio_monitor_fake.call_count, "Only %d pin events", interrupt_gpio_monitor_fake.call_count); } ZTEST(mkbp_event, no_ap_response) { /* Cause an event but do not send any host commands. This should cause * the EC to send the interrupt to the AP 3 times before giving up. * Use the GPIO emulator to monitor for interrupts. */ int ret; struct ec_response_get_next_event expected_event = { .event_type = EC_MKBP_EVENT_KEY_MATRIX, }; ret = mkbp_fifo_add(expected_event.event_type, (uint8_t *)&expected_event.data.key_matrix); activate_mkbp_with_events(BIT(expected_event.event_type)); zassert_equal(EC_SUCCESS, ret, "Got %d when adding to FIFO", ret); /* EC will attempt to signal the interrupt 3 times. Each attempt lasts * 1 second, so sleep for 5 and then count the number of times the * interrupt pin was asserted. (It does not get de-asserted) */ k_sleep(K_SECONDS(5)); zassert_equal(3, interrupt_gpio_monitor_fake.call_count, "Interrupt pin asserted only %d times.", interrupt_gpio_monitor_fake.call_count); } /* Set up a mock for mkbp_send_event(). This function is called by the MKBP * event sources to signal that a new event is available for servicing. Since we * are unit testing just event handling code, we do not want the various event * source tasks to raise unexpected events during testing and throw us off. * This mock will essentially cause mkbp_send_event() to become a no-op and * block the reset of the EC code from raising events and interfering. The test * code will bypass this by calling mkbp_event.c's internal * `activate_mkbp_with_events()` directly. */ FAKE_VALUE_FUNC(int, mkbp_send_event, uint8_t); static void *setup(void) { /* Add a callback to the EC->AP interrupt pin so we can log interrupt * attempts with an FFF fake. */ const struct gpio_dt_spec *interrupt_pin = GPIO_DT_FROM_NODELABEL(gpio_ap_ec_int_l); fixture.callback_config = (struct gpio_callback){ .pin_mask = BIT(interrupt_pin->pin), .handler = interrupt_gpio_monitor, }; zassume_ok(gpio_add_callback(interrupt_pin->port, &fixture.callback_config), "Could not configure GPIO callback."); return &fixture; } static void teardown(void *data) { /* Remove the GPIO callback on the interrupt pin */ struct event_fixture *f = (struct event_fixture *)data; const struct gpio_dt_spec *interrupt_pin = GPIO_DT_FROM_NODELABEL(gpio_ap_ec_int_l); gpio_remove_callback(interrupt_pin->port, &f->callback_config); } static void reset_events(void *data) { /* Clear any keyboard scan events (type EC_MKBP_EVENT_KEY_MATRIX) */ mkbp_clear_fifo(); /* Clear pending events */ mkbp_event_clear_all(); /* Mock reset */ RESET_FAKE(interrupt_gpio_monitor); RESET_FAKE(mkbp_send_event); mkbp_send_event_fake.return_val = 1; } ZTEST_SUITE(mkbp_event, drivers_predicate_post_main, setup, reset_events, reset_events, teardown);