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