Blame view

cmd/bootefi.c 15.7 KB
f739fcd83   Tom Rini   SPDX: Convert a f...
1
  // SPDX-License-Identifier: GPL-2.0+
b9939336d   Alexander Graf   efi_loader: Add "...
2
3
4
5
  /*
   *  EFI application loader
   *
   *  Copyright (c) 2016 Alexander Graf
b9939336d   Alexander Graf   efi_loader: Add "...
6
   */
d78e40d65   Heinrich Schuchardt   efi_selftest: all...
7
  #include <charset.h>
b9939336d   Alexander Graf   efi_loader: Add "...
8
9
  #include <common.h>
  #include <command.h>
9d922450a   Simon Glass   dm: Use dm.h head...
10
  #include <dm.h>
b9939336d   Alexander Graf   efi_loader: Add "...
11
  #include <efi_loader.h>
d78e40d65   Heinrich Schuchardt   efi_selftest: all...
12
  #include <efi_selftest.h>
b9939336d   Alexander Graf   efi_loader: Add "...
13
  #include <errno.h>
b08c8c487   Masahiro Yamada   libfdt: move head...
14
15
  #include <linux/libfdt.h>
  #include <linux/libfdt_env.h>
354264b31   Alexander Graf   efi_loader: Use m...
16
  #include <mapmem.h>
ad0c1a3d2   Alexander Graf   efi_loader: Put f...
17
  #include <memalign.h>
0d9d501f3   Alexander Graf   efi_loader: Use s...
18
  #include <asm/global_data.h>
e275458c2   Simon Glass   efi: Fix missing ...
19
  #include <asm-generic/sections.h>
c3b11dea7   Heinrich Schuchardt   efi_loader: allow...
20
  #include <asm-generic/unaligned.h>
e275458c2   Simon Glass   efi: Fix missing ...
21
  #include <linux/linkage.h>
0d9d501f3   Alexander Graf   efi_loader: Use s...
22

dc500c369   Mark Kettenis   efi_loader: ARM: ...
23
24
25
26
  #ifdef CONFIG_ARMV7_NONSEC
  #include <asm/armv7.h>
  #include <asm/secure.h>
  #endif
0d9d501f3   Alexander Graf   efi_loader: Use s...
27
  DECLARE_GLOBAL_DATA_PTR;
b9939336d   Alexander Graf   efi_loader: Add "...
28

fc225e608   Heinrich Schuchardt   efi_loader: check...
29
30
31
  #define OBJ_LIST_NOT_INITIALIZED 1
  
  static efi_status_t efi_obj_list_initialized = OBJ_LIST_NOT_INITIALIZED;
7cbc12415   Heinrich Schuchardt   efi_loader: inita...
32

95c5553ea   Rob Clark   efi_loader: refac...
33
34
  static struct efi_device_path *bootefi_image_path;
  static struct efi_device_path *bootefi_device_path;
b9939336d   Alexander Graf   efi_loader: Add "...
35

7cbc12415   Heinrich Schuchardt   efi_loader: inita...
36
  /* Initialize and populate EFI object list */
fc225e608   Heinrich Schuchardt   efi_loader: check...
37
  efi_status_t efi_init_obj_list(void)
