xref: /freebsd/stand/efi/boot1/boot1.c (revision 97cb52fa9aefd90fad38790fded50905aeeb9b9e)
1 /*-
2  * Copyright (c) 1998 Robert Nordier
3  * All rights reserved.
4  * Copyright (c) 2001 Robert Drehmel
5  * All rights reserved.
6  * Copyright (c) 2014 Nathan Whitehorn
7  * All rights reserved.
8  * Copyright (c) 2015 Eric McCorkle
9  * All rights reserved.
10  *
11  * Redistribution and use in source and binary forms are freely
12  * permitted provided that the above copyright notice and this
13  * paragraph and the following disclaimer are duplicated in all
14  * such forms.
15  *
16  * This software is provided "AS IS" and without any express or
17  * implied warranties, including, without limitation, the implied
18  * warranties of merchantability and fitness for a particular
19  * purpose.
20  */
21 
22 #include <sys/cdefs.h>
23 __FBSDID("$FreeBSD$");
24 
25 #include <sys/param.h>
26 #include <machine/elf.h>
27 #include <machine/stdarg.h>
28 #include <stand.h>
29 
30 #include <efi.h>
31 #include <eficonsctl.h>
32 typedef CHAR16 efi_char;
33 #include <efichar.h>
34 
35 #include "boot_module.h"
36 #include "paths.h"
37 
38 static void efi_panic(EFI_STATUS s, const char *fmt, ...) __dead2 __printflike(2, 3);
39 
40 static const boot_module_t *boot_modules[] =
41 {
42 #ifdef EFI_ZFS_BOOT
43 	&zfs_module,
44 #endif
45 #ifdef EFI_UFS_BOOT
46 	&ufs_module
47 #endif
48 };
49 
50 #define	NUM_BOOT_MODULES	nitems(boot_modules)
51 /* The initial number of handles used to query EFI for partitions. */
52 #define NUM_HANDLES_INIT	24
53 
54 static EFI_GUID BlockIoProtocolGUID = BLOCK_IO_PROTOCOL;
55 static EFI_GUID DevicePathGUID = DEVICE_PATH_PROTOCOL;
56 static EFI_GUID LoadedImageGUID = LOADED_IMAGE_PROTOCOL;
57 static EFI_GUID ConsoleControlGUID = EFI_CONSOLE_CONTROL_PROTOCOL_GUID;
58 static EFI_GUID FreeBSDBootVarGUID = FREEBSD_BOOT_VAR_GUID;
59 
60 /*
61  * Provide Malloc / Free backed by EFIs AllocatePool / FreePool which ensures
62  * memory is correctly aligned avoiding EFI_INVALID_PARAMETER returns from
63  * EFI methods.
64  */
65 void *
66 Malloc(size_t len, const char *file __unused, int line __unused)
67 {
68 	void *out;
69 
70 	if (BS->AllocatePool(EfiLoaderData, len, &out) == EFI_SUCCESS)
71 		return (out);
72 
73 	return (NULL);
74 }
75 
76 void
77 Free(void *buf, const char *file __unused, int line __unused)
78 {
79 	if (buf != NULL)
80 		(void)BS->FreePool(buf);
81 }
82 
83 static EFI_STATUS
84 efi_setenv_freebsd_wcs(const char *varname, CHAR16 *valstr)
85 {
86 	CHAR16 *var = NULL;
87 	size_t len;
88 	EFI_STATUS rv;
89 
90 	utf8_to_ucs2(varname, &var, &len);
91 	if (var == NULL)
92 		return (EFI_OUT_OF_RESOURCES);
93 	rv = RS->SetVariable(var, &FreeBSDBootVarGUID,
94 	    EFI_VARIABLE_BOOTSERVICE_ACCESS | EFI_VARIABLE_RUNTIME_ACCESS,
95 	    (ucs2len(valstr) + 1) * sizeof(efi_char), valstr);
96 	free(var);
97 	return (rv);
98 }
99 
100 /*
101  * nodes_match returns TRUE if the imgpath isn't NULL and the nodes match,
102  * FALSE otherwise.
103  */
104 static BOOLEAN
105 nodes_match(EFI_DEVICE_PATH *imgpath, EFI_DEVICE_PATH *devpath)
106 {
107 	size_t len;
108 
109 	if (imgpath == NULL || imgpath->Type != devpath->Type ||
110 	    imgpath->SubType != devpath->SubType)
111 		return (FALSE);
112 
113 	len = DevicePathNodeLength(imgpath);
114 	if (len != DevicePathNodeLength(devpath))
115 		return (FALSE);
116 
117 	return (memcmp(imgpath, devpath, (size_t)len) == 0);
118 }
119 
120 /*
121  * device_paths_match returns TRUE if the imgpath isn't NULL and all nodes
122  * in imgpath and devpath match up to their respective occurrences of a
123  * media node, FALSE otherwise.
124  */
125 static BOOLEAN
126 device_paths_match(EFI_DEVICE_PATH *imgpath, EFI_DEVICE_PATH *devpath)
127 {
128 
129 	if (imgpath == NULL)
130 		return (FALSE);
131 
132 	while (!IsDevicePathEnd(imgpath) && !IsDevicePathEnd(devpath)) {
133 		if (IsDevicePathType(imgpath, MEDIA_DEVICE_PATH) &&
134 		    IsDevicePathType(devpath, MEDIA_DEVICE_PATH))
135 			return (TRUE);
136 
137 		if (!nodes_match(imgpath, devpath))
138 			return (FALSE);
139 
140 		imgpath = NextDevicePathNode(imgpath);
141 		devpath = NextDevicePathNode(devpath);
142 	}
143 
144 	return (FALSE);
145 }
146 
147 /*
148  * devpath_last returns the last non-path end node in devpath.
149  */
150 static EFI_DEVICE_PATH *
151 devpath_last(EFI_DEVICE_PATH *devpath)
152 {
153 
154 	while (!IsDevicePathEnd(NextDevicePathNode(devpath)))
155 		devpath = NextDevicePathNode(devpath);
156 
157 	return (devpath);
158 }
159 
160 /*
161  * load_loader attempts to load the loader image data.
162  *
163  * It tries each module and its respective devices, identified by mod->probe,
164  * in order until a successful load occurs at which point it returns EFI_SUCCESS
165  * and EFI_NOT_FOUND otherwise.
166  *
167  * Only devices which have preferred matching the preferred parameter are tried.
168  */
169 static EFI_STATUS
170 load_loader(const boot_module_t **modp, dev_info_t **devinfop, void **bufp,
171     size_t *bufsize, BOOLEAN preferred)
172 {
173 	UINTN i;
174 	dev_info_t *dev;
175 	const boot_module_t *mod;
176 
177 	for (i = 0; i < NUM_BOOT_MODULES; i++) {
178 		mod = boot_modules[i];
179 		for (dev = mod->devices(); dev != NULL; dev = dev->next) {
180 			if (dev->preferred != preferred)
181 				continue;
182 
183 			if (mod->load(PATH_LOADER_EFI, dev, bufp, bufsize) ==
184 			    EFI_SUCCESS) {
185 				*devinfop = dev;
186 				*modp = mod;
187 				return (EFI_SUCCESS);
188 			}
189 		}
190 	}
191 
192 	return (EFI_NOT_FOUND);
193 }
194 
195 /*
196  * try_boot only returns if it fails to load the loader. If it succeeds
197  * it simply boots, otherwise it returns the status of last EFI call.
198  */
199 static EFI_STATUS
200 try_boot(void)
201 {
202 	size_t bufsize, loadersize, cmdsize;
203 	void *buf, *loaderbuf;
204 	char *cmd;
205 	dev_info_t *dev;
206 	const boot_module_t *mod;
207 	EFI_HANDLE loaderhandle;
208 	EFI_LOADED_IMAGE *loaded_image;
209 	EFI_STATUS status;
210 
211 	status = load_loader(&mod, &dev, &loaderbuf, &loadersize, TRUE);
212 	if (status != EFI_SUCCESS) {
213 		status = load_loader(&mod, &dev, &loaderbuf, &loadersize,
214 		    FALSE);
215 		if (status != EFI_SUCCESS) {
216 			printf("Failed to load '%s'\n", PATH_LOADER_EFI);
217 			return (status);
218 		}
219 	}
220 
221 	/*
222 	 * Read in and parse the command line from /boot.config or /boot/config,
223 	 * if present. We'll pass it the next stage via a simple ASCII
224 	 * string. loader.efi has a hack for ASCII strings, so we'll use that to
225 	 * keep the size down here. We only try to read the alternate file if
226 	 * we get EFI_NOT_FOUND because all other errors mean that the boot_module
227 	 * had troubles with the filesystem. We could return early, but we'll let
228 	 * loading the actual kernel sort all that out. Since these files are
229 	 * optional, we don't report errors in trying to read them.
230 	 */
231 	cmd = NULL;
232 	cmdsize = 0;
233 	status = mod->load(PATH_DOTCONFIG, dev, &buf, &bufsize);
234 	if (status == EFI_NOT_FOUND)
235 		status = mod->load(PATH_CONFIG, dev, &buf, &bufsize);
236 	if (status == EFI_SUCCESS) {
237 		cmdsize = bufsize + 1;
238 		cmd = malloc(cmdsize);
239 		if (cmd == NULL)
240 			goto errout;
241 		memcpy(cmd, buf, bufsize);
242 		cmd[bufsize] = '\0';
243 		free(buf);
244 		buf = NULL;
245 	}
246 
247 	if ((status = BS->LoadImage(TRUE, IH, devpath_last(dev->devpath),
248 	    loaderbuf, loadersize, &loaderhandle)) != EFI_SUCCESS) {
249 		printf("Failed to load image provided by %s, size: %zu, (%lu)\n",
250 		     mod->name, loadersize, EFI_ERROR_CODE(status));
251 		goto errout;
252 	}
253 
254 	if ((status = BS->HandleProtocol(loaderhandle, &LoadedImageGUID,
255 	    (VOID**)&loaded_image)) != EFI_SUCCESS) {
256 		printf("Failed to query LoadedImage provided by %s (%lu)\n",
257 		    mod->name, EFI_ERROR_CODE(status));
258 		goto errout;
259 	}
260 
261 	if (cmd != NULL)
262 		printf("    command args: %s\n", cmd);
263 
264 	loaded_image->DeviceHandle = dev->devhandle;
265 	loaded_image->LoadOptionsSize = cmdsize;
266 	loaded_image->LoadOptions = cmd;
267 
268 	DPRINTF("Starting '%s' in 5 seconds...", PATH_LOADER_EFI);
269 	DSTALL(1000000);
270 	DPRINTF(".");
271 	DSTALL(1000000);
272 	DPRINTF(".");
273 	DSTALL(1000000);
274 	DPRINTF(".");
275 	DSTALL(1000000);
276 	DPRINTF(".");
277 	DSTALL(1000000);
278 	DPRINTF(".\n");
279 
280 	if ((status = BS->StartImage(loaderhandle, NULL, NULL)) !=
281 	    EFI_SUCCESS) {
282 		printf("Failed to start image provided by %s (%lu)\n",
283 		    mod->name, EFI_ERROR_CODE(status));
284 		loaded_image->LoadOptionsSize = 0;
285 		loaded_image->LoadOptions = NULL;
286 	}
287 
288 errout:
289 	if (cmd != NULL)
290 		free(cmd);
291 	if (buf != NULL)
292 		free(buf);
293 	if (loaderbuf != NULL)
294 		free(loaderbuf);
295 
296 	return (status);
297 }
298 
299 /*
300  * probe_handle determines if the passed handle represents a logical partition
301  * if it does it uses each module in order to probe it and if successful it
302  * returns EFI_SUCCESS.
303  */
304 static EFI_STATUS
305 probe_handle(EFI_HANDLE h, EFI_DEVICE_PATH *imgpath, BOOLEAN *preferred)
306 {
307 	dev_info_t *devinfo;
308 	EFI_BLOCK_IO *blkio;
309 	EFI_DEVICE_PATH *devpath;
310 	EFI_STATUS status;
311 	UINTN i;
312 
313 	/* Figure out if we're dealing with an actual partition. */
314 	status = BS->HandleProtocol(h, &DevicePathGUID, (void **)&devpath);
315 	if (status == EFI_UNSUPPORTED)
316 		return (status);
317 
318 	if (status != EFI_SUCCESS) {
319 		DPRINTF("\nFailed to query DevicePath (%lu)\n",
320 		    EFI_ERROR_CODE(status));
321 		return (status);
322 	}
323 #ifdef EFI_DEBUG
324 	{
325 		CHAR16 *text = efi_devpath_name(devpath);
326 		DPRINTF("probing: %S\n", text);
327 		efi_free_devpath_name(text);
328 	}
329 #endif
330 	status = BS->HandleProtocol(h, &BlockIoProtocolGUID, (void **)&blkio);
331 	if (status == EFI_UNSUPPORTED)
332 		return (status);
333 
334 	if (status != EFI_SUCCESS) {
335 		DPRINTF("\nFailed to query BlockIoProtocol (%lu)\n",
336 		    EFI_ERROR_CODE(status));
337 		return (status);
338 	}
339 
340 	if (!blkio->Media->LogicalPartition)
341 		return (EFI_UNSUPPORTED);
342 
343 	*preferred = device_paths_match(imgpath, devpath);
344 
345 	/* Run through each module, see if it can load this partition */
346 	for (i = 0; i < NUM_BOOT_MODULES; i++) {
347 		devinfo = malloc(sizeof(*devinfo));
348 		if (devinfo == NULL) {
349 			DPRINTF("\nFailed to allocate devinfo\n");
350 			continue;
351 		}
352 		devinfo->dev = blkio;
353 		devinfo->devpath = devpath;
354 		devinfo->devhandle = h;
355 		devinfo->devdata = NULL;
356 		devinfo->preferred = *preferred;
357 		devinfo->next = NULL;
358 
359 		status = boot_modules[i]->probe(devinfo);
360 		if (status == EFI_SUCCESS)
361 			return (EFI_SUCCESS);
362 		free(devinfo);
363 	}
364 
365 	return (EFI_UNSUPPORTED);
366 }
367 
368 /*
369  * probe_handle_status calls probe_handle and outputs the returned status
370  * of the call.
371  */
372 static void
373 probe_handle_status(EFI_HANDLE h, EFI_DEVICE_PATH *imgpath)
374 {
375 	EFI_STATUS status;
376 	BOOLEAN preferred;
377 
378 	preferred = FALSE;
379 	status = probe_handle(h, imgpath, &preferred);
380 
381 	DPRINTF("probe: ");
382 	switch (status) {
383 	case EFI_UNSUPPORTED:
384 		printf(".");
385 		DPRINTF(" not supported\n");
386 		break;
387 	case EFI_SUCCESS:
388 		if (preferred) {
389 			printf("%c", '*');
390 			DPRINTF(" supported (preferred)\n");
391 		} else {
392 			printf("%c", '+');
393 			DPRINTF(" supported\n");
394 		}
395 		break;
396 	default:
397 		printf("x");
398 		DPRINTF(" error (%lu)\n", EFI_ERROR_CODE(status));
399 		break;
400 	}
401 	DSTALL(500000);
402 }
403 
404 EFI_STATUS
405 efi_main(EFI_HANDLE Ximage, EFI_SYSTEM_TABLE *Xsystab)
406 {
407 	EFI_HANDLE *handles;
408 	EFI_LOADED_IMAGE *img;
409 	EFI_DEVICE_PATH *imgpath;
410 	EFI_STATUS status;
411 	EFI_CONSOLE_CONTROL_PROTOCOL *ConsoleControl = NULL;
412 	SIMPLE_TEXT_OUTPUT_INTERFACE *conout = NULL;
413 	UINTN i, max_dim, best_mode, cols, rows, hsize, nhandles;
414 	CHAR16 *text;
415 
416 	/* Basic initialization*/
417 	ST = Xsystab;
418 	IH = Ximage;
419 	BS = ST->BootServices;
420 	RS = ST->RuntimeServices;
421 
422 	/* Set up the console, so printf works. */
423 	status = BS->LocateProtocol(&ConsoleControlGUID, NULL,
424 	    (VOID **)&ConsoleControl);
425 	if (status == EFI_SUCCESS)
426 		(void)ConsoleControl->SetMode(ConsoleControl,
427 		    EfiConsoleControlScreenText);
428 	/*
429 	 * Reset the console and find the best text mode.
430 	 */
431 	conout = ST->ConOut;
432 	conout->Reset(conout, TRUE);
433 	max_dim = best_mode = 0;
434 	for (i = 0; ; i++) {
435 		status = conout->QueryMode(conout, i, &cols, &rows);
436 		if (EFI_ERROR(status))
437 			break;
438 		if (cols * rows > max_dim) {
439 			max_dim = cols * rows;
440 			best_mode = i;
441 		}
442 	}
443 	if (max_dim > 0)
444 		conout->SetMode(conout, best_mode);
445 	conout->EnableCursor(conout, TRUE);
446 	conout->ClearScreen(conout);
447 
448 	printf("\n>> FreeBSD EFI boot block\n");
449 	printf("   Loader path: %s\n\n", PATH_LOADER_EFI);
450 	printf("   Initializing modules:");
451 	for (i = 0; i < NUM_BOOT_MODULES; i++) {
452 		printf(" %s", boot_modules[i]->name);
453 		if (boot_modules[i]->init != NULL)
454 			boot_modules[i]->init();
455 	}
456 	putchar('\n');
457 
458 	/* Determine the devpath of our image so we can prefer it. */
459 	status = BS->HandleProtocol(IH, &LoadedImageGUID, (VOID**)&img);
460 	imgpath = NULL;
461 	if (status == EFI_SUCCESS) {
462 		text = efi_devpath_name(img->FilePath);
463 		if (text != NULL) {
464 			printf("   Load Path: %S\n", text);
465 			efi_setenv_freebsd_wcs("Boot1Path", text);
466 			efi_free_devpath_name(text);
467 		}
468 
469 		status = BS->HandleProtocol(img->DeviceHandle, &DevicePathGUID,
470 		    (void **)&imgpath);
471 		if (status != EFI_SUCCESS) {
472 			DPRINTF("Failed to get image DevicePath (%lu)\n",
473 			    EFI_ERROR_CODE(status));
474 		} else {
475 			text = efi_devpath_name(imgpath);
476 			if (text != NULL) {
477 				printf("   Load Device: %S\n", text);
478 				efi_setenv_freebsd_wcs("Boot1Dev", text);
479 				efi_free_devpath_name(text);
480 			}
481 		}
482 	}
483 
484 	/* Get all the device handles */
485 	hsize = (UINTN)NUM_HANDLES_INIT * sizeof(EFI_HANDLE);
486 	handles = malloc(hsize);
487 	if (handles == NULL) {
488 		printf("Failed to allocate %d handles\n", NUM_HANDLES_INIT);
489 	}
490 
491 	status = BS->LocateHandle(ByProtocol, &BlockIoProtocolGUID, NULL,
492 	    &hsize, handles);
493 	switch (status) {
494 	case EFI_SUCCESS:
495 		break;
496 	case EFI_BUFFER_TOO_SMALL:
497 		free(handles);
498 		handles = malloc(hsize);
499 		if (handles == NULL)
500 			efi_panic(EFI_OUT_OF_RESOURCES, "Failed to allocate %d handles\n",
501 			    NUM_HANDLES_INIT);
502 		status = BS->LocateHandle(ByProtocol, &BlockIoProtocolGUID,
503 		    NULL, &hsize, handles);
504 		if (status != EFI_SUCCESS)
505 			efi_panic(status, "Failed to get device handles\n");
506 		break;
507 	default:
508 		efi_panic(status, "Failed to get device handles\n");
509 		break;
510 	}
511 
512 	/* Scan all partitions, probing with all modules. */
513 	nhandles = hsize / sizeof(*handles);
514 	printf("   Probing %zu block devices...", nhandles);
515 	DPRINTF("\n");
516 
517 	for (i = 0; i < nhandles; i++)
518 		probe_handle_status(handles[i], imgpath);
519 	printf(" done\n");
520 
521 	/* Status summary. */
522 	for (i = 0; i < NUM_BOOT_MODULES; i++) {
523 		printf("    ");
524 		boot_modules[i]->status();
525 	}
526 
527 	try_boot();
528 
529 	/* If we get here, we're out of luck... */
530 	efi_panic(EFI_LOAD_ERROR, "No bootable partitions found!");
531 }
532 
533 /*
534  * add_device adds a device to the passed devinfo list.
535  */
536 void
537 add_device(dev_info_t **devinfop, dev_info_t *devinfo)
538 {
539 	dev_info_t *dev;
540 
541 	if (*devinfop == NULL) {
542 		*devinfop = devinfo;
543 		return;
544 	}
545 
546 	for (dev = *devinfop; dev->next != NULL; dev = dev->next)
547 		;
548 
549 	dev->next = devinfo;
550 }
551 
552 /*
553  * OK. We totally give up. Exit back to EFI with a sensible status so
554  * it can try the next option on the list.
555  */
556 static void
557 efi_panic(EFI_STATUS s, const char *fmt, ...)
558 {
559 	va_list ap;
560 
561 	printf("panic: ");
562 	va_start(ap, fmt);
563 	vprintf(fmt, ap);
564 	va_end(ap);
565 	printf("\n");
566 
567 	BS->Exit(IH, s, 0, NULL);
568 }
569 
570 void
571 putchar(int c)
572 {
573 	CHAR16 buf[2];
574 
575 	if (c == '\n') {
576 		buf[0] = '\r';
577 		buf[1] = 0;
578 		ST->ConOut->OutputString(ST->ConOut, buf);
579 	}
580 	buf[0] = c;
581 	buf[1] = 0;
582 	ST->ConOut->OutputString(ST->ConOut, buf);
583 }
584