2 * Copyright (c) 2008 John Hay
3 * Copyright (c) 1998 Robert Nordier
6 * Redistribution and use in source and binary forms are freely
7 * permitted provided that the above copyright notice and this
8 * paragraph and the following disclaimer are duplicated in all
11 * This software is provided "AS IS" and without any express or
12 * implied warranties, including, without limitation, the implied
13 * warranties of merchantability and fitness for a particular
17 #include <sys/cdefs.h>
18 __FBSDID("$FreeBSD$");
20 #include <sys/param.h>
21 #include <sys/disklabel.h>
22 #include <sys/diskmbr.h>
23 #include <sys/dirent.h>
24 #include <sys/reboot.h>
26 #include <machine/elf.h>
32 #define RBX_ASKNAME 0x0 /* -a */
33 #define RBX_SINGLE 0x1 /* -s */
34 /* 0x2 is reserved for log2(RB_NOSYNC). */
35 /* 0x3 is reserved for log2(RB_HALT). */
36 /* 0x4 is reserved for log2(RB_INITNAME). */
37 #define RBX_DFLTROOT 0x5 /* -r */
38 /* #define RBX_KDB 0x6 -d */
39 /* 0x7 is reserved for log2(RB_RDONLY). */
40 /* 0x8 is reserved for log2(RB_DUMP). */
41 /* 0x9 is reserved for log2(RB_MINIROOT). */
42 #define RBX_CONFIG 0xa /* -c */
43 #define RBX_VERBOSE 0xb /* -v */
44 /* #define RBX_SERIAL 0xc -h */
45 /* #define RBX_CDROM 0xd -C */
46 /* 0xe is reserved for log2(RB_POWEROFF). */
47 #define RBX_GDB 0xf /* -g */
48 /* #define RBX_MUTE 0x10 -m */
49 /* 0x11 is reserved for log2(RB_SELFTEST). */
50 /* 0x12 is reserved for boot programs. */
51 /* 0x13 is reserved for boot programs. */
52 /* #define RBX_PAUSE 0x14 -p */
53 /* #define RBX_QUIET 0x15 -q */
54 #define RBX_NOINTR 0x1c /* -n */
55 /* 0x1d is reserved for log2(RB_MULTIPLE) and is just misnamed here. */
56 /* #define RBX_DUAL 0x1d -D */
57 /* 0x1f is reserved for log2(RB_BOOTINFO). */
59 /* pass: -a, -s, -r, -v, -g */
60 #define RBX_MASK (OPT_SET(RBX_ASKNAME) | OPT_SET(RBX_SINGLE) | \
61 OPT_SET(RBX_DFLTROOT) | \
62 OPT_SET(RBX_VERBOSE) | \
65 #define PATH_DOTCONFIG "/boot.config"
66 #define PATH_CONFIG "/boot/config"
67 #define PATH_KERNEL "/boot/kernel/kernel"
73 #define OPT_SET(opt) (1 << (opt))
74 #define OPT_CHECK(opt) ((opts) & OPT_SET(opt))
76 static const char optstr[NOPT] = "agnrsv";
77 static const unsigned char flags[NOPT] = {
86 static unsigned dsk_start;
88 static char kname[1024];
90 static uint8_t dsk_meta;
93 static int disk_layout;
95 #define DL_RAW 1 /* Dangerously dedicated */
96 #define DL_SLICE 2 /* Use only slices (DOS partitions) */
97 #define DL_SLICEPART 3 /* Use slices and partitions */
99 static void load(void);
100 static int parse(void);
101 static int xfsread(ino_t, void *, size_t);
102 static int dskread(void *, unsigned, unsigned);
103 static int drvread(void *, unsigned, unsigned);
104 #ifdef FIXUP_BOOT_DRV
105 static void fixup_boot_drv(caddr_t, int, int, int);
111 #define DPRINTF(fmt, ...) printf(fmt, __VA_ARGS__)
113 #define DPRINTF(fmt, ...)
117 xfsread(ino_t inode, void *buf, size_t nbyte)
119 if ((size_t)fsread(inode, buf, nbyte) != nbyte)
148 if (s - cmd < sizeof(cmd) - 1)
163 dmadat = (void *)(0x1c0000);
164 p_memset((char *)dmadat, 0, 32 * 1024);
167 printf("FreeBSD ARM (%s) boot2 v%d.%d\n", bt, 0, 4);
171 /* Process configuration file */
172 if ((ino = lookup(PATH_CONFIG)) ||
173 (ino = lookup(PATH_DOTCONFIG)))
174 fsread(ino, cmd, sizeof(cmd));
179 printf("%s: %s\n", PATH_CONFIG, cmd);
180 /* Do not process this command twice */
185 strcpy(kname, PATH_KERNEL);
187 /* Present the user with the boot2 prompt. */
189 printf("\nDefault: %s\nboot: ", kname);
191 (OPT_CHECK(RBX_NOINTR) == 0 && (c = getc(2)) != 0))
196 DPRINTF("cmd is '%s'\n", cmd);
208 static Elf32_Phdr ep[2];
213 #ifdef FIXUP_BOOT_DRV
217 staddr = (caddr_t)0xffffffff;
220 if (!(ino = lookup(kname))) {
222 printf("No %s\n", kname);
225 DPRINTF("Found %s\n", kname);
226 if (xfsread(ino, &eh, sizeof(eh)))
229 printf("Invalid %s\n", "format");
233 for (j = i = 0; i < eh.e_phnum && j < 2; i++) {
234 if (xfsread(ino, ep + j, sizeof(ep[0])))
236 if (ep[j].p_type == PT_LOAD)
239 for (i = 0; i < 2; i++) {
240 p = (caddr_t)(ep[i].p_paddr & 0x0fffffff);
241 fs_off = ep[i].p_offset;
242 #ifdef FIXUP_BOOT_DRV
243 if (staddr == (caddr_t)0xffffffff)
245 klen += ep[i].p_filesz;
247 if (xfsread(ino, p, ep[i].p_filesz))
250 addr = eh.e_entry & 0x0fffffff;
251 DPRINTF("Entry point %x for %s\n", addr, kname);
253 #ifdef FIXUP_BOOT_DRV
254 fixup_boot_drv(staddr, klen, bootslice, bootpart);
256 ((void(*)(int))addr)(RB_BOOTINFO /* XXX | (opts & RBX_MASK) */);
266 while ((c = *arg++)) {
267 if (c == ' ' || c == '\t' || c == '\n')
269 for (p = arg; *p && *p != '\n' && *p != ' ' && *p != '\t'; p++);
274 while ((c = *arg++)) {
275 for (i = 0; c != optstr[i]; i++)
278 opts ^= OPT_SET(flags[i]);
282 /* look for ad0s1a:... | ad0s1:... */
283 if (strlen(arg) > 6 && arg[0] == 'a' &&
284 arg[1] == 'd' && arg[3] == 's' &&
285 (arg[5] == ':' || arg[6] == ':')) {
286 /* XXX Should also handle disk. */
287 bootslice = arg[4] - '0';
288 if (bootslice < 1 || bootslice > 4)
292 bootpart = arg[5] - 'a';
293 if (bootpart < 0 || bootpart > 7)
300 /* look for ad0a:... */
301 } else if (strlen(arg) > 4 && arg[0] == 'a' &&
302 arg[1] == 'd' && arg[2] == '0' && arg[4] == ':') {
304 bootpart = arg[3] - 'a';
305 if (bootpart < 0 || bootpart > 7)
310 if ((i = ep - arg)) {
311 if ((size_t)i >= sizeof(kname))
313 memcpy(kname, arg, i + 1);
322 * dskread() will try to handle the disk layouts that are typically
324 * - raw or "Dangerously Dedicated" mode. No real slice table, just the
325 * default one that is included with bsdlabel -B. Typically this is
326 * used with ROOTDEVNAME=\"ufs:ad0a\".
327 * - slice only. Only a slice table is installed with no bsd label or
328 * bsd partition table. This is typically used with
329 * ROOTDEVNAME=\"ufs:ad0s1\".
330 * - slice + bsd label + partition table. This is typically done with
331 * with fdisk + bsdlabel and is used with ROOTDEVNAME=\"ufs:ad0s1a\".
334 dskread(void *buf, unsigned lba, unsigned nblk)
336 struct dos_partition *dp;
342 sec = dmadat->secbuf;
344 if (drvread(sec, DOSBBSECTOR, 1))
346 dp = (void *)(sec + DOSPARTOFF);
347 if (bootslice != 0) {
349 if (dp[i].dp_typ != DOSPTYP_386BSD)
352 for (i = 0; i < NDOSPART; i++) {
353 if ((dp[i].dp_typ == DOSPTYP_386BSD) &&
354 (dp[i].dp_flag == 0x80))
360 DPRINTF("Found an active fbsd slice. (%d)\n", i + 1);
362 * Although dp_start is aligned within the disk
363 * partition structure, DOSPARTOFF is 446, which
364 * is only word (2) aligned, not longword (4)
365 * aligned. Cope by using memcpy to fetch the
366 * start of this partition.
368 memcpy(&dsk_start, &dp[i].dp_start, 4);
369 dsk_start = swap32(dsk_start);
370 DPRINTF("dsk_start %x\n", dsk_start);
371 if ((bootslice == 4) && (dsk_start == 0)) {
372 disk_layout = DL_RAW;
376 if (drvread(sec, dsk_start + LABELSECTOR, 1))
378 d = (void *)(sec + LABELOFFSET);
379 if ((d->d_magic == DISKMAGIC && d->d_magic2 == DISKMAGIC) ||
380 (swap32(d->d_magic) == DISKMAGIC &&
381 swap32(d->d_magic2) == DISKMAGIC)) {
382 DPRINTF("p_size = %x\n",
383 !d->d_partitions[bootpart].p_size);
384 if (!d->d_partitions[bootpart].p_size) {
385 printf("Invalid partition\n");
388 DPRINTF("p_offset %x, RAW %x\n",
389 swap32(d->d_partitions[bootpart].p_offset),
390 swap32(d->d_partitions[RAW_PART].p_offset));
391 dsk_start += swap32(d->d_partitions[bootpart].p_offset);
392 dsk_start -= swap32(d->d_partitions[RAW_PART].p_offset);
393 if ((disk_layout == DL_UNKNOWN) && (bootslice == 0))
394 disk_layout = DL_RAW;
395 else if (disk_layout == DL_UNKNOWN)
396 disk_layout = DL_SLICEPART;
398 disk_layout = DL_SLICE;
399 DPRINTF("Invalid %s\n", "label");
401 DPRINTF("bootslice %d, bootpart %d, dsk_start %u\n", bootslice,
402 bootpart, dsk_start);
405 return drvread(buf, dsk_start + lba, nblk);
409 drvread(void *buf, unsigned lba, unsigned nblk)
411 static unsigned c = 0x2d5c7c2f;
413 printf("%c\b", c = c << 8 | c >> 24);
414 return (avila_read((char *)buf, lba, nblk));
417 #ifdef FIXUP_BOOT_DRV
419 * fixup_boot_drv() will try to find the ROOTDEVNAME spec in the kernel
420 * and change it to what was specified on the comandline or /boot.conf
421 * file or to what was encountered on the disk. It will try to handle 3
422 * different disk layouts, raw (dangerously dedicated), slice only and
423 * slice + partition. It will look for the following strings in the
424 * kernel, but if it is one of the first three, the string in the kernel
425 * must use the correct form to match the actual disk layout:
430 * In the case of the first three strings, only the "a" at the end and
431 * the "1" after the "s" will be modified, if they exist. The string
432 * length will not be changed. In the case of the last string, the
433 * whole string will be built up and nul, '\0' terminated.
436 fixup_boot_drv(caddr_t addr, int klen, int bs, int bp)
438 const u_int8_t op[] = "ufs:ROOTDEVNAME";
439 const u_int8_t op2[] = "ufs:ad0";
442 DPRINTF("fixup_boot_drv: 0x%x, %d, slice %d, partition %d\n",
443 (int)addr, klen, bs, bp);
448 ps = memmem(addr, klen, op, sizeof(op));
450 p = ps + 4; /* past ufs: */
451 DPRINTF("Found it at 0x%x\n", (int)ps);
452 p[0] = 'a'; p[1] = 'd'; p[2] = '0'; /* ad0 */
459 if (disk_layout != DL_SLICE) {
460 /* append partition */
465 ps = memmem(addr, klen, op2, sizeof(op2) - 1);
467 p = ps + sizeof(op2) - 1;
468 DPRINTF("Found it at 0x%x\n", (int)ps);
479 printf("Could not locate \"%s\" to fix kernel boot device, "
480 "check ROOTDEVNAME is set\n", op);
483 DPRINTF("Changed boot device to %s\n", ps);