xref: /freebsd/stand/i386/loader/main.c (revision ca987d4641cdcd7f27e153db17c5bf064934faf5)
1*ca987d46SWarner Losh /*-
2*ca987d46SWarner Losh  * Copyright (c) 1998 Michael Smith <msmith@freebsd.org>
3*ca987d46SWarner Losh  * All rights reserved.
4*ca987d46SWarner Losh  *
5*ca987d46SWarner Losh  * Redistribution and use in source and binary forms, with or without
6*ca987d46SWarner Losh  * modification, are permitted provided that the following conditions
7*ca987d46SWarner Losh  * are met:
8*ca987d46SWarner Losh  * 1. Redistributions of source code must retain the above copyright
9*ca987d46SWarner Losh  *    notice, this list of conditions and the following disclaimer.
10*ca987d46SWarner Losh  * 2. Redistributions in binary form must reproduce the above copyright
11*ca987d46SWarner Losh  *    notice, this list of conditions and the following disclaimer in the
12*ca987d46SWarner Losh  *    documentation and/or other materials provided with the distribution.
13*ca987d46SWarner Losh  *
14*ca987d46SWarner Losh  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15*ca987d46SWarner Losh  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16*ca987d46SWarner Losh  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17*ca987d46SWarner Losh  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18*ca987d46SWarner Losh  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19*ca987d46SWarner Losh  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20*ca987d46SWarner Losh  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21*ca987d46SWarner Losh  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22*ca987d46SWarner Losh  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23*ca987d46SWarner Losh  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24*ca987d46SWarner Losh  * SUCH DAMAGE.
25*ca987d46SWarner Losh  */
26*ca987d46SWarner Losh 
27*ca987d46SWarner Losh #include <sys/cdefs.h>
28*ca987d46SWarner Losh __FBSDID("$FreeBSD$");
29*ca987d46SWarner Losh 
30*ca987d46SWarner Losh /*
31*ca987d46SWarner Losh  * MD bootstrap main() and assorted miscellaneous
32*ca987d46SWarner Losh  * commands.
33*ca987d46SWarner Losh  */
34*ca987d46SWarner Losh 
35*ca987d46SWarner Losh #include <stand.h>
36*ca987d46SWarner Losh #include <stddef.h>
37*ca987d46SWarner Losh #include <string.h>
38*ca987d46SWarner Losh #include <machine/bootinfo.h>
39*ca987d46SWarner Losh #include <machine/cpufunc.h>
40*ca987d46SWarner Losh #include <machine/psl.h>
41*ca987d46SWarner Losh #include <sys/disk.h>
42*ca987d46SWarner Losh #include <sys/reboot.h>
43*ca987d46SWarner Losh #include <common/drv.h>
44*ca987d46SWarner Losh 
45*ca987d46SWarner Losh #include "bootstrap.h"
46*ca987d46SWarner Losh #include "common/bootargs.h"
47*ca987d46SWarner Losh #include "libi386/libi386.h"
48*ca987d46SWarner Losh #include "libi386/smbios.h"
49*ca987d46SWarner Losh #include "btxv86.h"
50*ca987d46SWarner Losh 
51*ca987d46SWarner Losh #ifdef LOADER_ZFS_SUPPORT
52*ca987d46SWarner Losh #include "../zfs/libzfs.h"
53*ca987d46SWarner Losh #endif
54*ca987d46SWarner Losh 
55*ca987d46SWarner Losh CTASSERT(sizeof(struct bootargs) == BOOTARGS_SIZE);
56*ca987d46SWarner Losh CTASSERT(offsetof(struct bootargs, bootinfo) == BA_BOOTINFO);
57*ca987d46SWarner Losh CTASSERT(offsetof(struct bootargs, bootflags) == BA_BOOTFLAGS);
58*ca987d46SWarner Losh CTASSERT(offsetof(struct bootinfo, bi_size) == BI_SIZE);
59*ca987d46SWarner Losh 
60*ca987d46SWarner Losh /* Arguments passed in from the boot1/boot2 loader */
61*ca987d46SWarner Losh static struct bootargs *kargs;
62*ca987d46SWarner Losh 
63*ca987d46SWarner Losh static u_int32_t	initial_howto;
64*ca987d46SWarner Losh static u_int32_t	initial_bootdev;
65*ca987d46SWarner Losh static struct bootinfo	*initial_bootinfo;
66*ca987d46SWarner Losh 
67*ca987d46SWarner Losh struct arch_switch	archsw;		/* MI/MD interface boundary */
68*ca987d46SWarner Losh 
69*ca987d46SWarner Losh static void		extract_currdev(void);
70*ca987d46SWarner Losh static int		isa_inb(int port);
71*ca987d46SWarner Losh static void		isa_outb(int port, int value);
72*ca987d46SWarner Losh void			exit(int code);
73*ca987d46SWarner Losh #ifdef LOADER_GELI_SUPPORT
74*ca987d46SWarner Losh #include "geliboot.h"
75*ca987d46SWarner Losh struct geli_boot_args	*gargs;
76*ca987d46SWarner Losh #endif
77*ca987d46SWarner Losh #ifdef LOADER_ZFS_SUPPORT
78*ca987d46SWarner Losh struct zfs_boot_args	*zargs;
79*ca987d46SWarner Losh static void		i386_zfs_probe(void);
80*ca987d46SWarner Losh #endif
81*ca987d46SWarner Losh 
82*ca987d46SWarner Losh /* from vers.c */
83*ca987d46SWarner Losh extern	char bootprog_info[];
84*ca987d46SWarner Losh 
85*ca987d46SWarner Losh /* XXX debugging */
86*ca987d46SWarner Losh extern char end[];
87*ca987d46SWarner Losh 
88*ca987d46SWarner Losh static void *heap_top;
89*ca987d46SWarner Losh static void *heap_bottom;
90*ca987d46SWarner Losh 
91*ca987d46SWarner Losh int
92*ca987d46SWarner Losh main(void)
93*ca987d46SWarner Losh {
94*ca987d46SWarner Losh     int			i;
95*ca987d46SWarner Losh 
96*ca987d46SWarner Losh     /* Pick up arguments */
97*ca987d46SWarner Losh     kargs = (void *)__args;
98*ca987d46SWarner Losh     initial_howto = kargs->howto;
99*ca987d46SWarner Losh     initial_bootdev = kargs->bootdev;
100*ca987d46SWarner Losh     initial_bootinfo = kargs->bootinfo ? (struct bootinfo *)PTOV(kargs->bootinfo) : NULL;
101*ca987d46SWarner Losh 
102*ca987d46SWarner Losh     /* Initialize the v86 register set to a known-good state. */
103*ca987d46SWarner Losh     bzero(&v86, sizeof(v86));
104*ca987d46SWarner Losh     v86.efl = PSL_RESERVED_DEFAULT | PSL_I;
105*ca987d46SWarner Losh 
106*ca987d46SWarner Losh     /*
107*ca987d46SWarner Losh      * Initialise the heap as early as possible.  Once this is done, malloc() is usable.
108*ca987d46SWarner Losh      */
109*ca987d46SWarner Losh     bios_getmem();
110*ca987d46SWarner Losh 
111*ca987d46SWarner Losh #if defined(LOADER_BZIP2_SUPPORT) || defined(LOADER_FIREWIRE_SUPPORT) || \
112*ca987d46SWarner Losh     defined(LOADER_GPT_SUPPORT) || defined(LOADER_ZFS_SUPPORT)
113*ca987d46SWarner Losh     if (high_heap_size > 0) {
114*ca987d46SWarner Losh 	heap_top = PTOV(high_heap_base + high_heap_size);
115*ca987d46SWarner Losh 	heap_bottom = PTOV(high_heap_base);
116*ca987d46SWarner Losh 	if (high_heap_base < memtop_copyin)
117*ca987d46SWarner Losh 	    memtop_copyin = high_heap_base;
118*ca987d46SWarner Losh     } else
119*ca987d46SWarner Losh #endif
120*ca987d46SWarner Losh     {
121*ca987d46SWarner Losh 	heap_top = (void *)PTOV(bios_basemem);
122*ca987d46SWarner Losh 	heap_bottom = (void *)end;
123*ca987d46SWarner Losh     }
124*ca987d46SWarner Losh     setheap(heap_bottom, heap_top);
125*ca987d46SWarner Losh 
126*ca987d46SWarner Losh     /*
127*ca987d46SWarner Losh      * XXX Chicken-and-egg problem; we want to have console output early, but some
128*ca987d46SWarner Losh      * console attributes may depend on reading from eg. the boot device, which we
129*ca987d46SWarner Losh      * can't do yet.
130*ca987d46SWarner Losh      *
131*ca987d46SWarner Losh      * We can use printf() etc. once this is done.
132*ca987d46SWarner Losh      * If the previous boot stage has requested a serial console, prefer that.
133*ca987d46SWarner Losh      */
134*ca987d46SWarner Losh     bi_setboothowto(initial_howto);
135*ca987d46SWarner Losh     if (initial_howto & RB_MULTIPLE) {
136*ca987d46SWarner Losh 	if (initial_howto & RB_SERIAL)
137*ca987d46SWarner Losh 	    setenv("console", "comconsole vidconsole", 1);
138*ca987d46SWarner Losh 	else
139*ca987d46SWarner Losh 	    setenv("console", "vidconsole comconsole", 1);
140*ca987d46SWarner Losh     } else if (initial_howto & RB_SERIAL)
141*ca987d46SWarner Losh 	setenv("console", "comconsole", 1);
142*ca987d46SWarner Losh     else if (initial_howto & RB_MUTE)
143*ca987d46SWarner Losh 	setenv("console", "nullconsole", 1);
144*ca987d46SWarner Losh     cons_probe();
145*ca987d46SWarner Losh 
146*ca987d46SWarner Losh     /*
147*ca987d46SWarner Losh      * Initialise the block cache. Set the upper limit.
148*ca987d46SWarner Losh      */
149*ca987d46SWarner Losh     bcache_init(32768, 512);
150*ca987d46SWarner Losh 
151*ca987d46SWarner Losh     /*
152*ca987d46SWarner Losh      * Special handling for PXE and CD booting.
153*ca987d46SWarner Losh      */
154*ca987d46SWarner Losh     if (kargs->bootinfo == 0) {
155*ca987d46SWarner Losh 	/*
156*ca987d46SWarner Losh 	 * We only want the PXE disk to try to init itself in the below
157*ca987d46SWarner Losh 	 * walk through devsw if we actually booted off of PXE.
158*ca987d46SWarner Losh 	 */
159*ca987d46SWarner Losh 	if (kargs->bootflags & KARGS_FLAGS_PXE)
160*ca987d46SWarner Losh 	    pxe_enable(kargs->pxeinfo ? PTOV(kargs->pxeinfo) : NULL);
161*ca987d46SWarner Losh 	else if (kargs->bootflags & KARGS_FLAGS_CD)
162*ca987d46SWarner Losh 	    bc_add(initial_bootdev);
163*ca987d46SWarner Losh     }
164*ca987d46SWarner Losh 
165*ca987d46SWarner Losh     archsw.arch_autoload = i386_autoload;
166*ca987d46SWarner Losh     archsw.arch_getdev = i386_getdev;
167*ca987d46SWarner Losh     archsw.arch_copyin = i386_copyin;
168*ca987d46SWarner Losh     archsw.arch_copyout = i386_copyout;
169*ca987d46SWarner Losh     archsw.arch_readin = i386_readin;
170*ca987d46SWarner Losh     archsw.arch_isainb = isa_inb;
171*ca987d46SWarner Losh     archsw.arch_isaoutb = isa_outb;
172*ca987d46SWarner Losh #ifdef LOADER_ZFS_SUPPORT
173*ca987d46SWarner Losh     archsw.arch_zfs_probe = i386_zfs_probe;
174*ca987d46SWarner Losh 
175*ca987d46SWarner Losh #ifdef LOADER_GELI_SUPPORT
176*ca987d46SWarner Losh     if ((kargs->bootflags & KARGS_FLAGS_EXTARG) != 0) {
177*ca987d46SWarner Losh 	zargs = (struct zfs_boot_args *)(kargs + 1);
178*ca987d46SWarner Losh 	if (zargs != NULL && zargs->size >= offsetof(struct zfs_boot_args, gelipw)) {
179*ca987d46SWarner Losh 	    if (zargs->size >= offsetof(struct zfs_boot_args, keybuf_sentinel) &&
180*ca987d46SWarner Losh 	      zargs->keybuf_sentinel == KEYBUF_SENTINEL) {
181*ca987d46SWarner Losh 		geli_save_keybuf(zargs->keybuf);
182*ca987d46SWarner Losh 	    }
183*ca987d46SWarner Losh 	    if (zargs->gelipw[0] != '\0') {
184*ca987d46SWarner Losh 		setenv("kern.geom.eli.passphrase", zargs->gelipw, 1);
185*ca987d46SWarner Losh 		explicit_bzero(zargs->gelipw, sizeof(zargs->gelipw));
186*ca987d46SWarner Losh 	    }
187*ca987d46SWarner Losh 	}
188*ca987d46SWarner Losh     }
189*ca987d46SWarner Losh #endif /* LOADER_GELI_SUPPORT */
190*ca987d46SWarner Losh #else /* !LOADER_ZFS_SUPPORT */
191*ca987d46SWarner Losh #ifdef LOADER_GELI_SUPPORT
192*ca987d46SWarner Losh     if ((kargs->bootflags & KARGS_FLAGS_EXTARG) != 0) {
193*ca987d46SWarner Losh 	gargs = (struct geli_boot_args *)(kargs + 1);
194*ca987d46SWarner Losh 	if (gargs != NULL && gargs->size >= offsetof(struct geli_boot_args, gelipw)) {
195*ca987d46SWarner Losh 	    if (gargs->keybuf_sentinel == KEYBUF_SENTINEL) {
196*ca987d46SWarner Losh 		geli_save_keybuf(gargs->keybuf);
197*ca987d46SWarner Losh 	    }
198*ca987d46SWarner Losh 	    if (gargs->gelipw[0] != '\0') {
199*ca987d46SWarner Losh 		setenv("kern.geom.eli.passphrase", gargs->gelipw, 1);
200*ca987d46SWarner Losh 		explicit_bzero(gargs->gelipw, sizeof(gargs->gelipw));
201*ca987d46SWarner Losh 	    }
202*ca987d46SWarner Losh 	}
203*ca987d46SWarner Losh     }
204*ca987d46SWarner Losh #endif /* LOADER_GELI_SUPPORT */
205*ca987d46SWarner Losh #endif /* LOADER_ZFS_SUPPORT */
206*ca987d46SWarner Losh 
207*ca987d46SWarner Losh     /*
208*ca987d46SWarner Losh      * March through the device switch probing for things.
209*ca987d46SWarner Losh      */
210*ca987d46SWarner Losh     for (i = 0; devsw[i] != NULL; i++)
211*ca987d46SWarner Losh 	if (devsw[i]->dv_init != NULL)
212*ca987d46SWarner Losh 	    (devsw[i]->dv_init)();
213*ca987d46SWarner Losh     printf("BIOS %dkB/%dkB available memory\n", bios_basemem / 1024, bios_extmem / 1024);
214*ca987d46SWarner Losh     if (initial_bootinfo != NULL) {
215*ca987d46SWarner Losh 	initial_bootinfo->bi_basemem = bios_basemem / 1024;
216*ca987d46SWarner Losh 	initial_bootinfo->bi_extmem = bios_extmem / 1024;
217*ca987d46SWarner Losh     }
218*ca987d46SWarner Losh 
219*ca987d46SWarner Losh     /* detect ACPI for future reference */
220*ca987d46SWarner Losh     biosacpi_detect();
221*ca987d46SWarner Losh 
222*ca987d46SWarner Losh     /* detect SMBIOS for future reference */
223*ca987d46SWarner Losh     smbios_detect(NULL);
224*ca987d46SWarner Losh 
225*ca987d46SWarner Losh     /* detect PCI BIOS for future reference */
226*ca987d46SWarner Losh     biospci_detect();
227*ca987d46SWarner Losh 
228*ca987d46SWarner Losh     printf("\n%s", bootprog_info);
229*ca987d46SWarner Losh 
230*ca987d46SWarner Losh     extract_currdev();				/* set $currdev and $loaddev */
231*ca987d46SWarner Losh     setenv("LINES", "24", 1);			/* optional */
232*ca987d46SWarner Losh 
233*ca987d46SWarner Losh     bios_getsmap();
234*ca987d46SWarner Losh 
235*ca987d46SWarner Losh     interact(NULL);
236*ca987d46SWarner Losh 
237*ca987d46SWarner Losh     /* if we ever get here, it is an error */
238*ca987d46SWarner Losh     return (1);
239*ca987d46SWarner Losh }
240*ca987d46SWarner Losh 
241*ca987d46SWarner Losh /*
242*ca987d46SWarner Losh  * Set the 'current device' by (if possible) recovering the boot device as
243*ca987d46SWarner Losh  * supplied by the initial bootstrap.
244*ca987d46SWarner Losh  *
245*ca987d46SWarner Losh  * XXX should be extended for netbooting.
246*ca987d46SWarner Losh  */
247*ca987d46SWarner Losh static void
248*ca987d46SWarner Losh extract_currdev(void)
249*ca987d46SWarner Losh {
250*ca987d46SWarner Losh     struct i386_devdesc		new_currdev;
251*ca987d46SWarner Losh #ifdef LOADER_ZFS_SUPPORT
252*ca987d46SWarner Losh     char			buf[20];
253*ca987d46SWarner Losh #endif
254*ca987d46SWarner Losh     int				biosdev = -1;
255*ca987d46SWarner Losh 
256*ca987d46SWarner Losh     /* Assume we are booting from a BIOS disk by default */
257*ca987d46SWarner Losh     new_currdev.d_dev = &biosdisk;
258*ca987d46SWarner Losh 
259*ca987d46SWarner Losh     /* new-style boot loaders such as pxeldr and cdldr */
260*ca987d46SWarner Losh     if (kargs->bootinfo == 0) {
261*ca987d46SWarner Losh         if ((kargs->bootflags & KARGS_FLAGS_CD) != 0) {
262*ca987d46SWarner Losh 	    /* we are booting from a CD with cdboot */
263*ca987d46SWarner Losh 	    new_currdev.d_dev = &bioscd;
264*ca987d46SWarner Losh 	    new_currdev.d_unit = bc_bios2unit(initial_bootdev);
265*ca987d46SWarner Losh 	} else if ((kargs->bootflags & KARGS_FLAGS_PXE) != 0) {
266*ca987d46SWarner Losh 	    /* we are booting from pxeldr */
267*ca987d46SWarner Losh 	    new_currdev.d_dev = &pxedisk;
268*ca987d46SWarner Losh 	    new_currdev.d_unit = 0;
269*ca987d46SWarner Losh 	} else {
270*ca987d46SWarner Losh 	    /* we don't know what our boot device is */
271*ca987d46SWarner Losh 	    new_currdev.d_kind.biosdisk.slice = -1;
272*ca987d46SWarner Losh 	    new_currdev.d_kind.biosdisk.partition = 0;
273*ca987d46SWarner Losh 	    biosdev = -1;
274*ca987d46SWarner Losh 	}
275*ca987d46SWarner Losh #ifdef LOADER_ZFS_SUPPORT
276*ca987d46SWarner Losh     } else if ((kargs->bootflags & KARGS_FLAGS_ZFS) != 0) {
277*ca987d46SWarner Losh 	zargs = NULL;
278*ca987d46SWarner Losh 	/* check for new style extended argument */
279*ca987d46SWarner Losh 	if ((kargs->bootflags & KARGS_FLAGS_EXTARG) != 0)
280*ca987d46SWarner Losh 	    zargs = (struct zfs_boot_args *)(kargs + 1);
281*ca987d46SWarner Losh 
282*ca987d46SWarner Losh 	if (zargs != NULL &&
283*ca987d46SWarner Losh 	    zargs->size >= offsetof(struct zfs_boot_args, primary_pool)) {
284*ca987d46SWarner Losh 	    /* sufficient data is provided */
285*ca987d46SWarner Losh 	    new_currdev.d_kind.zfs.pool_guid = zargs->pool;
286*ca987d46SWarner Losh 	    new_currdev.d_kind.zfs.root_guid = zargs->root;
287*ca987d46SWarner Losh 	    if (zargs->size >= sizeof(*zargs) && zargs->primary_vdev != 0) {
288*ca987d46SWarner Losh 		sprintf(buf, "%llu", zargs->primary_pool);
289*ca987d46SWarner Losh 		setenv("vfs.zfs.boot.primary_pool", buf, 1);
290*ca987d46SWarner Losh 		sprintf(buf, "%llu", zargs->primary_vdev);
291*ca987d46SWarner Losh 		setenv("vfs.zfs.boot.primary_vdev", buf, 1);
292*ca987d46SWarner Losh 	    }
293*ca987d46SWarner Losh 	} else {
294*ca987d46SWarner Losh 	    /* old style zfsboot block */
295*ca987d46SWarner Losh 	    new_currdev.d_kind.zfs.pool_guid = kargs->zfspool;
296*ca987d46SWarner Losh 	    new_currdev.d_kind.zfs.root_guid = 0;
297*ca987d46SWarner Losh 	}
298*ca987d46SWarner Losh 	new_currdev.d_dev = &zfs_dev;
299*ca987d46SWarner Losh #endif
300*ca987d46SWarner Losh     } else if ((initial_bootdev & B_MAGICMASK) != B_DEVMAGIC) {
301*ca987d46SWarner Losh 	/* The passed-in boot device is bad */
302*ca987d46SWarner Losh 	new_currdev.d_kind.biosdisk.slice = -1;
303*ca987d46SWarner Losh 	new_currdev.d_kind.biosdisk.partition = 0;
304*ca987d46SWarner Losh 	biosdev = -1;
305*ca987d46SWarner Losh     } else {
306*ca987d46SWarner Losh 	new_currdev.d_kind.biosdisk.slice = B_SLICE(initial_bootdev) - 1;
307*ca987d46SWarner Losh 	new_currdev.d_kind.biosdisk.partition = B_PARTITION(initial_bootdev);
308*ca987d46SWarner Losh 	biosdev = initial_bootinfo->bi_bios_dev;
309*ca987d46SWarner Losh 
310*ca987d46SWarner Losh 	/*
311*ca987d46SWarner Losh 	 * If we are booted by an old bootstrap, we have to guess at the BIOS
312*ca987d46SWarner Losh 	 * unit number.  We will lose if there is more than one disk type
313*ca987d46SWarner Losh 	 * and we are not booting from the lowest-numbered disk type
314*ca987d46SWarner Losh 	 * (ie. SCSI when IDE also exists).
315*ca987d46SWarner Losh 	 */
316*ca987d46SWarner Losh 	if ((biosdev == 0) && (B_TYPE(initial_bootdev) != 2))	/* biosdev doesn't match major */
317*ca987d46SWarner Losh 	    biosdev = 0x80 + B_UNIT(initial_bootdev);		/* assume harddisk */
318*ca987d46SWarner Losh     }
319*ca987d46SWarner Losh     new_currdev.d_type = new_currdev.d_dev->dv_type;
320*ca987d46SWarner Losh 
321*ca987d46SWarner Losh     /*
322*ca987d46SWarner Losh      * If we are booting off of a BIOS disk and we didn't succeed in determining
323*ca987d46SWarner Losh      * which one we booted off of, just use disk0: as a reasonable default.
324*ca987d46SWarner Losh      */
325*ca987d46SWarner Losh     if ((new_currdev.d_type == biosdisk.dv_type) &&
326*ca987d46SWarner Losh 	((new_currdev.d_unit = bd_bios2unit(biosdev)) == -1)) {
327*ca987d46SWarner Losh 	printf("Can't work out which disk we are booting from.\n"
328*ca987d46SWarner Losh 	       "Guessed BIOS device 0x%x not found by probes, defaulting to disk0:\n", biosdev);
329*ca987d46SWarner Losh 	new_currdev.d_unit = 0;
330*ca987d46SWarner Losh     }
331*ca987d46SWarner Losh 
332*ca987d46SWarner Losh #ifdef LOADER_ZFS_SUPPORT
333*ca987d46SWarner Losh     if (new_currdev.d_type == DEVT_ZFS)
334*ca987d46SWarner Losh 	init_zfs_bootenv(zfs_fmtdev(&new_currdev));
335*ca987d46SWarner Losh #endif
336*ca987d46SWarner Losh 
337*ca987d46SWarner Losh     env_setenv("currdev", EV_VOLATILE, i386_fmtdev(&new_currdev),
338*ca987d46SWarner Losh 	       i386_setcurrdev, env_nounset);
339*ca987d46SWarner Losh     env_setenv("loaddev", EV_VOLATILE, i386_fmtdev(&new_currdev), env_noset,
340*ca987d46SWarner Losh 	       env_nounset);
341*ca987d46SWarner Losh }
342*ca987d46SWarner Losh 
343*ca987d46SWarner Losh COMMAND_SET(reboot, "reboot", "reboot the system", command_reboot);
344*ca987d46SWarner Losh 
345*ca987d46SWarner Losh static int
346*ca987d46SWarner Losh command_reboot(int argc, char *argv[])
347*ca987d46SWarner Losh {
348*ca987d46SWarner Losh     int i;
349*ca987d46SWarner Losh 
350*ca987d46SWarner Losh     for (i = 0; devsw[i] != NULL; ++i)
351*ca987d46SWarner Losh 	if (devsw[i]->dv_cleanup != NULL)
352*ca987d46SWarner Losh 	    (devsw[i]->dv_cleanup)();
353*ca987d46SWarner Losh 
354*ca987d46SWarner Losh     printf("Rebooting...\n");
355*ca987d46SWarner Losh     delay(1000000);
356*ca987d46SWarner Losh     __exit(0);
357*ca987d46SWarner Losh }
358*ca987d46SWarner Losh 
359*ca987d46SWarner Losh /* provide this for panic, as it's not in the startup code */
360*ca987d46SWarner Losh void
361*ca987d46SWarner Losh exit(int code)
362*ca987d46SWarner Losh {
363*ca987d46SWarner Losh     __exit(code);
364*ca987d46SWarner Losh }
365*ca987d46SWarner Losh 
366*ca987d46SWarner Losh COMMAND_SET(heap, "heap", "show heap usage", command_heap);
367*ca987d46SWarner Losh 
368*ca987d46SWarner Losh static int
369*ca987d46SWarner Losh command_heap(int argc, char *argv[])
370*ca987d46SWarner Losh {
371*ca987d46SWarner Losh     mallocstats();
372*ca987d46SWarner Losh     printf("heap base at %p, top at %p, upper limit at %p\n", heap_bottom,
373*ca987d46SWarner Losh       sbrk(0), heap_top);
374*ca987d46SWarner Losh     return(CMD_OK);
375*ca987d46SWarner Losh }
376*ca987d46SWarner Losh 
377*ca987d46SWarner Losh #ifdef LOADER_ZFS_SUPPORT
378*ca987d46SWarner Losh COMMAND_SET(lszfs, "lszfs", "list child datasets of a zfs dataset",
379*ca987d46SWarner Losh     command_lszfs);
380*ca987d46SWarner Losh 
381*ca987d46SWarner Losh static int
382*ca987d46SWarner Losh command_lszfs(int argc, char *argv[])
383*ca987d46SWarner Losh {
384*ca987d46SWarner Losh     int err;
385*ca987d46SWarner Losh 
386*ca987d46SWarner Losh     if (argc != 2) {
387*ca987d46SWarner Losh 	command_errmsg = "wrong number of arguments";
388*ca987d46SWarner Losh 	return (CMD_ERROR);
389*ca987d46SWarner Losh     }
390*ca987d46SWarner Losh 
391*ca987d46SWarner Losh     err = zfs_list(argv[1]);
392*ca987d46SWarner Losh     if (err != 0) {
393*ca987d46SWarner Losh 	command_errmsg = strerror(err);
394*ca987d46SWarner Losh 	return (CMD_ERROR);
395*ca987d46SWarner Losh     }
396*ca987d46SWarner Losh 
397*ca987d46SWarner Losh     return (CMD_OK);
398*ca987d46SWarner Losh }
399*ca987d46SWarner Losh 
400*ca987d46SWarner Losh COMMAND_SET(reloadbe, "reloadbe", "refresh the list of ZFS Boot Environments",
401*ca987d46SWarner Losh     command_reloadbe);
402*ca987d46SWarner Losh 
403*ca987d46SWarner Losh static int
404*ca987d46SWarner Losh command_reloadbe(int argc, char *argv[])
405*ca987d46SWarner Losh {
406*ca987d46SWarner Losh     int err;
407*ca987d46SWarner Losh     char *root;
408*ca987d46SWarner Losh 
409*ca987d46SWarner Losh     if (argc > 2) {
410*ca987d46SWarner Losh 	command_errmsg = "wrong number of arguments";
411*ca987d46SWarner Losh 	return (CMD_ERROR);
412*ca987d46SWarner Losh     }
413*ca987d46SWarner Losh 
414*ca987d46SWarner Losh     if (argc == 2) {
415*ca987d46SWarner Losh 	err = zfs_bootenv(argv[1]);
416*ca987d46SWarner Losh     } else {
417*ca987d46SWarner Losh 	root = getenv("zfs_be_root");
418*ca987d46SWarner Losh 	if (root == NULL) {
419*ca987d46SWarner Losh 	    /* There does not appear to be a ZFS pool here, exit without error */
420*ca987d46SWarner Losh 	    return (CMD_OK);
421*ca987d46SWarner Losh 	}
422*ca987d46SWarner Losh 	err = zfs_bootenv(root);
423*ca987d46SWarner Losh     }
424*ca987d46SWarner Losh 
425*ca987d46SWarner Losh     if (err != 0) {
426*ca987d46SWarner Losh 	command_errmsg = strerror(err);
427*ca987d46SWarner Losh 	return (CMD_ERROR);
428*ca987d46SWarner Losh     }
429*ca987d46SWarner Losh 
430*ca987d46SWarner Losh     return (CMD_OK);
431*ca987d46SWarner Losh }
432*ca987d46SWarner Losh #endif
433*ca987d46SWarner Losh 
434*ca987d46SWarner Losh /* ISA bus access functions for PnP. */
435*ca987d46SWarner Losh static int
436*ca987d46SWarner Losh isa_inb(int port)
437*ca987d46SWarner Losh {
438*ca987d46SWarner Losh 
439*ca987d46SWarner Losh     return (inb(port));
440*ca987d46SWarner Losh }
441*ca987d46SWarner Losh 
442*ca987d46SWarner Losh static void
443*ca987d46SWarner Losh isa_outb(int port, int value)
444*ca987d46SWarner Losh {
445*ca987d46SWarner Losh 
446*ca987d46SWarner Losh     outb(port, value);
447*ca987d46SWarner Losh }
448*ca987d46SWarner Losh 
449*ca987d46SWarner Losh #ifdef LOADER_ZFS_SUPPORT
450*ca987d46SWarner Losh static void
451*ca987d46SWarner Losh i386_zfs_probe(void)
452*ca987d46SWarner Losh {
453*ca987d46SWarner Losh     char devname[32];
454*ca987d46SWarner Losh     int unit;
455*ca987d46SWarner Losh 
456*ca987d46SWarner Losh     /*
457*ca987d46SWarner Losh      * Open all the disks we can find and see if we can reconstruct
458*ca987d46SWarner Losh      * ZFS pools from them.
459*ca987d46SWarner Losh      */
460*ca987d46SWarner Losh     for (unit = 0; unit < MAXBDDEV; unit++) {
461*ca987d46SWarner Losh 	if (bd_unit2bios(unit) == -1)
462*ca987d46SWarner Losh 	    break;
463*ca987d46SWarner Losh 	sprintf(devname, "disk%d:", unit);
464*ca987d46SWarner Losh 	zfs_probe_dev(devname, NULL);
465*ca987d46SWarner Losh     }
466*ca987d46SWarner Losh }
467*ca987d46SWarner Losh 
468*ca987d46SWarner Losh uint64_t
469*ca987d46SWarner Losh ldi_get_size(void *priv)
470*ca987d46SWarner Losh {
471*ca987d46SWarner Losh 	int fd = (uintptr_t) priv;
472*ca987d46SWarner Losh 	uint64_t size;
473*ca987d46SWarner Losh 
474*ca987d46SWarner Losh 	ioctl(fd, DIOCGMEDIASIZE, &size);
475*ca987d46SWarner Losh 	return (size);
476*ca987d46SWarner Losh }
477*ca987d46SWarner Losh #endif
478