7cbc12415   Heinrich Schuchardt   efi_loader: inita...
38
  {
fc225e608   Heinrich Schuchardt   efi_loader: check...
39
  	efi_status_t ret = EFI_SUCCESS;
098a6cdd1   Heinrich Schuchardt   efi_loader: simpl...
40
  	/* Initialize once only */
fc225e608   Heinrich Schuchardt   efi_loader: check...
41
42
  	if (efi_obj_list_initialized != OBJ_LIST_NOT_INITIALIZED)
  		return efi_obj_list_initialized;
7cbc12415   Heinrich Schuchardt   efi_loader: inita...
43

640adadf8   Heinrich Schuchardt   efi_loader: calcu...
44
45
46
47
  	/* Initialize system table */
  	ret = efi_initialize_system_table();
  	if (ret != EFI_SUCCESS)
  		goto out;
05ef48a24   Heinrich Schuchardt   efi_driver: EFI b...
48
  	/* Initialize EFI driver uclass */
fc225e608   Heinrich Schuchardt   efi_loader: check...
49
50
51
  	ret = efi_driver_init();
  	if (ret != EFI_SUCCESS)
  		goto out;
05ef48a24   Heinrich Schuchardt   efi_driver: EFI b...
52

fc225e608   Heinrich Schuchardt   efi_loader: check...
53
54
55
  	ret = efi_console_register();
  	if (ret != EFI_SUCCESS)
  		goto out;
7cbc12415   Heinrich Schuchardt   efi_loader: inita...
56
  #ifdef CONFIG_PARTITIONS
fc225e608   Heinrich Schuchardt   efi_loader: check...
57
58
59
  	ret = efi_disk_register();
  	if (ret != EFI_SUCCESS)
  		goto out;
7cbc12415   Heinrich Schuchardt   efi_loader: inita...
60
61
  #endif
  #if defined(CONFIG_LCD) || defined(CONFIG_DM_VIDEO)
fc225e608   Heinrich Schuchardt   efi_loader: check...
62
63
64
  	ret = efi_gop_register();
  	if (ret != EFI_SUCCESS)
  		goto out;
7cbc12415   Heinrich Schuchardt   efi_loader: inita...
65
  #endif
092f2f35b   Joe Hershberger   Revert "Kconfig: ...
66
  #ifdef CONFIG_NET
fc225e608   Heinrich Schuchardt   efi_loader: check...
67
68
69
  	ret = efi_net_register();
  	if (ret != EFI_SUCCESS)
  		goto out;
7cbc12415   Heinrich Schuchardt   efi_loader: inita...
70
  #endif
86df34d42   Bin Meng   efi_loader: Insta...
71
72
73
74
75
  #ifdef CONFIG_GENERATE_ACPI_TABLE
  	ret = efi_acpi_register();
  	if (ret != EFI_SUCCESS)
  		goto out;
  #endif
7cbc12415   Heinrich Schuchardt   efi_loader: inita...
76
  #ifdef CONFIG_GENERATE_SMBIOS_TABLE
fc225e608   Heinrich Schuchardt   efi_loader: check...
77
78
79
  	ret = efi_smbios_register();
  	if (ret != EFI_SUCCESS)
  		goto out;
7cbc12415   Heinrich Schuchardt   efi_loader: inita...
80
  #endif
fc225e608   Heinrich Schuchardt   efi_loader: check...
81
82
83
  	ret = efi_watchdog_register();
  	if (ret != EFI_SUCCESS)
  		goto out;
7cbc12415   Heinrich Schuchardt   efi_loader: inita...
84
85
  
  	/* Initialize EFI runtime services */
fc225e608   Heinrich Schuchardt   efi_loader: check...
86
87
88
  	ret = efi_reset_system_init();
  	if (ret != EFI_SUCCESS)
  		goto out;
fc225e608   Heinrich Schuchardt   efi_loader: check...
89
90
91
92
  
  out:
  	efi_obj_list_initialized = ret;
  	return ret;
7cbc12415   Heinrich Schuchardt   efi_loader: inita...
93
  }
