1ca987d46SWarner Losh /*- 2ca987d46SWarner Losh * Copyright (c) 1998 Michael Smith <msmith@freebsd.org> 3ca987d46SWarner Losh * All rights reserved. 4ca987d46SWarner Losh * 5ca987d46SWarner Losh * Redistribution and use in source and binary forms, with or without 6ca987d46SWarner Losh * modification, are permitted provided that the following conditions 7ca987d46SWarner Losh * are met: 8ca987d46SWarner Losh * 1. Redistributions of source code must retain the above copyright 9ca987d46SWarner Losh * notice, this list of conditions and the following disclaimer. 10ca987d46SWarner Losh * 2. Redistributions in binary form must reproduce the above copyright 11ca987d46SWarner Losh * notice, this list of conditions and the following disclaimer in the 12ca987d46SWarner Losh * documentation and/or other materials provided with the distribution. 13ca987d46SWarner Losh * 14ca987d46SWarner Losh * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 15ca987d46SWarner Losh * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 16ca987d46SWarner Losh * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 17ca987d46SWarner Losh * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 18ca987d46SWarner Losh * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 19ca987d46SWarner Losh * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 20ca987d46SWarner Losh * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 21ca987d46SWarner Losh * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 22ca987d46SWarner Losh * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 23ca987d46SWarner Losh * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 24ca987d46SWarner Losh * SUCH DAMAGE. 25ca987d46SWarner Losh */ 26ca987d46SWarner Losh 27ca987d46SWarner Losh #include <sys/cdefs.h> 28ca987d46SWarner Losh __FBSDID("$FreeBSD$"); 29ca987d46SWarner Losh 30ca987d46SWarner Losh #include <stand.h> 31ca987d46SWarner Losh #include <sys/param.h> 32ca987d46SWarner Losh #include <sys/reboot.h> 33ca987d46SWarner Losh #include <sys/linker.h> 34ca987d46SWarner Losh #include <machine/bootinfo.h> 35ca987d46SWarner Losh #include <machine/cpufunc.h> 36ca987d46SWarner Losh #include <machine/metadata.h> 37ca987d46SWarner Losh #include <machine/psl.h> 38ca987d46SWarner Losh #include <machine/specialreg.h> 39ca987d46SWarner Losh #include "bootstrap.h" 40bca9c87bSWarner Losh #include "modinfo.h" 41ca987d46SWarner Losh #include "libi386.h" 42ca987d46SWarner Losh #include "btxv86.h" 43ca987d46SWarner Losh 44ca987d46SWarner Losh #ifdef LOADER_GELI_SUPPORT 45ca987d46SWarner Losh #include "geliboot.h" 46ca987d46SWarner Losh #endif 47ca987d46SWarner Losh 48ca987d46SWarner Losh /* 49ca987d46SWarner Losh * Check to see if this CPU supports long mode. 50ca987d46SWarner Losh */ 51ca987d46SWarner Losh static int 52ca987d46SWarner Losh bi_checkcpu(void) 53ca987d46SWarner Losh { 54ca987d46SWarner Losh char *cpu_vendor; 55ca987d46SWarner Losh int vendor[3]; 56ca987d46SWarner Losh int eflags; 57ca987d46SWarner Losh unsigned int regs[4]; 58ca987d46SWarner Losh 59ca987d46SWarner Losh /* Check for presence of "cpuid". */ 60ca987d46SWarner Losh eflags = read_eflags(); 61ca987d46SWarner Losh write_eflags(eflags ^ PSL_ID); 62ca987d46SWarner Losh if (!((eflags ^ read_eflags()) & PSL_ID)) 63ca987d46SWarner Losh return (0); 64ca987d46SWarner Losh 65ca987d46SWarner Losh /* Fetch the vendor string. */ 66ca987d46SWarner Losh do_cpuid(0, regs); 67ca987d46SWarner Losh vendor[0] = regs[1]; 68ca987d46SWarner Losh vendor[1] = regs[3]; 69ca987d46SWarner Losh vendor[2] = regs[2]; 70ca987d46SWarner Losh cpu_vendor = (char *)vendor; 71ca987d46SWarner Losh 72ca987d46SWarner Losh /* Check for vendors that support AMD features. */ 73ca987d46SWarner Losh if (strncmp(cpu_vendor, INTEL_VENDOR_ID, 12) != 0 && 74ca987d46SWarner Losh strncmp(cpu_vendor, AMD_VENDOR_ID, 12) != 0 && 752ee49facSKonstantin Belousov strncmp(cpu_vendor, HYGON_VENDOR_ID, 12) != 0 && 76ca987d46SWarner Losh strncmp(cpu_vendor, CENTAUR_VENDOR_ID, 12) != 0) 77ca987d46SWarner Losh return (0); 78ca987d46SWarner Losh 79ca987d46SWarner Losh /* Has to support AMD features. */ 80ca987d46SWarner Losh do_cpuid(0x80000000, regs); 81ca987d46SWarner Losh if (!(regs[0] >= 0x80000001)) 82ca987d46SWarner Losh return (0); 83ca987d46SWarner Losh 84ca987d46SWarner Losh /* Check for long mode. */ 85ca987d46SWarner Losh do_cpuid(0x80000001, regs); 86ca987d46SWarner Losh return (regs[3] & AMDID_LM); 87ca987d46SWarner Losh } 88ca987d46SWarner Losh 89ca987d46SWarner Losh /* 90ca987d46SWarner Losh * Load the information expected by an amd64 kernel. 91ca987d46SWarner Losh * 92ca987d46SWarner Losh * - The 'boothowto' argument is constructed 93ca987d46SWarner Losh * - The 'bootdev' argument is constructed 94ca987d46SWarner Losh * - The 'bootinfo' struct is constructed, and copied into the kernel space. 95ca987d46SWarner Losh * - The kernel environment is copied into kernel space. 96ca987d46SWarner Losh * - Module metadata are formatted and placed in kernel space. 97ca987d46SWarner Losh */ 98ca987d46SWarner Losh int 9945ecda8eSRoger Pau Monné bi_load64(char *args, vm_offset_t *modulep, 100ca987d46SWarner Losh vm_offset_t *kernendp, int add_smap) 101ca987d46SWarner Losh { 102ca987d46SWarner Losh struct preloaded_file *xp, *kfp; 103ca987d46SWarner Losh struct i386_devdesc *rootdev; 104ca987d46SWarner Losh struct file_metadata *md; 10556e53cb8SWarner Losh uint64_t kernend; 10656e53cb8SWarner Losh uint64_t envp; 10756e53cb8SWarner Losh uint64_t module; 10845ecda8eSRoger Pau Monné uint64_t addr; 109ca987d46SWarner Losh vm_offset_t size; 110ca987d46SWarner Losh char *rootdevname; 111ca987d46SWarner Losh int howto; 112ca987d46SWarner Losh 113ca987d46SWarner Losh if (!bi_checkcpu()) { 114ca987d46SWarner Losh printf("CPU doesn't support long mode\n"); 115ca987d46SWarner Losh return (EINVAL); 116ca987d46SWarner Losh } 117ca987d46SWarner Losh 118ca987d46SWarner Losh howto = bi_getboothowto(args); 119ca987d46SWarner Losh 120ca987d46SWarner Losh /* 121ca987d46SWarner Losh * Allow the environment variable 'rootdev' to override the supplied device 122ca987d46SWarner Losh * This should perhaps go to MI code and/or have $rootdev tested/set by 123ca987d46SWarner Losh * MI code before launching the kernel. 124ca987d46SWarner Losh */ 125ca987d46SWarner Losh rootdevname = getenv("rootdev"); 126ca987d46SWarner Losh i386_getdev((void **)(&rootdev), rootdevname, NULL); 127ca987d46SWarner Losh if (rootdev == NULL) { /* bad $rootdev/$currdev */ 128ca987d46SWarner Losh printf("can't determine root device\n"); 129ca987d46SWarner Losh return(EINVAL); 130ca987d46SWarner Losh } 131ca987d46SWarner Losh 132ca987d46SWarner Losh /* Try reading the /etc/fstab file to select the root device */ 13336232229SWarner Losh getrootmount(devformat(&rootdev->dd)); 134ca987d46SWarner Losh 13545ecda8eSRoger Pau Monné addr = 0; 136ca987d46SWarner Losh /* find the last module in the chain */ 137ca987d46SWarner Losh for (xp = file_findfile(NULL, NULL); xp != NULL; xp = xp->f_next) { 138ca987d46SWarner Losh if (addr < (xp->f_addr + xp->f_size)) 139ca987d46SWarner Losh addr = xp->f_addr + xp->f_size; 140ca987d46SWarner Losh } 141ca987d46SWarner Losh /* pad to a page boundary */ 142ca987d46SWarner Losh addr = roundup(addr, PAGE_SIZE); 143ca987d46SWarner Losh 1443630506bSToomas Soome addr = build_font_module(addr); 1453630506bSToomas Soome 146ca987d46SWarner Losh /* place the metadata before anything */ 147ca987d46SWarner Losh module = *modulep = addr; 148ca987d46SWarner Losh 149ca987d46SWarner Losh kfp = file_findfile(NULL, "elf kernel"); 150ca987d46SWarner Losh if (kfp == NULL) 151ca987d46SWarner Losh kfp = file_findfile(NULL, "elf64 kernel"); 152ca987d46SWarner Losh if (kfp == NULL) 153ca987d46SWarner Losh panic("can't find kernel file"); 154ca987d46SWarner Losh kernend = 0; /* fill it in later */ 155ca987d46SWarner Losh file_addmetadata(kfp, MODINFOMD_HOWTO, sizeof howto, &howto); 156ca987d46SWarner Losh file_addmetadata(kfp, MODINFOMD_ENVP, sizeof envp, &envp); 157ca987d46SWarner Losh file_addmetadata(kfp, MODINFOMD_KERNEND, sizeof kernend, &kernend); 158ca987d46SWarner Losh file_addmetadata(kfp, MODINFOMD_MODULEP, sizeof module, &module); 159ca987d46SWarner Losh if (add_smap != 0) 160ca987d46SWarner Losh bios_addsmapdata(kfp); 161ca987d46SWarner Losh #ifdef LOADER_GELI_SUPPORT 162c1418270SIan Lepore geli_export_key_metadata(kfp); 163ca987d46SWarner Losh #endif 1643630506bSToomas Soome bi_load_vbe_data(kfp); 165ca987d46SWarner Losh 1665d1531d9SWarner Losh size = md_copymodules(0, true); 167ca987d46SWarner Losh 168ca987d46SWarner Losh /* copy our environment */ 169ca987d46SWarner Losh envp = roundup(addr + size, PAGE_SIZE); 170*fc352701SWarner Losh addr = md_copyenv(envp); 171ca987d46SWarner Losh 172ca987d46SWarner Losh /* set kernend */ 173ca987d46SWarner Losh kernend = roundup(addr, PAGE_SIZE); 174ca987d46SWarner Losh *kernendp = kernend; 175ca987d46SWarner Losh 176ca987d46SWarner Losh /* patch MODINFOMD_KERNEND */ 177ca987d46SWarner Losh md = file_findmetadata(kfp, MODINFOMD_KERNEND); 178ca987d46SWarner Losh bcopy(&kernend, md->md_data, sizeof kernend); 179ca987d46SWarner Losh 180ca987d46SWarner Losh /* patch MODINFOMD_ENVP */ 181ca987d46SWarner Losh md = file_findmetadata(kfp, MODINFOMD_ENVP); 182ca987d46SWarner Losh bcopy(&envp, md->md_data, sizeof envp); 183ca987d46SWarner Losh 184ca987d46SWarner Losh /* copy module list and metadata */ 1855d1531d9SWarner Losh (void)md_copymodules(*modulep, true); 186ca987d46SWarner Losh 187ca987d46SWarner Losh return(0); 188ca987d46SWarner Losh } 189