mirror of
https://github.com/AsahiLinux/u-boot
synced 2025-01-10 04:08:52 +00:00
e102038641
gcc 4.8.4 wants to see all levels of braces when initializing a structure to zeros. Signed-off-by: Heinrich Schuchardt <xypron.glpk@gmx.de> Signed-off-by: Alexander Graf <agraf@suse.de>
198 lines
4.8 KiB
C
198 lines
4.8 KiB
C
// 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;
|
|
|
|
static void *efi_key_notify_handle;
|
|
static bool efi_running;
|
|
|
|
/**
|
|
* efi_key_notify_function() - key notification function
|
|
*
|
|
* This function is called when the registered key is hit.
|
|
*
|
|
* @key_data: next key
|
|
* Return: status code
|
|
*/
|
|
static efi_status_t EFIAPI efi_key_notify_function
|
|
(struct efi_key_data *key_data)
|
|
{
|
|
efi_running = false;
|
|
|
|
return EFI_SUCCESS;
|
|
}
|
|
|
|
/*
|
|
* 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;
|
|
struct efi_key_data key_data = {
|
|
.key = {
|
|
.scan_code = 0,
|
|
.unicode_char = 0x18
|
|
},
|
|
.key_state = {
|
|
.key_shift_state = EFI_SHIFT_STATE_VALID |
|
|
EFI_LEFT_CONTROL_PRESSED,
|
|
.key_toggle_state = EFI_TOGGLE_STATE_INVALID,
|
|
},
|
|
};
|
|
|
|
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;
|
|
}
|
|
|
|
ret = con_in_ex->register_key_notify(con_in_ex, &key_data,
|
|
efi_key_notify_function,
|
|
&efi_key_notify_handle);
|
|
if (ret != EFI_SUCCESS) {
|
|
efi_key_notify_handle = NULL;
|
|
efi_st_error
|
|
("Notify function could not be registered.\n");
|
|
return EFI_ST_FAILURE;
|
|
}
|
|
efi_running = true;
|
|
|
|
return EFI_ST_SUCCESS;
|
|
}
|
|
|
|
/*
|
|
* Tear down unit test.
|
|
*
|
|
* Unregister notify function.
|
|
*
|
|
* @return: EFI_ST_SUCCESS for success
|
|
*/
|
|
static int teardown(void)
|
|
{
|
|
efi_status_t ret;
|
|
|
|
ret = con_in_ex->unregister_key_notify
|
|
(con_in_ex, efi_key_notify_handle);
|
|
if (ret != EFI_SUCCESS) {
|
|
efi_st_error
|
|
("Notify function could not be registered.\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, 0}, {0, 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 'CTRL+x'\n");
|
|
|
|
while (efi_running) {
|
|
/* 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));
|
|
|
|
}
|
|
return EFI_ST_SUCCESS;
|
|
}
|
|
|
|
EFI_UNIT_TEST(textinputex) = {
|
|
.name = "extended text input",
|
|
.phase = EFI_EXECUTE_BEFORE_BOOTTIME_EXIT,
|
|
.setup = setup,
|
|
.execute = execute,
|
|
.teardown = teardown,
|
|
.on_request = true,
|
|
};
|