d78e40d65   Heinrich Schuchardt   efi_selftest: all...
94
  /*
c3b11dea7   Heinrich Schuchardt   efi_loader: allow...
95
96
97
98
99
100
101
102
103
   * Allow unaligned memory access.
   *
   * This routine is overridden by architectures providing this feature.
   */
  void __weak allow_unaligned(void)
  {
  }
  
  /*
d78e40d65   Heinrich Schuchardt   efi_selftest: all...
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
   * Set the load options of an image from an environment variable.
   *
   * @loaded_image_info:	the image
   * @env_var:		name of the environment variable
   */
  static void set_load_options(struct efi_loaded_image *loaded_image_info,
  			     const char *env_var)
  {
  	size_t size;
  	const char *env = env_get(env_var);
  
  	loaded_image_info->load_options = NULL;
  	loaded_image_info->load_options_size = 0;
  	if (!env)
  		return;
  	size = strlen(env) + 1;
  	loaded_image_info->load_options = calloc(size, sizeof(u16));
  	if (!loaded_image_info->load_options) {
  		printf("ERROR: Out of memory
  ");
  		return;
  	}
  	utf8_to_utf16(loaded_image_info->load_options, (u8 *)env, size);
  	loaded_image_info->load_options_size = size * 2;
  }
0d9d501f3   Alexander Graf   efi_loader: Use s...
129
130
131
  static void *copy_fdt(void *fdt)
  {
  	u64 fdt_size = fdt_totalsize(fdt);
ad0c1a3d2   Alexander Graf   efi_loader: Put f...
132
133
  	unsigned long fdt_ram_start = -1L, fdt_pages;
  	u64 new_fdt_addr;
0d9d501f3   Alexander Graf   efi_loader: Use s...
134
  	void *new_fdt;
ad0c1a3d2   Alexander Graf   efi_loader: Put f...
135
  	int i;
0d9d501f3   Alexander Graf   efi_loader: Use s...
136

ad0c1a3d2   Alexander Graf   efi_loader: Put f...
137
138
139
          for (i = 0; i < CONFIG_NR_DRAM_BANKS; i++) {
                  u64 ram_start = gd->bd->bi_dram[i].start;
                  u64 ram_size = gd->bd->bi_dram[i].size;
0d9d501f3   Alexander Graf   efi_loader: Use s...
140

ad0c1a3d2   Alexander Graf   efi_loader: Put f...
141
142
143
144
145
146
  		if (!ram_size)
  			continue;
  
  		if (ram_start < fdt_ram_start)
  			fdt_ram_start = ram_start;
  	}
bc9a638a1   Simon Glass   efi: Tidy up devi...
147
148
149
150
151
152
  	/*
  	 * Give us at least 4KB of breathing room in case the device tree needs
  	 * to be expanded later. Round up to the nearest EFI page boundary.
  	 */
  	fdt_size += 4096;
  	fdt_size = ALIGN(fdt_size + EFI_PAGE_SIZE - 1, EFI_PAGE_SIZE);
ad0c1a3d2   Alexander Graf   efi_loader: Put f...
153
154
155
156
  	fdt_pages = fdt_size >> EFI_PAGE_SHIFT;
  
  	/* Safe fdt location is at 128MB */
  	new_fdt_addr = fdt_ram_start + (128 * 1024 * 1024) + fdt_size;
e09159c86   Heinrich Schuchardt   efi_loader: avoid...
157
158
  	if (efi_allocate_pages(EFI_ALLOCATE_MAX_ADDRESS,
  			       EFI_RUNTIME_SERVICES_DATA, fdt_pages,
ad0c1a3d2   Alexander Graf   efi_loader: Put f...
159
160
  			       &new_fdt_addr) != EFI_SUCCESS) {
  		/* If we can't put it there, put it somewhere */
a44bffcc9   xypron.glpk@gmx.de   efi_loader: use E...
161
  		new_fdt_addr = (ulong)memalign(EFI_PAGE_SIZE, fdt_size);
e09159c86   Heinrich Schuchardt   efi_loader: avoid...
162
163
  		if (efi_allocate_pages(EFI_ALLOCATE_MAX_ADDRESS,
  				       EFI_RUNTIME_SERVICES_DATA, fdt_pages,
85a6e9b3c   Alexander Graf   efi_loader: Add c...
164
165
166
167
168
  				       &new_fdt_addr) != EFI_SUCCESS) {
  			printf("ERROR: Failed to reserve space for FDT
  ");
  			return NULL;
  		}
ad0c1a3d2   Alexander Graf   efi_loader: Put f...
169
  	}
85a6e9b3c   Alexander Graf   efi_loader: Add c...
170

ad0c1a3d2   Alexander Graf   efi_loader: Put f...
171
  	new_fdt = (void*)(ulong)new_fdt_addr;
0d9d501f3   Alexander Graf   efi_loader: Use s...
172
173
174
175
176
  	memcpy(new_fdt, fdt, fdt_totalsize(fdt));
  	fdt_set_totalsize(new_fdt, fdt_size);
  
  	return new_fdt;
  }
3eb0841be   Heinrich Schuchardt   efi_loader: consi...
177
  static efi_status_t efi_do_enter(
2074f7006   Heinrich Schuchardt   efi_loader: consi...
178
  			efi_handle_t image_handle, struct efi_system_table *st,
c6fa5df6a   Alexander Graf   efi_loader: Alway...
179
180
181
  			EFIAPI efi_status_t (*entry)(
  				efi_handle_t image_handle,
  				struct efi_system_table *st))
b06d8ac39   xypron.glpk@gmx.de   bootefi: allow re...
182
183
184
185
186
187
188
189
  {
  	efi_status_t ret = EFI_LOAD_ERROR;
  
  	if (entry)
  		ret = entry(image_handle, st);
  	st->boottime->exit(image_handle, ret, 0, NULL);
  	return ret;
  }
ec6617c39   Alison Wang   armv8: Support lo...
190
  #ifdef CONFIG_ARM64
c6fa5df6a   Alexander Graf   efi_loader: Alway...
191
  static efi_status_t efi_run_in_el2(EFIAPI efi_status_t (*entry)(
2074f7006   Heinrich Schuchardt   efi_loader: consi...
192
193
  			efi_handle_t image_handle, struct efi_system_table *st),
  			efi_handle_t image_handle, struct efi_system_table *st)
ec6617c39   Alison Wang   armv8: Support lo...
194
195
196
  {
  	/* Enable caches again */
  	dcache_enable();
b06d8ac39   xypron.glpk@gmx.de   bootefi: allow re...
197
  	return efi_do_enter(image_handle, st, entry);
ec6617c39   Alison Wang   armv8: Support lo...
198
199
  }
  #endif
dc500c369   Mark Kettenis   efi_loader: ARM: ...
200
  #ifdef CONFIG_ARMV7_NONSEC
f17f2001e   Mark Kettenis   efi_loader: ARM: ...
201
  static bool is_nonsec;
dc500c369   Mark Kettenis   efi_loader: ARM: ...
202
203
204
205
206
207
  static efi_status_t efi_run_in_hyp(EFIAPI efi_status_t (*entry)(
  			efi_handle_t image_handle, struct efi_system_table *st),
  			efi_handle_t image_handle, struct efi_system_table *st)
  {
  	/* Enable caches again */
  	dcache_enable();
f17f2001e   Mark Kettenis   efi_loader: ARM: ...
208
  	is_nonsec = true;
dc500c369   Mark Kettenis   efi_loader: ARM: ...
209
210
211
  	return efi_do_enter(image_handle, st, entry);
  }
  #endif
416e07e2c   Simon Glass   efi: Drop error r...
212
213
214
215
216
217
218
219
220
221
  /*
   * efi_carve_out_dt_rsv() - Carve out DT reserved memory ranges
   *
   * The mem_rsv entries of the FDT are added to the memory map. Any failures are
   * ignored because this is not critical and we would rather continue to try to
   * boot.
   *
   * @fdt: Pointer to device tree
   */
  static void efi_carve_out_dt_rsv(void *fdt)
806d2fa8e   Alexander Graf   efi_loader: Respe...
222
223
224
225
226
227
228
229
230
231
232
233
  {
  	int nr_rsv, i;
  	uint64_t addr, size, pages;
  
  	nr_rsv = fdt_num_mem_rsv(fdt);
  
  	/* Look for an existing entry and add it to the efi mem map. */
  	for (i = 0; i < nr_rsv; i++) {
  		if (fdt_get_mem_rsv(fdt, i, &addr, &size) != 0)
  			continue;
  
  		pages = ALIGN(size, EFI_PAGE_SIZE) >> EFI_PAGE_SHIFT;
416e07e2c   Simon Glass   efi: Drop error r...
234
235
236
237
  		if (!efi_add_memory_map(addr, pages, EFI_RESERVED_MEMORY_TYPE,
  					false))
  			printf("FDT memrsv map %d: Failed to add to map
  ", i);
806d2fa8e   Alexander Graf   efi_loader: Respe...
238
  	}
806d2fa8e   Alexander Graf   efi_loader: Respe...
239
  }
bc4f9133e   Heinrich Schuchardt   efi_loader: suppo...
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
  static efi_status_t efi_install_fdt(void *fdt)
  {
  	bootm_headers_t img = { 0 };
  	ulong fdt_pages, fdt_size, fdt_start, fdt_end;
  	efi_status_t ret;
  
  	if (fdt_check_header(fdt)) {
  		printf("ERROR: invalid device tree
  ");
  		return EFI_INVALID_PARAMETER;
  	}
  
  	/* Prepare fdt for payload */
  	fdt = copy_fdt(fdt);
  	if (!fdt)
  		return EFI_OUT_OF_RESOURCES;
  
  	if (image_setup_libfdt(&img, fdt, 0, NULL)) {
  		printf("ERROR: failed to process device tree
  ");
  		return EFI_LOAD_ERROR;
  	}
416e07e2c   Simon Glass   efi: Drop error r...
262
  	efi_carve_out_dt_rsv(fdt);
806d2fa8e   Alexander Graf   efi_loader: Respe...
263

bc4f9133e   Heinrich Schuchardt   efi_loader: suppo...
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
  	/* Link to it in the efi tables */
  	ret = efi_install_configuration_table(&efi_guid_fdt, fdt);
  	if (ret != EFI_SUCCESS)
  		return EFI_OUT_OF_RESOURCES;
  
  	/* And reserve the space in the memory map */
  	fdt_start = ((ulong)fdt) & ~EFI_PAGE_MASK;
  	fdt_end = ((ulong)fdt) + fdt_totalsize(fdt);
  	fdt_size = (fdt_end - fdt_start) + EFI_PAGE_MASK;
  	fdt_pages = fdt_size >> EFI_PAGE_SHIFT;
  	/* Give a bootloader the chance to modify the device tree */
  	fdt_pages += 2;
  	ret = efi_add_memory_map(fdt_start, fdt_pages,
  				 EFI_BOOT_SERVICES_DATA, true);
  	return ret;
  }
b9939336d   Alexander Graf   efi_loader: Add "...
280
281
282
283
  /*
   * Load an EFI payload into a newly allocated piece of memory, register all
   * EFI objects it would want to access and jump to it.
   */
bc4f9133e   Heinrich Schuchardt   efi_loader: suppo...
284
  static efi_status_t do_bootefi_exec(void *efi,
3eb0841be   Heinrich Schuchardt   efi_loader: consi...
285
286
  				    struct efi_device_path *device_path,
  				    struct efi_device_path *image_path)
b9939336d   Alexander Graf   efi_loader: Add "...
287
  {
95c5553ea   Rob Clark   efi_loader: refac...
288
289
  	struct efi_loaded_image loaded_image_info = {};
  	struct efi_object loaded_image_info_obj = {};
58bc69d20   Alexander Graf   efi_loader: Alloc...
290
  	struct efi_object mem_obj = {};
bf19273e8   Rob Clark   efi_loader: Add m...
291
  	struct efi_device_path *memdp = NULL;
45204b102   Heinrich Schuchardt   efi_loader: do_bo...
292
  	efi_status_t ret;
95c5553ea   Rob Clark   efi_loader: refac...
293

c6fa5df6a   Alexander Graf   efi_loader: Alway...
294
295
  	EFIAPI efi_status_t (*entry)(efi_handle_t image_handle,
  				     struct efi_system_table *st);
b9939336d   Alexander Graf   efi_loader: Add "...
296

bf19273e8   Rob Clark   efi_loader: Add m...
297
298
299
300
301
302
303
304
305
306
307
  	/*
  	 * Special case for efi payload not loaded from disk, such as
  	 * 'bootefi hello' or for example payload loaded directly into
  	 * memory via jtag/etc:
  	 */
  	if (!device_path && !image_path) {
  		printf("WARNING: using memory device/image path, this may confuse some payloads!
  ");
  		/* actual addresses filled in after efi_load_pe() */
  		memdp = efi_dp_from_mem(0, 0, 0);
  		device_path = image_path = memdp;
58bc69d20   Alexander Graf   efi_loader: Alloc...
308
309
310
311
312
313
  		efi_add_handle(&mem_obj);
  
  		ret = efi_add_protocol(mem_obj.handle, &efi_guid_device_path,
  				       device_path);
  		if (ret != EFI_SUCCESS)
  			goto exit;
bf19273e8   Rob Clark   efi_loader: Add m...
314
315
316
  	} else {
  		assert(device_path && image_path);
  	}
95c5553ea   Rob Clark   efi_loader: refac...
317
318
  	efi_setup_loaded_image(&loaded_image_info, &loaded_image_info_obj,
  			       device_path, image_path);
b9939336d   Alexander Graf   efi_loader: Add "...
319
320
321
322
323
  	/*
  	 * gd lives in a fixed register which may get clobbered while we execute
  	 * the payload. So save it here and restore it on every callback entry
  	 */
  	efi_save_gd();
b57f48a87   Heinrich Schuchardt   efi_loader: use b...
324
325
  	/* Transfer environment variable bootargs as load options */
  	set_load_options(&loaded_image_info, "bootargs");
b9939336d   Alexander Graf   efi_loader: Add "...
326
327
  	/* Load the EFI payload */
  	entry = efi_load_pe(efi, &loaded_image_info);
95c5553ea   Rob Clark   efi_loader: refac...
328
  	if (!entry) {
45204b102   Heinrich Schuchardt   efi_loader: do_bo...
329
  		ret = EFI_LOAD_ERROR;
95c5553ea   Rob Clark   efi_loader: refac...
330
331
  		goto exit;
  	}
80a4800ee   Alexander Graf   efi_loader: Allow...
332

bf19273e8   Rob Clark   efi_loader: Add m...
333
334
335
336
337
338
339
  	if (memdp) {
  		struct efi_device_path_memory *mdp = (void *)memdp;
  		mdp->memory_type = loaded_image_info.image_code_type;
  		mdp->start_address = (uintptr_t)loaded_image_info.image_base;
  		mdp->end_address = mdp->start_address +
  				loaded_image_info.image_size;
  	}
ad644e7c1   Rob Clark   efi_loader: efi v...
340
341
342
  	/* we don't support much: */
  	env_set("efi_8be4df61-93ca-11d2-aa0d-00e098032b8c_OsIndicationsSupported",
  		"{ro,boot}(blob)0000000000000000");
b9939336d   Alexander Graf   efi_loader: Add "...
343
  	/* Call our payload! */
edcef3ba1   Alexander Graf   efi_loader: Move ...
344
345
  	debug("%s:%d Jumping to 0x%lx
  ", __func__, __LINE__, (long)entry);
a86aeaf22   Alexander Graf   efi_loader: Add e...
346
347
  
  	if (setjmp(&loaded_image_info.exit_jmp)) {
95c5553ea   Rob Clark   efi_loader: refac...
348
  		ret = loaded_image_info.exit_status;
95c5553ea   Rob Clark   efi_loader: refac...
349
  		goto exit;
a86aeaf22   Alexander Graf   efi_loader: Add e...
350
  	}
69bd459d3   Alexander Graf   efi_loader: AArch...
351
352
353
354
355
  #ifdef CONFIG_ARM64
  	/* On AArch64 we need to make sure we call our payload in < EL3 */
  	if (current_el() == 3) {
  		smp_kick_all_cpus();
  		dcache_disable();	/* flush cache before switch to EL2 */
ec6617c39   Alison Wang   armv8: Support lo...
356
357
  
  		/* Move into EL2 and keep running there */
ea54ad592   Heinrich Schuchardt   efi_loader: pass ...
358
359
  		armv8_switch_to_el2((ulong)entry,
  				    (ulong)&loaded_image_info_obj.handle,
7c5e1feb1   Alison Wang   armv8: aarch64: F...
360
  				    (ulong)&systab, 0, (ulong)efi_run_in_el2,
ec6617c39   Alison Wang   armv8: Support lo...
361
362
363
364
  				    ES_TO_AARCH64);
  
  		/* Should never reach here, efi exits with longjmp */
  		while (1) { }
69bd459d3   Alexander Graf   efi_loader: AArch...
365
366
  	}
  #endif
dc500c369   Mark Kettenis   efi_loader: ARM: ...
367
  #ifdef CONFIG_ARMV7_NONSEC
f17f2001e   Mark Kettenis   efi_loader: ARM: ...
368
  	if (armv7_boot_nonsec() && !is_nonsec) {
dc500c369   Mark Kettenis   efi_loader: ARM: ...
369
370
371
372
373
374
375
376
377
378
379
380
381
  		dcache_disable();	/* flush cache before switch to HYP */
  
  		armv7_init_nonsec();
  		secure_ram_addr(_do_nonsec_entry)(
  					efi_run_in_hyp,
  					(uintptr_t)entry,
  					(uintptr_t)loaded_image_info_obj.handle,
  					(uintptr_t)&systab);
  
  		/* Should never reach here, efi exits with longjmp */
  		while (1) { }
  	}
  #endif
ea54ad592   Heinrich Schuchardt   efi_loader: pass ...
382
  	ret = efi_do_enter(loaded_image_info_obj.handle, &systab, entry);
95c5553ea   Rob Clark   efi_loader: refac...
383
384
385
386
  
  exit:
  	/* image has returned, loaded-image obj goes *poof*: */
  	list_del(&loaded_image_info_obj.link);
58bc69d20   Alexander Graf   efi_loader: Alloc...
387
388
  	if (mem_obj.handle)
  		list_del(&mem_obj.link);
95c5553ea   Rob Clark   efi_loader: refac...
389
390
  
  	return ret;
b9939336d   Alexander Graf   efi_loader: Add "...
391
  }
bc4f9133e   Heinrich Schuchardt   efi_loader: suppo...
392
  static int do_bootefi_bootmgr_exec(void)
9975fe96b   Rob Clark   efi_loader: add b...
393
394
395
396
  {
  	struct efi_device_path *device_path, *file_path;
  	void *addr;
  	efi_status_t r;
9975fe96b   Rob Clark   efi_loader: add b...
397
398
399
400
401
402
403
404
405
406
407
408
  	/*
  	 * gd lives in a fixed register which may get clobbered while we execute
  	 * the payload. So save it here and restore it on every callback entry
  	 */
  	efi_save_gd();
  
  	addr = efi_bootmgr_load(&device_path, &file_path);
  	if (!addr)
  		return 1;
  
  	printf("## Starting EFI application at %p ...
  ", addr);
bc4f9133e   Heinrich Schuchardt   efi_loader: suppo...
409
  	r = do_bootefi_exec(addr, device_path, file_path);
9975fe96b   Rob Clark   efi_loader: add b...
410
411
412
413
414
415
416
417
418
  	printf("## Application terminated, r = %lu
  ",
  	       r & ~EFI_ERROR_MASK);
  
  	if (r != EFI_SUCCESS)
  		return 1;
  
  	return 0;
  }
b9939336d   Alexander Graf   efi_loader: Add "...
419
420
421
  /* Interpreter command to boot an arbitrary EFI image from memory */
  static int do_bootefi(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  {
bc4f9133e   Heinrich Schuchardt   efi_loader: suppo...
422
423
  	unsigned long addr;
  	char *saddr;
3eb0841be   Heinrich Schuchardt   efi_loader: consi...
424
  	efi_status_t r;
354264b31   Alexander Graf   efi_loader: Use m...
425
426
  	unsigned long fdt_addr;
  	void *fdt;
b9939336d   Alexander Graf   efi_loader: Add "...
427

c3b11dea7   Heinrich Schuchardt   efi_loader: allow...
428
429
  	/* Allow unaligned memory access */
  	allow_unaligned();
fc225e608   Heinrich Schuchardt   efi_loader: check...
430
431
432
433
434
435
436
437
  	/* Initialize EFI drivers */
  	r = efi_init_obj_list();
  	if (r != EFI_SUCCESS) {
  		printf("Error: Cannot set up EFI drivers, r = %lu
  ",
  		       r & ~EFI_ERROR_MASK);
  		return CMD_RET_FAILURE;
  	}
b9939336d   Alexander Graf   efi_loader: Add "...
438
  	if (argc < 2)
3c1dcef62   Bin Meng   cmd: efi_loader: ...
439
  		return CMD_RET_USAGE;
bc4f9133e   Heinrich Schuchardt   efi_loader: suppo...
440
441
  
  	if (argc > 2) {
354264b31   Alexander Graf   efi_loader: Use m...
442
  		fdt_addr = simple_strtoul(argv[2], NULL, 16);
bc4f9133e   Heinrich Schuchardt   efi_loader: suppo...
443
444
445
  		if (!fdt_addr && *argv[2] != '0')
  			return CMD_RET_USAGE;
  		/* Install device tree */
354264b31   Alexander Graf   efi_loader: Use m...
446
447
  		fdt = map_sysmem(fdt_addr, 0);
  		r = efi_install_fdt(fdt);
bc4f9133e   Heinrich Schuchardt   efi_loader: suppo...
448
449
450
451
452
453
454
455
456
457
458
  		if (r != EFI_SUCCESS) {
  			printf("ERROR: failed to install device tree
  ");
  			return CMD_RET_FAILURE;
  		}
  	} else {
  		/* Remove device tree. EFI_NOT_FOUND can be ignored here */
  		efi_install_configuration_table(&efi_guid_fdt, NULL);
  		printf("WARNING: booting without device tree
  ");
  	}
c7ae3dfdc   Simon Glass   efi: Add support ...
459
460
  #ifdef CONFIG_CMD_BOOTEFI_HELLO
  	if (!strcmp(argv[1], "hello")) {
5e44489bc   Heinrich Schuchardt   efi_loader: renam...
461
  		ulong size = __efi_helloworld_end - __efi_helloworld_begin;
b9939336d   Alexander Graf   efi_loader: Add "...
462

51c533fdc   Heinrich Schuchardt   efi_loader: boote...
463
464
465
466
467
  		saddr = env_get("loadaddr");
  		if (saddr)
  			addr = simple_strtoul(saddr, NULL, 16);
  		else
  			addr = CONFIG_SYS_LOAD_ADDR;
354264b31   Alexander Graf   efi_loader: Use m...
468
  		memcpy(map_sysmem(addr, size), __efi_helloworld_begin, size);
c7ae3dfdc   Simon Glass   efi: Add support ...
469
470
  	} else
  #endif
623b3a579   Heinrich Schuchardt   efi_selftest: pro...
471
472
  #ifdef CONFIG_CMD_BOOTEFI_SELFTEST
  	if (!strcmp(argv[1], "selftest")) {
7aca68ca0   Heinrich Schuchardt   efi_loader: reena...
473
474
  		struct efi_loaded_image loaded_image_info = {};
  		struct efi_object loaded_image_info_obj = {};
f972dc140   Heinrich Schuchardt   efi_selftest: pro...
475
476
477
478
479
  		/* Construct a dummy device path. */
  		bootefi_device_path = efi_dp_from_mem(EFI_RESERVED_MEMORY_TYPE,
  						      (uintptr_t)&efi_selftest,
  						      (uintptr_t)&efi_selftest);
  		bootefi_image_path = efi_dp_from_file(NULL, 0, "\\selftest");
7aca68ca0   Heinrich Schuchardt   efi_loader: reena...
480
481
482
  		efi_setup_loaded_image(&loaded_image_info,
  				       &loaded_image_info_obj,
  				       bootefi_device_path, bootefi_image_path);
623b3a579   Heinrich Schuchardt   efi_selftest: pro...
483
484
485
486
487
488
  		/*
  		 * gd lives in a fixed register which may get clobbered while we
  		 * execute the payload. So save it here and restore it on every
  		 * callback entry
  		 */
  		efi_save_gd();
d78e40d65   Heinrich Schuchardt   efi_selftest: all...
489
490
491
  		/* Transfer environment variable efi_selftest as load options */
  		set_load_options(&loaded_image_info, "efi_selftest");
  		/* Execute the test */
ea54ad592   Heinrich Schuchardt   efi_loader: pass ...
492
  		r = efi_selftest(loaded_image_info_obj.handle, &systab);
c2b53902c   Heinrich Schuchardt   efi_selftest: cor...
493
  		efi_restore_gd();
d78e40d65   Heinrich Schuchardt   efi_selftest: all...
494
  		free(loaded_image_info.load_options);
c2b53902c   Heinrich Schuchardt   efi_selftest: cor...
495
496
  		list_del(&loaded_image_info_obj.link);
  		return r != EFI_SUCCESS;
623b3a579   Heinrich Schuchardt   efi_selftest: pro...
497
498
  	} else
  #endif
9975fe96b   Rob Clark   efi_loader: add b...
499
  	if (!strcmp(argv[1], "bootmgr")) {
bc4f9133e   Heinrich Schuchardt   efi_loader: suppo...
500
  		return do_bootefi_bootmgr_exec();
9975fe96b   Rob Clark   efi_loader: add b...
501
  	} else {
c7ae3dfdc   Simon Glass   efi: Add support ...
502
  		saddr = argv[1];
b9939336d   Alexander Graf   efi_loader: Add "...
503

c7ae3dfdc   Simon Glass   efi: Add support ...
504
  		addr = simple_strtoul(saddr, NULL, 16);
49db1cb8c   Heinrich Schuchardt   efi_loader: catch...
505
506
507
  		/* Check that a numeric value was passed */
  		if (!addr && *saddr != '0')
  			return CMD_RET_USAGE;
c7ae3dfdc   Simon Glass   efi: Add support ...
508

1c39809b9   Alexander Graf   efi_loader: Pass ...
509
  	}
5ee31baf8   Simon Glass   efi: Fix debug me...
510
511
  	printf("## Starting EFI application at %08lx ...
  ", addr);
354264b31   Alexander Graf   efi_loader: Use m...
512
  	r = do_bootefi_exec(map_sysmem(addr, 0), bootefi_device_path,
bc4f9133e   Heinrich Schuchardt   efi_loader: suppo...
513
  			    bootefi_image_path);
1da1bac47   xypron.glpk@gmx.de   efi_loader: provi...
514
515
516
  	printf("## Application terminated, r = %lu
  ",
  	       r & ~EFI_ERROR_MASK);
b9939336d   Alexander Graf   efi_loader: Add "...
517

1da1bac47   xypron.glpk@gmx.de   efi_loader: provi...
518
519
520
521
  	if (r != EFI_SUCCESS)
  		return 1;
  	else
  		return 0;
b9939336d   Alexander Graf   efi_loader: Add "...
522
523
524
525
  }
  
  #ifdef CONFIG_SYS_LONGHELP
  static char bootefi_help_text[] =
1c39809b9   Alexander Graf   efi_loader: Pass ...
526
527
528
529
530
531
  	"<image address> [fdt address]
  "
  	"  - boot EFI payload stored at address <image address>.
  "
  	"    If specified, the device tree located at <fdt address> gets
  "
c7ae3dfdc   Simon Glass   efi: Add support ...
532
533
534
  	"    exposed as EFI configuration table.
  "
  #ifdef CONFIG_CMD_BOOTEFI_HELLO
623b3a579   Heinrich Schuchardt   efi_selftest: pro...
535
536
537
538
539
540
  	"bootefi hello
  "
  	"  - boot a sample Hello World application stored within U-Boot
  "
  #endif
  #ifdef CONFIG_CMD_BOOTEFI_SELFTEST
bc4f9133e   Heinrich Schuchardt   efi_loader: suppo...
541
542
  	"bootefi selftest [fdt address]
  "
623b3a579   Heinrich Schuchardt   efi_selftest: pro...
543
544
  	"  - boot an EFI selftest application stored within U-Boot
  "
d78e40d65   Heinrich Schuchardt   efi_selftest: all...
545
546
547
548
  	"    Use environment variable efi_selftest to select a single test.
  "
  	"    Use 'setenv efi_selftest list' to enumerate all tests.
  "
c7ae3dfdc   Simon Glass   efi: Add support ...
549
  #endif
f623e07f0   Heinrich Schuchardt   efi_loader: fix t...
550
551
  	"bootefi bootmgr [fdt addr]
  "
9975fe96b   Rob Clark   efi_loader: add b...
552
553
554
555
556
557
558
559
  	"  - load and boot EFI payload based on BootOrder/BootXXXX variables.
  "
  	"
  "
  	"    If specified, the device tree located at <fdt address> gets
  "
  	"    exposed as EFI configuration table.
  ";
b9939336d   Alexander Graf   efi_loader: Add "...
560
561
562
  #endif
  
  U_BOOT_CMD(
1c39809b9   Alexander Graf   efi_loader: Pass ...
563
  	bootefi, 3, 0, do_bootefi,
92dfd9221   Sergey Kubushyn   cmd: bootefi: cos...
564
  	"Boots an EFI payload from memory",
b9939336d   Alexander Graf   efi_loader: Add "...
565
566
  	bootefi_help_text
  );
0f4060ebc   Alexander Graf   efi_loader: Pass ...
567

95c5553ea   Rob Clark   efi_loader: refac...
568
569
570
571
  void efi_set_bootdev(const char *dev, const char *devnr, const char *path)
  {
  	char filename[32] = { 0 }; /* dp->str is u16[32] long */
  	char *s;
f9d334bdf   Alexander Graf   efi_loader: disk:...
572

95c5553ea   Rob Clark   efi_loader: refac...
573
574
  	if (strcmp(dev, "Net")) {
  		struct blk_desc *desc;
2db1eba1c   Heinrich Schuchardt   efi_loader: corre...
575
  		disk_partition_t fs_partition;
95c5553ea   Rob Clark   efi_loader: refac...
576
  		int part;
8c3df0bf2   Alexander Graf   efi_loader: Add e...
577

2db1eba1c   Heinrich Schuchardt   efi_loader: corre...
578
579
580
  		part = blk_get_device_part_str(dev, devnr, &desc, &fs_partition,
  					       1);
  		if (part < 0)
8300be612   Stefan Roese   efi_loader: Exit ...
581
  			return;
0f4060ebc   Alexander Graf   efi_loader: Pass ...
582

95c5553ea   Rob Clark   efi_loader: refac...
583
584
  		bootefi_device_path = efi_dp_from_part(desc, part);
  	} else {
092f2f35b   Joe Hershberger   Revert "Kconfig: ...
585
  #ifdef CONFIG_NET
95c5553ea   Rob Clark   efi_loader: refac...
586
587
588
  		bootefi_device_path = efi_dp_from_eth();
  #endif
  	}
c07ad7c03   Alexander Graf   efi_loader: Pass ...
589

9975fe96b   Rob Clark   efi_loader: add b...
590
591
  	if (!path)
  		return;
492716662   Alexander Graf   efi_loader: Make ...
592
593
  	if (strcmp(dev, "Net")) {
  		/* Add leading / to fs paths, because they're absolute */
95c5553ea   Rob Clark   efi_loader: refac...
594
  		snprintf(filename, sizeof(filename), "/%s", path);
492716662   Alexander Graf   efi_loader: Make ...
595
  	} else {
95c5553ea   Rob Clark   efi_loader: refac...
596
  		snprintf(filename, sizeof(filename), "%s", path);
492716662   Alexander Graf   efi_loader: Make ...
597
  	}
3e433e960   Rob Clark   efi_loader: EFI f...
598
  	/* DOS style file path: */
95c5553ea   Rob Clark   efi_loader: refac...
599
  	s = filename;
3e433e960   Rob Clark   efi_loader: EFI f...
600
601
  	while ((s = strchr(s, '/')))
  		*s++ = '\\';
95c5553ea   Rob Clark   efi_loader: refac...
602
  	bootefi_image_path = efi_dp_from_file(NULL, 0, filename);
0f4060ebc   Alexander Graf   efi_loader: Pass ...
603
  }