diff options
author | Todd C. Miller <millert@cvs.openbsd.org> | 2004-08-11 18:51:05 +0000 |
---|---|---|
committer | Todd C. Miller <millert@cvs.openbsd.org> | 2004-08-11 18:51:05 +0000 |
commit | fec58ae8c4bc4002d602ce897ee4f88767e23679 (patch) | |
tree | e53b89ab6350db64b6d90a958cbbf40685d7d5c2 /sys/arch/i386 | |
parent | 59be772d42b86c228dc34c00c6c7be3a6b9bc0a9 (diff) |
In dkcsumattach() avoid modifying bootdev until we are done matching.
Also deal with mixed scsi/ide/whatever setups. Based on a diff from
Fred de Jong. OK deraadt@ and tested by several people.
Diffstat (limited to 'sys/arch/i386')
-rw-r--r-- | sys/arch/i386/i386/dkcsum.c | 29 |
1 files changed, 24 insertions, 5 deletions
diff --git a/sys/arch/i386/i386/dkcsum.c b/sys/arch/i386/i386/dkcsum.c index 3122a8b2612..7071689ad1f 100644 --- a/sys/arch/i386/i386/dkcsum.c +++ b/sys/arch/i386/i386/dkcsum.c @@ -1,4 +1,4 @@ -/* $OpenBSD: dkcsum.c,v 1.11 2003/06/03 20:31:07 deraadt Exp $ */ +/* $OpenBSD: dkcsum.c,v 1.12 2004/08/11 18:51:03 millert Exp $ */ /*- * Copyright (c) 1997 Niklas Hallqvist. All rights reserved. @@ -58,7 +58,7 @@ dkcsumattach() struct device *dv; struct buf *bp; struct bdevsw *bdsw; - dev_t dev; + dev_t dev, pribootdev, altbootdev; int error; u_int32_t csum; bios_diskinfo_t *bdi, *hit; @@ -67,6 +67,8 @@ dkcsumattach() if (bios_diskinfo == NULL || bios_cksumlen * DEV_BSIZE > MAXBSIZE) return; + pribootdev = altbootdev = 0; + /* * XXX Whatif DEV_BSIZE is changed to something else than the BIOS * blocksize? Today, /boot doesn't cover that case so neither need @@ -163,22 +165,37 @@ dkcsumattach() continue; } - /* Fixup bootdev if units match. This means that all of + /* + * Fixup bootdev if units match. This means that all of * hd*, sd*, wd*, will be interpreted the same. Not 100% * backwards compatible, but sd* and wd* should be phased- * out in the bootblocks. */ + + /* B_TYPE dependent hd unit counting bootblocks */ + if ((B_TYPE(bootdev) == B_TYPE(hit->bsd_dev)) && + (B_UNIT(bootdev) == B_UNIT(hit->bsd_dev))) { + int type, ctrl, adap, part, unit; + + type = major(bp->b_dev); + adap = B_ADAPTOR(bootdev); + ctrl = B_CONTROLLER(bootdev); + unit = DISKUNIT(bp->b_dev); + part = B_PARTITION(bootdev); + + pribootdev = MAKEBOOTDEV(type, ctrl, adap, unit, part); + } + /* B_TYPE independent hd unit counting bootblocks */ if (B_UNIT(bootdev) == (hit->bios_number & 0x7F)) { int type, ctrl, adap, part, unit; - /* Translate to MAKEBOOTDEV() style */ type = major(bp->b_dev); adap = B_ADAPTOR(bootdev); ctrl = B_CONTROLLER(bootdev); unit = DISKUNIT(bp->b_dev); part = B_PARTITION(bootdev); - bootdev = MAKEBOOTDEV(type, ctrl, adap, unit, part); + altbootdev = MAKEBOOTDEV(type, ctrl, adap, unit, part); } /* This will overwrite /boot's guess, just so you remember */ @@ -186,6 +203,8 @@ dkcsumattach() DISKUNIT(bp->b_dev), RAW_PART); hit->flags |= BDI_PICKED; } + bootdev = pribootdev ? pribootdev : altbootdev ? altbootdev : bootdev; + bp->b_flags |= B_INVAL; brelse(bp); } |