// SPDX-License-Identifier: GPL-2.0+ /* * efi_selftest_controllers * * Copyright (c) 2017 Heinrich Schuchardt <xypron.glpk@gmx.de> * * This unit test checks the following protocol services: * ConnectController, DisconnectController, * InstallProtocol, ReinstallProtocol, UninstallProtocol, * OpenProtocol, CloseProtcol, OpenProtocolInformation */ #include <efi_selftest.h> #define NUMBER_OF_CHILD_CONTROLLERS 4 static int interface1 = 1; static int interface2 = 2; static struct efi_boot_services *boottime; const efi_guid_t guid_driver_binding_protocol = EFI_DRIVER_BINDING_PROTOCOL_GUID; static efi_guid_t guid_controller = EFI_GUID(0xe6ab1d96, 0x6bff, 0xdb42, 0xaa, 0x05, 0xc8, 0x1f, 0x7f, 0x45, 0x26, 0x34); static efi_guid_t guid_child_controller = EFI_GUID(0x1d41f6f5, 0x2c41, 0xddfb, 0xe2, 0x9b, 0xb8, 0x0e, 0x2e, 0xe8, 0x3a, 0x85); static efi_handle_t handle_controller; static efi_handle_t handle_child_controller[NUMBER_OF_CHILD_CONTROLLERS]; static efi_handle_t handle_driver; /* * Count child controllers * * @handle handle on which child controllers are installed * @protocol protocol for which the child controlles where installed * @count number of child controllers * @return status code */ static efi_status_t count_child_controllers(efi_handle_t handle, efi_guid_t *protocol, efi_uintn_t *count) { efi_status_t ret; efi_uintn_t entry_count; struct efi_open_protocol_info_entry *entry_buffer; *count = 0; ret = boottime->open_protocol_information(handle, protocol, &entry_buffer, &entry_count); if (ret != EFI_SUCCESS) return ret; if (!entry_count) return EFI_SUCCESS; while (entry_count) { if (entry_buffer[--entry_count].attributes & EFI_OPEN_PROTOCOL_BY_CHILD_CONTROLLER) ++*count; } ret = boottime->free_pool(entry_buffer); if (ret != EFI_SUCCESS) efi_st_error("Cannot free buffer\n"); return ret; } /* * Check if the driver supports the controller. * * @this driver binding protocol * @controller_handle handle of the controller * @remaining_device_path path specifying the child controller * @return status code */ static efi_status_t EFIAPI supported( struct efi_driver_binding_protocol *this, efi_handle_t controller_handle, struct efi_device_path *remaining_device_path) { efi_status_t ret; void *interface; ret = boottime->open_protocol( controller_handle, &guid_controller, &interface, handle_driver, controller_handle, EFI_OPEN_PROTOCOL_BY_DRIVER); switch (ret) { case EFI_ACCESS_DENIED: case EFI_ALREADY_STARTED: return ret; case EFI_SUCCESS: break; default: return EFI_UNSUPPORTED; } ret = boottime->close_protocol( controller_handle, &guid_controller, handle_driver, controller_handle); if (ret != EFI_SUCCESS) ret = EFI_UNSUPPORTED; return ret; } /* * Create child controllers and attach driver. * * @this driver binding protocol * @controller_handle handle of the controller * @remaining_device_path path specifying the child controller * @return status code */ static efi_status_t EFIAPI start( struct efi_driver_binding_protocol *this, efi_handle_t controller_handle, struct efi_device_path *remaining_device_path) { size_t i; efi_status_t ret; void *interface; /* Attach driver to controller */ ret = boottime->open_protocol( controller_handle, &guid_controller, &interface, handle_driver, controller_handle, EFI_OPEN_PROTOCOL_BY_DRIVER); switch (ret) { case EFI_ACCESS_DENIED: case EFI_ALREADY_STARTED: return ret; case EFI_SUCCESS: break; default: return EFI_UNSUPPORTED; } /* Create child controllers */ for (i = 0; i < NUMBER_OF_CHILD_CONTROLLERS; ++i) { ret = boottime->install_protocol_interface( &handle_child_controller[i], &guid_child_controller, EFI_NATIVE_INTERFACE, NULL); if (ret != EFI_SUCCESS) { efi_st_error("InstallProtocolInterface failed\n"); return EFI_ST_FAILURE; } ret = boottime->open_protocol( controller_handle, &guid_controller, &interface, handle_child_controller[i], handle_child_controller[i], EFI_OPEN_PROTOCOL_BY_CHILD_CONTROLLER); if (ret != EFI_SUCCESS) { efi_st_error("OpenProtocol failed\n"); return EFI_ST_FAILURE; } } return ret; } /* * Remove a single child controller from the parent controller. * * @controller_handle parent controller * @child_handle child controller * @return status code */ static efi_status_t disconnect_child(efi_handle_t controller_handle, efi_handle_t child_handle) { efi_status_t ret; ret = boottime->close_protocol( controller_handle, &guid_controller, child_handle, child_handle); if (ret != EFI_SUCCESS) { efi_st_error("Cannot close protocol\n"); return ret; } ret = boottime->uninstall_protocol_interface( child_handle, &guid_child_controller, NULL); if (ret != EFI_SUCCESS) { efi_st_error("Cannot uninstall protocol interface\n"); return ret; } return ret; } /* * Remove child controllers and disconnect the controller. * * @this driver binding protocol * @controller_handle handle of the controller * @number_of_children number of child controllers to remove * @child_handle_buffer handles of the child controllers to remove * @return status code */ static efi_status_t EFIAPI stop( struct efi_driver_binding_protocol *this, efi_handle_t controller_handle, size_t number_of_children, efi_handle_t *child_handle_buffer) { efi_status_t ret; efi_uintn_t count; struct efi_open_protocol_info_entry *entry_buffer; /* Destroy provided child controllers */ if (number_of_children) { efi_uintn_t i; for (i = 0; i < number_of_children; ++i) { ret = disconnect_child(controller_handle, child_handle_buffer[i]); if (ret != EFI_SUCCESS) return ret; } return EFI_SUCCESS; } /* Destroy all children */ ret = boottime->open_protocol_information( controller_handle, &guid_controller, &entry_buffer, &count); if (ret != EFI_SUCCESS) { efi_st_error("OpenProtocolInformation failed\n"); return ret; } while (count) { if (entry_buffer[--count].attributes & EFI_OPEN_PROTOCOL_BY_CHILD_CONTROLLER) { ret = disconnect_child( controller_handle, entry_buffer[count].agent_handle); if (ret != EFI_SUCCESS) return ret; } } ret = boottime->free_pool(entry_buffer); if (ret != EFI_SUCCESS) efi_st_error("Cannot free buffer\n"); /* Detach driver from controller */ ret = boottime->close_protocol( controller_handle, &guid_controller, handle_driver, controller_handle); if (ret != EFI_SUCCESS) { efi_st_error("Cannot close protocol\n"); return ret; } return EFI_SUCCESS; } /* Driver binding protocol interface */ static struct efi_driver_binding_protocol binding_interface = { supported, start, stop, 0xffffffff, NULL, NULL, }; /* * Setup unit test. * * @handle handle of the loaded image * @systable system table */ static int setup(const efi_handle_t img_handle, const struct efi_system_table *systable) { efi_status_t ret; boottime = systable->boottime; /* Create controller handle */ ret = boottime->install_protocol_interface( &handle_controller, &guid_controller, EFI_NATIVE_INTERFACE, &interface1); if (ret != EFI_SUCCESS) { efi_st_error("InstallProtocolInterface failed\n"); return EFI_ST_FAILURE; } /* Create driver handle */ ret = boottime->install_protocol_interface( &handle_driver, &guid_driver_binding_protocol, EFI_NATIVE_INTERFACE, &binding_interface); if (ret != EFI_SUCCESS) { efi_st_error("InstallProtocolInterface failed\n"); return EFI_ST_FAILURE; } return EFI_ST_SUCCESS; } /* * Execute unit test. * * The number of child controllers is checked after each of the following * actions: * * Connect a controller to a driver. * Disconnect and destroy a child controller. * Disconnect and destroy the remaining child controllers. * * Connect a controller to a driver. * Reinstall the driver protocol on the controller. * Uninstall the driver protocol from the controller. */ static int execute(void) { efi_status_t ret; efi_uintn_t count; /* Connect controller to driver */ ret = boottime->connect_controller(handle_controller, NULL, NULL, 1); if (ret != EFI_SUCCESS) { efi_st_error("Failed to connect controller\n"); return EFI_ST_FAILURE; } /* Check number of child controllers */ ret = count_child_controllers(handle_controller, &guid_controller, &count); if (ret != EFI_SUCCESS || count != NUMBER_OF_CHILD_CONTROLLERS) { efi_st_error("Number of children %u != %u\n", (unsigned int)count, NUMBER_OF_CHILD_CONTROLLERS); } /* Destroy second child controller */ ret = boottime->disconnect_controller(handle_controller, handle_driver, handle_child_controller[1]); if (ret != EFI_SUCCESS) { efi_st_error("Failed to disconnect child controller\n"); return EFI_ST_FAILURE; } /* Check number of child controllers */ ret = count_child_controllers(handle_controller, &guid_controller, &count); if (ret != EFI_SUCCESS || count != NUMBER_OF_CHILD_CONTROLLERS - 1) { efi_st_error("Destroying single child controller failed\n"); return EFI_ST_FAILURE; } /* Destroy remaining child controllers and disconnect controller */ ret = boottime->disconnect_controller(handle_controller, NULL, NULL); if (ret != EFI_SUCCESS) { efi_st_error("Failed to disconnect controller\n"); return EFI_ST_FAILURE; } /* Check number of child controllers */ ret = count_child_controllers(handle_controller, &guid_controller, &count); if (ret != EFI_SUCCESS || count) { efi_st_error("Destroying child controllers failed\n"); return EFI_ST_FAILURE; } /* Connect controller to driver */ ret = boottime->connect_controller(handle_controller, NULL, NULL, 1); if (ret != EFI_SUCCESS) { efi_st_error("Failed to connect controller\n"); return EFI_ST_FAILURE; } /* Check number of child controllers */ ret = count_child_controllers(handle_controller, &guid_controller, &count); if (ret != EFI_SUCCESS || count != NUMBER_OF_CHILD_CONTROLLERS) { efi_st_error("Number of children %u != %u\n", (unsigned int)count, NUMBER_OF_CHILD_CONTROLLERS); } /* Try to uninstall controller protocol using the wrong interface */ ret = boottime->uninstall_protocol_interface(handle_controller, &guid_controller, &interface2); if (ret == EFI_SUCCESS) { efi_st_error( "Interface not checked when uninstalling protocol\n"); return EFI_ST_FAILURE; } /* Reinstall controller protocol */ ret = boottime->reinstall_protocol_interface(handle_controller, &guid_controller, &interface1, &interface2); if (ret != EFI_SUCCESS) { efi_st_error("Failed to reinstall protocols\n"); return EFI_ST_FAILURE; } /* Check number of child controllers */ ret = count_child_controllers(handle_controller, &guid_controller, &count); if (ret != EFI_SUCCESS || count != NUMBER_OF_CHILD_CONTROLLERS) { efi_st_error("Number of children %u != %u\n", (unsigned int)count, NUMBER_OF_CHILD_CONTROLLERS); } /* Uninstall controller protocol */ ret = boottime->uninstall_protocol_interface(handle_controller, &guid_controller, &interface2); if (ret != EFI_SUCCESS) { efi_st_error("Failed to uninstall protocols\n"); return EFI_ST_FAILURE; } /* Check number of child controllers */ ret = count_child_controllers(handle_controller, &guid_controller, &count); if (ret == EFI_SUCCESS) efi_st_error("Uninstall failed\n"); return EFI_ST_SUCCESS; } EFI_UNIT_TEST(controllers) = { .name = "controllers", .phase = EFI_EXECUTE_BEFORE_BOOTTIME_EXIT, .setup = setup, .execute = execute, };