2017-09-15 08:06:19 +00:00
|
|
|
/*
|
2017-10-04 13:31:26 +00:00
|
|
|
* efi_selftest_exitbootservices
|
2017-09-15 08:06:19 +00:00
|
|
|
*
|
|
|
|
* Copyright (c) 2017 Heinrich Schuchardt <xypron.glpk@gmx.de>
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: GPL-2.0+
|
|
|
|
*
|
|
|
|
* This unit test checks that the notification function of an
|
|
|
|
* EVT_SIGNAL_EXIT_BOOT_SERVICES event is called exactly once.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <efi_selftest.h>
|
|
|
|
|
|
|
|
static struct efi_boot_services *boottime;
|
|
|
|
static struct efi_event *event_notify;
|
2017-10-04 13:31:26 +00:00
|
|
|
static unsigned int notification_count;
|
2017-09-15 08:06:19 +00:00
|
|
|
|
|
|
|
/*
|
2017-10-04 13:31:26 +00:00
|
|
|
* Notification function, increments the notification count.
|
2017-09-15 08:06:19 +00:00
|
|
|
*
|
|
|
|
* @event notified event
|
2017-10-04 13:31:26 +00:00
|
|
|
* @context pointer to the notification count
|
2017-09-15 08:06:19 +00:00
|
|
|
*/
|
|
|
|
static void EFIAPI notify(struct efi_event *event, void *context)
|
|
|
|
{
|
2017-10-04 13:31:26 +00:00
|
|
|
unsigned int *count = context;
|
|
|
|
|
|
|
|
++*count;
|
2017-09-15 08:06:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Setup unit test.
|
|
|
|
*
|
|
|
|
* Create an EVT_SIGNAL_EXIT_BOOT_SERVICES event.
|
|
|
|
*
|
|
|
|
* @handle: handle of the loaded image
|
|
|
|
* @systable: system table
|
2017-10-04 13:31:26 +00:00
|
|
|
* @return: EFI_ST_SUCCESS for success
|
2017-09-15 08:06:19 +00:00
|
|
|
*/
|
|
|
|
static int setup(const efi_handle_t handle,
|
|
|
|
const struct efi_system_table *systable)
|
|
|
|
{
|
|
|
|
efi_status_t ret;
|
|
|
|
|
|
|
|
boottime = systable->boottime;
|
|
|
|
|
2017-10-04 13:31:26 +00:00
|
|
|
notification_count = 0;
|
2017-09-15 08:06:19 +00:00
|
|
|
ret = boottime->create_event(EVT_SIGNAL_EXIT_BOOT_SERVICES,
|
2017-10-04 13:31:26 +00:00
|
|
|
TPL_CALLBACK, notify,
|
|
|
|
(void *)¬ification_count,
|
2017-09-15 08:06:19 +00:00
|
|
|
&event_notify);
|
|
|
|
if (ret != EFI_SUCCESS) {
|
|
|
|
efi_st_error("could not create event\n");
|
2017-10-04 13:31:26 +00:00
|
|
|
return EFI_ST_FAILURE;
|
2017-09-15 08:06:19 +00:00
|
|
|
}
|
2017-10-04 13:31:26 +00:00
|
|
|
return EFI_ST_SUCCESS;
|
2017-09-15 08:06:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Tear down unit test.
|
|
|
|
*
|
|
|
|
* Close the event created in setup.
|
2017-10-04 13:31:26 +00:00
|
|
|
*
|
|
|
|
* @return: EFI_ST_SUCCESS for success
|
2017-09-15 08:06:19 +00:00
|
|
|
*/
|
|
|
|
static int teardown(void)
|
|
|
|
{
|
|
|
|
efi_status_t ret;
|
|
|
|
|
|
|
|
if (event_notify) {
|
|
|
|
ret = boottime->close_event(event_notify);
|
|
|
|
event_notify = NULL;
|
|
|
|
if (ret != EFI_SUCCESS) {
|
|
|
|
efi_st_error("could not close event\n");
|
2017-10-04 13:31:26 +00:00
|
|
|
return EFI_ST_FAILURE;
|
2017-09-15 08:06:19 +00:00
|
|
|
}
|
|
|
|
}
|
2017-10-04 13:31:26 +00:00
|
|
|
return EFI_ST_SUCCESS;
|
2017-09-15 08:06:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Execute unit test.
|
|
|
|
*
|
|
|
|
* Check that the notification function of the EVT_SIGNAL_EXIT_BOOT_SERVICES
|
|
|
|
* event has been called.
|
|
|
|
*
|
|
|
|
* Call ExitBootServices again and check that the notification function is
|
|
|
|
* not called again.
|
2017-10-04 13:31:26 +00:00
|
|
|
*
|
|
|
|
* @return: EFI_ST_SUCCESS for success
|
2017-09-15 08:06:19 +00:00
|
|
|
*/
|
|
|
|
static int execute(void)
|
|
|
|
{
|
2017-10-04 13:31:26 +00:00
|
|
|
if (notification_count != 1) {
|
|
|
|
efi_st_error("ExitBootServices was not notified\n");
|
|
|
|
return EFI_ST_FAILURE;
|
2017-09-15 08:06:19 +00:00
|
|
|
}
|
|
|
|
efi_st_exit_boot_services();
|
2017-10-04 13:31:26 +00:00
|
|
|
if (notification_count != 1) {
|
|
|
|
efi_st_error("ExitBootServices was notified twice\n");
|
|
|
|
return EFI_ST_FAILURE;
|
2017-09-15 08:06:19 +00:00
|
|
|
}
|
2017-10-04 13:31:26 +00:00
|
|
|
return EFI_ST_SUCCESS;
|
2017-09-15 08:06:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
EFI_UNIT_TEST(exitbootservices) = {
|
|
|
|
.name = "ExitBootServices",
|
|
|
|
.phase = EFI_SETUP_BEFORE_BOOTTIME_EXIT,
|
|
|
|
.setup = setup,
|
|
|
|
.execute = execute,
|
|
|
|
.teardown = teardown,
|
|
|
|
};
|