| // SPDX-License-Identifier: GPL-2.0+ |
| /* |
| * efi_selftest_devicepath_util |
| * |
| * Copyright (c) 2018 Heinrich Schuchardt <xypron.glpk@gmx.de> |
| * |
| * This unit test checks the device path utilities protocol. |
| */ |
| |
| #include <efi_selftest.h> |
| |
| static struct efi_boot_services *boottime; |
| |
| static efi_guid_t guid_device_path_utilities_protocol = |
| EFI_DEVICE_PATH_UTILITIES_PROTOCOL_GUID; |
| |
| struct efi_device_path_utilities_protocol *dpu; |
| |
| /* |
| * Setup unit test. |
| * |
| * Locate the device path utilities protocol. |
| * |
| * @handle: handle of the loaded image |
| * @systable: system table |
| */ |
| static int setup(const efi_handle_t img_handle, |
| const struct efi_system_table *systable) |
| { |
| int ret; |
| |
| boottime = systable->boottime; |
| |
| ret = boottime->locate_protocol(&guid_device_path_utilities_protocol, |
| NULL, (void **)&dpu); |
| if (ret != EFI_SUCCESS) { |
| dpu = NULL; |
| efi_st_error( |
| "Device path to text protocol is not available.\n"); |
| return EFI_ST_FAILURE; |
| } |
| |
| return EFI_ST_SUCCESS; |
| } |
| |
| /* |
| * Create a device path consisting of a single media device node followed by an |
| * end node. |
| * |
| * @length: length of the media device node |
| * @dp: device path |
| * @return: status code |
| */ |
| static int create_single_node_device_path(unsigned int length, |
| struct efi_device_path **dp) |
| { |
| struct efi_device_path *node; |
| efi_uintn_t len; |
| int ret; |
| |
| node = dpu->create_device_node(DEVICE_PATH_TYPE_MEDIA_DEVICE, |
| DEVICE_PATH_SUB_TYPE_FILE_PATH, length); |
| if (!node) { |
| efi_st_error("CreateDeviceNode failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| *dp = dpu->append_device_node(NULL, node); |
| if (!*dp) { |
| efi_st_error("AppendDeviceNode failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| ret = boottime->free_pool(node); |
| if (ret != EFI_ST_SUCCESS) { |
| efi_st_error("FreePool failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| len = dpu->get_device_path_size(*dp); |
| if (len != length + 4) { |
| efi_st_error("Wrong device path length %u, expected %u\n", |
| (unsigned int)len, length); |
| return EFI_ST_FAILURE; |
| } |
| return EFI_ST_SUCCESS; |
| } |
| |
| /* |
| * Execute unit test. |
| * |
| * In the test device paths are created, copied, and concatenated. The device |
| * path length is used as a measure of success. |
| */ |
| static int execute(void) |
| { |
| struct efi_device_path *dp1; |
| struct efi_device_path *dp2; |
| struct efi_device_path *dp3; |
| |
| efi_uintn_t len; |
| int ret; |
| |
| /* IsDevicePathMultiInstance(NULL) */ |
| if (dpu->is_device_path_multi_instance(NULL)) { |
| efi_st_error("IsDevicePathMultiInstance(NULL) returned true\n"); |
| return EFI_ST_FAILURE; |
| } |
| /* GetDevicePathSize(NULL) */ |
| len = dpu->get_device_path_size(NULL); |
| if (len) { |
| efi_st_error("Wrong device path length %u, expected 0\n", |
| (unsigned int)len); |
| return EFI_ST_FAILURE; |
| } |
| /* DuplicateDevicePath(NULL) */ |
| dp1 = dpu->duplicate_device_path(NULL); |
| if (dp1) { |
| efi_st_error("DuplicateDevicePath(NULL) failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| /* AppendDevicePath(NULL, NULL) */ |
| dp1 = dpu->append_device_path(NULL, NULL); |
| if (!dp1) { |
| efi_st_error("AppendDevicePath(NULL, NULL) failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| len = dpu->get_device_path_size(dp1); |
| if (len != 4) { |
| efi_st_error("Wrong device path length %u, expected 4\n", |
| (unsigned int)len); |
| return EFI_ST_FAILURE; |
| } |
| ret = boottime->free_pool(dp1); |
| if (ret != EFI_ST_SUCCESS) { |
| efi_st_error("FreePool failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| /* CreateDeviceNode */ |
| ret = create_single_node_device_path(21, &dp1); |
| if (ret != EFI_ST_SUCCESS) |
| return ret; |
| ret = create_single_node_device_path(17, &dp2); |
| if (ret != EFI_ST_SUCCESS) |
| return ret; |
| /* AppendDevicePath */ |
| dp3 = dpu->append_device_path(dp1, dp2); |
| if (!dp3) { |
| efi_st_error("AppendDevicePath failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| if (dp3 == dp1 || dp3 == dp2) { |
| efi_st_error("AppendDevicePath reused buffer\n"); |
| return EFI_ST_FAILURE; |
| } |
| len = dpu->get_device_path_size(dp3); |
| /* 21 + 17 + 4 */ |
| if (len != 42) { |
| efi_st_error("Wrong device path length %u, expected 42\n", |
| (unsigned int)len); |
| return EFI_ST_FAILURE; |
| } |
| ret = boottime->free_pool(dp2); |
| if (ret != EFI_ST_SUCCESS) { |
| efi_st_error("FreePool failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| /* AppendDeviceNode */ |
| dp2 = dpu->append_device_node(dp1, dp3); |
| if (!dp2) { |
| efi_st_error("AppendDevicePath failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| len = dpu->get_device_path_size(dp2); |
| /* 21 + 21 + 4 */ |
| if (len != 46) { |
| printf("%s(%d) %s\n", __FILE__, __LINE__, __func__); |
| efi_st_error("Wrong device path length %u, expected 46\n", |
| (unsigned int)len); |
| return EFI_ST_FAILURE; |
| } |
| ret = boottime->free_pool(dp1); |
| if (ret != EFI_ST_SUCCESS) { |
| efi_st_error("FreePool failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| /* IsDevicePathMultiInstance */ |
| if (dpu->is_device_path_multi_instance(dp2)) { |
| printf("%s(%d) %s\n", __FILE__, __LINE__, __func__); |
| efi_st_error("IsDevicePathMultiInstance returned true\n"); |
| return EFI_ST_FAILURE; |
| } |
| /* AppendDevicePathInstance */ |
| dp1 = dpu->append_device_path_instance(dp2, dp3); |
| if (!dp1) { |
| efi_st_error("AppendDevicePathInstance failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| len = dpu->get_device_path_size(dp1); |
| /* 46 + 42 */ |
| if (len != 88) { |
| efi_st_error("Wrong device path length %u, expected 88\n", |
| (unsigned int)len); |
| return EFI_ST_FAILURE; |
| } |
| /* IsDevicePathMultiInstance */ |
| if (!dpu->is_device_path_multi_instance(dp1)) { |
| efi_st_error("IsDevicePathMultiInstance returned false\n"); |
| return EFI_ST_FAILURE; |
| } |
| ret = boottime->free_pool(dp2); |
| if (ret != EFI_ST_SUCCESS) { |
| efi_st_error("FreePool failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| ret = boottime->free_pool(dp3); |
| if (ret != EFI_ST_SUCCESS) { |
| efi_st_error("FreePool failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| /* GetNextDevicePathInstance */ |
| dp3 = dp1; |
| dp2 = dpu->get_next_device_path_instance(&dp1, &len); |
| if (!dp2) { |
| efi_st_error("GetNextDevicePathInstance failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| if (!dp1) { |
| efi_st_error("GetNextDevicePathInstance no 2nd instance\n"); |
| return EFI_ST_FAILURE; |
| } |
| if (len != 46) { |
| efi_st_error("Wrong device path length %u, expected 46\n", |
| (unsigned int)len); |
| return EFI_ST_FAILURE; |
| } |
| len = dpu->get_device_path_size(dp1); |
| if (len != 42) { |
| efi_st_error("Wrong device path length %u, expected 42\n", |
| (unsigned int)len); |
| return EFI_ST_FAILURE; |
| } |
| ret = boottime->free_pool(dp2); |
| if (ret != EFI_ST_SUCCESS) { |
| efi_st_error("FreePool failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| dp2 = dpu->get_next_device_path_instance(&dp1, &len); |
| if (!dp2) { |
| efi_st_error("GetNextDevicePathInstance failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| if (len != 42) { |
| efi_st_error("Wrong device path length %u, expected 46\n", |
| (unsigned int)len); |
| return EFI_ST_FAILURE; |
| } |
| if (dp1) { |
| efi_st_error("GetNextDevicePathInstance did not signal end\n"); |
| return EFI_ST_FAILURE; |
| } |
| |
| /* Clean up */ |
| ret = boottime->free_pool(dp2); |
| if (ret != EFI_ST_SUCCESS) { |
| efi_st_error("FreePool failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| ret = boottime->free_pool(dp3); |
| if (ret != EFI_ST_SUCCESS) { |
| efi_st_error("FreePool failed\n"); |
| return EFI_ST_FAILURE; |
| } |
| |
| return EFI_ST_SUCCESS; |
| } |
| |
| EFI_UNIT_TEST(dputil) = { |
| .name = "device path utilities protocol", |
| .phase = EFI_EXECUTE_BEFORE_BOOTTIME_EXIT, |
| .setup = setup, |
| .execute = execute, |
| }; |