// SPDX-License-Identifier: GPL-2.0+ /* * efi_selftest_textinput * * Copyright (c) 2018 Heinrich Schuchardt * * Provides a unit test for the EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL. * The unicode character and the scan code are printed for text * input. To run the test: * * setenv efi_selftest extended text input * bootefi selftest */ #include static const efi_guid_t text_input_ex_protocol_guid = EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL_GUID; static struct efi_simple_text_input_ex_protocol *con_in_ex; static struct efi_boot_services *boottime; /* * Setup unit test. * * @handle: handle of the loaded image * @systable: system table * @return: EFI_ST_SUCCESS for success */ static int setup(const efi_handle_t handle, const struct efi_system_table *systable) { efi_status_t ret; boottime = systable->boottime; ret = boottime->locate_protocol(&text_input_ex_protocol_guid, NULL, (void **)&con_in_ex); if (ret != EFI_SUCCESS) { con_in_ex = NULL; efi_st_error ("Extended text input protocol is not available.\n"); return EFI_ST_FAILURE; } return EFI_ST_SUCCESS; } /* * Execute unit test. * * @return: EFI_ST_SUCCESS for success */ static int execute(void) { struct efi_key_data input_key = {0,}; efi_status_t ret; efi_uintn_t index; if (!con_in_ex) { efi_st_printf("Setup failed\n"); return EFI_ST_FAILURE; } /* Drain the console input */ ret = con_in_ex->reset(con_in_ex, true); if (ret != EFI_SUCCESS) { efi_st_error("Reset failed\n"); return EFI_ST_FAILURE; } ret = con_in_ex->read_key_stroke_ex(con_in_ex, &input_key); if (ret != EFI_NOT_READY) { efi_st_error("Empty buffer not reported\n"); return EFI_ST_FAILURE; } efi_st_printf("Waiting for your input\n"); efi_st_printf("To terminate type 'x'\n"); for (;;) { /* Wait for next key */ ret = boottime->wait_for_event(1, &con_in_ex->wait_for_key_ex, &index); if (ret != EFI_ST_SUCCESS) { efi_st_error("WaitForEvent failed\n"); return EFI_ST_FAILURE; } ret = con_in_ex->read_key_stroke_ex(con_in_ex, &input_key); if (ret != EFI_SUCCESS) { efi_st_error("ReadKeyStroke failed\n"); return EFI_ST_FAILURE; } /* Allow 5 minutes until time out */ boottime->set_watchdog_timer(300, 0, 0, NULL); efi_st_printf("Unicode char %u (%ps), scan code %u (", (unsigned int)input_key.key.unicode_char, efi_st_translate_char(input_key.key.unicode_char), (unsigned int)input_key.key.scan_code); if (input_key.key_state.key_shift_state & EFI_SHIFT_STATE_VALID) { if (input_key.key_state.key_shift_state & (EFI_LEFT_SHIFT_PRESSED | EFI_RIGHT_SHIFT_PRESSED)) efi_st_printf("SHIFT+"); if (input_key.key_state.key_shift_state & (EFI_LEFT_ALT_PRESSED | EFI_RIGHT_ALT_PRESSED)) efi_st_printf("ALT+"); if (input_key.key_state.key_shift_state & (EFI_LEFT_CONTROL_PRESSED | EFI_RIGHT_CONTROL_PRESSED)) efi_st_printf("CTRL+"); if (input_key.key_state.key_shift_state & (EFI_LEFT_LOGO_PRESSED | EFI_RIGHT_LOGO_PRESSED)) efi_st_printf("META+"); if (input_key.key_state.key_shift_state == EFI_SHIFT_STATE_VALID) efi_st_printf("+"); } efi_st_printf("%ps)\n", efi_st_translate_code(input_key.key.scan_code)); switch (input_key.key.unicode_char) { case 'x': case 'X': return EFI_ST_SUCCESS; } } } EFI_UNIT_TEST(textinputex) = { .name = "extended text input", .phase = EFI_EXECUTE_BEFORE_BOOTTIME_EXIT, .setup = setup, .execute = execute, .on_request = true, };