summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/arch/loongson/loongson/machdep.c944
1 files changed, 944 insertions, 0 deletions
diff --git a/sys/arch/loongson/loongson/machdep.c b/sys/arch/loongson/loongson/machdep.c
new file mode 100644
index 00000000000..989a92bb584
--- /dev/null
+++ b/sys/arch/loongson/loongson/machdep.c
@@ -0,0 +1,944 @@
+/* $OpenBSD: machdep.c,v 1.1 2009/12/25 22:06:03 miod Exp $ */
+
+/*
+ * Copyright (c) 2009 Miodrag Vallat.
+ *
+ * Permission to use, copy, modify, and distribute this software for any
+ * purpose with or without fee is hereby granted, provided that the above
+ * copyright notice and this permission notice appear in all copies.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ */
+/*
+ * Copyright (c) 2003-2004 Opsycon AB (www.opsycon.se / www.opsycon.com)
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
+ * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ *
+ */
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/kernel.h>
+#include <sys/proc.h>
+#include <sys/buf.h>
+#include <sys/reboot.h>
+#include <sys/conf.h>
+#include <sys/file.h>
+#include <sys/msgbuf.h>
+#include <sys/tty.h>
+#include <sys/user.h>
+#include <sys/exec.h>
+#include <sys/sysctl.h>
+#include <sys/mount.h>
+#include <sys/syscallargs.h>
+#include <sys/exec_elf.h>
+#ifdef SYSVSHM
+#include <sys/shm.h>
+#endif
+#ifdef SYSVSEM
+#include <sys/sem.h>
+#endif
+
+#include <uvm/uvm_extern.h>
+
+#include <machine/db_machdep.h>
+#include <ddb/db_interface.h>
+
+#include <machine/autoconf.h>
+#include <machine/cpu.h>
+#include <machine/memconf.h>
+#include <machine/pmon.h>
+
+#include <dev/cons.h>
+
+#include <mips64/archtype.h>
+
+#include <loongson/dev/bonitoreg.h>
+
+/* The following is used externally (sysctl_hw) */
+char machine[] = MACHINE; /* Machine "architecture" */
+char cpu_model[30];
+
+/*
+ * Declare these as initialized data so we can patch them.
+ */
+#ifndef BUFCACHEPERCENT
+#define BUFCACHEPERCENT 5 /* Can be changed in config. */
+#endif
+#ifndef BUFPAGES
+#define BUFPAGES 0 /* Can be changed in config. */
+#endif
+int bufpages = BUFPAGES;
+int bufcachepercent = BUFCACHEPERCENT;
+
+vm_map_t exec_map;
+vm_map_t phys_map;
+
+caddr_t msgbufbase;
+vaddr_t uncached_base;
+
+int physmem; /* Max supported memory, changes to actual. */
+int ncpu = 1; /* At least one CPU in the system. */
+struct user *proc0paddr;
+int kbd_reset;
+
+struct sys_rec sys_config;
+
+/* Pointers to the start and end of the symbol table. */
+caddr_t ssym;
+caddr_t esym;
+caddr_t ekern;
+
+struct phys_mem_desc mem_layout[MAXMEMSEGS];
+
+static u_long atoi(const char *, uint);
+static void dobootopts(int);
+
+void build_trampoline(vaddr_t, vaddr_t);
+void dumpsys(void);
+void dumpconf(void);
+vaddr_t mips_init(int32_t, int32_t, int32_t, int32_t);
+
+extern void loongson2e_setup(u_long, u_long);
+extern void loongson2f_setup(u_long, u_long);
+
+cons_decl(pmon);
+
+struct consdev pmoncons = {
+ NULL,
+ NULL,
+ pmoncngetc,
+ pmoncnputc,
+ nullcnpollc,
+ NULL,
+ makedev(0, 0),
+ CN_DEAD
+};
+
+/*
+ * Do all the stuff that locore normally does before calling main().
+ * Reset mapping and set up mapping to hardware and init "wired" reg.
+ */
+
+vaddr_t
+mips_init(int32_t argc, int32_t argv, int32_t envp, int32_t cv)
+{
+ uint prid, loongson_ver;
+ u_long memlo, memhi, cpuspeed;
+ vaddr_t xtlb_handler;
+ const char *envvar;
+ int i;
+
+ extern char start[], edata[], end[];
+ extern char exception[], e_exception[];
+ extern char *hw_vendor, *hw_prod;
+ extern void xtlb_miss;
+
+ /*
+ * Clear the compiled BSS segment in OpenBSD code.
+ * PMON is supposed to have done this, though.
+ */
+
+ bzero(edata, end - edata);
+
+ /*
+ * Set up early console output.
+ */
+
+ pmon_init(argc, argv, envp, cv);
+ cn_tab = &pmoncons;
+
+ /*
+ * Attempt to locate ELF header and symbol table after kernel,
+ * and reserve space for the symbol table, if it exists.
+ */
+
+ ssym = (char *)(vaddr_t)*(int32_t *)end;
+ if (((long)ssym - (long)end) >= 0 &&
+ ((long)ssym - (long)end) <= 0x1000 &&
+ ssym[0] == ELFMAG0 && ssym[1] == ELFMAG1 &&
+ ssym[2] == ELFMAG2 && ssym[3] == ELFMAG3 ) {
+ /* Pointers exist directly after kernel. */
+ esym = (char *)(vaddr_t)*((int32_t *)end + 1);
+ ekern = esym;
+ } else {
+ /* Pointers aren't setup either... */
+ ssym = esym = NULL;
+ ekern = end;
+ }
+
+ /*
+ * Try and figure out what kind of hardware we are.
+ */
+
+ envvar = pmon_getenv("systype");
+ if (envvar == NULL) {
+ pmon_printf("Unable to figure out system type!\n");
+ goto unsupported;
+ }
+ if (strcmp(envvar, "Bonito") != 0) {
+ pmon_printf("This kernel doesn't support system type \"%s\".\n",
+ envvar);
+ goto unsupported;
+ }
+
+ /*
+ * While the kernel supports other processor types than Loongson,
+ * we are not expecting a Bonito-based system with a different
+ * processor. Just to be on the safe side, refuse to run on
+ * non Loongson2 processors for now.
+ */
+
+ prid = cp0_get_prid();
+ switch ((prid >> 8) & 0xff) {
+ case MIPS_LOONGSON2:
+ loongson_ver = 0x2c + (prid & 0xff);
+ if (loongson_ver == 0x2e || loongson_ver == 0x2f)
+ break;
+ /* FALLTHROUGH */
+ default:
+ pmon_printf("This kernel doesn't support processor type 0x%x"
+ ", version %d.%d.\n",
+ (prid >> 8) & 0xff, (prid >> 4) & 0x0f, prid & 0x0f);
+ goto unsupported;
+ }
+
+ /*
+ * Try to figure out what particular machine we run on, depending
+ * on the PMON version information.
+ */
+
+ envvar = pmon_getenv("Version");
+ if (envvar == NULL) {
+ pmon_printf("Unable to figure out model!\n");
+ goto unsupported;
+ }
+
+ /* Lemote Yeelong 8089 */
+ if (strncmp(envvar, "LM8089", 6) == 0) {
+ sys_config.system_type = LOONGSON_YEELONG;
+ }
+
+ if (sys_config.system_type == 0) {
+ pmon_printf("This kernel doesn't support model \"%s\".\n",
+ envvar);
+ goto unsupported;
+ }
+
+ switch (sys_config.system_type) {
+ case LOONGSON_YEELONG:
+ hw_vendor = "Lemote";
+ hw_prod = "Yeelong";
+ break;
+ default: /* won't happen */
+ goto unsupported;
+ }
+
+ pmon_printf("Found %s %s, setting up.\n", hw_vendor, hw_prod);
+
+ snprintf(cpu_model, sizeof cpu_model, "Loongson %X", loongson_ver);
+
+ /*
+ * Figure out processor clock speed.
+ * Hopefully the processor speed, in Hertz, will not overflow
+ * uint32_t...
+ */
+
+ cpuspeed = 0;
+ envvar = pmon_getenv("cpuclock");
+ if (envvar != NULL)
+ cpuspeed = atoi(envvar, 10); /* speed in Hz */
+ if (cpuspeed < 100 * 1000000)
+ cpuspeed = 797000000; /* Reasonable default */
+ sys_config.cpu[0].clock = cpuspeed;
+
+ /*
+ * Look at arguments passed to us and compute boothowto.
+ */
+
+ boothowto = RB_AUTOBOOT;
+ dobootopts(argc);
+
+ /*
+ * Figure out memory information.
+ * PMON reports it in two chunks, the memory under the 256MB
+ * CKSEG limit, and memory above that limit. We need to do the
+ * math ourselves.
+ */
+
+ envvar = pmon_getenv("memsize");
+ if (envvar == NULL) {
+ pmon_printf("Could not get memory information"
+ " from the firmware\n");
+ goto unsupported;
+ }
+ memlo = atoi(envvar, 10); /* size in MB */
+ if (memlo < 0 || memlo > 256) {
+ pmon_printf("Incorrect low memory size `%s'\n", envvar);
+ goto unsupported;
+ }
+
+ if (memlo == 256) {
+ envvar = pmon_getenv("highmemsize");
+ if (envvar == NULL)
+ memhi = 0;
+ else
+ memhi = atoi(envvar, 10); /* size in MB */
+ if (memhi < 0 || memhi > (64 * 1024) - 256) {
+ pmon_printf("Incorrect high memory size `%s'\n",
+ envvar);
+ /* better expose the problem than limit to 256MB */
+ goto unsupported;
+ }
+ }
+
+ uncached_base = PHYS_TO_XKPHYS(0, CCA_NC);
+
+ switch (loongson_ver) {
+ case 0x2e:
+ loongson2e_setup(memlo, memhi);
+ break;
+ default:
+ case 0x2f:
+ loongson2f_setup(memlo, memhi);
+ break;
+ }
+
+ /*
+ * The above call might have altered address mappings,
+ * so pmon_printf() should no longer be used from now on.
+ */
+
+ /*
+ * Set pagesize to enable use of page macros and functions.
+ * Commit available memory to UVM system.
+ */
+
+ uvmexp.pagesize = PAGE_SIZE;
+ uvm_setpagesize();
+
+ for (i = 0; i < MAXMEMSEGS && mem_layout[i].mem_last_page != 0; i++) {
+ uint64_t fp, lp;
+ uint64_t firstkernpage, lastkernpage;
+ unsigned int freelist;
+ paddr_t firstkernpa, lastkernpa;
+
+ /* kernel is linked in CKSEG0 */
+ firstkernpa = CKSEG0_TO_PHYS((vaddr_t)start);
+ lastkernpa = CKSEG0_TO_PHYS((vaddr_t)ekern);
+ if (loongson_ver == 0x2f) {
+ firstkernpa |= 0x80000000;
+ lastkernpa |= 0x80000000;
+ }
+
+ firstkernpage = atop(trunc_page(firstkernpa));
+ lastkernpage = atop(round_page(lastkernpa));
+
+ fp = mem_layout[i].mem_first_page;
+ lp = mem_layout[i].mem_last_page;
+ freelist = mem_layout[i].mem_freelist;
+
+ /* Account for kernel and kernel symbol table. */
+ if (fp >= firstkernpage && lp < lastkernpage)
+ continue; /* In kernel. */
+
+ if (lp < firstkernpage || fp > lastkernpage) {
+ uvm_page_physload(fp, lp, fp, lp, freelist);
+ continue; /* Outside kernel. */
+ }
+
+ if (fp >= firstkernpage)
+ fp = lastkernpage;
+ else if (lp < lastkernpage)
+ lp = firstkernpage;
+ else { /* Need to split! */
+ uint64_t xp = firstkernpage;
+ uvm_page_physload(fp, xp, fp, xp, freelist);
+ fp = lastkernpage;
+ }
+ if (lp > fp) {
+ uvm_page_physload(fp, lp, fp, lp, freelist);
+ }
+ }
+
+ sys_config.cpu[0].type = (prid >> 8) & 0xff;
+ sys_config.cpu[0].vers_maj = (prid >> 4) & 0x0f;
+ sys_config.cpu[0].vers_min = prid & 0x0f;
+ /* FPU reports itself as type 5, version 0.1... */
+ sys_config.cpu[0].fptype = sys_config.cpu[0].type;
+ sys_config.cpu[0].fpvers_maj = sys_config.cpu[0].vers_maj;
+ sys_config.cpu[0].fpvers_min = sys_config.cpu[0].vers_min;
+
+ sys_config.cpu[0].tlbsize = 64;
+
+ /*
+ * Configure cache.
+ * Note that the caches being physically tagged, there is no
+ * need to invalidate or flush it.
+ */
+
+ Loongson2_ConfigCache();
+
+ sys_config.cpu[0].tlbwired = UPAGES / 2;
+ tlb_set_page_mask(TLB_PAGE_MASK);
+ tlb_set_wired(0);
+ tlb_flush(sys_config.cpu[0].tlbsize);
+ tlb_set_wired(sys_config.cpu[0].tlbwired);
+
+ /*
+ * Get a console, very early but after initial mapping setup.
+ */
+
+ consinit();
+ printf("Initial setup done, switching console.\n");
+
+ /*
+ * Init message buffer.
+ */
+
+ msgbufbase = (caddr_t)pmap_steal_memory(MSGBUFSIZE, NULL,NULL);
+ initmsgbuf(msgbufbase, MSGBUFSIZE);
+
+ /*
+ * Allocate U page(s) for proc[0], pm_tlbpid 1.
+ */
+
+ proc0.p_addr = proc0paddr = curcpu()->ci_curprocpaddr =
+ (struct user *)pmap_steal_memory(USPACE, NULL, NULL);
+ proc0.p_md.md_regs = (struct trap_frame *)&proc0paddr->u_pcb.pcb_regs;
+ tlb_set_pid(1);
+
+ /*
+ * Bootstrap VM system.
+ */
+
+ pmap_bootstrap();
+
+ /*
+ * Copy down exception vector code.
+ */
+
+ bcopy(exception, (char *)CACHE_ERR_EXC_VEC, e_exception - exception);
+ bcopy(exception, (char *)GEN_EXC_VEC, e_exception - exception);
+
+ /*
+ * Build proper TLB refill handler trampolines.
+ */
+
+ xtlb_handler = (vaddr_t)&xtlb_miss;
+ build_trampoline(TLB_MISS_EXC_VEC, xtlb_handler);
+
+ /*
+ * Turn off bootstrap exception vectors.
+ * (this is done by PMON already, but it doesn't hurt to be safe)
+ */
+
+ setsr(getsr() & ~SR_BOOT_EXC_VEC);
+ proc0.p_md.md_regs->sr = getsr();
+
+#ifdef DDB
+ db_machine_init();
+ if (boothowto & RB_KDB)
+ Debugger();
+#endif
+
+ /*
+ * Return the new kernel stack pointer.
+ */
+
+ return ((vaddr_t)proc0paddr + USPACE - 64);
+
+unsupported:
+ pmon_printf("Halting system.\nPress enter to return to PMON\n");
+ cngetc();
+ return 0; /* causes us to return to pmon */
+}
+
+/*
+ * Build a tlb trampoline
+ */
+void
+build_trampoline(vaddr_t addr, vaddr_t dest)
+{
+ const uint32_t insns[] = {
+ 0x3c1a0000, /* lui k0, imm16 */
+ 0x675a0000, /* daddiu k0, k0, imm16 */
+ 0x001ad438, /* dsll k0, k0, 0x10 */
+ 0x675a0000, /* daddiu k0, k0, imm16 */
+ 0x001ad438, /* dsll k0, k0, 0x10 */
+ 0x675a0000, /* daddiu k0, k0, imm16 */
+ 0x03400008, /* jr k0 */
+ 0x00000000 /* nop */
+ };
+ uint32_t *dst = (uint32_t *)addr;
+ const uint32_t *src = insns;
+ uint32_t a, b, c, d;
+
+ /*
+ * Decompose the handler address in the four components which,
+ * added with sign extension, will produce the correct address.
+ */
+ d = dest & 0xffff;
+ dest >>= 16;
+ if (d & 0x8000)
+ dest++;
+ c = dest & 0xffff;
+ dest >>= 16;
+ if (c & 0x8000)
+ dest++;
+ b = dest & 0xffff;
+ dest >>= 16;
+ if (b & 0x8000)
+ dest++;
+ a = dest & 0xffff;
+
+ /*
+ * Build the trampoline, skipping noop computations.
+ */
+ *dst++ = *src++ | a;
+ if (b != 0)
+ *dst++ = *src++ | b;
+ else
+ src++;
+ *dst++ = *src++;
+ if (c != 0)
+ *dst++ = *src++ | c;
+ else
+ src++;
+ *dst++ = *src++;
+ if (d != 0)
+ *dst++ = *src++ | d;
+ else
+ src++;
+ *dst++ = *src++;
+ *dst++ = *src++;
+
+ /*
+ * Note that we keep the delay slot instruction a nop, instead
+ * of branching to the second instruction of the handler and
+ * having its first instruction in the delay slot, so that the
+ * tlb handler is free to use k0 immediately.
+ */
+}
+
+/*
+ * Decode boot options.
+ */
+static void
+dobootopts(int argc)
+{
+ const char *arg;
+ const char *cp;
+ int ignore = 1;
+ int i;
+
+ /*
+ * Parse the boot command line.
+ *
+ * It should be of the form `boot [flags] filename [args]', so we
+ * need to ignore flags to the boot command.
+ * To achieve this, we ignore argc[0], which is the `boot' command
+ * itself, and ignore arguments starting with dashes until the
+ * boot file has been found.
+ */
+
+ for (i = 1; i < argc; i++) {
+ arg = pmon_getarg(i);
+ if (arg == NULL)
+ continue;
+
+ if (*arg != '-') {
+ /* found filename or non-option argument */
+ ignore = 0;
+ continue;
+ }
+
+ if (ignore)
+ continue;
+
+ for (cp = arg + 1; *cp != '\0'; cp++)
+ switch (*cp) {
+ case '-':
+ break;
+ case 'a':
+ boothowto |= RB_ASKNAME;
+ break;
+ case 'c':
+ boothowto |= RB_CONFIG;
+ break;
+ case 'd':
+ boothowto |= RB_KDB;
+ break;
+ case 's':
+ boothowto |= RB_SINGLE;
+ break;
+ default:
+ pmon_printf("unrecognized option `%c'", *cp);
+ break;
+ }
+ }
+
+ /*
+ * Consider parsing the `karg' environment variable here too?
+ */
+}
+
+
+/*
+ * Console initialization: called early on from main, before vm init or startup.
+ * Do enough configuration to choose and initialize a console.
+ */
+void
+consinit()
+{
+ static int console_ok = 0;
+
+ if (console_ok == 0) {
+ cninit();
+ console_ok = 1;
+ }
+}
+
+/*
+ * cpu_startup: allocate memory for variable-sized tables, initialize CPU, and
+ * do auto-configuration.
+ */
+void
+cpu_startup()
+{
+ vaddr_t minaddr, maxaddr;
+
+ /*
+ * Good {morning,afternoon,evening,night}.
+ */
+ printf(version);
+ printf("real mem = %u (%uMB)\n", ptoa(physmem),
+ ptoa(physmem)/1024/1024);
+
+ /*
+ * Allocate a submap for exec arguments. This map effectively
+ * limits the number of processes exec'ing at any time.
+ */
+ minaddr = vm_map_min(kernel_map);
+ exec_map = uvm_km_suballoc(kernel_map, &minaddr, &maxaddr,
+ 16 * NCARGS, VM_MAP_PAGEABLE, FALSE, NULL);
+ /* Allocate a submap for physio. */
+ phys_map = uvm_km_suballoc(kernel_map, &minaddr, &maxaddr,
+ VM_PHYS_SIZE, 0, FALSE, NULL);
+
+ printf("avail mem = %u (%uMB)\n", ptoa(uvmexp.free),
+ ptoa(uvmexp.free)/1024/1024);
+
+ /*
+ * Set up buffers, so they can be used to read disk labels.
+ */
+ bufinit();
+
+ /*
+ * Configure the system.
+ */
+ if (boothowto & RB_CONFIG) {
+#ifdef BOOT_CONFIG
+ user_config();
+#else
+ printf("kernel does not support -c; continuing..\n");
+#endif
+ }
+}
+
+/*
+ * Machine dependent system variables.
+ */
+int
+cpu_sysctl(name, namelen, oldp, oldlenp, newp, newlen, p)
+ int *name;
+ u_int namelen;
+ void *oldp;
+ size_t *oldlenp;
+ void *newp;
+ size_t newlen;
+ struct proc *p;
+{
+ /* All sysctl names at this level are terminal. */
+ if (namelen != 1)
+ return ENOTDIR; /* Overloaded */
+
+ switch (name[0]) {
+ case CPU_KBDRESET:
+ if (securelevel > 0)
+ return (sysctl_rdint(oldp, oldlenp, newp, kbd_reset));
+ return (sysctl_int(oldp, oldlenp, newp, newlen, &kbd_reset));
+ default:
+ return EOPNOTSUPP;
+ }
+}
+
+/*
+ * Set registers on exec for native exec format. For o64/64.
+ */
+void
+setregs(p, pack, stack, retval)
+ struct proc *p;
+ struct exec_package *pack;
+ u_long stack;
+ register_t *retval;
+{
+ extern struct proc *machFPCurProcPtr;
+
+ bzero((caddr_t)p->p_md.md_regs, sizeof(struct trap_frame));
+ p->p_md.md_regs->sp = stack;
+ p->p_md.md_regs->pc = pack->ep_entry & ~3;
+ p->p_md.md_regs->t9 = pack->ep_entry & ~3; /* abicall req */
+ p->p_md.md_regs->sr = SR_FR_32 | SR_XX | SR_KSU_USER | SR_KX | SR_UX |
+ SR_EXL | SR_INT_ENAB;
+ p->p_md.md_regs->sr |= idle_mask & SR_INT_MASK;
+ p->p_md.md_regs->ic = (idle_mask << 8) & IC_INT_MASK;
+ p->p_md.md_flags &= ~MDP_FPUSED;
+ if (machFPCurProcPtr == p)
+ machFPCurProcPtr = NULL;
+ p->p_md.md_ss_addr = 0;
+ p->p_md.md_pc_ctrl = 0;
+ p->p_md.md_watch_1 = 0;
+ p->p_md.md_watch_2 = 0;
+
+ retval[1] = 0;
+}
+
+
+int waittime = -1;
+
+void
+boot(int howto)
+{
+
+ /* Take a snapshot before clobbering any registers. */
+ if (curproc)
+ savectx(curproc->p_addr, 0);
+
+ if (cold) {
+ /*
+ * If the system is cold, just halt, unless the user
+ * explicitly asked for reboot.
+ */
+ if ((howto & RB_USERREQ) == 0)
+ howto |= RB_HALT;
+ goto haltsys;
+ }
+
+ boothowto = howto;
+ if ((howto & RB_NOSYNC) == 0 && waittime < 0) {
+ extern struct proc proc0;
+ /* fill curproc with live object */
+ if (curproc == NULL)
+ curproc = &proc0;
+ /*
+ * Synchronize the disks...
+ */
+ waittime = 0;
+ vfs_shutdown();
+
+ /*
+ * If we've been adjusting the clock, the todr will be out of
+ * sync; adjust it now.
+ */
+ if ((howto & RB_TIMEBAD) == 0) {
+ resettodr();
+ } else {
+ printf("WARNING: not updating battery clock\n");
+ }
+ }
+
+ uvm_shutdown();
+ (void) splhigh(); /* Extreme priority. */
+
+ if (howto & RB_DUMP)
+ dumpsys();
+
+haltsys:
+ doshutdownhooks();
+
+ if (howto & RB_HALT) {
+ if (howto & RB_POWERDOWN) {
+ printf("System Power Down.\n");
+ REGVAL(BONITO_GPIODATA) &= ~0x00000001;
+ REGVAL(BONITO_GPIOIE) &= ~0x00000001;
+ } else
+ printf("System Halt.\n");
+ } else {
+ void (*__reset)(void) = (void (*)(void))RESET_EXC_VEC;
+ printf("System restart.\n");
+ __reset();
+ }
+
+ for (;;) ;
+ /*NOTREACHED*/
+}
+
+u_long dumpmag = 0x8fca0101; /* Magic number for savecore. */
+int dumpsize = 0; /* Also for savecore. */
+long dumplo = 0;
+
+void
+dumpconf(void)
+{
+ int nblks;
+
+ if (dumpdev == NODEV ||
+ (nblks = (bdevsw[major(dumpdev)].d_psize)(dumpdev)) == 0)
+ return;
+ if (nblks <= ctod(1))
+ return;
+
+ dumpsize = ptoa(physmem);
+ if (dumpsize > atop(round_page(dbtob(nblks - dumplo))))
+ dumpsize = atop(round_page(dbtob(nblks - dumplo)));
+ else if (dumplo == 0)
+ dumplo = nblks - btodb(ptoa(physmem));
+
+ /*
+ * Don't dump on the first page in case the dump device includes a
+ * disk label.
+ */
+ if (dumplo < btodb(PAGE_SIZE))
+ dumplo = btodb(PAGE_SIZE);
+}
+
+void
+dumpsys()
+{
+ /* XXX TBD */
+}
+
+/*
+ * Convert an ASCII string into an integer.
+ */
+static u_long
+atoi(const char *s, uint b)
+{
+ int c;
+ uint base = b, d;
+ int neg = 0;
+ u_long val = 0;
+
+ if (s == NULL || *s == '\0')
+ return 0;
+
+ /* Skip spaces if any. */
+ do {
+ c = *s++;
+ } while (c == ' ' || c == '\t');
+
+ /* Parse sign, allow more than one (compat). */
+ while (c == '-') {
+ neg = !neg;
+ c = *s++;
+ }
+
+ /* Parse base specification, if any. */
+ if (base == 0 && c == '0') {
+ c = *s++;
+ switch (c) {
+ case 'X':
+ case 'x':
+ base = 16;
+ c = *s++;
+ break;
+ case 'B':
+ case 'b':
+ base = 2;
+ c = *s++;
+ break;
+ default:
+ base = 8;
+ break;
+ }
+ }
+
+ /* Parse number proper. */
+ for (;;) {
+ if (c >= '0' && c <= '9')
+ d = c - '0';
+ else if (c >= 'a' && c <= 'z')
+ d = c - 'a' + 10;
+ else if (c >= 'A' && c <= 'Z')
+ d = c - 'A' + 10;
+ else
+ break;
+ if (d >= base)
+ break;
+ val *= base;
+ val += d;
+ c = *s++;
+ }
+
+ return neg ? -val : val;
+}
+
+/*
+ * Early console through pmon routines.
+ */
+
+int
+pmoncngetc(dev_t dev)
+{
+ /*
+ * PMON does not give us a getc routine. So try to get a whole line
+ * and return it char by char, trying not to lose the \n. Kind
+ * of ugly but should work.
+ */
+ static char buf[1 + PMON_MAXLN];
+ static char *bufpos = buf;
+ int c;
+
+ if (*bufpos == '\0') {
+ bufpos = buf;
+ if (pmon_gets(buf) == NULL) {
+ /* either an empty line or EOF. assume the former */
+ return (int)'\n';
+ } else {
+ /* put back the \n sign */
+ buf[strlen(buf)] = '\n';
+ }
+ }
+
+ c = (int)*bufpos++;
+ if (bufpos - buf > PMON_MAXLN) {
+ bufpos = buf;
+ *bufpos = '\0';
+ }
+
+ return c;
+}
+
+void
+pmoncnputc(dev_t dev, int c)
+{
+ if (c == '\n')
+ pmon_printf("\n");
+ else
+ pmon_printf("%c", c);
+}