/* $NetBSD: bootxx.c,v 1.9 2014/03/26 17:43:11 christos Exp $ */ /*- * Copyright (C) 1999 Tsubai Masanari. All rights reserved. * * 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. * 3. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * 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 #include #include #include #include struct shared_bbinfo bbinfo = { { NEWSMIPS_BBINFO_MAGIC }, /* bbi_magic[] */ 0, /* bbi_block_size */ SHARED_BBINFO_MAXBLOCKS, /* bbi_block_count */ { 0 } /* bbi_block_tagle[] */ }; #ifndef DEFAULT_ENTRY_POINT #define DEFAULT_ENTRY_POINT 0xa0700000 #endif void bootxx(uint32_t, uint32_t, uint32_t, uint32_t, uint32_t, uint32_t); void (*entry_point)(uint32_t, uint32_t, uint32_t, uint32_t, void *) = (void *)DEFAULT_ENTRY_POINT; #ifdef BOOTXX_DEBUG # define DPRINTF printf #else # define DPRINTF while (0) printf #endif char *devs[] = { "sd", "fh", "fd", NULL, NULL, "rd", "st" }; struct apbus_sysinfo *_sip; int apbus = 0; void bootxx(uint32_t a0, uint32_t a1, uint32_t a2, uint32_t a3, uint32_t a4, uint32_t a5) { int fd, blk, bs; int ctlr, unit, part, type; int i; int bootdev = a1; char *addr; char devname[32]; /* * XXX a3 contains: * maxmem (nws-3xxx) * argv (apbus-based machine) */ if (a3 & 0x80000000) apbus = 1; else apbus = 0; printf("NetBSD/newsmips Primary Boot\n"); DPRINTF("\n"); DPRINTF("a0 %x\n", a0); DPRINTF("a1 %x\n", a1); DPRINTF("a2 %x (%s)\n", a2, (char *)a2); DPRINTF("a3 %x\n", a3); DPRINTF("a4 %x\n", a4); DPRINTF("a5 %x\n", a5); DPRINTF("block_size = %d\n", bbinfo.bbi_block_size); DPRINTF("block_count = %d\n", bbinfo.bbi_block_count); DPRINTF("entry_point = %p\n", entry_point); if (apbus) { strcpy(devname, (char *)a1); fd = apcall_open(devname, 0); } else { /* sd(ctlr, lun, part, bus?, host) */ ctlr = BOOTDEV_CTLR(bootdev); unit = BOOTDEV_UNIT(bootdev); part = BOOTDEV_PART(bootdev); type = BOOTDEV_TYPE(bootdev); if (devs[type] == NULL) { printf("unknown bootdev (0x%x)\n", bootdev); return; } snprintf(devname, sizeof(devname), "%s(%d,%d,%d)", devs[type], ctlr, unit, part); fd = rom_open(devname, 0); } if (fd == -1) { printf("cannot open %s\n", devname); return; } addr = (char *)entry_point; bs = bbinfo.bbi_block_size; DPRINTF("reading block:"); for (i = 0; i < bbinfo.bbi_block_count; i++) { blk = bbinfo.bbi_block_table[i]; DPRINTF(" %d", blk); if (apbus) { apcall_lseek(fd, blk * 512, 0); apcall_read(fd, addr, bs); } else { rom_lseek(fd, blk * 512, 0); rom_read(fd, addr, bs); } addr += bs; } DPRINTF(" done\n"); if (apbus) apcall_close(fd); else rom_close(fd); (*entry_point)(a0, a1, a2, a3, _sip); } void putchar(int x) { char c = x; if (apbus) apcall_write(1, &c, 1); else rom_write(1, &c, 1); }