/* $NetBSD: disksubr.c,v 1.35.4.1 2020/06/30 18:20:53 martin Exp $ */
/*
* Copyright (c) 1982, 1986, 1988 Regents of the University of California.
* 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. Neither the name of the University nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``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 REGENTS OR CONTRIBUTORS 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.
*
* @(#)ufs_disksubr.c 7.16 (Berkeley) 5/4/91
*/
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: disksubr.c,v 1.35.4.1 2020/06/30 18:20:53 martin Exp $");
#include "opt_compat_netbsd.h"
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/buf.h>
#include <sys/disklabel.h>
#include <sys/syslog.h>
#include <sys/disk.h>
/* get rid of DEV_BSIZE dependency */
#define DEF_BSIZE DEV_BSIZE /* default sector size = 512 */
static void parttbl_consistency_check(struct disklabel *,
struct dos_partition *);
/*
* Attempt to read a disk label from a device
* using the indicated strategy routine.
* The label must be partly set up before this:
* secpercyl, secsize and anything required for a block i/o read
* operation in the driver's strategy/start routines
* must be filled in before calling us.
*
* Returns null on success and an error string on failure.
*/
const char *
readdisklabel(dev_t dev, void (*strat)(struct buf *),
struct disklabel *lp, struct cpu_disklabel *osdep)
{
struct dos_partition *dp = 0;
struct dkbad *bdp = &osdep->bad;
struct buf *bp;
struct disklabel *dlp;
const char *msg = NULL;
int i, bsdlabelsz, humanlabelsz;
if (osdep)
dp = osdep->dosparts;
/* minimal requirements for archtypal disk label */
if (lp->d_secsize == 0)
lp->d_secsize = DEF_BSIZE;
if (lp->d_secperunit == 0)
lp->d_secperunit = 0x1fffffff;
lp->d_npartitions = RAW_PART + 1;
for (i = 0; i < RAW_PART; i++) {
lp->d_partitions[i].p_size = 0;
lp->d_partitions[i].p_offset = 0;
}
if (lp->d_partitions[RAW_PART].p_size == 0)
lp->d_partitions[RAW_PART].p_size = lp->d_secperunit;
lp->d_partitions[RAW_PART].p_offset = 0;
lp->d_partitions[0].p_size = lp->d_partitions[RAW_PART].p_size;
/* get a buffer and initialize it */
bsdlabelsz =
howmany(LABELOFFSET + sizeof(struct disklabel), lp->d_secsize)
* lp->d_secsize;
humanlabelsz =
howmany(sizeof(struct cpu_disklabel), lp->d_secsize)
* lp->d_secsize;
bp = geteblk(MAX(bsdlabelsz, humanlabelsz));
bp->b_dev = dev;
/* read BSD disklabel first */
bp->b_blkno = LABELSECTOR;
bp->b_cylinder = LABELSECTOR/lp->d_secpercyl;
bp->b_bcount = bsdlabelsz; /* to support < 512B/sector disks */
bp->b_flags |= B_READ;
(*strat)(bp);
/* if successful, locate disk label within block and validate */
if (biowait(bp)) {
msg = "disk label I/O error";
goto dodospart;
}
for (dlp = (struct disklabel *)bp->b_data;
dlp <= (struct disklabel *)
((char *)bp->b_data + bsdlabelsz - sizeof(*dlp));
dlp = (struct disklabel *)((uint8_t *)dlp + sizeof(long))) {
if (dlp->d_magic != DISKMAGIC || dlp->d_magic2 != DISKMAGIC) {
if (msg == NULL)
msg = "no disk label";
} else if (dlp->d_npartitions > MAXPARTITIONS ||
dkcksum(dlp) != 0)
msg = "disk label corrupted";
else {
*lp = *dlp;
msg = NULL;
break;
}
}
dodospart:
/* next do the Human68k-style partition table */
/* Human68k does not support > 2048B/sector devices (?) */
if (lp->d_secsize >= 2048) {
if (msg)
goto done;
goto dobadsect;
}
bp->b_blkno = DOSPARTOFF * DEF_BSIZE / lp->d_secsize;
/* DOSPARTOFF in DEV_BSIZE unit */
bp->b_cylinder = DOSBBSECTOR / lp->d_secpercyl;
bp->b_bcount = humanlabelsz; /* to support < 512B/sector disks */
bp->b_oflags &= ~(BO_DONE);
(*strat)(bp);
/* if successful, wander through Human68k partition table */
if (biowait(bp))
goto done;
if (strncmp(bp->b_data, "X68K", 4) != 0) {
/* Human68k-style partition table does not exist */
if (msg)
goto done;
goto dobadsect;
}
/* XXX how do we check veracity/bounds of this? */
if (dp)
memcpy(dp, (char *)bp->b_data + sizeof(*dp) /*DOSPARTOFF*/,
NDOSPART * sizeof(*dp));
else
dp = (void *)((char *)bp->b_data + sizeof(*dp) /*DOSPARTOFF*/);
/* if BSD disklabel does not exist, fall back to Human68k partition */
if (msg != NULL) {
msg = NULL;
lp->d_bbsize = 8192;
lp->d_sbsize = 2048;
for (i = 0; i < NDOSPART; i++, dp++)
/* is this ours? */
if (dp->dp_size) {
u_char fstype;
int part = i + (i < RAW_PART ? 0 : 1);
int start = dp->dp_start * 2;
int size = dp->dp_size * 2;
/* update disklabel with details */
lp->d_partitions[part].p_size = size;
lp->d_partitions[part].p_offset = start;
/* get partition type */
#ifndef COMPAT_10
if (dp->dp_flag == 1)
fstype = FS_UNUSED;
else
#endif
if (!memcmp(dp->dp_typname, "Human68k", 8))
fstype = FS_MSDOS;
else if (!memcmp(dp->dp_typname,
"BSD ffs ", 8))
fstype = FS_BSDFFS;
else if (!memcmp(dp->dp_typname,
"BSD lfs ", 8))
fstype = FS_BSDLFS;
else if (!memcmp(dp->dp_typname,
"BSD swap", 8))
fstype = FS_SWAP;
#ifndef COMPAT_14
else if (part == 1)
fstype = FS_SWAP;
#endif
else
fstype = FS_BSDFFS; /* XXX */
lp->d_partitions[part].p_fstype = fstype; /* XXX */
if (lp->d_npartitions <= part)
lp->d_npartitions = part + 1;
}
} else {
parttbl_consistency_check(lp, dp);
}
dobadsect:
/* obtain bad sector table if requested and present */
if (bdp && (lp->d_flags & D_BADSECT)) {
struct dkbad *db;
i = 0;
do {
/* read a bad sector table */
bp->b_oflags &= ~(BO_DONE);
bp->b_flags |= B_READ;
bp->b_blkno = lp->d_secperunit - lp->d_nsectors + i;
if (lp->d_secsize > DEF_BSIZE)
bp->b_blkno *= lp->d_secsize / DEF_BSIZE;
else
bp->b_blkno /= DEF_BSIZE / lp->d_secsize;
bp->b_bcount = lp->d_secsize;
bp->b_cylinder = lp->d_ncylinders - 1;
(*strat)(bp);
/* if successful, validate, otherwise try another */
if (biowait(bp)) {
msg = "bad sector table I/O error";
} else {
db = (struct dkbad *)(bp->b_data);
#define DKBAD_MAGIC 0x4321
if (db->bt_mbz == 0
&& db->bt_flag == DKBAD_MAGIC) {
msg = NULL;
*bdp = *db;
break;
} else
msg = "bad sector table corrupted";
}
} while (bp->b_error != 0 && (i += 2) < 10 &&
i < lp->d_nsectors);
}
done:
brelse(bp, 0);
return (msg);
}
/*
* Write disk label back to device after modification.
*/
int
writedisklabel(dev_t dev, void (*strat)(struct buf *),
struct disklabel *lp, struct cpu_disklabel *osdep)
{
struct dos_partition *dp = 0;
struct buf *bp;
struct disklabel *dlp;
int error, bsdlabelsz, humanlabelsz, i;
const char *np;
if (osdep)
dp = osdep->dosparts;
/* sanity clause */
if (lp->d_secpercyl == 0 || lp->d_secsize == 0
/*|| (lp->d_secsize % DEF_BSIZE) != 0*/)
return(EINVAL);
if (dp)
parttbl_consistency_check(lp, dp);
/* get a buffer and initialize it */
bsdlabelsz =
howmany(LABELOFFSET + sizeof(struct disklabel), lp->d_secsize)
* lp->d_secsize;
humanlabelsz =
howmany(sizeof(struct cpu_disklabel), lp->d_secsize)
* lp->d_secsize;
bp = geteblk(MAX(bsdlabelsz, humanlabelsz));
bp->b_dev = dev;
/* attempt to write BSD disklabel first */
bp->b_blkno = LABELSECTOR;
bp->b_cylinder = LABELSECTOR / lp->d_secpercyl;
bp->b_bcount = bsdlabelsz; /* to support < 512B/sector disks */
bp->b_flags |= B_READ;
(*strat)(bp);
/* if successful, locate disk label within block and validate */
if (biowait(bp))
goto dodospart;
error = ESRCH;
for (dlp = (struct disklabel *)bp->b_data;
dlp <= (struct disklabel *)
((char *)bp->b_data + bsdlabelsz - sizeof(*dlp));
dlp = (struct disklabel *)((char *)dlp + sizeof(long))) {
if (dlp->d_magic == DISKMAGIC && dlp->d_magic2 == DISKMAGIC &&
dkcksum(dlp) == 0) {
*dlp = *lp;
bp->b_oflags &= ~(BO_DONE);
bp->b_flags &= ~(B_READ);
bp->b_flags |= B_WRITE;
(*strat)(bp);
error = biowait(bp);
break;
}
}
/* do dos partitions in the process of getting disklabel? */
if (error) {
dodospart:
if (lp->d_secsize >= 2048) {
error = ESRCH;
goto done;
}
#if 0 /* there is no mark on floppies */
/* read the x68k disk magic */
bp->b_blkno = DOSBBSECTOR;
bp->b_bcount = lp->d_secsize;
bp->b_oflags &= ~(BO_DONE);
bp->b_flags &= ~(B_WRITE);
bp->b_flags |= B_READ;
bp->b_cylinder = DOSBBSECTOR / lp->d_secpercyl;
(*strat)(bp);
if ((error = biowait(bp)) || memcmp(bp->b_data, "X68SCSI1", 8))
printf("warning: disk not marked for x68k");
#endif
/* read the partition table */
bp->b_blkno = DOSPARTOFF;
bp->b_bcount = humanlabelsz;
bp->b_oflags &= ~(BO_DONE);
bp->b_flags &= ~(B_WRITE);
bp->b_flags |= B_READ;
bp->b_cylinder = DOSPARTOFF / lp->d_secpercyl;
(*strat)(bp);
if ((error = biowait(bp)) == 0) {
/* XXX how do we check veracity/bounds of this? */
dp = (struct dos_partition *)bp->b_data + 1;
for (i = 0; i < NDOSPART; i++, dp++) {
int part = i + (i < RAW_PART ? 0 : 1);
int start, size;
start = lp->d_partitions[part].p_offset >> 1;
size = lp->d_partitions[part].p_size >> 1;
switch (lp->d_partitions[part].p_fstype) {
case FS_MSDOS:
np = "Human68k";
dp->dp_flag = 0; /* autoboot */
break;
case FS_SWAP:
np = "BSD swap";
dp->dp_flag = 2; /* in use */
break;
case FS_BSDFFS:
np = "BSD ffs ";
if (part == 0)
dp->dp_flag = 0; /* autoboot */
else
dp->dp_flag = 2; /* in use */
break;
case FS_BSDLFS:
np = "BSD lfs ";
if (part == 0)
dp->dp_flag = 0; /* autoboot */
else
dp->dp_flag = 2; /* in use */
break;
case FS_UNUSED:
np = "\0\0\0\0\0\0\0\0";
start = size = 0;
if (part < lp->d_npartitions) {
dp->dp_flag = 1;
} else {
dp->dp_flag = 0;
}
break;
default:
/* XXX OS-9, MINIX etc. */
continue;
}
memcpy(dp->dp_typname, np, 8);
dp->dp_start = start;
dp->dp_size = size;
}
bp->b_oflags &= ~(BO_DONE);
bp->b_flags &= ~(B_READ);
bp->b_flags |= B_WRITE;
(*strat)(bp);
error = biowait(bp);
}
}
#ifdef maybe
/* disklabel in appropriate location? */
if (lp->d_partitions[0].p_offset != 0
&& lp->d_partitions[0].p_offset != dospartoff) {
error = EXDEV;
goto done;
}
#endif
done:
brelse(bp, 0);
return (error);
}
static void
parttbl_consistency_check(struct disklabel *lp, struct dos_partition *dp)
{
int i, j;
int f = (lp->d_secsize >= 1024) ? lp->d_secsize/1024 : 1;
int g = (lp->d_secsize >= 1024) ? 1 : 1024/lp->d_secsize;
/* 1. overlapping check on partition table */
for (i = 0; i < NDOSPART; i++) {
if (dp[i].dp_size == 0)
continue;
for (j = i+1; j < NDOSPART; j++) {
if (dp[j].dp_size == 0)
continue;
if (((dp[i].dp_start <= dp[j].dp_start) &&
(dp[i].dp_start + dp[i].dp_size > dp[j].dp_start))||
((dp[j].dp_start <= dp[i].dp_start) &&
(dp[j].dp_start + dp[j].dp_size > dp[i].dp_start))) {
printf("warning: Human68k partition %d and %d"
" are overlapping\n", i+1, j+1);
return;
}
}
}
/* 2. scan disklabel partitions */
#define bp lp->d_partitions
for (i = 0; i < lp->d_npartitions; i++) {
int c = 0;
if (lp->d_partitions[i].p_fstype == FS_UNUSED ||
lp->d_partitions[i].p_size == 0)
continue;
for (j = 0; j < NDOSPART; j++) {
if (dp[j].dp_size == 0)
continue;
if ((bp[i].p_offset * f < (dp[j].dp_start + dp[j].dp_size) * g) &&
((bp[i].p_offset + bp[i].p_size) * f >= (dp[j].dp_start + dp[j].dp_size) * g))
c++;
if ((bp[i].p_offset * f > dp[j].dp_start * g) &&
((bp[i].p_offset + bp[i].p_size) * f < (dp[j].dp_start + dp[j].dp_size) * g))
c++;
if ((bp[i].p_offset * f >= dp[j].dp_start * g) &&
((bp[i].p_offset + bp[i].p_size) * f < dp[j].dp_start * g))
c++;
}
if (c > 1)
printf ("warning: partition %c spans for 2 or more"
" partitions in Human68k partition table.\n",
i+'a');
}
#undef bp
/* more checks? */
}