xref: /freebsd/sys/kern/subr_boot.c (revision 2cb49090114108d594195b9b32c762391340484c)
152379d36SWarner Losh /*-
24d846d26SWarner Losh  * SPDX-License-Identifier: BSD-2-Clause
352379d36SWarner Losh  *
452379d36SWarner Losh  * Copyright (c) 1998 Michael Smith <msmith@freebsd.org>
552379d36SWarner Losh  * All Rights Reserved.
652379d36SWarner Losh  * Copyright (c) 1998 Robert Nordier
752379d36SWarner Losh  * All Rights Reserved.
852379d36SWarner Losh  * Copyright (c) 2009, Oleksandr Tymoshenko <gonzo@FreeBSD.org>
952379d36SWarner Losh  * All rights reserved.
1052379d36SWarner Losh  * Copyright (c) 2014 Roger Pau Monné <roger.pau@citrix.com>
1152379d36SWarner Losh  * All Rights Reserved.
1252379d36SWarner Losh  * Copyright (c) 2018 Kyle Evans <kevans@FreeBSD.org>
1352467047SWarner Losh  * Copyright (c) 2018 Netflix, Inc.
1452379d36SWarner Losh  *
1552379d36SWarner Losh  * Redistribution and use in source and binary forms, with or without
1652379d36SWarner Losh  * modification, are permitted provided that the following conditions
1752379d36SWarner Losh  * are met:
1852379d36SWarner Losh  * 1. Redistributions of source code must retain the above copyright
1952379d36SWarner Losh  *    notice, this list of conditions and the following disclaimer
2052379d36SWarner Losh  *    in this position and unchanged.
2152379d36SWarner Losh  * 2. Redistributions in binary form must reproduce the above copyright
2252379d36SWarner Losh  *    notice, this list of conditions and the following disclaimer in the
2352379d36SWarner Losh  *    documentation and/or other materials provided with the distribution.
2452379d36SWarner Losh  *
2552379d36SWarner Losh  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
2652379d36SWarner Losh  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
2752379d36SWarner Losh  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
2852379d36SWarner Losh  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
2952379d36SWarner Losh  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
3052379d36SWarner Losh  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
3152379d36SWarner Losh  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
3252379d36SWarner Losh  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
3352379d36SWarner Losh  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
3452379d36SWarner Losh  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
3552379d36SWarner Losh  * SUCH DAMAGE.
3652379d36SWarner Losh  */
3752379d36SWarner Losh 
3852379d36SWarner Losh #include <sys/cdefs.h>
3952379d36SWarner Losh /* Note: This is compiled in both the kernel and boot loader contexts */
4052379d36SWarner Losh 
4152379d36SWarner Losh #include <sys/param.h>
4252379d36SWarner Losh #ifdef _KERNEL
4352379d36SWarner Losh #include <sys/systm.h>
4452379d36SWarner Losh #else
4552379d36SWarner Losh #include <stand.h>
4652379d36SWarner Losh #endif
4752379d36SWarner Losh #include <sys/reboot.h>
4852379d36SWarner Losh #include <sys/boot.h>
499d6ae1e3SColin Percival #include <sys/tslog.h>
5052379d36SWarner Losh 
5152379d36SWarner Losh #ifdef _KERNEL
5252379d36SWarner Losh #define SETENV(k, v)	kern_setenv(k, v)
5352379d36SWarner Losh #define GETENV(k)	kern_getenv(k)
5452379d36SWarner Losh #define FREE(v)		freeenv(v)
5552379d36SWarner Losh #else	/* Boot loader */
5652379d36SWarner Losh #define SETENV(k, v)	setenv(k, v, 1)
5752379d36SWarner Losh #define GETENV(k)	getenv(k)
5852379d36SWarner Losh #define	FREE(v)
5952379d36SWarner Losh #endif
6052379d36SWarner Losh 
6152379d36SWarner Losh static struct
6252379d36SWarner Losh {
6352379d36SWarner Losh 	const char	*ev;
6452379d36SWarner Losh 	int		mask;
6552379d36SWarner Losh } howto_names[] = {
6652379d36SWarner Losh 	{ "boot_askname",	RB_ASKNAME},
6752379d36SWarner Losh 	{ "boot_cdrom",		RB_CDROM},
6852379d36SWarner Losh 	{ "boot_ddb",		RB_KDB},
6952379d36SWarner Losh 	{ "boot_dfltroot",	RB_DFLTROOT},
7052379d36SWarner Losh 	{ "boot_gdb",		RB_GDB},
7152379d36SWarner Losh 	{ "boot_multicons",	RB_MULTIPLE},
7252379d36SWarner Losh 	{ "boot_mute",		RB_MUTE},
73*2cb49090SJustin Hibbits 	{ "boot_mutemsgs",	RB_MUTEMSGS},
7452379d36SWarner Losh 	{ "boot_pause",		RB_PAUSE},
7552379d36SWarner Losh 	{ "boot_serial",	RB_SERIAL},
7652379d36SWarner Losh 	{ "boot_single",	RB_SINGLE},
7752379d36SWarner Losh 	{ "boot_verbose",	RB_VERBOSE},
7852379d36SWarner Losh 	{ NULL,	0}
7952379d36SWarner Losh };
8052379d36SWarner Losh 
8152379d36SWarner Losh /*
8252379d36SWarner Losh  * In the boot environment, we often parse a command line and have to throw away
8352379d36SWarner Losh  * its contents. As we do so, we set environment variables that correspond to
8452379d36SWarner Losh  * the flags we encounter. Later, to get a howto mask, we grovel through these
8552379d36SWarner Losh  * to reconstruct it. This also allows users in their loader.conf to set them
8652379d36SWarner Losh  * and have the kernel see them.
8752379d36SWarner Losh  */
8852379d36SWarner Losh 
8952379d36SWarner Losh /**
9052379d36SWarner Losh  * @brief convert the env vars in howto_names into a howto mask
9152379d36SWarner Losh  */
9252379d36SWarner Losh int
9352379d36SWarner Losh boot_env_to_howto(void)
9452379d36SWarner Losh {
9552379d36SWarner Losh 	int i, howto;
9652379d36SWarner Losh 	char *val;
9752379d36SWarner Losh 
989d6ae1e3SColin Percival 	TSENTER();
9952379d36SWarner Losh 	for (howto = 0, i = 0; howto_names[i].ev != NULL; i++) {
10052379d36SWarner Losh 		val = GETENV(howto_names[i].ev);
10152379d36SWarner Losh 		if (val != NULL && strcasecmp(val, "no") != 0)
10252379d36SWarner Losh 			howto |= howto_names[i].mask;
10352379d36SWarner Losh 		FREE(val);
10452379d36SWarner Losh 	}
1059d6ae1e3SColin Percival 	TSEXIT();
10652379d36SWarner Losh 	return (howto);
10752379d36SWarner Losh }
10852379d36SWarner Losh 
10952379d36SWarner Losh /**
11052379d36SWarner Losh  * @brief Set env vars from howto_names based on howto passed in
11152379d36SWarner Losh  */
11252379d36SWarner Losh void
11352379d36SWarner Losh boot_howto_to_env(int howto)
11452379d36SWarner Losh {
11552379d36SWarner Losh 	int i;
11652379d36SWarner Losh 
11752379d36SWarner Losh 	for (i = 0; howto_names[i].ev != NULL; i++)
11852379d36SWarner Losh 		if (howto & howto_names[i].mask)
11952379d36SWarner Losh 			SETENV(howto_names[i].ev, "YES");
12052379d36SWarner Losh }
12152379d36SWarner Losh 
12252379d36SWarner Losh /**
12352379d36SWarner Losh  * @brief Helper routine to parse a single arg and return its mask
12452379d36SWarner Losh  *
12552379d36SWarner Losh  * Parse all the - options to create a mask (or a serial speed in the
12652379d36SWarner Losh  * case of -S). If the arg doesn't start with '-' assume it's an env
12752379d36SWarner Losh  * variable and set that instead.
12852379d36SWarner Losh  */
12952379d36SWarner Losh int
130ed56dcfcSWarner Losh boot_parse_arg(const char *v)
13152379d36SWarner Losh {
13252379d36SWarner Losh 	char *n;
13352379d36SWarner Losh 	int howto;
13452379d36SWarner Losh 
13552379d36SWarner Losh #if 0
13652379d36SWarner Losh /* Need to see if this is better or worse than the meat of the #else */
137*2cb49090SJustin Hibbits static const char howto_switches[] = "aCdrgDmMphsv";
13852379d36SWarner Losh static int howto_masks[] = {
13952379d36SWarner Losh 	RB_ASKNAME, RB_CDROM, RB_KDB, RB_DFLTROOT, RB_GDB, RB_MULTIPLE,
140*2cb49090SJustin Hibbits 	RB_MUTE, RB_MUTEMSGS, RB_PAUSE, RB_SERIAL, RB_SINGLE, RB_VERBOSE
14152379d36SWarner Losh };
14252379d36SWarner Losh 
14352379d36SWarner Losh 	opts = strchr(kargs, '-');
14452379d36SWarner Losh 	while (opts != NULL) {
14552379d36SWarner Losh 		while (*(++opts) != '\0') {
14652379d36SWarner Losh 			sw = strchr(howto_switches, *opts);
14752379d36SWarner Losh 			if (sw == NULL)
14852379d36SWarner Losh 				break;
14952379d36SWarner Losh 			howto |= howto_masks[sw - howto_switches];
15052379d36SWarner Losh 		}
15152379d36SWarner Losh 		opts = strchr(opts, '-');
15252379d36SWarner Losh 	}
15352379d36SWarner Losh #else
15452379d36SWarner Losh 	howto = 0;
15552379d36SWarner Losh 	if (*v == '-') {
15652379d36SWarner Losh 		while (*v != '\0') {
15752379d36SWarner Losh 			v++;
15852379d36SWarner Losh 			switch (*v) {
15952379d36SWarner Losh 			case 'a': howto |= RB_ASKNAME; break;
16052379d36SWarner Losh 			case 'C': howto |= RB_CDROM; break;
16152379d36SWarner Losh 			case 'd': howto |= RB_KDB; break;
16252379d36SWarner Losh 			case 'D': howto |= RB_MULTIPLE; break;
16352379d36SWarner Losh 			case 'm': howto |= RB_MUTE; break;
164*2cb49090SJustin Hibbits 			case 'M': howto |= RB_MUTEMSGS; break;
16552379d36SWarner Losh 			case 'g': howto |= RB_GDB; break;
16652379d36SWarner Losh 			case 'h': howto |= RB_SERIAL; break;
16752379d36SWarner Losh 			case 'p': howto |= RB_PAUSE; break;
16852379d36SWarner Losh 			case 'P': howto |= RB_PROBE; break;
16952379d36SWarner Losh 			case 'r': howto |= RB_DFLTROOT; break;
17052379d36SWarner Losh 			case 's': howto |= RB_SINGLE; break;
17152379d36SWarner Losh 			case 'S': SETENV("comconsole_speed", v + 1); v += strlen(v); break;
17252379d36SWarner Losh 			case 'v': howto |= RB_VERBOSE; break;
17352379d36SWarner Losh 			}
17452379d36SWarner Losh 		}
17552379d36SWarner Losh 	} else {
176ed56dcfcSWarner Losh 		char buf[128];
177ed56dcfcSWarner Losh 		char *vv = buf;
178ed56dcfcSWarner Losh 
179ed56dcfcSWarner Losh 		strlcpy(buf, v, sizeof(buf));
180ed56dcfcSWarner Losh 		n = strsep(&vv, "=");
181ed56dcfcSWarner Losh 		if (vv == NULL)
18252379d36SWarner Losh 			SETENV(n, "1");
18352379d36SWarner Losh 		else
184ed56dcfcSWarner Losh 			SETENV(n, vv);
18552379d36SWarner Losh 	}
18652379d36SWarner Losh #endif
18752379d36SWarner Losh 	return (howto);
18852379d36SWarner Losh }
18952379d36SWarner Losh 
19052379d36SWarner Losh /**
19152379d36SWarner Losh  * @brief breakup the command line into args, and pass to boot_parse_arg
19252379d36SWarner Losh  */
19352379d36SWarner Losh int
19452379d36SWarner Losh boot_parse_cmdline_delim(char *cmdline, const char *delim)
19552379d36SWarner Losh {
19652379d36SWarner Losh 	char *v;
19752379d36SWarner Losh 	int howto;
19852379d36SWarner Losh 
1999d6ae1e3SColin Percival 	TSENTER();
20052379d36SWarner Losh 	howto = 0;
20152379d36SWarner Losh 	while ((v = strsep(&cmdline, delim)) != NULL) {
20252379d36SWarner Losh 		if (*v == '\0')
20352379d36SWarner Losh 			continue;
20452379d36SWarner Losh 		howto |= boot_parse_arg(v);
20552379d36SWarner Losh 	}
2069d6ae1e3SColin Percival 	TSEXIT();
20752379d36SWarner Losh 	return (howto);
20852379d36SWarner Losh }
20952379d36SWarner Losh 
21052379d36SWarner Losh /**
2112834e42cSWarner Losh  * @brief Simplified interface for common 'space or tab separated' args
21252379d36SWarner Losh  */
21352379d36SWarner Losh int
21452379d36SWarner Losh boot_parse_cmdline(char *cmdline)
21552379d36SWarner Losh {
21652379d36SWarner Losh 
2172834e42cSWarner Losh 	return (boot_parse_cmdline_delim(cmdline, " \t\n"));
21852379d36SWarner Losh }
21952379d36SWarner Losh 
22052379d36SWarner Losh /**
22152379d36SWarner Losh  * @brief Pass a vector of strings to boot_parse_arg
22252379d36SWarner Losh  */
22352379d36SWarner Losh int
22452379d36SWarner Losh boot_parse_args(int argc, char *argv[])
22552379d36SWarner Losh {
22652379d36SWarner Losh         int i, howto;
22752379d36SWarner Losh 
22852379d36SWarner Losh 	howto = 0;
22952379d36SWarner Losh         for (i = 1; i < argc; i++)
23052379d36SWarner Losh                 howto |= boot_parse_arg(argv[i]);
23152379d36SWarner Losh 	return (howto);
23252379d36SWarner Losh }
233