blob: f1a136044d42ff07eaa996781cc447c0e34e651d [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Heinrich Schuchardt630d1342017-09-15 10:06:19 +02002/*
Heinrich Schuchardt1afe5f72017-10-04 15:31:26 +02003 * efi_selftest_exitbootservices
Heinrich Schuchardt630d1342017-09-15 10:06:19 +02004 *
5 * Copyright (c) 2017 Heinrich Schuchardt <xypron.glpk@gmx.de>
6 *
Heinrich Schuchardt630d1342017-09-15 10:06:19 +02007 * This unit test checks that the notification function of an
8 * EVT_SIGNAL_EXIT_BOOT_SERVICES event is called exactly once.
9 */
10
11#include <efi_selftest.h>
12
13static struct efi_boot_services *boottime;
14static struct efi_event *event_notify;
Heinrich Schuchardt1afe5f72017-10-04 15:31:26 +020015static unsigned int notification_count;
Heinrich Schuchardt630d1342017-09-15 10:06:19 +020016
17/*
Heinrich Schuchardt1afe5f72017-10-04 15:31:26 +020018 * Notification function, increments the notification count.
Heinrich Schuchardt630d1342017-09-15 10:06:19 +020019 *
20 * @event notified event
Heinrich Schuchardt1afe5f72017-10-04 15:31:26 +020021 * @context pointer to the notification count
Heinrich Schuchardt630d1342017-09-15 10:06:19 +020022 */
23static void EFIAPI notify(struct efi_event *event, void *context)
24{
Heinrich Schuchardt1afe5f72017-10-04 15:31:26 +020025 unsigned int *count = context;
26
27 ++*count;
Heinrich Schuchardt630d1342017-09-15 10:06:19 +020028}
29
30/*
31 * Setup unit test.
32 *
33 * Create an EVT_SIGNAL_EXIT_BOOT_SERVICES event.
34 *
35 * @handle: handle of the loaded image
36 * @systable: system table
Heinrich Schuchardt1afe5f72017-10-04 15:31:26 +020037 * @return: EFI_ST_SUCCESS for success
Heinrich Schuchardt630d1342017-09-15 10:06:19 +020038 */
39static int setup(const efi_handle_t handle,
40 const struct efi_system_table *systable)
41{
42 efi_status_t ret;
43
44 boottime = systable->boottime;
45
Heinrich Schuchardt1afe5f72017-10-04 15:31:26 +020046 notification_count = 0;
Heinrich Schuchardt630d1342017-09-15 10:06:19 +020047 ret = boottime->create_event(EVT_SIGNAL_EXIT_BOOT_SERVICES,
Heinrich Schuchardt1afe5f72017-10-04 15:31:26 +020048 TPL_CALLBACK, notify,
49 (void *)&notification_count,
Heinrich Schuchardt630d1342017-09-15 10:06:19 +020050 &event_notify);
51 if (ret != EFI_SUCCESS) {
52 efi_st_error("could not create event\n");
Heinrich Schuchardt1afe5f72017-10-04 15:31:26 +020053 return EFI_ST_FAILURE;
Heinrich Schuchardt630d1342017-09-15 10:06:19 +020054 }
Heinrich Schuchardt1afe5f72017-10-04 15:31:26 +020055 return EFI_ST_SUCCESS;
Heinrich Schuchardt630d1342017-09-15 10:06:19 +020056}
57
58/*
59 * Tear down unit test.
60 *
61 * Close the event created in setup.
Heinrich Schuchardt1afe5f72017-10-04 15:31:26 +020062 *
63 * @return: EFI_ST_SUCCESS for success
Heinrich Schuchardt630d1342017-09-15 10:06:19 +020064 */
65static int teardown(void)
66{
67 efi_status_t ret;
68
69 if (event_notify) {
70 ret = boottime->close_event(event_notify);
71 event_notify = NULL;
72 if (ret != EFI_SUCCESS) {
73 efi_st_error("could not close event\n");
Heinrich Schuchardt1afe5f72017-10-04 15:31:26 +020074 return EFI_ST_FAILURE;
Heinrich Schuchardt630d1342017-09-15 10:06:19 +020075 }
76 }
Heinrich Schuchardt1afe5f72017-10-04 15:31:26 +020077 return EFI_ST_SUCCESS;
Heinrich Schuchardt630d1342017-09-15 10:06:19 +020078}
79
80/*
81 * Execute unit test.
82 *
83 * Check that the notification function of the EVT_SIGNAL_EXIT_BOOT_SERVICES
84 * event has been called.
85 *
86 * Call ExitBootServices again and check that the notification function is
87 * not called again.
Heinrich Schuchardt1afe5f72017-10-04 15:31:26 +020088 *
89 * @return: EFI_ST_SUCCESS for success
Heinrich Schuchardt630d1342017-09-15 10:06:19 +020090 */
91static int execute(void)
92{
Heinrich Schuchardt1afe5f72017-10-04 15:31:26 +020093 if (notification_count != 1) {
94 efi_st_error("ExitBootServices was not notified\n");
95 return EFI_ST_FAILURE;
Heinrich Schuchardt630d1342017-09-15 10:06:19 +020096 }
97 efi_st_exit_boot_services();
Heinrich Schuchardt1afe5f72017-10-04 15:31:26 +020098 if (notification_count != 1) {
99 efi_st_error("ExitBootServices was notified twice\n");
100 return EFI_ST_FAILURE;
Heinrich Schuchardt630d1342017-09-15 10:06:19 +0200101 }
Heinrich Schuchardt1afe5f72017-10-04 15:31:26 +0200102 return EFI_ST_SUCCESS;
Heinrich Schuchardt630d1342017-09-15 10:06:19 +0200103}
104
105EFI_UNIT_TEST(exitbootservices) = {
106 .name = "ExitBootServices",
107 .phase = EFI_SETUP_BEFORE_BOOTTIME_EXIT,
108 .setup = setup,
109 .execute = execute,
110 .teardown = teardown,
111};