xref: /titanic_44/usr/src/uts/i86pc/os/fakebop.c (revision 021ceb9857f712f245edda92c7eb82f0a3a83acb)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 
22 /*
23  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
24  * Use is subject to license terms.
25  */
26 
27 /*
28  * This file contains the functionality that mimics the boot operations
29  * on SPARC systems or the old boot.bin/multiboot programs on x86 systems.
30  * The x86 kernel now does everything on its own.
31  */
32 
33 #include <sys/types.h>
34 #include <sys/bootconf.h>
35 #include <sys/bootsvcs.h>
36 #include <sys/bootinfo.h>
37 #include <sys/multiboot.h>
38 #include <sys/bootvfs.h>
39 #include <sys/bootprops.h>
40 #include <sys/varargs.h>
41 #include <sys/param.h>
42 #include <sys/machparam.h>
43 #include <sys/archsystm.h>
44 #include <sys/boot_console.h>
45 #include <sys/cmn_err.h>
46 #include <sys/systm.h>
47 #include <sys/promif.h>
48 #include <sys/archsystm.h>
49 #include <sys/x86_archext.h>
50 #include <sys/kobj.h>
51 #include <sys/privregs.h>
52 #include <sys/sysmacros.h>
53 #include <sys/ctype.h>
54 #include <sys/fastboot.h>
55 #ifdef __xpv
56 #include <sys/hypervisor.h>
57 #include <net/if.h>
58 #endif
59 #include <vm/kboot_mmu.h>
60 #include <vm/hat_pte.h>
61 #include <sys/dmar_acpi.h>
62 #include <sys/kobj.h>
63 #include <sys/kobj_lex.h>
64 #include "acpi_fw.h"
65 
66 static int have_console = 0;	/* set once primitive console is initialized */
67 static char *boot_args = "";
68 
69 /*
70  * Debugging macros
71  */
72 static uint_t kbm_debug = 0;
73 #define	DBG_MSG(s)	{ if (kbm_debug) bop_printf(NULL, "%s", s); }
74 #define	DBG(x)		{ if (kbm_debug)			\
75 	bop_printf(NULL, "%s is %" PRIx64 "\n", #x, (uint64_t)(x));	\
76 	}
77 
78 #define	PUT_STRING(s) {				\
79 	char *cp;				\
80 	for (cp = (s); *cp; ++cp)		\
81 		bcons_putchar(*cp);		\
82 	}
83 
84 struct xboot_info *xbootp;	/* boot info from "glue" code in low memory */
85 bootops_t bootop;	/* simple bootops we'll pass on to kernel */
86 struct bsys_mem bm;
87 
88 static uintptr_t next_virt;	/* next available virtual address */
89 static paddr_t next_phys;	/* next available physical address from dboot */
90 static paddr_t high_phys = -(paddr_t)1;	/* last used physical address */
91 
92 /*
93  * buffer for vsnprintf for console I/O
94  */
95 #define	BUFFERSIZE	256
96 static char buffer[BUFFERSIZE];
97 /*
98  * stuff to store/report/manipulate boot property settings.
99  */
100 typedef struct bootprop {
101 	struct bootprop *bp_next;
102 	char *bp_name;
103 	uint_t bp_vlen;
104 	char *bp_value;
105 } bootprop_t;
106 
107 static bootprop_t *bprops = NULL;
108 static char *curr_page = NULL;		/* ptr to avail bprop memory */
109 static int curr_space = 0;		/* amount of memory at curr_page */
110 
111 #ifdef __xpv
112 start_info_t *xen_info;
113 shared_info_t *HYPERVISOR_shared_info;
114 #endif
115 
116 /*
117  * some allocator statistics
118  */
119 static ulong_t total_bop_alloc_scratch = 0;
120 static ulong_t total_bop_alloc_kernel = 0;
121 
122 static void build_firmware_properties(void);
123 
124 static int early_allocation = 1;
125 
126 int force_fastreboot = 0;
127 volatile int fastreboot_onpanic = 0;
128 int post_fastreboot = 0;
129 #ifdef	__xpv
130 int fastreboot_capable = 0;
131 #else
132 int fastreboot_capable = 1;
133 #endif
134 
135 /*
136  * Information saved from current boot for fast reboot.
137  * If the information size exceeds what we have allocated, fast reboot
138  * will not be supported.
139  */
140 multiboot_info_t saved_mbi;
141 mb_memory_map_t saved_mmap[FASTBOOT_SAVED_MMAP_COUNT];
142 uint8_t saved_drives[FASTBOOT_SAVED_DRIVES_SIZE];
143 char saved_cmdline[FASTBOOT_SAVED_CMDLINE_LEN];
144 int saved_cmdline_len = 0;
145 size_t saved_file_size[FASTBOOT_MAX_FILES_MAP];
146 
147 /*
148  * Turn off fastreboot_onpanic to avoid panic loop.
149  */
150 char fastreboot_onpanic_cmdline[FASTBOOT_SAVED_CMDLINE_LEN];
151 static const char fastreboot_onpanic_args[] = " -B fastreboot_onpanic=0";
152 
153 /*
154  * Pointers to where System Resource Affinity Table (SRAT) and
155  * System Locality Information Table (SLIT) are mapped into virtual memory
156  */
157 struct srat	*srat_ptr = NULL;
158 struct slit	*slit_ptr = NULL;
159 
160 
161 /*
162  * Allocate aligned physical memory at boot time. This allocator allocates
163  * from the highest possible addresses. This avoids exhausting memory that
164  * would be useful for DMA buffers.
165  */
166 paddr_t
167 do_bop_phys_alloc(uint64_t size, uint64_t align)
168 {
169 	paddr_t	pa = 0;
170 	paddr_t	start;
171 	paddr_t	end;
172 	struct memlist	*ml = (struct memlist *)xbootp->bi_phys_install;
173 
174 	/*
175 	 * Be careful if high memory usage is limited in startup.c
176 	 * Since there are holes in the low part of the physical address
177 	 * space we can treat physmem as a pfn (not just a pgcnt) and
178 	 * get a conservative upper limit.
179 	 */
180 	if (physmem != 0 && high_phys > pfn_to_pa(physmem))
181 		high_phys = pfn_to_pa(physmem);
182 
183 	/*
184 	 * find the lowest or highest available memory in physinstalled
185 	 * On 32 bit avoid physmem above 4Gig if PAE isn't enabled
186 	 */
187 #if defined(__i386)
188 	if (xbootp->bi_use_pae == 0 && high_phys > FOUR_GIG)
189 		high_phys = FOUR_GIG;
190 #endif
191 
192 	/*
193 	 * find the highest available memory in physinstalled
194 	 */
195 	size = P2ROUNDUP(size, align);
196 	for (; ml; ml = ml->next) {
197 		start = P2ROUNDUP(ml->address, align);
198 		end = P2ALIGN(ml->address + ml->size, align);
199 		if (start < next_phys)
200 			start = P2ROUNDUP(next_phys, align);
201 		if (end > high_phys)
202 			end = P2ALIGN(high_phys, align);
203 
204 		if (end <= start)
205 			continue;
206 		if (end - start < size)
207 			continue;
208 
209 		/*
210 		 * Early allocations need to use low memory, since
211 		 * physmem might be further limited by bootenv.rc
212 		 */
213 		if (early_allocation) {
214 			if (pa == 0 || start < pa)
215 				pa = start;
216 		} else {
217 			if (end - size > pa)
218 				pa = end - size;
219 		}
220 	}
221 	if (pa != 0) {
222 		if (early_allocation)
223 			next_phys = pa + size;
224 		else
225 			high_phys = pa;
226 		return (pa);
227 	}
228 	bop_panic("do_bop_phys_alloc(0x%" PRIx64 ", 0x%" PRIx64
229 	    ") Out of memory\n", size, align);
230 	/*NOTREACHED*/
231 }
232 
233 static uintptr_t
234 alloc_vaddr(size_t size, paddr_t align)
235 {
236 	uintptr_t rv;
237 
238 	next_virt = P2ROUNDUP(next_virt, (uintptr_t)align);
239 	rv = (uintptr_t)next_virt;
240 	next_virt += size;
241 	return (rv);
242 }
243 
244 /*
245  * Allocate virtual memory. The size is always rounded up to a multiple
246  * of base pagesize.
247  */
248 
249 /*ARGSUSED*/
250 static caddr_t
251 do_bsys_alloc(bootops_t *bop, caddr_t virthint, size_t size, int align)
252 {
253 	paddr_t a = align;	/* same type as pa for masking */
254 	uint_t pgsize;
255 	paddr_t pa;
256 	uintptr_t va;
257 	ssize_t s;		/* the aligned size */
258 	uint_t level;
259 	uint_t is_kernel = (virthint != 0);
260 
261 	if (a < MMU_PAGESIZE)
262 		a = MMU_PAGESIZE;
263 	else if (!ISP2(a))
264 		prom_panic("do_bsys_alloc() incorrect alignment");
265 	size = P2ROUNDUP(size, MMU_PAGESIZE);
266 
267 	/*
268 	 * Use the next aligned virtual address if we weren't given one.
269 	 */
270 	if (virthint == NULL) {
271 		virthint = (caddr_t)alloc_vaddr(size, a);
272 		total_bop_alloc_scratch += size;
273 	} else {
274 		total_bop_alloc_kernel += size;
275 	}
276 
277 	/*
278 	 * allocate the physical memory
279 	 */
280 	pa = do_bop_phys_alloc(size, a);
281 
282 	/*
283 	 * Add the mappings to the page tables, try large pages first.
284 	 */
285 	va = (uintptr_t)virthint;
286 	s = size;
287 	level = 1;
288 	pgsize = xbootp->bi_use_pae ? TWO_MEG : FOUR_MEG;
289 	if (xbootp->bi_use_largepage && a == pgsize) {
290 		while (IS_P2ALIGNED(pa, pgsize) && IS_P2ALIGNED(va, pgsize) &&
291 		    s >= pgsize) {
292 			kbm_map(va, pa, level, is_kernel);
293 			va += pgsize;
294 			pa += pgsize;
295 			s -= pgsize;
296 		}
297 	}
298 
299 	/*
300 	 * Map remaining pages use small mappings
301 	 */
302 	level = 0;
303 	pgsize = MMU_PAGESIZE;
304 	while (s > 0) {
305 		kbm_map(va, pa, level, is_kernel);
306 		va += pgsize;
307 		pa += pgsize;
308 		s -= pgsize;
309 	}
310 	return (virthint);
311 }
312 
313 /*
314  * Free virtual memory - we'll just ignore these.
315  */
316 /*ARGSUSED*/
317 static void
318 do_bsys_free(bootops_t *bop, caddr_t virt, size_t size)
319 {
320 	bop_printf(NULL, "do_bsys_free(virt=0x%p, size=0x%lx) ignored\n",
321 	    (void *)virt, size);
322 }
323 
324 /*
325  * Old interface
326  */
327 /*ARGSUSED*/
328 static caddr_t
329 do_bsys_ealloc(
330 	bootops_t *bop,
331 	caddr_t virthint,
332 	size_t size,
333 	int align,
334 	int flags)
335 {
336 	prom_panic("unsupported call to BOP_EALLOC()\n");
337 	return (0);
338 }
339 
340 
341 static void
342 bsetprop(char *name, int nlen, void *value, int vlen)
343 {
344 	uint_t size;
345 	uint_t need_size;
346 	bootprop_t *b;
347 
348 	/*
349 	 * align the size to 16 byte boundary
350 	 */
351 	size = sizeof (bootprop_t) + nlen + 1 + vlen;
352 	size = (size + 0xf) & ~0xf;
353 	if (size > curr_space) {
354 		need_size = (size + (MMU_PAGEOFFSET)) & MMU_PAGEMASK;
355 		curr_page = do_bsys_alloc(NULL, 0, need_size, MMU_PAGESIZE);
356 		curr_space = need_size;
357 	}
358 
359 	/*
360 	 * use a bootprop_t at curr_page and link into list
361 	 */
362 	b = (bootprop_t *)curr_page;
363 	curr_page += sizeof (bootprop_t);
364 	curr_space -=  sizeof (bootprop_t);
365 	b->bp_next = bprops;
366 	bprops = b;
367 
368 	/*
369 	 * follow by name and ending zero byte
370 	 */
371 	b->bp_name = curr_page;
372 	bcopy(name, curr_page, nlen);
373 	curr_page += nlen;
374 	*curr_page++ = 0;
375 	curr_space -= nlen + 1;
376 
377 	/*
378 	 * copy in value, but no ending zero byte
379 	 */
380 	b->bp_value = curr_page;
381 	b->bp_vlen = vlen;
382 	if (vlen > 0) {
383 		bcopy(value, curr_page, vlen);
384 		curr_page += vlen;
385 		curr_space -= vlen;
386 	}
387 
388 	/*
389 	 * align new values of curr_page, curr_space
390 	 */
391 	while (curr_space & 0xf) {
392 		++curr_page;
393 		--curr_space;
394 	}
395 }
396 
397 static void
398 bsetprops(char *name, char *value)
399 {
400 	bsetprop(name, strlen(name), value, strlen(value) + 1);
401 }
402 
403 static void
404 bsetprop64(char *name, uint64_t value)
405 {
406 	bsetprop(name, strlen(name), (void *)&value, sizeof (value));
407 }
408 
409 static void
410 bsetpropsi(char *name, int value)
411 {
412 	char prop_val[32];
413 
414 	(void) snprintf(prop_val, sizeof (prop_val), "%d", value);
415 	bsetprops(name, prop_val);
416 }
417 
418 /*
419  * to find the size of the buffer to allocate
420  */
421 /*ARGSUSED*/
422 int
423 do_bsys_getproplen(bootops_t *bop, const char *name)
424 {
425 	bootprop_t *b;
426 
427 	for (b = bprops; b; b = b->bp_next) {
428 		if (strcmp(name, b->bp_name) != 0)
429 			continue;
430 		return (b->bp_vlen);
431 	}
432 	return (-1);
433 }
434 
435 /*
436  * get the value associated with this name
437  */
438 /*ARGSUSED*/
439 int
440 do_bsys_getprop(bootops_t *bop, const char *name, void *value)
441 {
442 	bootprop_t *b;
443 
444 	for (b = bprops; b; b = b->bp_next) {
445 		if (strcmp(name, b->bp_name) != 0)
446 			continue;
447 		bcopy(b->bp_value, value, b->bp_vlen);
448 		return (0);
449 	}
450 	return (-1);
451 }
452 
453 /*
454  * get the name of the next property in succession from the standalone
455  */
456 /*ARGSUSED*/
457 static char *
458 do_bsys_nextprop(bootops_t *bop, char *name)
459 {
460 	bootprop_t *b;
461 
462 	/*
463 	 * A null name is a special signal for the 1st boot property
464 	 */
465 	if (name == NULL || strlen(name) == 0) {
466 		if (bprops == NULL)
467 			return (NULL);
468 		return (bprops->bp_name);
469 	}
470 
471 	for (b = bprops; b; b = b->bp_next) {
472 		if (name != b->bp_name)
473 			continue;
474 		b = b->bp_next;
475 		if (b == NULL)
476 			return (NULL);
477 		return (b->bp_name);
478 	}
479 	return (NULL);
480 }
481 
482 /*
483  * Parse numeric value from a string. Understands decimal, hex, octal, - and ~
484  */
485 static int
486 parse_value(char *p, uint64_t *retval)
487 {
488 	int adjust = 0;
489 	uint64_t tmp = 0;
490 	int digit;
491 	int radix = 10;
492 
493 	*retval = 0;
494 	if (*p == '-' || *p == '~')
495 		adjust = *p++;
496 
497 	if (*p == '0') {
498 		++p;
499 		if (*p == 0)
500 			return (0);
501 		if (*p == 'x' || *p == 'X') {
502 			radix = 16;
503 			++p;
504 		} else {
505 			radix = 8;
506 			++p;
507 		}
508 	}
509 	while (*p) {
510 		if ('0' <= *p && *p <= '9')
511 			digit = *p - '0';
512 		else if ('a' <= *p && *p <= 'f')
513 			digit = 10 + *p - 'a';
514 		else if ('A' <= *p && *p <= 'F')
515 			digit = 10 + *p - 'A';
516 		else
517 			return (-1);
518 		if (digit >= radix)
519 			return (-1);
520 		tmp = tmp * radix + digit;
521 		++p;
522 	}
523 	if (adjust == '-')
524 		tmp = -tmp;
525 	else if (adjust == '~')
526 		tmp = ~tmp;
527 	*retval = tmp;
528 	return (0);
529 }
530 
531 /*
532  * 2nd part of building the table of boot properties. This includes:
533  * - values from /boot/solaris/bootenv.rc (ie. eeprom(1m) values)
534  *
535  * lines look like one of:
536  * ^$
537  * ^# comment till end of line
538  * setprop name 'value'
539  * setprop name value
540  * setprop name "value"
541  *
542  * we do single character I/O since this is really just looking at memory
543  */
544 void
545 boot_prop_finish(void)
546 {
547 	int fd;
548 	char *line;
549 	int c;
550 	int bytes_read;
551 	char *name;
552 	int n_len;
553 	char *value;
554 	int v_len;
555 	char *inputdev;	/* these override the command line if serial ports */
556 	char *outputdev;
557 	char *consoledev;
558 	uint64_t lvalue;
559 	int use_xencons = 0;
560 
561 #ifdef __xpv
562 	if (!DOMAIN_IS_INITDOMAIN(xen_info))
563 		use_xencons = 1;
564 #endif /* __xpv */
565 
566 	DBG_MSG("Opening /boot/solaris/bootenv.rc\n");
567 	fd = BRD_OPEN(bfs_ops, "/boot/solaris/bootenv.rc", 0);
568 	DBG(fd);
569 
570 	line = do_bsys_alloc(NULL, NULL, MMU_PAGESIZE, MMU_PAGESIZE);
571 	while (fd >= 0) {
572 
573 		/*
574 		 * get a line
575 		 */
576 		for (c = 0; ; ++c) {
577 			bytes_read = BRD_READ(bfs_ops, fd, line + c, 1);
578 			if (bytes_read == 0) {
579 				if (c == 0)
580 					goto done;
581 				break;
582 			}
583 			if (line[c] == '\n')
584 				break;
585 		}
586 		line[c] = 0;
587 
588 		/*
589 		 * ignore comment lines
590 		 */
591 		c = 0;
592 		while (ISSPACE(line[c]))
593 			++c;
594 		if (line[c] == '#' || line[c] == 0)
595 			continue;
596 
597 		/*
598 		 * must have "setprop " or "setprop\t"
599 		 */
600 		if (strncmp(line + c, "setprop ", 8) != 0 &&
601 		    strncmp(line + c, "setprop\t", 8) != 0)
602 			continue;
603 		c += 8;
604 		while (ISSPACE(line[c]))
605 			++c;
606 		if (line[c] == 0)
607 			continue;
608 
609 		/*
610 		 * gather up the property name
611 		 */
612 		name = line + c;
613 		n_len = 0;
614 		while (line[c] && !ISSPACE(line[c]))
615 			++n_len, ++c;
616 
617 		/*
618 		 * gather up the value, if any
619 		 */
620 		value = "";
621 		v_len = 0;
622 		while (ISSPACE(line[c]))
623 			++c;
624 		if (line[c] != 0) {
625 			value = line + c;
626 			while (line[c] && !ISSPACE(line[c]))
627 				++v_len, ++c;
628 		}
629 
630 		if (v_len >= 2 && value[0] == value[v_len - 1] &&
631 		    (value[0] == '\'' || value[0] == '"')) {
632 			++value;
633 			v_len -= 2;
634 		}
635 		name[n_len] = 0;
636 		if (v_len > 0)
637 			value[v_len] = 0;
638 		else
639 			continue;
640 
641 		/*
642 		 * ignore "boot-file" property, it's now meaningless
643 		 */
644 		if (strcmp(name, "boot-file") == 0)
645 			continue;
646 		if (strcmp(name, "boot-args") == 0 &&
647 		    strlen(boot_args) > 0)
648 			continue;
649 
650 		/*
651 		 * If a property was explicitly set on the command line
652 		 * it will override a setting in bootenv.rc
653 		 */
654 		if (do_bsys_getproplen(NULL, name) > 0)
655 			continue;
656 
657 		bsetprop(name, n_len, value, v_len + 1);
658 	}
659 done:
660 	if (fd >= 0)
661 		BRD_CLOSE(bfs_ops, fd);
662 
663 	/*
664 	 * Check if we have to limit the boot time allocator
665 	 */
666 	if (do_bsys_getproplen(NULL, "physmem") != -1 &&
667 	    do_bsys_getprop(NULL, "physmem", line) >= 0 &&
668 	    parse_value(line, &lvalue) != -1) {
669 		if (0 < lvalue && (lvalue < physmem || physmem == 0)) {
670 			physmem = (pgcnt_t)lvalue;
671 			DBG(physmem);
672 		}
673 	}
674 	early_allocation = 0;
675 
676 	/*
677 	 * check to see if we have to override the default value of the console
678 	 */
679 	if (!use_xencons) {
680 		inputdev = line;
681 		v_len = do_bsys_getproplen(NULL, "input-device");
682 		if (v_len > 0)
683 			(void) do_bsys_getprop(NULL, "input-device", inputdev);
684 		else
685 			v_len = 0;
686 		inputdev[v_len] = 0;
687 
688 		outputdev = inputdev + v_len + 1;
689 		v_len = do_bsys_getproplen(NULL, "output-device");
690 		if (v_len > 0)
691 			(void) do_bsys_getprop(NULL, "output-device",
692 			    outputdev);
693 		else
694 			v_len = 0;
695 		outputdev[v_len] = 0;
696 
697 		consoledev = outputdev + v_len + 1;
698 		v_len = do_bsys_getproplen(NULL, "console");
699 		if (v_len > 0) {
700 			(void) do_bsys_getprop(NULL, "console", consoledev);
701 			if (post_fastreboot &&
702 			    strcmp(consoledev, "graphics") == 0) {
703 				bsetprops("console", "text");
704 				v_len = strlen("text");
705 				bcopy("text", consoledev, v_len);
706 			}
707 		} else {
708 			v_len = 0;
709 		}
710 		consoledev[v_len] = 0;
711 		bcons_init2(inputdev, outputdev, consoledev);
712 	} else {
713 		/*
714 		 * Ensure console property exists
715 		 * If not create it as "hypervisor"
716 		 */
717 		v_len = do_bsys_getproplen(NULL, "console");
718 		if (v_len < 0)
719 			bsetprops("console", "hypervisor");
720 		inputdev = outputdev = consoledev = "hypervisor";
721 		bcons_init2(inputdev, outputdev, consoledev);
722 	}
723 
724 	if (strstr((char *)xbootp->bi_cmdline, "prom_debug") || kbm_debug) {
725 		value = line;
726 		bop_printf(NULL, "\nBoot properties:\n");
727 		name = "";
728 		while ((name = do_bsys_nextprop(NULL, name)) != NULL) {
729 			bop_printf(NULL, "\t0x%p %s = ", (void *)name, name);
730 			(void) do_bsys_getprop(NULL, name, value);
731 			v_len = do_bsys_getproplen(NULL, name);
732 			bop_printf(NULL, "len=%d ", v_len);
733 			value[v_len] = 0;
734 			bop_printf(NULL, "%s\n", value);
735 		}
736 	}
737 }
738 
739 /*
740  * print formatted output
741  */
742 /*PRINTFLIKE2*/
743 /*ARGSUSED*/
744 void
745 bop_printf(bootops_t *bop, const char *fmt, ...)
746 {
747 	va_list	ap;
748 
749 	if (have_console == 0)
750 		return;
751 
752 	va_start(ap, fmt);
753 	(void) vsnprintf(buffer, BUFFERSIZE, fmt, ap);
754 	va_end(ap);
755 	PUT_STRING(buffer);
756 }
757 
758 /*
759  * Another panic() variant; this one can be used even earlier during boot than
760  * prom_panic().
761  */
762 /*PRINTFLIKE1*/
763 void
764 bop_panic(const char *fmt, ...)
765 {
766 	va_list ap;
767 
768 	va_start(ap, fmt);
769 	bop_printf(NULL, fmt, ap);
770 	va_end(ap);
771 
772 	bop_printf(NULL, "\nPress any key to reboot.\n");
773 	(void) bcons_getchar();
774 	bop_printf(NULL, "Resetting...\n");
775 	pc_reset();
776 }
777 
778 /*
779  * Do a real mode interrupt BIOS call
780  */
781 typedef struct bios_regs {
782 	unsigned short ax, bx, cx, dx, si, di, bp, es, ds;
783 } bios_regs_t;
784 typedef int (*bios_func_t)(int, bios_regs_t *);
785 
786 /*ARGSUSED*/
787 static void
788 do_bsys_doint(bootops_t *bop, int intnum, struct bop_regs *rp)
789 {
790 #if defined(__xpv)
791 	prom_panic("unsupported call to BOP_DOINT()\n");
792 #else	/* __xpv */
793 	static int firsttime = 1;
794 	bios_func_t bios_func = (bios_func_t)(void *)(uintptr_t)0x5000;
795 	bios_regs_t br;
796 
797 	/*
798 	 * The first time we do this, we have to copy the pre-packaged
799 	 * low memory bios call code image into place.
800 	 */
801 	if (firsttime) {
802 		extern char bios_image[];
803 		extern uint32_t bios_size;
804 
805 		bcopy(bios_image, (void *)bios_func, bios_size);
806 		firsttime = 0;
807 	}
808 
809 	br.ax = rp->eax.word.ax;
810 	br.bx = rp->ebx.word.bx;
811 	br.cx = rp->ecx.word.cx;
812 	br.dx = rp->edx.word.dx;
813 	br.bp = rp->ebp.word.bp;
814 	br.si = rp->esi.word.si;
815 	br.di = rp->edi.word.di;
816 	br.ds = rp->ds;
817 	br.es = rp->es;
818 
819 	DBG_MSG("Doing BIOS call...");
820 	DBG(br.ax);
821 	DBG(br.bx);
822 	DBG(br.dx);
823 	rp->eflags = bios_func(intnum, &br);
824 	DBG_MSG("done\n");
825 
826 	rp->eax.word.ax = br.ax;
827 	rp->ebx.word.bx = br.bx;
828 	rp->ecx.word.cx = br.cx;
829 	rp->edx.word.dx = br.dx;
830 	rp->ebp.word.bp = br.bp;
831 	rp->esi.word.si = br.si;
832 	rp->edi.word.di = br.di;
833 	rp->ds = br.ds;
834 	rp->es = br.es;
835 #endif /* __xpv */
836 }
837 
838 static struct boot_syscalls bop_sysp = {
839 	bcons_getchar,
840 	bcons_putchar,
841 	bcons_ischar,
842 };
843 
844 static char *whoami;
845 
846 #define	BUFLEN	64
847 
848 #if defined(__xpv)
849 
850 static char namebuf[32];
851 
852 static void
853 xen_parse_props(char *s, char *prop_map[], int n_prop)
854 {
855 	char **prop_name = prop_map;
856 	char *cp = s, *scp;
857 
858 	do {
859 		scp = cp;
860 		while ((*cp != NULL) && (*cp != ':'))
861 			cp++;
862 
863 		if ((scp != cp) && (*prop_name != NULL)) {
864 			*cp = NULL;
865 			bsetprops(*prop_name, scp);
866 		}
867 
868 		cp++;
869 		prop_name++;
870 		n_prop--;
871 	} while (n_prop > 0);
872 }
873 
874 #define	VBDPATHLEN	64
875 
876 /*
877  * parse the 'xpv-root' property to create properties used by
878  * ufs_mountroot.
879  */
880 static void
881 xen_vbdroot_props(char *s)
882 {
883 	char vbdpath[VBDPATHLEN] = "/xpvd/xdf@";
884 	const char lnamefix[] = "/dev/dsk/c0d";
885 	char *pnp;
886 	char *prop_p;
887 	char mi;
888 	short minor;
889 	long addr = 0;
890 
891 	pnp = vbdpath + strlen(vbdpath);
892 	prop_p = s + strlen(lnamefix);
893 	while ((*prop_p != '\0') && (*prop_p != 's') && (*prop_p != 'p'))
894 		addr = addr * 10 + *prop_p++ - '0';
895 	(void) snprintf(pnp, VBDPATHLEN, "%lx", addr);
896 	pnp = vbdpath + strlen(vbdpath);
897 	if (*prop_p == 's')
898 		mi = 'a';
899 	else if (*prop_p == 'p')
900 		mi = 'q';
901 	else
902 		ASSERT(0); /* shouldn't be here */
903 	prop_p++;
904 	ASSERT(*prop_p != '\0');
905 	if (ISDIGIT(*prop_p)) {
906 		minor = *prop_p - '0';
907 		prop_p++;
908 		if (ISDIGIT(*prop_p)) {
909 			minor = minor * 10 + *prop_p - '0';
910 		}
911 	} else {
912 		/* malformed root path, use 0 as default */
913 		minor = 0;
914 	}
915 	ASSERT(minor < 16); /* at most 16 partitions */
916 	mi += minor;
917 	*pnp++ = ':';
918 	*pnp++ = mi;
919 	*pnp++ = '\0';
920 	bsetprops("fstype", "ufs");
921 	bsetprops("bootpath", vbdpath);
922 
923 	DBG_MSG("VBD bootpath set to ");
924 	DBG_MSG(vbdpath);
925 	DBG_MSG("\n");
926 }
927 
928 /*
929  * parse the xpv-nfsroot property to create properties used by
930  * nfs_mountroot.
931  */
932 static void
933 xen_nfsroot_props(char *s)
934 {
935 	char *prop_map[] = {
936 		BP_SERVER_IP,	/* server IP address */
937 		BP_SERVER_NAME,	/* server hostname */
938 		BP_SERVER_PATH,	/* root path */
939 	};
940 	int n_prop = sizeof (prop_map) / sizeof (prop_map[0]);
941 
942 	bsetprop("fstype", 6, "nfs", 4);
943 
944 	xen_parse_props(s, prop_map, n_prop);
945 
946 	/*
947 	 * If a server name wasn't specified, use a default.
948 	 */
949 	if (do_bsys_getproplen(NULL, BP_SERVER_NAME) == -1)
950 		bsetprops(BP_SERVER_NAME, "unknown");
951 }
952 
953 /*
954  * Extract our IP address, etc. from the "xpv-ip" property.
955  */
956 static void
957 xen_ip_props(char *s)
958 {
959 	char *prop_map[] = {
960 		BP_HOST_IP,		/* IP address */
961 		NULL,			/* NFS server IP address (ignored in */
962 					/* favour of xpv-nfsroot) */
963 		BP_ROUTER_IP,		/* IP gateway */
964 		BP_SUBNET_MASK,		/* IP subnet mask */
965 		"xpv-hostname",		/* hostname (ignored) */
966 		BP_NETWORK_INTERFACE,	/* interface name */
967 		"xpv-hcp",		/* host configuration protocol */
968 	};
969 	int n_prop = sizeof (prop_map) / sizeof (prop_map[0]);
970 	char ifname[IFNAMSIZ];
971 
972 	xen_parse_props(s, prop_map, n_prop);
973 
974 	/*
975 	 * A Linux dom0 administrator expects all interfaces to be
976 	 * called "ethX", which is not the case here.
977 	 *
978 	 * If the interface name specified is "eth0", presume that
979 	 * this is really intended to be "xnf0" (the first domU ->
980 	 * dom0 interface for this domain).
981 	 */
982 	if ((do_bsys_getprop(NULL, BP_NETWORK_INTERFACE, ifname) == 0) &&
983 	    (strcmp("eth0", ifname) == 0)) {
984 		bsetprops(BP_NETWORK_INTERFACE, "xnf0");
985 		bop_printf(NULL,
986 		    "network interface name 'eth0' replaced with 'xnf0'\n");
987 	}
988 }
989 
990 #else	/* __xpv */
991 
992 static void
993 setup_rarp_props(struct sol_netinfo *sip)
994 {
995 	char buf[BUFLEN];	/* to hold ip/mac addrs */
996 	uint8_t *val;
997 
998 	val = (uint8_t *)&sip->sn_ciaddr;
999 	(void) snprintf(buf, BUFLEN, "%d.%d.%d.%d",
1000 	    val[0], val[1], val[2], val[3]);
1001 	bsetprops(BP_HOST_IP, buf);
1002 
1003 	val = (uint8_t *)&sip->sn_siaddr;
1004 	(void) snprintf(buf, BUFLEN, "%d.%d.%d.%d",
1005 	    val[0], val[1], val[2], val[3]);
1006 	bsetprops(BP_SERVER_IP, buf);
1007 
1008 	if (sip->sn_giaddr != 0) {
1009 		val = (uint8_t *)&sip->sn_giaddr;
1010 		(void) snprintf(buf, BUFLEN, "%d.%d.%d.%d",
1011 		    val[0], val[1], val[2], val[3]);
1012 		bsetprops(BP_ROUTER_IP, buf);
1013 	}
1014 
1015 	if (sip->sn_netmask != 0) {
1016 		val = (uint8_t *)&sip->sn_netmask;
1017 		(void) snprintf(buf, BUFLEN, "%d.%d.%d.%d",
1018 		    val[0], val[1], val[2], val[3]);
1019 		bsetprops(BP_SUBNET_MASK, buf);
1020 	}
1021 
1022 	if (sip->sn_mactype != 4 || sip->sn_maclen != 6) {
1023 		bop_printf(NULL, "unsupported mac type %d, mac len %d\n",
1024 		    sip->sn_mactype, sip->sn_maclen);
1025 	} else {
1026 		val = sip->sn_macaddr;
1027 		(void) snprintf(buf, BUFLEN, "%x:%x:%x:%x:%x:%x",
1028 		    val[0], val[1], val[2], val[3], val[4], val[5]);
1029 		bsetprops(BP_BOOT_MAC, buf);
1030 	}
1031 }
1032 
1033 #endif	/* __xpv */
1034 
1035 static void
1036 build_panic_cmdline(const char *cmd, int cmdlen)
1037 {
1038 	int proplen;
1039 	size_t arglen;
1040 
1041 	arglen = sizeof (fastreboot_onpanic_args);
1042 	/*
1043 	 * If we allready have fastreboot-onpanic set to zero,
1044 	 * don't add them again.
1045 	 */
1046 	if ((proplen = do_bsys_getproplen(NULL, FASTREBOOT_ONPANIC)) > 0 &&
1047 	    proplen <=  sizeof (fastreboot_onpanic_cmdline)) {
1048 		(void) do_bsys_getprop(NULL, FASTREBOOT_ONPANIC,
1049 		    fastreboot_onpanic_cmdline);
1050 		if (FASTREBOOT_ONPANIC_NOTSET(fastreboot_onpanic_cmdline))
1051 			arglen = 1;
1052 	}
1053 
1054 	/*
1055 	 * construct fastreboot_onpanic_cmdline
1056 	 */
1057 	if (cmdlen + arglen > sizeof (fastreboot_onpanic_cmdline)) {
1058 		DBG_MSG("Command line too long: clearing "
1059 		    FASTREBOOT_ONPANIC "\n");
1060 		fastreboot_onpanic = 0;
1061 	} else {
1062 		bcopy(cmd, fastreboot_onpanic_cmdline, cmdlen);
1063 		if (arglen != 1)
1064 			bcopy(fastreboot_onpanic_args,
1065 			    fastreboot_onpanic_cmdline + cmdlen, arglen);
1066 		else
1067 			fastreboot_onpanic_cmdline[cmdlen] = 0;
1068 	}
1069 }
1070 
1071 
1072 #ifndef	__xpv
1073 /*
1074  * Construct boot command line for Fast Reboot
1075  */
1076 static void
1077 build_fastboot_cmdline(void)
1078 {
1079 	saved_cmdline_len =  strlen(xbootp->bi_cmdline) + 1;
1080 	if (saved_cmdline_len > FASTBOOT_SAVED_CMDLINE_LEN) {
1081 		DBG(saved_cmdline_len);
1082 		DBG_MSG("Command line too long: clearing fastreboot_capable\n");
1083 		fastreboot_capable = 0;
1084 	} else {
1085 		bcopy((void *)(xbootp->bi_cmdline), (void *)saved_cmdline,
1086 		    saved_cmdline_len);
1087 		saved_cmdline[saved_cmdline_len - 1] = '\0';
1088 		build_panic_cmdline(saved_cmdline, saved_cmdline_len - 1);
1089 	}
1090 }
1091 
1092 /*
1093  * Save memory layout, disk drive information, unix and boot archive sizes for
1094  * Fast Reboot.
1095  */
1096 static void
1097 save_boot_info(multiboot_info_t *mbi, struct xboot_info *xbi)
1098 {
1099 	struct boot_modules *modp;
1100 	int i;
1101 
1102 	bcopy(mbi, &saved_mbi, sizeof (multiboot_info_t));
1103 	if (mbi->mmap_length > sizeof (saved_mmap)) {
1104 		DBG_MSG("mbi->mmap_length too big: clearing "
1105 		    "fastreboot_capable\n");
1106 		fastreboot_capable = 0;
1107 	} else {
1108 		bcopy((void *)(uintptr_t)mbi->mmap_addr, (void *)saved_mmap,
1109 		    mbi->mmap_length);
1110 	}
1111 
1112 	if ((mbi->flags & MB_INFO_DRIVE_INFO) != 0) {
1113 		if (mbi->drives_length > sizeof (saved_drives)) {
1114 			DBG(mbi->drives_length);
1115 			DBG_MSG("mbi->drives_length too big: clearing "
1116 			    "fastreboot_capable\n");
1117 			fastreboot_capable = 0;
1118 		} else {
1119 			bcopy((void *)(uintptr_t)mbi->drives_addr,
1120 			    (void *)saved_drives, mbi->drives_length);
1121 		}
1122 	} else {
1123 		saved_mbi.drives_length = 0;
1124 		saved_mbi.drives_addr = NULL;
1125 	}
1126 
1127 	/*
1128 	 * Current file sizes.  Used by fastboot.c to figure out how much
1129 	 * memory to reserve for panic reboot.
1130 	 * Use the module list from the dboot-constructed xboot_info
1131 	 * instead of the list referenced by the multiboot structure
1132 	 * because that structure may not be addressable now.
1133 	 */
1134 	saved_file_size[FASTBOOT_NAME_UNIX] = FOUR_MEG - PAGESIZE;
1135 	for (i = 0, modp = (struct boot_modules *)(uintptr_t)xbi->bi_modules;
1136 	    i < xbi->bi_module_cnt; i++, modp++) {
1137 		saved_file_size[FASTBOOT_NAME_BOOTARCHIVE] += modp->bm_size;
1138 	}
1139 }
1140 #endif	/* __xpv */
1141 
1142 
1143 /*
1144  * 1st pass at building the table of boot properties. This includes:
1145  * - values set on the command line: -B a=x,b=y,c=z ....
1146  * - known values we just compute (ie. from xbootp)
1147  * - values from /boot/solaris/bootenv.rc (ie. eeprom(1m) values)
1148  *
1149  * the grub command line looked like:
1150  * kernel boot-file [-B prop=value[,prop=value]...] [boot-args]
1151  *
1152  * whoami is the same as boot-file
1153  */
1154 static void
1155 build_boot_properties(void)
1156 {
1157 	char *name;
1158 	int name_len;
1159 	char *value;
1160 	int value_len;
1161 	struct boot_modules *bm;
1162 	char *propbuf;
1163 	int quoted = 0;
1164 	int boot_arg_len;
1165 #ifndef __xpv
1166 	static int stdout_val = 0;
1167 	uchar_t boot_device;
1168 	char str[3];
1169 	multiboot_info_t *mbi;
1170 	int netboot;
1171 	struct sol_netinfo *sip;
1172 #endif
1173 
1174 	/*
1175 	 * These have to be done first, so that kobj_mount_root() works
1176 	 */
1177 	DBG_MSG("Building boot properties\n");
1178 	propbuf = do_bsys_alloc(NULL, NULL, MMU_PAGESIZE, 0);
1179 	DBG((uintptr_t)propbuf);
1180 	if (xbootp->bi_module_cnt > 0) {
1181 		bm = xbootp->bi_modules;
1182 		bsetprop64("ramdisk_start", (uint64_t)(uintptr_t)bm->bm_addr);
1183 		bsetprop64("ramdisk_end", (uint64_t)(uintptr_t)bm->bm_addr +
1184 		    bm->bm_size);
1185 	}
1186 
1187 	DBG_MSG("Parsing command line for boot properties\n");
1188 	value = xbootp->bi_cmdline;
1189 
1190 	/*
1191 	 * allocate memory to collect boot_args into
1192 	 */
1193 	boot_arg_len = strlen(xbootp->bi_cmdline) + 1;
1194 	boot_args = do_bsys_alloc(NULL, NULL, boot_arg_len, MMU_PAGESIZE);
1195 	boot_args[0] = 0;
1196 	boot_arg_len = 0;
1197 
1198 #ifdef __xpv
1199 	/*
1200 	 * Xen puts a lot of device information in front of the kernel name
1201 	 * let's grab them and make them boot properties.  The first
1202 	 * string w/o an "=" in it will be the boot-file property.
1203 	 */
1204 	(void) strcpy(namebuf, "xpv-");
1205 	for (;;) {
1206 		/*
1207 		 * get to next property
1208 		 */
1209 		while (ISSPACE(*value))
1210 			++value;
1211 		name = value;
1212 		/*
1213 		 * look for an "="
1214 		 */
1215 		while (*value && !ISSPACE(*value) && *value != '=') {
1216 			value++;
1217 		}
1218 		if (*value != '=') { /* no "=" in the property */
1219 			value = name;
1220 			break;
1221 		}
1222 		name_len = value - name;
1223 		value_len = 0;
1224 		/*
1225 		 * skip over the "="
1226 		 */
1227 		value++;
1228 		while (value[value_len] && !ISSPACE(value[value_len])) {
1229 			++value_len;
1230 		}
1231 		/*
1232 		 * build property name with "xpv-" prefix
1233 		 */
1234 		if (name_len + 4 > 32) { /* skip if name too long */
1235 			value += value_len;
1236 			continue;
1237 		}
1238 		bcopy(name, &namebuf[4], name_len);
1239 		name_len += 4;
1240 		namebuf[name_len] = 0;
1241 		bcopy(value, propbuf, value_len);
1242 		propbuf[value_len] = 0;
1243 		bsetprops(namebuf, propbuf);
1244 
1245 		/*
1246 		 * xpv-root is set to the logical disk name of the xen
1247 		 * VBD when booting from a disk-based filesystem.
1248 		 */
1249 		if (strcmp(namebuf, "xpv-root") == 0)
1250 			xen_vbdroot_props(propbuf);
1251 		/*
1252 		 * While we're here, if we have a "xpv-nfsroot" property
1253 		 * then we need to set "fstype" to "nfs" so we mount
1254 		 * our root from the nfs server.  Also parse the xpv-nfsroot
1255 		 * property to create the properties that nfs_mountroot will
1256 		 * need to find the root and mount it.
1257 		 */
1258 		if (strcmp(namebuf, "xpv-nfsroot") == 0)
1259 			xen_nfsroot_props(propbuf);
1260 
1261 		if (strcmp(namebuf, "xpv-ip") == 0)
1262 			xen_ip_props(propbuf);
1263 		value += value_len;
1264 	}
1265 #endif
1266 
1267 	while (ISSPACE(*value))
1268 		++value;
1269 	/*
1270 	 * value now points at the boot-file
1271 	 */
1272 	value_len = 0;
1273 	while (value[value_len] && !ISSPACE(value[value_len]))
1274 		++value_len;
1275 	if (value_len > 0) {
1276 		whoami = propbuf;
1277 		bcopy(value, whoami, value_len);
1278 		whoami[value_len] = 0;
1279 		bsetprops("boot-file", whoami);
1280 		/*
1281 		 * strip leading path stuff from whoami, so running from
1282 		 * PXE/miniroot makes sense.
1283 		 */
1284 		if (strstr(whoami, "/platform/") != NULL)
1285 			whoami = strstr(whoami, "/platform/");
1286 		bsetprops("whoami", whoami);
1287 	}
1288 
1289 	/*
1290 	 * Values forcibly set boot properties on the command line via -B.
1291 	 * Allow use of quotes in values. Other stuff goes on kernel
1292 	 * command line.
1293 	 */
1294 	name = value + value_len;
1295 	while (*name != 0) {
1296 		/*
1297 		 * anything not " -B" is copied to the command line
1298 		 */
1299 		if (!ISSPACE(name[0]) || name[1] != '-' || name[2] != 'B') {
1300 			boot_args[boot_arg_len++] = *name;
1301 			boot_args[boot_arg_len] = 0;
1302 			++name;
1303 			continue;
1304 		}
1305 
1306 		/*
1307 		 * skip the " -B" and following white space
1308 		 */
1309 		name += 3;
1310 		while (ISSPACE(*name))
1311 			++name;
1312 		while (*name && !ISSPACE(*name)) {
1313 			value = strstr(name, "=");
1314 			if (value == NULL)
1315 				break;
1316 			name_len = value - name;
1317 			++value;
1318 			value_len = 0;
1319 			quoted = 0;
1320 			for (; ; ++value_len) {
1321 				if (!value[value_len])
1322 					break;
1323 
1324 				/*
1325 				 * is this value quoted?
1326 				 */
1327 				if (value_len == 0 &&
1328 				    (value[0] == '\'' || value[0] == '"')) {
1329 					quoted = value[0];
1330 					++value_len;
1331 				}
1332 
1333 				/*
1334 				 * In the quote accept any character,
1335 				 * but look for ending quote.
1336 				 */
1337 				if (quoted) {
1338 					if (value[value_len] == quoted)
1339 						quoted = 0;
1340 					continue;
1341 				}
1342 
1343 				/*
1344 				 * a comma or white space ends the value
1345 				 */
1346 				if (value[value_len] == ',' ||
1347 				    ISSPACE(value[value_len]))
1348 					break;
1349 			}
1350 
1351 			if (value_len == 0) {
1352 				bsetprop(name, name_len, "true", 5);
1353 			} else {
1354 				char *v = value;
1355 				int l = value_len;
1356 				if (v[0] == v[l - 1] &&
1357 				    (v[0] == '\'' || v[0] == '"')) {
1358 					++v;
1359 					l -= 2;
1360 				}
1361 				bcopy(v, propbuf, l);
1362 				propbuf[l] = '\0';
1363 				bsetprop(name, name_len, propbuf,
1364 				    l + 1);
1365 			}
1366 			name = value + value_len;
1367 			while (*name == ',')
1368 				++name;
1369 		}
1370 	}
1371 
1372 	/*
1373 	 * set boot-args property
1374 	 * 1275 name is bootargs, so set
1375 	 * that too
1376 	 */
1377 	bsetprops("boot-args", boot_args);
1378 	bsetprops("bootargs", boot_args);
1379 
1380 #ifndef __xpv
1381 	/*
1382 	 * set the BIOS boot device from GRUB
1383 	 */
1384 	netboot = 0;
1385 	mbi = xbootp->bi_mb_info;
1386 
1387 	/*
1388 	 * Build boot command line for Fast Reboot
1389 	 */
1390 	build_fastboot_cmdline();
1391 
1392 	/*
1393 	 * Save various boot information for Fast Reboot
1394 	 */
1395 	save_boot_info(mbi, xbootp);
1396 
1397 	if (mbi != NULL && mbi->flags & MB_INFO_BOOTDEV) {
1398 		boot_device = mbi->boot_device >> 24;
1399 		if (boot_device == 0x20)
1400 			netboot++;
1401 		str[0] = (boot_device >> 4) + '0';
1402 		str[1] = (boot_device & 0xf) + '0';
1403 		str[2] = 0;
1404 		bsetprops("bios-boot-device", str);
1405 	} else {
1406 		netboot = 1;
1407 	}
1408 
1409 	/*
1410 	 * In the netboot case, drives_info is overloaded with the dhcp ack.
1411 	 * This is not multiboot compliant and requires special pxegrub!
1412 	 */
1413 	if (netboot && mbi->drives_length != 0) {
1414 		sip = (struct sol_netinfo *)(uintptr_t)mbi->drives_addr;
1415 		if (sip->sn_infotype == SN_TYPE_BOOTP)
1416 			bsetprop("bootp-response", sizeof ("bootp-response"),
1417 			    (void *)(uintptr_t)mbi->drives_addr,
1418 			    mbi->drives_length);
1419 		else if (sip->sn_infotype == SN_TYPE_RARP)
1420 			setup_rarp_props(sip);
1421 	}
1422 	bsetprop("stdout", strlen("stdout"),
1423 	    &stdout_val, sizeof (stdout_val));
1424 #endif /* __xpv */
1425 
1426 	/*
1427 	 * more conjured up values for made up things....
1428 	 */
1429 #if defined(__xpv)
1430 	bsetprops("mfg-name", "i86xpv");
1431 	bsetprops("impl-arch-name", "i86xpv");
1432 #else
1433 	bsetprops("mfg-name", "i86pc");
1434 	bsetprops("impl-arch-name", "i86pc");
1435 #endif
1436 
1437 	/*
1438 	 * Build firmware-provided system properties
1439 	 */
1440 	build_firmware_properties();
1441 
1442 	/*
1443 	 * XXPV
1444 	 *
1445 	 * Find out what these are:
1446 	 * - cpuid_feature_ecx_include
1447 	 * - cpuid_feature_ecx_exclude
1448 	 * - cpuid_feature_edx_include
1449 	 * - cpuid_feature_edx_exclude
1450 	 *
1451 	 * Find out what these are in multiboot:
1452 	 * - netdev-path
1453 	 * - fstype
1454 	 */
1455 }
1456 
1457 #ifdef __xpv
1458 /*
1459  * Under the Hypervisor, memory usable for DMA may be scarce. One
1460  * very likely large pool of DMA friendly memory is occupied by
1461  * the boot_archive, as it was loaded by grub into low MFNs.
1462  *
1463  * Here we free up that memory by copying the boot archive to what are
1464  * likely higher MFN pages and then swapping the mfn/pfn mappings.
1465  */
1466 #define	PFN_2GIG	0x80000
1467 static void
1468 relocate_boot_archive(void)
1469 {
1470 	mfn_t max_mfn = HYPERVISOR_memory_op(XENMEM_maximum_ram_page, NULL);
1471 	struct boot_modules *bm = xbootp->bi_modules;
1472 	uintptr_t va;
1473 	pfn_t va_pfn;
1474 	mfn_t va_mfn;
1475 	caddr_t copy;
1476 	pfn_t copy_pfn;
1477 	mfn_t copy_mfn;
1478 	size_t	len;
1479 	int slop;
1480 	int total = 0;
1481 	int relocated = 0;
1482 	int mmu_update_return;
1483 	mmu_update_t t[2];
1484 	x86pte_t pte;
1485 
1486 	/*
1487 	 * If all MFN's are below 2Gig, don't bother doing this.
1488 	 */
1489 	if (max_mfn < PFN_2GIG)
1490 		return;
1491 	if (xbootp->bi_module_cnt < 1) {
1492 		DBG_MSG("no boot_archive!");
1493 		return;
1494 	}
1495 
1496 	DBG_MSG("moving boot_archive to high MFN memory\n");
1497 	va = (uintptr_t)bm->bm_addr;
1498 	len = bm->bm_size;
1499 	slop = va & MMU_PAGEOFFSET;
1500 	if (slop) {
1501 		va += MMU_PAGESIZE - slop;
1502 		len -= MMU_PAGESIZE - slop;
1503 	}
1504 	len = P2ALIGN(len, MMU_PAGESIZE);
1505 
1506 	/*
1507 	 * Go through all boot_archive pages, swapping any low MFN pages
1508 	 * with memory at next_phys.
1509 	 */
1510 	while (len != 0) {
1511 		++total;
1512 		va_pfn = mmu_btop(va - ONE_GIG);
1513 		va_mfn = mfn_list[va_pfn];
1514 		if (mfn_list[va_pfn] < PFN_2GIG) {
1515 			copy = kbm_remap_window(next_phys, 1);
1516 			bcopy((void *)va, copy, MMU_PAGESIZE);
1517 			copy_pfn = mmu_btop(next_phys);
1518 			copy_mfn = mfn_list[copy_pfn];
1519 
1520 			pte = mfn_to_ma(copy_mfn) | PT_NOCONSIST | PT_VALID;
1521 			if (HYPERVISOR_update_va_mapping(va, pte,
1522 			    UVMF_INVLPG | UVMF_LOCAL))
1523 				bop_panic("relocate_boot_archive():  "
1524 				    "HYPERVISOR_update_va_mapping() failed");
1525 
1526 			mfn_list[va_pfn] = copy_mfn;
1527 			mfn_list[copy_pfn] = va_mfn;
1528 
1529 			t[0].ptr = mfn_to_ma(copy_mfn) | MMU_MACHPHYS_UPDATE;
1530 			t[0].val = va_pfn;
1531 			t[1].ptr = mfn_to_ma(va_mfn) | MMU_MACHPHYS_UPDATE;
1532 			t[1].val = copy_pfn;
1533 			if (HYPERVISOR_mmu_update(t, 2, &mmu_update_return,
1534 			    DOMID_SELF) != 0 || mmu_update_return != 2)
1535 				bop_panic("relocate_boot_archive():  "
1536 				    "HYPERVISOR_mmu_update() failed");
1537 
1538 			next_phys += MMU_PAGESIZE;
1539 			++relocated;
1540 		}
1541 		len -= MMU_PAGESIZE;
1542 		va += MMU_PAGESIZE;
1543 	}
1544 	DBG_MSG("Relocated pages:\n");
1545 	DBG(relocated);
1546 	DBG_MSG("Out of total pages:\n");
1547 	DBG(total);
1548 }
1549 #endif /* __xpv */
1550 
1551 #if !defined(__xpv)
1552 /*
1553  * Install a temporary IDT that lets us catch errors in the boot time code.
1554  * We shouldn't get any faults at all while this is installed, so we'll
1555  * just generate a traceback and exit.
1556  */
1557 #ifdef __amd64
1558 static const int bcode_sel = B64CODE_SEL;
1559 #else
1560 static const int bcode_sel = B32CODE_SEL;
1561 #endif
1562 
1563 /*
1564  * simple description of a stack frame (args are 32 bit only currently)
1565  */
1566 typedef struct bop_frame {
1567 	struct bop_frame *old_frame;
1568 	pc_t retaddr;
1569 	long arg[1];
1570 } bop_frame_t;
1571 
1572 void
1573 bop_traceback(bop_frame_t *frame)
1574 {
1575 	pc_t pc;
1576 	int cnt;
1577 	int a;
1578 	char *ksym;
1579 	ulong_t off;
1580 
1581 	bop_printf(NULL, "Stack traceback:\n");
1582 	for (cnt = 0; cnt < 30; ++cnt) {	/* up to 30 frames */
1583 		pc = frame->retaddr;
1584 		if (pc == 0)
1585 			break;
1586 		ksym = kobj_getsymname(pc, &off);
1587 		if (ksym)
1588 			bop_printf(NULL, "  %s+%lx", ksym, off);
1589 		else
1590 			bop_printf(NULL, "  0x%lx", pc);
1591 
1592 		frame = frame->old_frame;
1593 		if (frame == 0) {
1594 			bop_printf(NULL, "\n");
1595 			break;
1596 		}
1597 		for (a = 0; a < 6; ++a) {	/* try for 6 args */
1598 #if defined(__i386)
1599 			if ((void *)&frame->arg[a] == (void *)frame->old_frame)
1600 				break;
1601 			if (a == 0)
1602 				bop_printf(NULL, "(");
1603 			else
1604 				bop_printf(NULL, ",");
1605 			bop_printf(NULL, "0x%lx", frame->arg[a]);
1606 #endif
1607 		}
1608 		bop_printf(NULL, ")\n");
1609 	}
1610 }
1611 
1612 struct trapframe {
1613 	ulong_t frame_ptr;	/* %[er]bp pushed by our code */
1614 	ulong_t error_code;	/* optional */
1615 	ulong_t inst_ptr;
1616 	ulong_t code_seg;
1617 	ulong_t flags_reg;
1618 #ifdef __amd64
1619 	ulong_t stk_ptr;
1620 	ulong_t stk_seg;
1621 #endif
1622 };
1623 
1624 void
1625 bop_trap(struct trapframe *tf)
1626 {
1627 	bop_frame_t fakeframe;
1628 	static int depth = 0;
1629 
1630 	/*
1631 	 * Check for an infinite loop of traps.
1632 	 */
1633 	if (++depth > 2)
1634 		bop_panic("Nested trap");
1635 
1636 	/*
1637 	 * adjust the tf for optional error_code by detecting the code selector
1638 	 */
1639 	if (tf->code_seg != bcode_sel)
1640 		tf = (struct trapframe *)((uintptr_t)tf - sizeof (ulong_t));
1641 
1642 	bop_printf(NULL, "Unexpected trap\n");
1643 	bop_printf(NULL, "instruction pointer  0x%lx\n", tf->inst_ptr);
1644 	bop_printf(NULL, "error code, optional 0x%lx\n",
1645 	    tf->error_code & 0xffffffff);
1646 	bop_printf(NULL, "code segment         0x%lx\n", tf->code_seg & 0xffff);
1647 	bop_printf(NULL, "flags register       0x%lx\n", tf->flags_reg);
1648 #ifdef __amd64
1649 	bop_printf(NULL, "return %%rsp         0x%lx\n", tf->stk_ptr);
1650 	bop_printf(NULL, "return %%ss          0x%lx\n", tf->stk_seg & 0xffff);
1651 #endif
1652 	fakeframe.old_frame = (bop_frame_t *)tf->frame_ptr;
1653 	fakeframe.retaddr = (pc_t)tf->inst_ptr;
1654 	bop_printf(NULL, "Attempting stack backtrace:\n");
1655 	bop_traceback(&fakeframe);
1656 	bop_panic("unexpected trap in early boot");
1657 }
1658 
1659 extern void bop_trap_handler(void);
1660 
1661 static gate_desc_t *bop_idt;
1662 
1663 static desctbr_t bop_idt_info;
1664 
1665 static void
1666 bop_idt_init(void)
1667 {
1668 	int t;
1669 
1670 	bop_idt = (gate_desc_t *)
1671 	    do_bsys_alloc(NULL, NULL, MMU_PAGESIZE, MMU_PAGESIZE);
1672 	bzero(bop_idt, MMU_PAGESIZE);
1673 	for (t = 0; t < NIDT; ++t) {
1674 		/*
1675 		 * Note that since boot runs without a TSS, the
1676 		 * double fault handler cannot use an alternate stack
1677 		 * (64-bit) or a task gate (32-bit).
1678 		 */
1679 		set_gatesegd(&bop_idt[t], &bop_trap_handler, bcode_sel,
1680 		    SDT_SYSIGT, TRP_KPL, 0);
1681 	}
1682 	bop_idt_info.dtr_limit = (NIDT * sizeof (gate_desc_t)) - 1;
1683 	bop_idt_info.dtr_base = (uintptr_t)bop_idt;
1684 	wr_idtr(&bop_idt_info);
1685 }
1686 #endif	/* !defined(__xpv) */
1687 
1688 /*
1689  * This is where we enter the kernel. It dummies up the boot_ops and
1690  * boot_syscalls vectors and jumps off to _kobj_boot()
1691  */
1692 void
1693 _start(struct xboot_info *xbp)
1694 {
1695 	bootops_t *bops = &bootop;
1696 	extern void _kobj_boot();
1697 
1698 	/*
1699 	 * 1st off - initialize the console for any error messages
1700 	 */
1701 	xbootp = xbp;
1702 #ifdef __xpv
1703 	HYPERVISOR_shared_info = (void *)xbootp->bi_shared_info;
1704 	xen_info = xbootp->bi_xen_start_info;
1705 #endif
1706 
1707 #ifndef __xpv
1708 	if (*((uint32_t *)(FASTBOOT_SWTCH_PA + FASTBOOT_STACK_OFFSET)) ==
1709 	    FASTBOOT_MAGIC) {
1710 		post_fastreboot = 1;
1711 		*((uint32_t *)(FASTBOOT_SWTCH_PA + FASTBOOT_STACK_OFFSET)) = 0;
1712 	}
1713 #endif
1714 
1715 	bcons_init((void *)xbootp->bi_cmdline);
1716 	have_console = 1;
1717 
1718 	/*
1719 	 * enable debugging
1720 	 */
1721 	if (strstr((char *)xbootp->bi_cmdline, "kbm_debug"))
1722 		kbm_debug = 1;
1723 
1724 	DBG_MSG("\n\n*** Entered Solaris in _start() cmdline is: ");
1725 	DBG_MSG((char *)xbootp->bi_cmdline);
1726 	DBG_MSG("\n\n\n");
1727 
1728 	/*
1729 	 * physavail is no longer used by startup
1730 	 */
1731 	bm.physinstalled = xbp->bi_phys_install;
1732 	bm.pcimem = xbp->bi_pcimem;
1733 	bm.rsvdmem = xbp->bi_rsvdmem;
1734 	bm.physavail = NULL;
1735 
1736 	/*
1737 	 * initialize the boot time allocator
1738 	 */
1739 	next_phys = xbootp->bi_next_paddr;
1740 	DBG(next_phys);
1741 	next_virt = (uintptr_t)xbootp->bi_next_vaddr;
1742 	DBG(next_virt);
1743 	DBG_MSG("Initializing boot time memory management...");
1744 #ifdef __xpv
1745 	{
1746 		xen_platform_parameters_t p;
1747 
1748 		/* This call shouldn't fail, dboot already did it once. */
1749 		(void) HYPERVISOR_xen_version(XENVER_platform_parameters, &p);
1750 		mfn_to_pfn_mapping = (pfn_t *)(xen_virt_start = p.virt_start);
1751 		DBG(xen_virt_start);
1752 	}
1753 #endif
1754 	kbm_init(xbootp);
1755 	DBG_MSG("done\n");
1756 
1757 	/*
1758 	 * Fill in the bootops vector
1759 	 */
1760 	bops->bsys_version = BO_VERSION;
1761 	bops->boot_mem = &bm;
1762 	bops->bsys_alloc = do_bsys_alloc;
1763 	bops->bsys_free = do_bsys_free;
1764 	bops->bsys_getproplen = do_bsys_getproplen;
1765 	bops->bsys_getprop = do_bsys_getprop;
1766 	bops->bsys_nextprop = do_bsys_nextprop;
1767 	bops->bsys_printf = bop_printf;
1768 	bops->bsys_doint = do_bsys_doint;
1769 
1770 	/*
1771 	 * BOP_EALLOC() is no longer needed
1772 	 */
1773 	bops->bsys_ealloc = do_bsys_ealloc;
1774 
1775 #ifdef __xpv
1776 	/*
1777 	 * On domain 0 we need to free up some physical memory that is
1778 	 * usable for DMA. Since GRUB loaded the boot_archive, it is
1779 	 * sitting in low MFN memory. We'll relocated the boot archive
1780 	 * pages to high PFN memory.
1781 	 */
1782 	if (DOMAIN_IS_INITDOMAIN(xen_info))
1783 		relocate_boot_archive();
1784 #endif
1785 
1786 #ifndef __xpv
1787 	/*
1788 	 * Install an IDT to catch early pagefaults (shouldn't have any).
1789 	 * Also needed for kmdb.
1790 	 */
1791 	bop_idt_init();
1792 #endif
1793 
1794 	/*
1795 	 * Start building the boot properties from the command line
1796 	 */
1797 	DBG_MSG("Initializing boot properties:\n");
1798 	build_boot_properties();
1799 
1800 	if (strstr((char *)xbootp->bi_cmdline, "prom_debug") || kbm_debug) {
1801 		char *name;
1802 		char *value;
1803 		char *cp;
1804 		int len;
1805 
1806 		value = do_bsys_alloc(NULL, NULL, MMU_PAGESIZE, MMU_PAGESIZE);
1807 		bop_printf(NULL, "\nBoot properties:\n");
1808 		name = "";
1809 		while ((name = do_bsys_nextprop(NULL, name)) != NULL) {
1810 			bop_printf(NULL, "\t0x%p %s = ", (void *)name, name);
1811 			(void) do_bsys_getprop(NULL, name, value);
1812 			len = do_bsys_getproplen(NULL, name);
1813 			bop_printf(NULL, "len=%d ", len);
1814 			value[len] = 0;
1815 			for (cp = value; *cp; ++cp) {
1816 				if (' ' <= *cp && *cp <= '~')
1817 					bop_printf(NULL, "%c", *cp);
1818 				else
1819 					bop_printf(NULL, "-0x%x-", *cp);
1820 			}
1821 			bop_printf(NULL, "\n");
1822 		}
1823 	}
1824 
1825 	/*
1826 	 * jump into krtld...
1827 	 */
1828 	_kobj_boot(&bop_sysp, NULL, bops, NULL);
1829 }
1830 
1831 
1832 /*ARGSUSED*/
1833 static caddr_t
1834 no_more_alloc(bootops_t *bop, caddr_t virthint, size_t size, int align)
1835 {
1836 	panic("Attempt to bsys_alloc() too late\n");
1837 	return (NULL);
1838 }
1839 
1840 /*ARGSUSED*/
1841 static void
1842 no_more_free(bootops_t *bop, caddr_t virt, size_t size)
1843 {
1844 	panic("Attempt to bsys_free() too late\n");
1845 }
1846 
1847 void
1848 bop_no_more_mem(void)
1849 {
1850 	DBG(total_bop_alloc_scratch);
1851 	DBG(total_bop_alloc_kernel);
1852 	bootops->bsys_alloc = no_more_alloc;
1853 	bootops->bsys_free = no_more_free;
1854 }
1855 
1856 
1857 #ifndef __xpv
1858 /*
1859  * Set ACPI firmware properties
1860  */
1861 
1862 static caddr_t
1863 vmap_phys(size_t length, paddr_t pa)
1864 {
1865 	paddr_t	start, end;
1866 	caddr_t	va;
1867 	size_t	len, page;
1868 
1869 	start = P2ALIGN(pa, MMU_PAGESIZE);
1870 	end = P2ROUNDUP(pa + length, MMU_PAGESIZE);
1871 	len = end - start;
1872 	va = (caddr_t)alloc_vaddr(len, MMU_PAGESIZE);
1873 	for (page = 0; page < len; page += MMU_PAGESIZE)
1874 		kbm_map((uintptr_t)va + page, start + page, 0, 0);
1875 	return (va + (pa & MMU_PAGEOFFSET));
1876 }
1877 
1878 static uint8_t
1879 checksum_table(uint8_t *tp, size_t len)
1880 {
1881 	uint8_t sum = 0;
1882 
1883 	while (len-- > 0)
1884 		sum += *tp++;
1885 
1886 	return (sum);
1887 }
1888 
1889 static int
1890 valid_rsdp(struct rsdp *rp)
1891 {
1892 
1893 	/* validate the V1.x checksum */
1894 	if (checksum_table((uint8_t *)&rp->v1, sizeof (struct rsdp_v1)) != 0)
1895 		return (0);
1896 
1897 	/* If pre-ACPI 2.0, this is a valid RSDP */
1898 	if (rp->v1.revision < 2)
1899 		return (1);
1900 
1901 	/* validate the V2.x checksum */
1902 	if (checksum_table((uint8_t *)rp, sizeof (struct rsdp)) != 0)
1903 		return (0);
1904 
1905 	return (1);
1906 }
1907 
1908 /*
1909  * Scan memory range for an RSDP;
1910  * see ACPI 3.0 Spec, 5.2.5.1
1911  */
1912 static struct rsdp *
1913 scan_rsdp(paddr_t start, paddr_t end)
1914 {
1915 	size_t len  = end - start + 1;
1916 	caddr_t ptr;
1917 
1918 	ptr = vmap_phys(len, start);
1919 	while (len > 0) {
1920 		if (strncmp(ptr, ACPI_RSDP_SIG, ACPI_RSDP_SIG_LEN) == 0)
1921 			if (valid_rsdp((struct rsdp *)ptr))
1922 				return ((struct rsdp *)ptr);
1923 		ptr += 16;
1924 		len -= 16;
1925 	}
1926 
1927 	return (NULL);
1928 }
1929 
1930 /*
1931  * Refer to ACPI 3.0 Spec, section 5.2.5.1 to understand this function
1932  */
1933 static struct rsdp *
1934 find_rsdp() {
1935 	struct rsdp *rsdp;
1936 	uint16_t *ebda_seg;
1937 	paddr_t  ebda_addr;
1938 
1939 	/*
1940 	 * Get the EBDA segment and scan the first 1K
1941 	 */
1942 	ebda_seg = (uint16_t *)vmap_phys(sizeof (uint16_t), ACPI_EBDA_SEG_ADDR);
1943 	ebda_addr = *ebda_seg << 4;
1944 	rsdp = scan_rsdp(ebda_addr, ebda_addr + ACPI_EBDA_LEN - 1);
1945 	if (rsdp == NULL)
1946 		/* if EBDA doesn't contain RSDP, look in BIOS memory */
1947 		rsdp = scan_rsdp(0xe0000, 0xfffff);
1948 	return (rsdp);
1949 }
1950 
1951 static struct table_header *
1952 map_fw_table(paddr_t table_addr)
1953 {
1954 	struct table_header *tp;
1955 	size_t len = MAX(sizeof (struct table_header), MMU_PAGESIZE);
1956 
1957 	/*
1958 	 * Map at least a page; if the table is larger than this, remap it
1959 	 */
1960 	tp = (struct table_header *)vmap_phys(len, table_addr);
1961 	if (tp->len > len)
1962 		tp = (struct table_header *)vmap_phys(tp->len, table_addr);
1963 	return (tp);
1964 }
1965 
1966 static struct table_header *
1967 find_fw_table(char *signature)
1968 {
1969 	static int revision = 0;
1970 	static struct xsdt *xsdt;
1971 	static int len;
1972 	paddr_t xsdt_addr;
1973 	struct rsdp *rsdp;
1974 	struct table_header *tp;
1975 	paddr_t table_addr;
1976 	int	n;
1977 
1978 	if (strlen(signature) != ACPI_TABLE_SIG_LEN)
1979 		return (NULL);
1980 
1981 	/*
1982 	 * Reading the ACPI 3.0 Spec, section 5.2.5.3 will help
1983 	 * understand this code.  If we haven't already found the RSDT/XSDT,
1984 	 * revision will be 0. Find the RSDP and check the revision
1985 	 * to find out whether to use the RSDT or XSDT.  If revision is
1986 	 * 0 or 1, use the RSDT and set internal revision to 1; if it is 2,
1987 	 * use the XSDT.  If the XSDT address is 0, though, fall back to
1988 	 * revision 1 and use the RSDT.
1989 	 */
1990 	if (revision == 0) {
1991 		if ((rsdp = (struct rsdp *)find_rsdp()) != NULL) {
1992 			revision = rsdp->v1.revision;
1993 			switch (revision) {
1994 			case 2:
1995 				/*
1996 				 * Use the XSDT unless BIOS is buggy and
1997 				 * claims to be rev 2 but has a null XSDT
1998 				 * address
1999 				 */
2000 				xsdt_addr = rsdp->xsdt;
2001 				if (xsdt_addr != 0)
2002 					break;
2003 				/* FALLTHROUGH */
2004 			case 0:
2005 				/* treat RSDP rev 0 as revision 1 internally */
2006 				revision = 1;
2007 				/* FALLTHROUGH */
2008 			case 1:
2009 				/* use the RSDT for rev 0/1 */
2010 				xsdt_addr = rsdp->v1.rsdt;
2011 				break;
2012 			default:
2013 				/* unknown revision */
2014 				revision = 0;
2015 				break;
2016 			}
2017 		}
2018 		if (revision == 0)
2019 			return (NULL);
2020 
2021 		/* cache the XSDT info */
2022 		xsdt = (struct xsdt *)map_fw_table(xsdt_addr);
2023 		len = (xsdt->hdr.len - sizeof (xsdt->hdr)) /
2024 		    ((revision == 1) ? sizeof (uint32_t) : sizeof (uint64_t));
2025 	}
2026 
2027 	/*
2028 	 * Scan the table headers looking for a signature match
2029 	 */
2030 	for (n = 0; n < len; n++) {
2031 		table_addr = (revision == 1) ? xsdt->p.r[n] : xsdt->p.x[n];
2032 		if (table_addr == 0)
2033 			continue;
2034 		tp = map_fw_table(table_addr);
2035 		if (strncmp(tp->sig, signature, ACPI_TABLE_SIG_LEN) == 0) {
2036 			return (tp);
2037 		}
2038 	}
2039 	return (NULL);
2040 }
2041 
2042 static void
2043 process_madt(struct madt *tp)
2044 {
2045 	struct madt_processor *cpu, *end;
2046 	uint32_t cpu_count = 0;
2047 	uint8_t cpu_apicid_array[UINT8_MAX + 1];
2048 
2049 	if (tp != NULL) {
2050 		/*
2051 		 * Determine number of CPUs and keep track of "final" APIC ID
2052 		 * for each CPU by walking through ACPI MADT processor list
2053 		 */
2054 		end = (struct madt_processor *)(tp->hdr.len + (uintptr_t)tp);
2055 		cpu = tp->list;
2056 		while (cpu < end) {
2057 			if (cpu->type == MADT_PROCESSOR) {
2058 				if (cpu->flags & 1) {
2059 					if (cpu_count < UINT8_MAX)
2060 						cpu_apicid_array[cpu_count] =
2061 						    cpu->apic_id;
2062 					cpu_count++;
2063 				}
2064 			}
2065 
2066 			cpu = (struct madt_processor *)
2067 			    (cpu->len + (uintptr_t)cpu);
2068 		}
2069 
2070 		/*
2071 		 * Make boot property for array of "final" APIC IDs for each
2072 		 * CPU
2073 		 */
2074 		bsetprop(BP_CPU_APICID_ARRAY, strlen(BP_CPU_APICID_ARRAY),
2075 		    cpu_apicid_array, cpu_count * sizeof (uint8_t));
2076 	}
2077 
2078 	/*
2079 	 * User-set boot-ncpus overrides firmware count
2080 	 */
2081 	if (do_bsys_getproplen(NULL, "boot-ncpus") >= 0)
2082 		return;
2083 
2084 	/*
2085 	 * Set boot property for boot-ncpus to number of CPUs given in MADT
2086 	 * if user hasn't set the property already
2087 	 */
2088 	if (tp != NULL)
2089 		bsetpropsi("boot-ncpus", cpu_count);
2090 }
2091 
2092 static void
2093 process_srat(struct srat *tp)
2094 {
2095 	struct srat_item *item, *end;
2096 	int i;
2097 	int proc_num, mem_num;
2098 #pragma pack(1)
2099 	struct {
2100 		uint32_t domain;
2101 		uint32_t apic_id;
2102 		uint32_t sapic_id;
2103 	} processor;
2104 	struct {
2105 		uint32_t domain;
2106 		uint32_t x2apic_id;
2107 	} x2apic;
2108 	struct {
2109 		uint32_t domain;
2110 		uint64_t addr;
2111 		uint64_t length;
2112 		uint32_t flags;
2113 	} memory;
2114 #pragma pack()
2115 	char prop_name[30];
2116 
2117 	if (tp == NULL)
2118 		return;
2119 
2120 	proc_num = mem_num = 0;
2121 	end = (struct srat_item *)(tp->hdr.len + (uintptr_t)tp);
2122 	item = tp->list;
2123 	while (item < end) {
2124 		switch (item->type) {
2125 		case SRAT_PROCESSOR:
2126 			if (!(item->i.p.flags & SRAT_ENABLED))
2127 				break;
2128 			processor.domain = item->i.p.domain1;
2129 			for (i = 0; i < 3; i++)
2130 				processor.domain +=
2131 				    item->i.p.domain2[i] << ((i + 1) * 8);
2132 			processor.apic_id = item->i.p.apic_id;
2133 			processor.sapic_id = item->i.p.local_sapic_eid;
2134 			(void) snprintf(prop_name, 30, "acpi-srat-processor-%d",
2135 			    proc_num);
2136 			bsetprop(prop_name, strlen(prop_name), &processor,
2137 			    sizeof (processor));
2138 			proc_num++;
2139 			break;
2140 		case SRAT_MEMORY:
2141 			if (!(item->i.m.flags & SRAT_ENABLED))
2142 				break;
2143 			memory.domain = item->i.m.domain;
2144 			memory.addr = item->i.m.base_addr;
2145 			memory.length = item->i.m.len;
2146 			memory.flags = item->i.m.flags;
2147 			(void) snprintf(prop_name, 30, "acpi-srat-memory-%d",
2148 			    mem_num);
2149 			bsetprop(prop_name, strlen(prop_name), &memory,
2150 			    sizeof (memory));
2151 			mem_num++;
2152 			break;
2153 		case SRAT_X2APIC:
2154 			if (!(item->i.xp.flags & SRAT_ENABLED))
2155 				break;
2156 			x2apic.domain = item->i.xp.domain;
2157 			x2apic.x2apic_id = item->i.xp.x2apic_id;
2158 			(void) snprintf(prop_name, 30, "acpi-srat-processor-%d",
2159 			    proc_num);
2160 			bsetprop(prop_name, strlen(prop_name), &x2apic,
2161 			    sizeof (x2apic));
2162 			proc_num++;
2163 			break;
2164 		}
2165 
2166 		item = (struct srat_item *)
2167 		    (item->len + (caddr_t)item);
2168 	}
2169 }
2170 
2171 static void
2172 process_slit(struct slit *tp)
2173 {
2174 
2175 	/*
2176 	 * Check the number of localities; if it's too huge, we just
2177 	 * return and locality enumeration code will handle this later,
2178 	 * if possible.
2179 	 *
2180 	 * Note that the size of the table is the square of the
2181 	 * number of localities; if the number of localities exceeds
2182 	 * UINT16_MAX, the table size may overflow an int when being
2183 	 * passed to bsetprop() below.
2184 	 */
2185 	if (tp->number >= SLIT_LOCALITIES_MAX)
2186 		return;
2187 
2188 	bsetprop(SLIT_NUM_PROPNAME, strlen(SLIT_NUM_PROPNAME), &tp->number,
2189 	    sizeof (tp->number));
2190 	bsetprop(SLIT_PROPNAME, strlen(SLIT_PROPNAME), &tp->entry,
2191 	    tp->number * tp->number);
2192 }
2193 
2194 static void
2195 process_dmar(struct dmar *tp)
2196 {
2197 	bsetprop(DMAR_TABLE_PROPNAME, strlen(DMAR_TABLE_PROPNAME),
2198 	    tp, tp->hdr.len);
2199 }
2200 
2201 #else /* __xpv */
2202 static void
2203 enumerate_xen_cpus()
2204 {
2205 	processorid_t	id, max_id;
2206 
2207 	/*
2208 	 * User-set boot-ncpus overrides enumeration
2209 	 */
2210 	if (do_bsys_getproplen(NULL, "boot-ncpus") >= 0)
2211 		return;
2212 
2213 	/*
2214 	 * Probe every possible virtual CPU id and remember the
2215 	 * highest id present; the count of CPUs is one greater
2216 	 * than this.  This tacitly assumes at least cpu 0 is present.
2217 	 */
2218 	max_id = 0;
2219 	for (id = 0; id < MAX_VIRT_CPUS; id++)
2220 		if (HYPERVISOR_vcpu_op(VCPUOP_is_up, id, NULL) == 0)
2221 			max_id = id;
2222 
2223 	bsetpropsi("boot-ncpus", max_id+1);
2224 
2225 }
2226 #endif /* __xpv */
2227 
2228 static void
2229 build_firmware_properties(void)
2230 {
2231 #ifndef __xpv
2232 	struct table_header *tp;
2233 
2234 	if ((tp = find_fw_table("APIC")) != NULL)
2235 		process_madt((struct madt *)tp);
2236 
2237 	if ((srat_ptr = (struct srat *)find_fw_table("SRAT")) != NULL)
2238 		process_srat(srat_ptr);
2239 
2240 	if (slit_ptr = (struct slit *)find_fw_table("SLIT"))
2241 		process_slit(slit_ptr);
2242 
2243 	if (tp = find_fw_table("DMAR"))
2244 		process_dmar((struct dmar *)tp);
2245 #else /* __xpv */
2246 	enumerate_xen_cpus();
2247 #endif /* __xpv */
2248 }
2249 
2250 /*
2251  * fake up a boot property for deferred early console output
2252  * this is used by both graphical boot and the (developer only)
2253  * USB serial console
2254  */
2255 void *
2256 defcons_init(size_t size)
2257 {
2258 	static char *p = NULL;
2259 
2260 	p = do_bsys_alloc(NULL, NULL, size, MMU_PAGESIZE);
2261 	*p = 0;
2262 	bsetprop("deferred-console-buf", strlen("deferred-console-buf") + 1,
2263 	    &p, sizeof (p));
2264 	return (p);
2265 }
2266 
2267 /*ARGSUSED*/
2268 int
2269 boot_compinfo(int fd, struct compinfo *cbp)
2270 {
2271 	cbp->iscmp = 0;
2272 	cbp->blksize = MAXBSIZE;
2273 	return (0);
2274 }
2275 
2276 #define	BP_MAX_STRLEN	32
2277 
2278 /*
2279  * Get value for given boot property
2280  */
2281 int
2282 bootprop_getval(const char *prop_name, u_longlong_t *prop_value)
2283 {
2284 	int		boot_prop_len;
2285 	char		str[BP_MAX_STRLEN];
2286 	u_longlong_t	value;
2287 
2288 	boot_prop_len = BOP_GETPROPLEN(bootops, prop_name);
2289 	if (boot_prop_len < 0 || boot_prop_len > sizeof (str) ||
2290 	    BOP_GETPROP(bootops, prop_name, str) < 0 ||
2291 	    kobj_getvalue(str, &value) == -1)
2292 		return (-1);
2293 
2294 	if (prop_value)
2295 		*prop_value = value;
2296 
2297 	return (0);
2298 }
2299