summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorTheo de Raadt <deraadt@cvs.openbsd.org>1997-02-12 02:15:30 +0000
committerTheo de Raadt <deraadt@cvs.openbsd.org>1997-02-12 02:15:30 +0000
commit2553ce04767666557e49a10098dade45ad5305db (patch)
tree6a810907dbe6d2fb7c2c11d72a772d86b374f853
parent62eb7e4d9fe3833073fcd412d31d7b35a5c1ae7a (diff)
The original disklabel code from dale had an unaligned-aligned problem
in it. This new code from rahnds and nivas will read either style of disklabel and hence will work on the mvme88k as well.
-rw-r--r--sys/arch/mvme68k/include/disklabel.h152
-rw-r--r--sys/arch/mvme68k/mvme68k/disksubr.c343
2 files changed, 303 insertions, 192 deletions
diff --git a/sys/arch/mvme68k/include/disklabel.h b/sys/arch/mvme68k/include/disklabel.h
index bf4bfe1765f..b3cc39689b6 100644
--- a/sys/arch/mvme68k/include/disklabel.h
+++ b/sys/arch/mvme68k/include/disklabel.h
@@ -1,6 +1,5 @@
-/* $OpenBSD: disklabel.h,v 1.3 1996/05/10 03:18:43 chuck Exp $ */
-
/*
+ * Copyright (c) 1996 Nivas Madhur
* Copyright (c) 1995 Dale Rahn.
* All rights reserved.
*
@@ -34,7 +33,7 @@
#define _MACHINE_DISKLABEL_H_
/* number of boot pieces , ie xxboot bootxx */
-#define NUMBOOT 0
+#define NUMBOOT 2
#define PARTITIONSHIFT 4
@@ -69,92 +68,85 @@
*
* disksubr.c translates between cpu_disklabel and BSD disklabel.
*
+ * Note: this structure is exactly 512 bytes in size. If you move fields
+ * around, make sure the various members are properly aligned and the
+ * compiler won't do any additional padding.
*/
struct cpu_disklabel {
/* VID */
- u_char vid_id[4]; /* volume ID */
-#define VID_ID "NBSD"
- u_char vid_0[16];
- u_int vid_oss; /* starting block # of bootstrap */
-#define VID_OSS 2
- u_short vid_osl; /* bootstrap length (30 blocks) */
-#define VID_OSL 30
- u_char vid_1[4];
- u_short vid_osa_u; /* bootstrap start address (upper) */
- u_short vid_osa_l; /* bootstrap start address (lower) */
-#define VID_OSA 0x3f0000 /* MUST match bootstrap code */
-#define VID_OSAU ((VID_OSA >> 16) & 0xffff)
-#define VID_OSAL (VID_OSA & 0xffff)
- u_char vid_2[2];
- u_short partitions;
- u_char vid_vd[16];
- u_long bbsize;
- u_long magic1; /* 4 */
- u_short type; /* 2 */
- u_short subtype; /* 2 */
- u_char packname[16]; /* 16 */
- u_long flags; /* 4 */
- u_long drivedata[5]; /* 4 */
- u_long spare[5]; /* 4 */
- u_short checksum; /* 2 */
+ u_char vid_id[4];
+ u_char vid_0[16];
+ u_int vid_oss;
+ u_short vid_osl;
+ u_char vid_1[4];
+ u_short vid_osa_u;
+ u_short vid_osa_l;
+ u_char version;
+ u_char vid_2[1];
+ u_short checksum; /* 2 */
+ u_short partitions;
+ u_char vid_vd[16];
+ u_long bbsize;
+ u_long magic1; /* 4 */
+ u_short type; /* 2 */
+ u_short subtype; /* 2 */
+ u_char packname[16]; /* 16 */
+ u_long flags; /* 4 */
+ u_long drivedata[5]; /* 4 */
+ u_long spare[5]; /* 4 */
- u_long secpercyl; /* 4 */
- u_long secperunit; /* 4 */
- u_long headswitch; /* 4 */
+ u_long secpercyl; /* 4 */
+ u_long secperunit; /* 4 */
+ u_long headswitch; /* 4 */
- u_char vid_3[4];
- u_int vid_cas; /* block # of CFG area, hardwired at 1 */
-#define VID_CAS 1
- u_char vid_cal; /* length of CFG area, in blocks (1) */
-#define VID_CAL 1
- u_char vid_4_0[3];
- u_char vid_4[64];
- u_char vid_4_1[28];
- u_long sbsize;
- u_char vid_mot[8]; /* must contain "MOTOROLA" */
-#define VID_MOT "MOTOROLA"
+ u_char vid_3[4];
+ u_int vid_cas;
+ u_char vid_cal;
+ u_char vid_4_0[3];
+ u_char vid_4[64];
+ u_char vid_4_1[28];
+ u_long sbsize;
+ u_char vid_mot[8];
/* CFG */
- u_char cfg_0[4];
- u_short cfg_atm;
- u_short cfg_prm;
- u_short cfg_atw;
- u_short cfg_rec; /* block size (256) */
-#define CFG_REC 256
+ u_char cfg_0[4];
+ u_short cfg_atm;
+ u_short cfg_prm;
+ u_short cfg_atw;
+ u_short cfg_rec;
- u_short sparespertrack;
- u_short sparespercyl;
- u_long acylinders;
- u_short rpm;
- u_short cylskew;
+ u_short sparespertrack;
+ u_short sparespercyl;
+ u_long acylinders;
+ u_short rpm;
+ u_short cylskew;
- u_char cfg_spt;
- u_char cfg_hds;
- u_short cfg_trk;
- u_char cfg_ilv;
- u_char cfg_sof;
- u_short cfg_psm; /* physical sector size (512) */
-#define CFG_PSM 512
- u_short cfg_shd;
- u_char cfg_2[2];
- u_short cfg_pcom;
- u_char cfg_3;
- u_char cfg_ssr;
- u_short cfg_rwcc;
- u_short cfg_ecc;
- u_short cfg_eatm;
- u_short cfg_eprm;
- u_short cfg_eatw;
- u_char cfg_gpb1;
- u_char cfg_gpb2;
- u_char cfg_gpb3;
- u_char cfg_gpb4;
- u_char cfg_ssc;
- u_char cfg_runit;
- u_short cfg_rsvc1;
- u_short cfg_rsvc2;
- u_long magic2;
- u_char cfg_4[192];
+ u_char cfg_spt;
+ u_char cfg_hds;
+ u_short cfg_trk;
+ u_char cfg_ilv;
+ u_char cfg_sof;
+ u_short cfg_psm;
+ u_short cfg_shd;
+ u_char cfg_2[2];
+ u_short cfg_pcom;
+ u_char cfg_3;
+ u_char cfg_ssr;
+ u_short cfg_rwcc;
+ u_short cfg_ecc;
+ u_short cfg_eatm;
+ u_short cfg_eprm;
+ u_short cfg_eatw;
+ u_char cfg_gpb1;
+ u_char cfg_gpb2;
+ u_char cfg_gpb3;
+ u_char cfg_gpb4;
+ u_char cfg_ssc;
+ u_char cfg_runit;
+ u_short cfg_rsvc1;
+ u_short cfg_rsvc2;
+ u_long magic2;
+ u_char cfg_4[192];
};
#endif _MACHINE_DISKLABEL_H_
diff --git a/sys/arch/mvme68k/mvme68k/disksubr.c b/sys/arch/mvme68k/mvme68k/disksubr.c
index 1e636cb34ff..2b0130c17b3 100644
--- a/sys/arch/mvme68k/mvme68k/disksubr.c
+++ b/sys/arch/mvme68k/mvme68k/disksubr.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: disksubr.c,v 1.8 1996/11/08 20:49:12 chuck Exp $ */
+/* $OpenBSD: disksubr.c,v 1.9 1997/02/12 02:15:29 deraadt Exp $ */
/*
* Copyright (c) 1995 Dale Rahn.
@@ -70,33 +70,31 @@ dk_establish(dk, dev)
return;
/*
- * scsi: sd,cd
- */
+ * scsi: sd,cd
+ */
if (strncmp("sd", dev->dv_xname, 2) == 0 ||
strncmp("cd", dev->dv_xname, 2) == 0) {
- sbsc = (struct scsibus_softc *)dev->dv_parent;
+ sbsc = (struct scsibus_softc *)dev->dv_parent;
if (cputyp == CPU_147) {
target = bootctrllun % 8; /* XXX: 147 only */
lun = bootdevlun; /* XXX: 147, untested */
} else {
/*
- * XXX: on the 167:
- * ignore bootctrllun
- */
- target = bootdevlun / 10;
- lun = bootdevlun % 10;
+ * XXX: on the 167:
+ * ignore bootctrllun
+ */
+ target = bootdevlun / 10;
+ lun = bootdevlun % 10;
}
- if (sbsc->sc_link[target][lun] != NULL &&
- sbsc->sc_link[target][lun]->device_softc == (void *)dev) {
+ if (sbsc->sc_link[target][lun] != NULL &&
+ sbsc->sc_link[target][lun]->device_softc == (void *)dev) {
bootdv = dev;
- return;
+ return;
}
- }
-
- return;
+ }
}
/*
@@ -137,10 +135,9 @@ readdisklabel(dev, strat, lp, clp)
bp->b_flags = B_INVAL | B_AGE | B_READ;
brelse(bp);
- if (msg || clp->magic1 != DISKMAGIC || clp->magic2 != DISKMAGIC) {
+ if (msg) {
return (msg);
}
-
cputobsdlabel(lp, clp);
#ifdef DEBUG
if(disksubr_debug > 0) {
@@ -301,37 +298,39 @@ bounds_check_with_label(bp, lp, wlabel)
int wlabel;
{
struct partition *p = lp->d_partitions + DISKPART(bp->b_dev);
+ int labelsect = lp->d_partitions[0].p_offset;
int maxsz = p->p_size;
int sz = (bp->b_bcount + DEV_BSIZE - 1) >> DEV_BSHIFT;
/* overwriting disk label ? */
/* XXX should also protect bootstrap in first 8K */
- /* XXX this assumes everything <=LABELSECTOR is label! */
- /* (but since LABELSECTOR is zero, this is ok) */
- if (bp->b_blkno + p->p_offset <= LABELSECTOR &&
- (bp->b_flags & B_READ) == 0 && wlabel == 0) {
- bp->b_error = EROFS;
- goto bad;
- }
+ if (bp->b_blkno + p->p_offset <= LABELSECTOR + labelsect &&
+#if LABELSECTOR != 0
+ bp->b_blkno + p->p_offset + sz > LABELSECTOR + labelsect &&
+#endif
+ (bp->b_flags & B_READ) == 0 && wlabel == 0) {
+ bp->b_error = EROFS;
+ goto bad;
+ }
/* beyond partition? */
- if (bp->b_blkno < 0 || bp->b_blkno + sz > maxsz) {
- /* if exactly at end of disk, return an EOF */
- if (bp->b_blkno == maxsz) {
- bp->b_resid = bp->b_bcount;
- return(0);
- }
- /* or truncate if part of it fits */
- sz = maxsz - bp->b_blkno;
- if (sz <= 0) {
+ if (bp->b_blkno < 0 || bp->b_blkno + sz > maxsz) {
+ /* if exactly at end of disk, return an EOF */
+ if (bp->b_blkno == maxsz) {
+ bp->b_resid = bp->b_bcount;
+ return(0);
+ }
+ /* or truncate if part of it fits */
+ sz = maxsz - bp->b_blkno;
+ if (sz <= 0) {
bp->b_error = EINVAL;
- goto bad;
+ goto bad;
}
- bp->b_bcount = sz << DEV_BSHIFT;
- }
+ bp->b_bcount = sz << DEV_BSHIFT;
+ }
/* calculate cylinder for disksort to order transfers with */
- bp->b_cylin = (bp->b_blkno + p->p_offset) / lp->d_secpercyl;
+ bp->b_cylin = (bp->b_blkno + p->p_offset) / lp->d_secpercyl;
return(1);
bad:
@@ -396,25 +395,85 @@ bsdtocpulabel(lp, clp)
clp->checksum = lp->d_checksum;
bcopy(&lp->d_partitions[0], clp->vid_4, sizeof(struct partition) * 4);
bcopy(&lp->d_partitions[4], clp->cfg_4, sizeof(struct partition) * 12);
-
- /*
- * here are the parts of the cpu_disklabel the kernel must init.
- * see disklabel.h for more details
- * [note: this used to be handled by 'wrtvid']
- */
- bcopy(VID_ID, clp->vid_id, sizeof(clp->vid_id));
- clp->vid_oss = VID_OSS;
- clp->vid_osl = VID_OSL;
- clp->vid_osa_u = VID_OSAU;
- clp->vid_osa_l = VID_OSAL;
- clp->vid_cas = VID_CAS;
- clp->vid_cal = VID_CAL;
- bcopy(VID_MOT, clp->vid_mot, sizeof(clp->vid_mot));
- clp->cfg_rec = CFG_REC;
- clp->cfg_psm = CFG_PSM;
-
+ clp->version = 1;
}
+struct cpu_disklabel_old {
+ /* VID */
+ u_char vid_id[4];
+ u_char vid_0[16];
+ u_int vid_oss;
+ u_short vid_osl;
+ u_char vid_1[4];
+ u_short vid_osa_u;
+ u_short vid_osa_l;
+ u_char vid_2[2];
+ u_short partitions;
+ u_char vid_vd[16];
+ u_long bbsize;
+ u_long magic1; /* 4 */
+ u_short type; /* 2 */
+ u_short subtype; /* 2 */
+ u_char packname[16]; /* 16 */
+ u_long flags; /* 4 */
+ u_long drivedata[5]; /* 4 */
+ u_long spare[5]; /* 4 */
+ u_short checksum; /* 2 */
+
+ u_long secpercyl; /* 4 */
+ u_long secperunit; /* 4 */
+ u_long headswitch; /* 4 */
+
+ u_char vid_3[4];
+ u_int vid_cas;
+ u_char vid_cal;
+ u_char vid_4_0[3];
+ u_char vid_4[64];
+ u_char vid_4_1[28];
+ u_long sbsize;
+ u_char vid_mot[8];
+
+ /* CFG */
+ u_char cfg_0[4];
+ u_short cfg_atm;
+ u_short cfg_prm;
+ u_short cfg_atw;
+ u_short cfg_rec;
+
+ u_short sparespertrack;
+ u_short sparespercyl;
+ u_long acylinders;
+ u_short rpm;
+ u_short cylskew;
+
+ u_char cfg_spt;
+ u_char cfg_hds;
+ u_short cfg_trk;
+ u_char cfg_ilv;
+ u_char cfg_sof;
+ u_short cfg_psm;
+ u_short cfg_shd;
+ u_char cfg_2[2];
+ u_short cfg_pcom;
+ u_char cfg_3;
+ u_char cfg_ssr;
+ u_short cfg_rwcc;
+ u_short cfg_ecc;
+ u_short cfg_eatm;
+ u_short cfg_eprm;
+ u_short cfg_eatw;
+ u_char cfg_gpb1;
+ u_char cfg_gpb2;
+ u_char cfg_gpb3;
+ u_char cfg_gpb4;
+ u_char cfg_ssc;
+ u_char cfg_runit;
+ u_short cfg_rsvc1;
+ u_short cfg_rsvc2;
+ u_long magic2;
+ u_char cfg_4[192];
+};
+
static void
cputobsdlabel(lp, clp)
struct disklabel *lp;
@@ -422,65 +481,125 @@ cputobsdlabel(lp, clp)
{
int i;
- lp->d_magic = clp->magic1;
- lp->d_type = clp->type;
- lp->d_subtype = clp->subtype;
- strncpy(lp->d_typename, clp->vid_vd, 16);
- strncpy(lp->d_packname, clp->packname, 16);
- lp->d_secsize = clp->cfg_psm;
- lp->d_nsectors = clp->cfg_spt;
- lp->d_ncylinders = clp->cfg_trk; /* trk is really num of cyl! */
- lp->d_ntracks = clp->cfg_hds;
-
- lp->d_secpercyl = clp->secpercyl;
- lp->d_secperunit = clp->secperunit;
- lp->d_secpercyl = clp->secpercyl;
- lp->d_secperunit = clp->secperunit;
- lp->d_sparespertrack = clp->sparespertrack;
- lp->d_sparespercyl = clp->sparespercyl;
- lp->d_acylinders = clp->acylinders;
- lp->d_rpm = clp->rpm;
- lp->d_interleave = clp->cfg_ilv;
- lp->d_trackskew = clp->cfg_sof;
- lp->d_cylskew = clp->cylskew;
- lp->d_headswitch = clp->headswitch;
-
- /* this silly table is for winchester drives */
- switch (clp->cfg_ssr) {
- case 0:
- lp->d_trkseek = 0;
- break;
- case 1:
- lp->d_trkseek = 6;
- break;
- case 2:
- lp->d_trkseek = 10;
- break;
- case 3:
- lp->d_trkseek = 15;
- break;
- case 4:
- lp->d_trkseek = 20;
- break;
- default:
- lp->d_trkseek = 0;
+ if (clp->version == 0) {
+ struct cpu_disklabel_old *clpo = (void *) clp;
+ lp->d_magic = DISKMAGIC;
+ lp->d_type = clpo->type;
+ lp->d_subtype = clpo->subtype;
+ strncpy(lp->d_typename, clpo->vid_vd, 16);
+ strncpy(lp->d_packname, clpo->packname, 16);
+ lp->d_secsize = clpo->cfg_psm;
+ lp->d_nsectors = clpo->cfg_spt;
+ lp->d_ncylinders = clpo->cfg_trk; /* trk is really num of cyl! */
+ lp->d_ntracks = clpo->cfg_hds;
+ lp->d_secpercyl = clpo->secpercyl;
+ lp->d_secperunit = clpo->secperunit;
+ lp->d_secpercyl = clpo->secpercyl;
+ lp->d_secperunit = clpo->secperunit;
+ lp->d_sparespertrack = clpo->sparespertrack;
+ lp->d_sparespercyl = clpo->sparespercyl;
+ lp->d_acylinders = clpo->acylinders;
+ lp->d_rpm = clpo->rpm;
+ lp->d_interleave = clpo->cfg_ilv;
+ lp->d_trackskew = clpo->cfg_sof;
+ lp->d_cylskew = clpo->cylskew;
+ lp->d_headswitch = clpo->headswitch;
+ /* this silly table is for winchester drives */
+ switch (clpo->cfg_ssr) {
+ case 0:
+ lp->d_trkseek = 0;
+ break;
+ case 1:
+ lp->d_trkseek = 6;
+ break;
+ case 2:
+ lp->d_trkseek = 10;
+ break;
+ case 3:
+ lp->d_trkseek = 15;
+ break;
+ case 4:
+ lp->d_trkseek = 20;
+ break;
+ default:
+ lp->d_trkseek = 0;
+ }
+ lp->d_flags = clpo->flags;
+ for (i = 0; i < NDDATA; i++)
+ lp->d_drivedata[i] = clpo->drivedata[i];
+ for (i = 0; i < NSPARE; i++)
+ lp->d_spare[i] = clpo->spare[i];
+
+ lp->d_magic2 = DISKMAGIC;
+ lp->d_checksum = clpo->checksum;
+ lp->d_npartitions = clpo->partitions;
+ lp->d_bbsize = clpo->bbsize;
+ lp->d_sbsize = clpo->sbsize;
+ bcopy(clpo->vid_4, &lp->d_partitions[0], sizeof(struct partition) * 4);
+ bcopy(clpo->cfg_4, &lp->d_partitions[4], sizeof(struct partition) * 12);
+ lp->d_checksum = 0;
+ lp->d_checksum = dkcksum(lp);
+ } else {
+ lp->d_magic = clp->magic1;
+ lp->d_type = clp->type;
+ lp->d_subtype = clp->subtype;
+ strncpy(lp->d_typename, clp->vid_vd, 16);
+ strncpy(lp->d_packname, clp->packname, 16);
+ lp->d_secsize = clp->cfg_psm;
+ lp->d_nsectors = clp->cfg_spt;
+ lp->d_ncylinders = clp->cfg_trk; /* trk is really num of cyl! */
+ lp->d_ntracks = clp->cfg_hds;
+
+ lp->d_secpercyl = clp->secpercyl;
+ lp->d_secperunit = clp->secperunit;
+ lp->d_secpercyl = clp->secpercyl;
+ lp->d_secperunit = clp->secperunit;
+ lp->d_sparespertrack = clp->sparespertrack;
+ lp->d_sparespercyl = clp->sparespercyl;
+ lp->d_acylinders = clp->acylinders;
+ lp->d_rpm = clp->rpm;
+ lp->d_interleave = clp->cfg_ilv;
+ lp->d_trackskew = clp->cfg_sof;
+ lp->d_cylskew = clp->cylskew;
+ lp->d_headswitch = clp->headswitch;
+
+ /* this silly table is for winchester drives */
+ switch (clp->cfg_ssr) {
+ case 0:
+ lp->d_trkseek = 0;
+ break;
+ case 1:
+ lp->d_trkseek = 6;
+ break;
+ case 2:
+ lp->d_trkseek = 10;
+ break;
+ case 3:
+ lp->d_trkseek = 15;
+ break;
+ case 4:
+ lp->d_trkseek = 20;
+ break;
+ default:
+ lp->d_trkseek = 0;
+ }
+ lp->d_flags = clp->flags;
+ for (i = 0; i < NDDATA; i++)
+ lp->d_drivedata[i] = clp->drivedata[i];
+ for (i = 0; i < NSPARE; i++)
+ lp->d_spare[i] = clp->spare[i];
+
+ lp->d_magic2 = clp->magic2;
+ lp->d_checksum = clp->checksum;
+ lp->d_npartitions = clp->partitions;
+ lp->d_bbsize = clp->bbsize;
+ lp->d_sbsize = clp->sbsize;
+ bcopy(clp->vid_4, &lp->d_partitions[0], sizeof(struct partition) * 4);
+ bcopy(clp->cfg_4, &lp->d_partitions[4], sizeof(struct partition) * 12);
+ lp->d_checksum = 0;
+ lp->d_checksum = dkcksum(lp);
}
- lp->d_flags = clp->flags;
- for (i = 0; i < NDDATA; i++)
- lp->d_drivedata[i] = clp->drivedata[i];
- for (i = 0; i < NSPARE; i++)
- lp->d_spare[i] = clp->spare[i];
-
- lp->d_magic2 = clp->magic2;
- lp->d_checksum = clp->checksum;
- lp->d_npartitions = clp->partitions;
- lp->d_bbsize = clp->bbsize;
- lp->d_sbsize = clp->sbsize;
- bcopy(clp->vid_4, &lp->d_partitions[0], sizeof(struct partition) * 4);
- bcopy(clp->cfg_4, &lp->d_partitions[4], sizeof(struct partition) * 12);
- lp->d_checksum = 0;
- lp->d_checksum = dkcksum(lp);
-#if DEBUG
+#ifdef DEBUG
if (disksubr_debug > 0) {
printlp(lp, "translated label read from disk\n");
}