blob: bf38392fac38dd71476542868bba0b3337ca5d0c [file] [log] [blame]
AKASHI Takahiro9b08b9a2024-01-17 13:39:41 +09001// SPDX-License-Identifier: GPL-2.0+
2/*
3 * For the code moved from cmd/bootefi.c
4 * Copyright (c) 2016 Alexander Graf
5 */
6
7#define LOG_CATEGORY LOGC_EFI
8
9#include <charset.h>
10#include <efi.h>
11#include <efi_loader.h>
12#include <env.h>
13#include <image.h>
14#include <log.h>
15#include <malloc.h>
16
17static struct efi_device_path *bootefi_image_path;
18static struct efi_device_path *bootefi_device_path;
19static void *image_addr;
20static size_t image_size;
21
22/**
23 * efi_get_image_parameters() - return image parameters
24 *
25 * @img_addr: address of loaded image in memory
26 * @img_size: size of loaded image
27 */
28void efi_get_image_parameters(void **img_addr, size_t *img_size)
29{
30 *img_addr = image_addr;
31 *img_size = image_size;
32}
33
34/**
35 * efi_clear_bootdev() - clear boot device
36 */
37void efi_clear_bootdev(void)
38{
39 efi_free_pool(bootefi_device_path);
40 efi_free_pool(bootefi_image_path);
41 bootefi_device_path = NULL;
42 bootefi_image_path = NULL;
43 image_addr = NULL;
44 image_size = 0;
45}
46
47/**
48 * efi_set_bootdev() - set boot device
49 *
50 * This function is called when a file is loaded, e.g. via the 'load' command.
51 * We use the path to this file to inform the UEFI binary about the boot device.
52 *
53 * @dev: device, e.g. "MMC"
54 * @devnr: number of the device, e.g. "1:2"
55 * @path: path to file loaded
56 * @buffer: buffer with file loaded
57 * @buffer_size: size of file loaded
58 */
59void efi_set_bootdev(const char *dev, const char *devnr, const char *path,
60 void *buffer, size_t buffer_size)
61{
62 struct efi_device_path *device, *image;
63 efi_status_t ret;
64
65 log_debug("dev=%s, devnr=%s, path=%s, buffer=%p, size=%zx\n", dev,
66 devnr, path, buffer, buffer_size);
67
68 /* Forget overwritten image */
69 if (buffer + buffer_size >= image_addr &&
70 image_addr + image_size >= buffer)
71 efi_clear_bootdev();
72
73 /* Remember only PE-COFF and FIT images */
74 if (efi_check_pe(buffer, buffer_size, NULL) != EFI_SUCCESS) {
75 if (IS_ENABLED(CONFIG_FIT) &&
76 !fit_check_format(buffer, IMAGE_SIZE_INVAL)) {
77 /*
78 * FIT images of type EFI_OS are started via command
79 * bootm. We should not use their boot device with the
80 * bootefi command.
81 */
82 buffer = 0;
83 buffer_size = 0;
84 } else {
85 log_debug("- not remembering image\n");
86 return;
87 }
88 }
89
90 /* efi_set_bootdev() is typically called repeatedly, recover memory */
91 efi_clear_bootdev();
92
93 image_addr = buffer;
94 image_size = buffer_size;
95
96 ret = efi_dp_from_name(dev, devnr, path, &device, &image);
97 if (ret == EFI_SUCCESS) {
98 bootefi_device_path = device;
99 if (image) {
100 /* FIXME: image should not contain device */
101 struct efi_device_path *image_tmp = image;
102
103 efi_dp_split_file_path(image, &device, &image);
104 efi_free_pool(image_tmp);
105 }
106 bootefi_image_path = image;
107 log_debug("- boot device %pD\n", device);
108 if (image)
109 log_debug("- image %pD\n", image);
110 } else {
111 log_debug("- efi_dp_from_name() failed, err=%lx\n", ret);
112 efi_clear_bootdev();
113 }
114}
115
116/**
117 * efi_run_image() - run loaded UEFI image
118 *
119 * @source_buffer: memory address of the UEFI image
120 * @source_size: size of the UEFI image
121 * Return: status code
122 */
123efi_status_t efi_run_image(void *source_buffer, efi_uintn_t source_size)
124{
125 efi_handle_t mem_handle = NULL, handle;
126 struct efi_device_path *file_path = NULL;
127 struct efi_device_path *msg_path;
Heinrich Schuchardtb41a33f2024-03-16 10:36:42 +0100128 efi_status_t ret;
AKASHI Takahiro9b08b9a2024-01-17 13:39:41 +0900129 u16 *load_options;
130
131 if (!bootefi_device_path || !bootefi_image_path) {
132 log_debug("Not loaded from disk\n");
133 /*
134 * Special case for efi payload not loaded from disk,
135 * such as 'bootefi hello' or for example payload
136 * loaded directly into memory via JTAG, etc:
137 */
138 file_path = efi_dp_from_mem(EFI_RESERVED_MEMORY_TYPE,
139 (uintptr_t)source_buffer,
Moritz Fischer4786f952024-10-30 21:17:48 +0000140 (uintptr_t)source_buffer +
AKASHI Takahiro9b08b9a2024-01-17 13:39:41 +0900141 source_size);
142 /*
143 * Make sure that device for device_path exist
144 * in load_image(). Otherwise, shell and grub will fail.
145 */
146 ret = efi_install_multiple_protocol_interfaces(&mem_handle,
147 &efi_guid_device_path,
148 file_path, NULL);
149 if (ret != EFI_SUCCESS)
150 goto out;
151 msg_path = file_path;
152 } else {
153 file_path = efi_dp_concat(bootefi_device_path,
Heinrich Schuchardtf8de0092024-05-24 14:54:26 +0200154 bootefi_image_path, 0);
AKASHI Takahiro9b08b9a2024-01-17 13:39:41 +0900155 msg_path = bootefi_image_path;
156 log_debug("Loaded from disk\n");
157 }
158
159 log_info("Booting %pD\n", msg_path);
160
161 ret = EFI_CALL(efi_load_image(false, efi_root, file_path, source_buffer,
162 source_size, &handle));
163 if (ret != EFI_SUCCESS) {
164 log_err("Loading image failed\n");
165 goto out;
166 }
167
168 /* Transfer environment variable as load options */
169 ret = efi_env_set_load_options(handle, "bootargs", &load_options);
170 if (ret != EFI_SUCCESS)
171 goto out;
172
173 ret = do_bootefi_exec(handle, load_options);
174
175out:
Heinrich Schuchardtb41a33f2024-03-16 10:36:42 +0100176 if (mem_handle) {
177 efi_status_t r;
178
179 r = efi_uninstall_multiple_protocol_interfaces(
180 mem_handle, &efi_guid_device_path, file_path, NULL);
181 if (r != EFI_SUCCESS)
182 log_err("Uninstalling protocol interfaces failed\n");
183 }
AKASHI Takahiro9b08b9a2024-01-17 13:39:41 +0900184 efi_free_pool(file_path);
Heinrich Schuchardtb41a33f2024-03-16 10:36:42 +0100185
186 return ret;
AKASHI Takahiro9b08b9a2024-01-17 13:39:41 +0900187}
188
189/**
190 * efi_binary_run() - run loaded UEFI image
191 *
192 * @image: memory address of the UEFI image
193 * @size: size of the UEFI image
194 * @fdt: device-tree
195 *
196 * Execute an EFI binary image loaded at @image.
197 * @size may be zero if the binary is loaded with U-Boot load command.
198 *
199 * Return: status code
200 */
201efi_status_t efi_binary_run(void *image, size_t size, void *fdt)
202{
203 efi_status_t ret;
204
205 /* Initialize EFI drivers */
206 ret = efi_init_obj_list();
207 if (ret != EFI_SUCCESS) {
208 log_err("Error: Cannot initialize UEFI sub-system, r = %lu\n",
209 ret & ~EFI_ERROR_MASK);
210 return -1;
211 }
212
213 ret = efi_install_fdt(fdt);
214 if (ret != EFI_SUCCESS)
215 return ret;
216
217 return efi_run_image(image, size);
218}