1 /*- 2 * SPDX-License-Identifier: BSD-2-Clause 3 * 4 * Copyright (c) 2011 NetApp, Inc. 5 * All rights reserved. 6 * 7 * Redistribution and use in source and binary forms, with or without 8 * modification, are permitted provided that the following conditions 9 * are met: 10 * 1. Redistributions of source code must retain the above copyright 11 * notice, this list of conditions and the following disclaimer. 12 * 2. Redistributions in binary form must reproduce the above copyright 13 * notice, this list of conditions and the following disclaimer in the 14 * documentation and/or other materials provided with the distribution. 15 * 16 * THIS SOFTWARE IS PROVIDED BY NETAPP, INC ``AS IS'' AND 17 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 19 * ARE DISCLAIMED. IN NO EVENT SHALL NETAPP, INC OR CONTRIBUTORS BE LIABLE 20 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 21 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 22 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 23 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 24 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 25 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 26 * SUCH DAMAGE. 27 * 28 * $FreeBSD$ 29 */ 30 31 /*- 32 * Copyright (c) 2011 Google, Inc. 33 * All rights reserved. 34 * 35 * Redistribution and use in source and binary forms, with or without 36 * modification, are permitted provided that the following conditions 37 * are met: 38 * 1. Redistributions of source code must retain the above copyright 39 * notice, this list of conditions and the following disclaimer. 40 * 2. Redistributions in binary form must reproduce the above copyright 41 * notice, this list of conditions and the following disclaimer in the 42 * documentation and/or other materials provided with the distribution. 43 * 44 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 45 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 46 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 47 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 48 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 49 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 50 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 51 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 52 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 53 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 54 * SUCH DAMAGE. 55 * 56 * $FreeBSD$ 57 */ 58 59 #include <sys/cdefs.h> 60 __FBSDID("$FreeBSD$"); 61 62 #include <sys/ioctl.h> 63 #include <sys/stat.h> 64 #include <sys/disk.h> 65 #include <sys/queue.h> 66 67 #include <machine/specialreg.h> 68 #include <machine/vmm.h> 69 70 #include <assert.h> 71 #include <dirent.h> 72 #include <dlfcn.h> 73 #include <errno.h> 74 #include <err.h> 75 #include <fcntl.h> 76 #include <getopt.h> 77 #include <libgen.h> 78 #include <limits.h> 79 #include <setjmp.h> 80 #include <stdio.h> 81 #include <stdlib.h> 82 #include <string.h> 83 #include <sysexits.h> 84 #include <termios.h> 85 #include <unistd.h> 86 87 #include <vmmapi.h> 88 89 #include "userboot.h" 90 91 #define MB (1024 * 1024UL) 92 #define GB (1024 * 1024 * 1024UL) 93 #define BSP 0 94 95 #define NDISKS 32 96 97 static char *host_base; 98 static struct termios term, oldterm; 99 static int disk_fd[NDISKS]; 100 static int ndisks; 101 static int consin_fd, consout_fd; 102 103 static int need_reinit; 104 105 static void *loader_hdl; 106 static char *loader; 107 static int explicit_loader; 108 static jmp_buf jb; 109 110 static char *vmname, *progname; 111 static struct vmctx *ctx; 112 static struct vcpu *vcpu; 113 114 static uint64_t gdtbase, cr3, rsp; 115 116 static void cb_exit(void *arg, int v); 117 118 /* 119 * Console i/o callbacks 120 */ 121 122 static void 123 cb_putc(void *arg __unused, int ch) 124 { 125 char c = ch; 126 127 (void) write(consout_fd, &c, 1); 128 } 129 130 static int 131 cb_getc(void *arg __unused) 132 { 133 char c; 134 135 if (read(consin_fd, &c, 1) == 1) 136 return (c); 137 return (-1); 138 } 139 140 static int 141 cb_poll(void *arg __unused) 142 { 143 int n; 144 145 if (ioctl(consin_fd, FIONREAD, &n) >= 0) 146 return (n > 0); 147 return (0); 148 } 149 150 /* 151 * Host filesystem i/o callbacks 152 */ 153 154 struct cb_file { 155 int cf_isdir; 156 size_t cf_size; 157 struct stat cf_stat; 158 union { 159 int fd; 160 DIR *dir; 161 } cf_u; 162 }; 163 164 static int 165 cb_open(void *arg __unused, const char *filename, void **hp) 166 { 167 struct cb_file *cf; 168 char path[PATH_MAX]; 169 170 if (!host_base) 171 return (ENOENT); 172 173 strlcpy(path, host_base, PATH_MAX); 174 if (path[strlen(path) - 1] == '/') 175 path[strlen(path) - 1] = 0; 176 strlcat(path, filename, PATH_MAX); 177 cf = malloc(sizeof(struct cb_file)); 178 if (stat(path, &cf->cf_stat) < 0) { 179 free(cf); 180 return (errno); 181 } 182 183 cf->cf_size = cf->cf_stat.st_size; 184 if (S_ISDIR(cf->cf_stat.st_mode)) { 185 cf->cf_isdir = 1; 186 cf->cf_u.dir = opendir(path); 187 if (!cf->cf_u.dir) 188 goto out; 189 *hp = cf; 190 return (0); 191 } 192 if (S_ISREG(cf->cf_stat.st_mode)) { 193 cf->cf_isdir = 0; 194 cf->cf_u.fd = open(path, O_RDONLY); 195 if (cf->cf_u.fd < 0) 196 goto out; 197 *hp = cf; 198 return (0); 199 } 200 201 out: 202 free(cf); 203 return (EINVAL); 204 } 205 206 static int 207 cb_close(void *arg __unused, void *h) 208 { 209 struct cb_file *cf = h; 210 211 if (cf->cf_isdir) 212 closedir(cf->cf_u.dir); 213 else 214 close(cf->cf_u.fd); 215 free(cf); 216 217 return (0); 218 } 219 220 static int 221 cb_isdir(void *arg __unused, void *h) 222 { 223 struct cb_file *cf = h; 224 225 return (cf->cf_isdir); 226 } 227 228 static int 229 cb_read(void *arg __unused, void *h, void *buf, size_t size, size_t *resid) 230 { 231 struct cb_file *cf = h; 232 ssize_t sz; 233 234 if (cf->cf_isdir) 235 return (EINVAL); 236 sz = read(cf->cf_u.fd, buf, size); 237 if (sz < 0) 238 return (EINVAL); 239 *resid = size - sz; 240 return (0); 241 } 242 243 static int 244 cb_readdir(void *arg __unused, void *h, uint32_t *fileno_return, 245 uint8_t *type_return, size_t *namelen_return, char *name) 246 { 247 struct cb_file *cf = h; 248 struct dirent *dp; 249 250 if (!cf->cf_isdir) 251 return (EINVAL); 252 253 dp = readdir(cf->cf_u.dir); 254 if (!dp) 255 return (ENOENT); 256 257 /* 258 * Note: d_namlen is in the range 0..255 and therefore less 259 * than PATH_MAX so we don't need to test before copying. 260 */ 261 *fileno_return = dp->d_fileno; 262 *type_return = dp->d_type; 263 *namelen_return = dp->d_namlen; 264 memcpy(name, dp->d_name, dp->d_namlen); 265 name[dp->d_namlen] = 0; 266 267 return (0); 268 } 269 270 static int 271 cb_seek(void *arg __unused, void *h, uint64_t offset, int whence) 272 { 273 struct cb_file *cf = h; 274 275 if (cf->cf_isdir) 276 return (EINVAL); 277 if (lseek(cf->cf_u.fd, offset, whence) < 0) 278 return (errno); 279 return (0); 280 } 281 282 static int 283 cb_stat(void *arg __unused, void *h, struct stat *sbp) 284 { 285 struct cb_file *cf = h; 286 287 memset(sbp, 0, sizeof(struct stat)); 288 sbp->st_mode = cf->cf_stat.st_mode; 289 sbp->st_uid = cf->cf_stat.st_uid; 290 sbp->st_gid = cf->cf_stat.st_gid; 291 sbp->st_size = cf->cf_stat.st_size; 292 sbp->st_mtime = cf->cf_stat.st_mtime; 293 sbp->st_dev = cf->cf_stat.st_dev; 294 sbp->st_ino = cf->cf_stat.st_ino; 295 296 return (0); 297 } 298 299 /* 300 * Disk image i/o callbacks 301 */ 302 303 static int 304 cb_diskread(void *arg __unused, int unit, uint64_t from, void *to, size_t size, 305 size_t *resid) 306 { 307 ssize_t n; 308 309 if (unit < 0 || unit >= ndisks) 310 return (EIO); 311 n = pread(disk_fd[unit], to, size, from); 312 if (n < 0) 313 return (errno); 314 *resid = size - n; 315 return (0); 316 } 317 318 static int 319 cb_diskwrite(void *arg __unused, int unit, uint64_t offset, void *src, 320 size_t size, size_t *resid) 321 { 322 ssize_t n; 323 324 if (unit < 0 || unit >= ndisks) 325 return (EIO); 326 n = pwrite(disk_fd[unit], src, size, offset); 327 if (n < 0) 328 return (errno); 329 *resid = size - n; 330 return (0); 331 } 332 333 static int 334 cb_diskioctl(void *arg __unused, int unit, u_long cmd, void *data) 335 { 336 struct stat sb; 337 338 if (unit < 0 || unit >= ndisks) 339 return (EBADF); 340 341 switch (cmd) { 342 case DIOCGSECTORSIZE: 343 *(u_int *)data = 512; 344 break; 345 case DIOCGMEDIASIZE: 346 if (fstat(disk_fd[unit], &sb) != 0) 347 return (ENOTTY); 348 if (S_ISCHR(sb.st_mode) && 349 ioctl(disk_fd[unit], DIOCGMEDIASIZE, &sb.st_size) != 0) 350 return (ENOTTY); 351 *(off_t *)data = sb.st_size; 352 break; 353 default: 354 return (ENOTTY); 355 } 356 357 return (0); 358 } 359 360 /* 361 * Guest virtual machine i/o callbacks 362 */ 363 static int 364 cb_copyin(void *arg __unused, const void *from, uint64_t to, size_t size) 365 { 366 char *ptr; 367 368 to &= 0x7fffffff; 369 370 ptr = vm_map_gpa(ctx, to, size); 371 if (ptr == NULL) 372 return (EFAULT); 373 374 memcpy(ptr, from, size); 375 return (0); 376 } 377 378 static int 379 cb_copyout(void *arg __unused, uint64_t from, void *to, size_t size) 380 { 381 char *ptr; 382 383 from &= 0x7fffffff; 384 385 ptr = vm_map_gpa(ctx, from, size); 386 if (ptr == NULL) 387 return (EFAULT); 388 389 memcpy(to, ptr, size); 390 return (0); 391 } 392 393 static void 394 cb_setreg(void *arg __unused, int r, uint64_t v) 395 { 396 int error; 397 enum vm_reg_name vmreg; 398 399 vmreg = VM_REG_LAST; 400 401 switch (r) { 402 case 4: 403 vmreg = VM_REG_GUEST_RSP; 404 rsp = v; 405 break; 406 default: 407 break; 408 } 409 410 if (vmreg == VM_REG_LAST) { 411 printf("test_setreg(%d): not implemented\n", r); 412 cb_exit(NULL, USERBOOT_EXIT_QUIT); 413 } 414 415 error = vm_set_register(vcpu, vmreg, v); 416 if (error) { 417 perror("vm_set_register"); 418 cb_exit(NULL, USERBOOT_EXIT_QUIT); 419 } 420 } 421 422 static void 423 cb_setmsr(void *arg __unused, int r, uint64_t v) 424 { 425 int error; 426 enum vm_reg_name vmreg; 427 428 vmreg = VM_REG_LAST; 429 430 switch (r) { 431 case MSR_EFER: 432 vmreg = VM_REG_GUEST_EFER; 433 break; 434 default: 435 break; 436 } 437 438 if (vmreg == VM_REG_LAST) { 439 printf("test_setmsr(%d): not implemented\n", r); 440 cb_exit(NULL, USERBOOT_EXIT_QUIT); 441 } 442 443 error = vm_set_register(vcpu, vmreg, v); 444 if (error) { 445 perror("vm_set_msr"); 446 cb_exit(NULL, USERBOOT_EXIT_QUIT); 447 } 448 } 449 450 static void 451 cb_setcr(void *arg __unused, int r, uint64_t v) 452 { 453 int error; 454 enum vm_reg_name vmreg; 455 456 vmreg = VM_REG_LAST; 457 458 switch (r) { 459 case 0: 460 vmreg = VM_REG_GUEST_CR0; 461 break; 462 case 3: 463 vmreg = VM_REG_GUEST_CR3; 464 cr3 = v; 465 break; 466 case 4: 467 vmreg = VM_REG_GUEST_CR4; 468 break; 469 default: 470 break; 471 } 472 473 if (vmreg == VM_REG_LAST) { 474 printf("test_setcr(%d): not implemented\n", r); 475 cb_exit(NULL, USERBOOT_EXIT_QUIT); 476 } 477 478 error = vm_set_register(vcpu, vmreg, v); 479 if (error) { 480 perror("vm_set_cr"); 481 cb_exit(NULL, USERBOOT_EXIT_QUIT); 482 } 483 } 484 485 static void 486 cb_setgdt(void *arg __unused, uint64_t base, size_t size) 487 { 488 int error; 489 490 error = vm_set_desc(vcpu, VM_REG_GUEST_GDTR, base, size - 1, 0); 491 if (error != 0) { 492 perror("vm_set_desc(gdt)"); 493 cb_exit(NULL, USERBOOT_EXIT_QUIT); 494 } 495 496 gdtbase = base; 497 } 498 499 static void 500 cb_exec(void *arg __unused, uint64_t rip) 501 { 502 int error; 503 504 if (cr3 == 0) 505 error = vm_setup_freebsd_registers_i386(vcpu, rip, gdtbase, 506 rsp); 507 else 508 error = vm_setup_freebsd_registers(vcpu, rip, cr3, gdtbase, 509 rsp); 510 if (error) { 511 perror("vm_setup_freebsd_registers"); 512 cb_exit(NULL, USERBOOT_EXIT_QUIT); 513 } 514 515 cb_exit(NULL, 0); 516 } 517 518 /* 519 * Misc 520 */ 521 522 static void 523 cb_delay(void *arg __unused, int usec) 524 { 525 526 usleep(usec); 527 } 528 529 static void 530 cb_exit(void *arg __unused, int v) 531 { 532 533 tcsetattr(consout_fd, TCSAFLUSH, &oldterm); 534 exit(v); 535 } 536 537 static void 538 cb_getmem(void *arg __unused, uint64_t *ret_lowmem, uint64_t *ret_highmem) 539 { 540 541 *ret_lowmem = vm_get_lowmem_size(ctx); 542 *ret_highmem = vm_get_highmem_size(ctx); 543 } 544 545 struct env { 546 char *str; /* name=value */ 547 SLIST_ENTRY(env) next; 548 }; 549 550 static SLIST_HEAD(envhead, env) envhead; 551 552 static void 553 addenv(const char *str) 554 { 555 struct env *env; 556 557 env = malloc(sizeof(struct env)); 558 if (env == NULL) 559 err(EX_OSERR, "malloc"); 560 env->str = strdup(str); 561 if (env->str == NULL) 562 err(EX_OSERR, "strdup"); 563 SLIST_INSERT_HEAD(&envhead, env, next); 564 } 565 566 static char * 567 cb_getenv(void *arg __unused, int num) 568 { 569 int i; 570 struct env *env; 571 572 i = 0; 573 SLIST_FOREACH(env, &envhead, next) { 574 if (i == num) 575 return (env->str); 576 i++; 577 } 578 579 return (NULL); 580 } 581 582 static int 583 cb_vm_set_register(void *arg __unused, int vcpuid, int reg, uint64_t val) 584 { 585 586 assert(vcpuid == BSP); 587 return (vm_set_register(vcpu, reg, val)); 588 } 589 590 static int 591 cb_vm_set_desc(void *arg __unused, int vcpuid, int reg, uint64_t base, 592 u_int limit, u_int access) 593 { 594 595 assert(vcpuid == BSP); 596 return (vm_set_desc(vcpu, reg, base, limit, access)); 597 } 598 599 static void 600 cb_swap_interpreter(void *arg __unused, const char *interp_req) 601 { 602 603 /* 604 * If the user specified a loader but we detected a mismatch, we should 605 * not try to pivot to a different loader on them. 606 */ 607 free(loader); 608 if (explicit_loader == 1) { 609 perror("requested loader interpreter does not match guest userboot"); 610 cb_exit(NULL, 1); 611 } 612 if (interp_req == NULL || *interp_req == '\0') { 613 perror("guest failed to request an interpreter"); 614 cb_exit(NULL, 1); 615 } 616 617 if (asprintf(&loader, "/boot/userboot_%s.so", interp_req) == -1) 618 err(EX_OSERR, "malloc"); 619 need_reinit = 1; 620 longjmp(jb, 1); 621 } 622 623 static struct loader_callbacks cb = { 624 .getc = cb_getc, 625 .putc = cb_putc, 626 .poll = cb_poll, 627 628 .open = cb_open, 629 .close = cb_close, 630 .isdir = cb_isdir, 631 .read = cb_read, 632 .readdir = cb_readdir, 633 .seek = cb_seek, 634 .stat = cb_stat, 635 636 .diskread = cb_diskread, 637 .diskwrite = cb_diskwrite, 638 .diskioctl = cb_diskioctl, 639 640 .copyin = cb_copyin, 641 .copyout = cb_copyout, 642 .setreg = cb_setreg, 643 .setmsr = cb_setmsr, 644 .setcr = cb_setcr, 645 .setgdt = cb_setgdt, 646 .exec = cb_exec, 647 648 .delay = cb_delay, 649 .exit = cb_exit, 650 .getmem = cb_getmem, 651 652 .getenv = cb_getenv, 653 654 /* Version 4 additions */ 655 .vm_set_register = cb_vm_set_register, 656 .vm_set_desc = cb_vm_set_desc, 657 658 /* Version 5 additions */ 659 .swap_interpreter = cb_swap_interpreter, 660 }; 661 662 static int 663 altcons_open(char *path) 664 { 665 struct stat sb; 666 int err; 667 int fd; 668 669 /* 670 * Allow stdio to be passed in so that the same string 671 * can be used for the bhyveload console and bhyve com-port 672 * parameters 673 */ 674 if (!strcmp(path, "stdio")) 675 return (0); 676 677 err = stat(path, &sb); 678 if (err == 0) { 679 if (!S_ISCHR(sb.st_mode)) 680 err = ENOTSUP; 681 else { 682 fd = open(path, O_RDWR | O_NONBLOCK); 683 if (fd < 0) 684 err = errno; 685 else 686 consin_fd = consout_fd = fd; 687 } 688 } 689 690 return (err); 691 } 692 693 static int 694 disk_open(char *path) 695 { 696 int fd; 697 698 if (ndisks >= NDISKS) 699 return (ERANGE); 700 701 fd = open(path, O_RDWR); 702 if (fd < 0) 703 return (errno); 704 705 disk_fd[ndisks] = fd; 706 ndisks++; 707 708 return (0); 709 } 710 711 static void 712 usage(void) 713 { 714 715 fprintf(stderr, 716 "usage: %s [-S][-c <console-device>] [-d <disk-path>] [-e <name=value>]\n" 717 " %*s [-h <host-path>] [-m memsize[K|k|M|m|G|g|T|t]] <vmname>\n", 718 progname, 719 (int)strlen(progname), ""); 720 exit(1); 721 } 722 723 int 724 main(int argc, char** argv) 725 { 726 void (*func)(struct loader_callbacks *, void *, int, int); 727 uint64_t mem_size; 728 int opt, error, memflags; 729 730 progname = basename(argv[0]); 731 732 memflags = 0; 733 mem_size = 256 * MB; 734 735 consin_fd = STDIN_FILENO; 736 consout_fd = STDOUT_FILENO; 737 738 while ((opt = getopt(argc, argv, "CSc:d:e:h:l:m:")) != -1) { 739 switch (opt) { 740 case 'c': 741 error = altcons_open(optarg); 742 if (error != 0) 743 errx(EX_USAGE, "Could not open '%s'", optarg); 744 break; 745 746 case 'd': 747 error = disk_open(optarg); 748 if (error != 0) 749 errx(EX_USAGE, "Could not open '%s'", optarg); 750 break; 751 752 case 'e': 753 addenv(optarg); 754 break; 755 756 case 'h': 757 host_base = optarg; 758 break; 759 760 case 'l': 761 if (loader != NULL) 762 errx(EX_USAGE, "-l can only be given once"); 763 loader = strdup(optarg); 764 if (loader == NULL) 765 err(EX_OSERR, "malloc"); 766 explicit_loader = 1; 767 break; 768 769 case 'm': 770 error = vm_parse_memsize(optarg, &mem_size); 771 if (error != 0) 772 errx(EX_USAGE, "Invalid memsize '%s'", optarg); 773 break; 774 case 'C': 775 memflags |= VM_MEM_F_INCORE; 776 break; 777 case 'S': 778 memflags |= VM_MEM_F_WIRED; 779 break; 780 case '?': 781 usage(); 782 } 783 } 784 785 argc -= optind; 786 argv += optind; 787 788 if (argc != 1) 789 usage(); 790 791 vmname = argv[0]; 792 793 need_reinit = 0; 794 error = vm_create(vmname); 795 if (error) { 796 if (errno != EEXIST) { 797 perror("vm_create"); 798 exit(1); 799 } 800 need_reinit = 1; 801 } 802 803 ctx = vm_open(vmname); 804 if (ctx == NULL) { 805 perror("vm_open"); 806 exit(1); 807 } 808 809 vcpu = vm_vcpu_open(ctx, BSP); 810 811 /* 812 * setjmp in the case the guest wants to swap out interpreter, 813 * cb_swap_interpreter will swap out loader as appropriate and set 814 * need_reinit so that we end up in a clean state once again. 815 */ 816 setjmp(jb); 817 818 if (need_reinit) { 819 error = vm_reinit(ctx); 820 if (error) { 821 perror("vm_reinit"); 822 exit(1); 823 } 824 } 825 826 vm_set_memflags(ctx, memflags); 827 error = vm_setup_memory(ctx, mem_size, VM_MMAP_ALL); 828 if (error) { 829 perror("vm_setup_memory"); 830 exit(1); 831 } 832 833 if (loader == NULL) { 834 loader = strdup("/boot/userboot.so"); 835 if (loader == NULL) 836 err(EX_OSERR, "malloc"); 837 } 838 if (loader_hdl != NULL) 839 dlclose(loader_hdl); 840 loader_hdl = dlopen(loader, RTLD_LOCAL); 841 if (!loader_hdl) { 842 printf("%s\n", dlerror()); 843 free(loader); 844 return (1); 845 } 846 func = dlsym(loader_hdl, "loader_main"); 847 if (!func) { 848 printf("%s\n", dlerror()); 849 free(loader); 850 return (1); 851 } 852 853 tcgetattr(consout_fd, &term); 854 oldterm = term; 855 cfmakeraw(&term); 856 term.c_cflag |= CLOCAL; 857 858 tcsetattr(consout_fd, TCSAFLUSH, &term); 859 860 addenv("smbios.bios.vendor=BHYVE"); 861 addenv("boot_serial=1"); 862 863 func(&cb, NULL, USERBOOT_VERSION_5, ndisks); 864 865 free(loader); 866 return (0); 867 } 868