summaryrefslogtreecommitdiff
path: root/sys/arch
diff options
context:
space:
mode:
Diffstat (limited to 'sys/arch')
-rw-r--r--sys/arch/i386/stand/installboot/installboot.c151
1 files changed, 86 insertions, 65 deletions
diff --git a/sys/arch/i386/stand/installboot/installboot.c b/sys/arch/i386/stand/installboot/installboot.c
index a1a068c5bd3..3dfaa3164c9 100644
--- a/sys/arch/i386/stand/installboot/installboot.c
+++ b/sys/arch/i386/stand/installboot/installboot.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: installboot.c,v 1.6 1997/08/23 14:31:49 mickey Exp $ */
+/* $OpenBSD: installboot.c,v 1.7 1997/08/29 20:08:36 mickey Exp $ */
/* $NetBSD: installboot.c,v 1.5 1995/11/17 23:23:50 gwr Exp $ */
/*
@@ -56,9 +56,10 @@
#include <stdio.h>
#include <string.h>
#include <unistd.h>
+#include <util.h>
int verbose, nowrite, heads, nsectors;
-char *boot, *proto, *dev;
+char *boot, *proto, *dev, *realdev;
struct nlist nl[] = {
#define X_BLOCK_COUNT 0
{{"_block_count"}},
@@ -69,14 +70,14 @@ struct nlist nl[] = {
u_int8_t *block_count_p; /* block count var. in prototype image */
u_int8_t *block_table_p; /* block number array in prototype image */
-int maxblocknum; /* size of this array */
+int maxblocknum; /* size of this array */
#ifdef CPU_BIOSDEV
int biosdev;
#endif
char *loadprotoblocks __P((char *, long *));
-int loadblocknums __P((char *, int));
+int loadblocknums __P((char *, int, struct disklabel *));
static void devread __P((int, void *, daddr_t, size_t, char *));
static void usage __P((void));
static int record_block
@@ -100,6 +101,9 @@ main(argc, argv)
char *protostore;
long protosize;
struct stat sb;
+ struct disklabel dl;
+ struct dos_mbr mbr;
+ struct dos_partition *dp;
nsectors = heads = -1;
while ((c = getopt(argc, argv, "vnh:s:")) != EOF) {
@@ -129,13 +133,7 @@ main(argc, argv)
boot = argv[optind];
proto = argv[optind + 1];
- dev = argv[optind + 2];
-
- if (verbose) {
- printf("boot: %s\n", boot);
- printf("proto: %s\n", proto);
- printf("device: %s\n", dev);
- }
+ realdev = dev = argv[optind + 2];
#ifdef CPU_BIOSDEV
if (heads == -1 || nsectors == -1) {
@@ -163,6 +161,24 @@ main(argc, argv)
}
#endif
+ /* Open and check raw disk device */
+ if ((devfd = opendev(dev, (nowrite? O_RDONLY:O_RDWR),
+ OPENDEV_PART, &realdev)) < 0)
+ err(1, "open: %s", realdev);
+
+ if (verbose) {
+ fprintf(stderr, "boot: %s\n", boot);
+ fprintf(stderr, "proto: %s\n", proto);
+ fprintf(stderr, "device: %s\n", realdev);
+ }
+
+ if (ioctl(devfd, DIOCGDINFO, &dl) != 0)
+ err(1, "disklabel: %s", realdev);
+
+ /* check disklabel */
+ if (dl.d_magic != DISKMAGIC)
+ err(1, "bad disklabel magic=%0x8x", dl.d_magic);
+
/* Load proto blocks into core */
if ((protostore = loadprotoblocks(proto, &protosize)) == NULL)
exit(1);
@@ -171,38 +187,53 @@ main(argc, argv)
if (protosize & (DEV_BSIZE - 1))
err(1, "proto bootblock bad size=%ld", protosize);
- /* Open and check raw disk device */
- if ((devfd = open(dev, O_RDONLY, 0)) < 0)
- err(1, "open: %s", dev);
+ /* Write patched proto bootblocks into the superblock */
+ if (protosize > SBSIZE - DEV_BSIZE)
+ errx(1, "proto bootblocks too big");
if (fstat(devfd, &sb) < 0)
- err(1, "stat: %s", dev);
+ err(1, "stat: %s", realdev);
if (!S_ISCHR(sb.st_mode))
- errx(1, "%s: Not a character device", dev);
+ errx(1, "%s: Not a character device", realdev);
/* Extract and load block numbers */
- if (loadblocknums(boot, devfd) != 0)
+ if (loadblocknums(boot, devfd, &dl) != 0)
exit(1);
- (void)close(devfd);
-
- if (nowrite)
- return 0;
-
- /* Write patched proto bootblocks into the superblock */
- if (protosize > SBSIZE - DEV_BSIZE)
- errx(1, "proto bootblocks too big");
-
- if ((devfd = open(dev, (nowrite? O_RDONLY:O_RDWR), 0)) < 0)
- err(1, "open: %s", dev);
-
/* Sync filesystems (to clean in-memory superblock?) */
- sync();
+ sync(); sleep(1);
+
+ if (lseek(devfd, (off_t)DOSBBSECTOR, SEEK_SET) < 0 ||
+ read(devfd, &mbr, sizeof(mbr)) < sizeof(mbr))
+ err(4, "can't read master boot record");
+
+ if (mbr.dmbr_sign != DOSMBR_SIGNATURE)
+ errx(1, "broken MBR");
+
+ /* Find OpenBSD partition. */
+ for (dp = mbr.dmbr_parts; dp < &mbr.dmbr_parts[NDOSPART]; dp++) {
+ if (dp->dp_size && dp->dp_typ == DOSPTYP_OPENBSD) {
+ fprintf(stderr, "using MBR partition %d: "
+ "type %d (0x%02x) offset %d (0x%x)\n",
+ dp - mbr.dmbr_parts,
+ dp->dp_typ, dp->dp_typ,
+ dp->dp_start, dp->dp_start);
+ break;
+ }
+ }
+ /* don't check for old part number, that is ;-p */
+ if (dp >= &mbr.dmbr_parts[NDOSPART])
+ errx(1, "no OpenBSD partition");
+
+ if (!nowrite) {
+ if (lseek(devfd, dp->dp_start * dl.d_secsize, SEEK_SET) < 0 ||
+ write(devfd, protostore, protosize) != protosize)
+ err(1, "write bootstrap");
+ }
- if (write(devfd, protostore, protosize) != protosize)
- err(1, "write bootstrap");
(void)close(devfd);
+
return 0;
}
@@ -275,9 +306,9 @@ loadprotoblocks(fname, size)
maxblocknum = *block_count_p;
if (verbose) {
- printf("%s: entry point %#lx\n", fname, eh.a_entry);
- printf("proto bootblock size %ld\n", *size);
- printf("room for %d filesystem blocks at %#lx\n",
+ fprintf(stderr, "%s: entry point %#lx\n", fname, eh.a_entry);
+ fprintf(stderr, "proto bootblock size %ld\n", *size);
+ fprintf(stderr, "room for %d filesystem blocks at %#lx\n",
maxblocknum, nl[X_BLOCK_TABLE].n_value);
}
@@ -310,14 +341,14 @@ devread(fd, buf, blk, size, msg)
static char sblock[SBSIZE];
int
-loadblocknums(boot, devfd)
+loadblocknums(boot, devfd, dl)
char *boot;
int devfd;
+ struct disklabel *dl;
{
int i, fd;
struct stat statbuf;
struct statfs statfsbuf;
- struct disklabel dl;
struct partition *pl;
struct fs *fs;
char *buf;
@@ -335,19 +366,6 @@ loadblocknums(boot, devfd)
/* Make sure the (probably new) boot file is on disk. */
sync(); sleep(1);
- if (ioctl(devfd, DIOCGDINFO, &dl) != 0)
- err(1, "disklabel: %s", dev);
-
- /* check disklabel */
- if (dl.d_magic != DISKMAGIC)
- err(1, "bad disklabel magic=%0x8x", dl.d_magic);
-
- /* get partition pointer */
- if (fstat(devfd, &statbuf) < 0)
- err(1, "stat: %s", dev);
-
- pl = &dl.d_partitions[DISKPART(statbuf.st_rdev)];
-
if ((fd = open(boot, O_RDONLY)) < 0)
err(1, "open: %s", boot);
@@ -373,10 +391,11 @@ loadblocknums(boot, devfd)
if (fstat(fd, &statbuf) != 0)
err(1, "fstat: %s", boot);
+ pl = &dl->d_partitions[DISKPART(statbuf.st_dev)];
close(fd);
/* Read superblock */
- devread(devfd, sblock, SBLOCK, SBSIZE, "superblock");
+ devread(devfd, sblock, pl->p_offset + SBLOCK, SBSIZE, "superblock");
fs = (struct fs *)sblock;
/* Sanity-check super-block. */
@@ -390,7 +409,7 @@ loadblocknums(boot, devfd)
errx(1, "No memory for filesystem block");
blk = fsbtodb(fs, ino_to_fsba(fs, statbuf.st_ino));
- devread(devfd, buf, blk, fs->fs_bsize, "inode");
+ devread(devfd, buf, pl->p_offset + blk, fs->fs_bsize, "inode");
ip = (struct dinode *)(buf) + ino_to_fsbo(fs, statbuf.st_ino);
/*
@@ -400,15 +419,15 @@ loadblocknums(boot, devfd)
if (ndb > maxblocknum)
errx(1, "Too many blocks");
if (verbose)
- printf("Will load %d blocks of size %d each.\n",
- ndb, fs->fs_bsize);
+ fprintf(stderr, "Will load %d blocks of size %d each.\n",
+ ndb, fs->fs_bsize);
- if (dl.d_type != 0 && dl.d_type < DTYPE_FLOPPY) {
+ if (dl->d_type != 0 && dl->d_type < DTYPE_FLOPPY) {
/* adjust disklabel w/ synthetic geometry */
if (nsectors > 0)
- dl.d_nsectors = nsectors;
+ dl->d_nsectors = nsectors;
if (heads > 0)
- dl.d_secpercyl = dl.d_nsectors * heads;
+ dl->d_secpercyl = dl->d_nsectors * heads;
}
/*
@@ -418,7 +437,7 @@ loadblocknums(boot, devfd)
bt = block_table_p;
for (i = 0; i < NDADDR && *ap && ndb; i++, ap++, ndb--)
bt += record_block(bt, pl->p_offset + fsbtodb(fs, *ap),
- fs->fs_bsize / 512, &dl);
+ fs->fs_bsize / 512, dl);
if (ndb != 0) {
/*
@@ -426,21 +445,23 @@ loadblocknums(boot, devfd)
* for more in the 2nd-level /boot anyway.
*/
blk = fsbtodb(fs, ip->di_ib[0]);
- devread(devfd, buf, blk, fs->fs_bsize, "indirect block");
+ devread(devfd, buf, pl->p_offset + blk, fs->fs_bsize,
+ "indirect block");
ap = (daddr_t *)buf;
for (; i < NINDIR(fs) && *ap && ndb; i++, ap++, ndb--)
bt += record_block(bt, pl->p_offset + fsbtodb(fs, *ap),
- fs->fs_bsize / 512, &dl);
+ fs->fs_bsize / 512, dl);
}
/* write out remaining piece */
- bt += record_block(bt, 0, 0, &dl);
+ bt += record_block(bt, 0, 0, dl);
/* and again */
- bt += record_block(bt, 0, 0, &dl);
+ bt += record_block(bt, 0, 0, dl);
*block_count_p = (bt - block_table_p) / 4;
if (verbose)
- printf("%s: %d entries total\n", boot, *block_count_p);
+ fprintf(stderr, "%s: %d entries total\n",
+ boot, *block_count_p);
return 0;
}
@@ -479,8 +500,8 @@ record_block(bt, blk, bs, dl)
((c & 0xff) << 8) | ((c & 0x300) >> 2);
if (verbose)
- printf("%2d: %2d @(%d %d %d) (%d-%d)\n",
- i, bt[3], c, bt[2], s, ss, ss + bt[3] - 1);
+ fprintf(stderr, "%2d: %2d @(%d %d %d) (%d-%d)\n",
+ i, bt[3], c, bt[2], s, ss, ss + bt[3] - 1);
if ((ss % dl->d_nsectors + l) >= dl->d_nsectors) {
ss += bt[3];