123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139 |
- // SPDX-License-Identifier: GPL-2.0+
- /*
- * efi_selftest_textinput
- *
- * Copyright (c) 2018 Heinrich Schuchardt <xypron.glpk@gmx.de>
- *
- * 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 <efi_selftest.h>
- 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,
- };
|