MFMBR: Add ioctls for writing an IPL and a boot menu.

This commit is contained in:
Yoshihiro Takahashi 2003-01-03 07:13:36 +00:00
parent f61e88f901
commit 05f0a5a323
2 changed files with 197 additions and 102 deletions

View File

@ -77,60 +77,11 @@ g_dec_dos_partition(u_char *ptr, struct dos_partition *d)
}
struct g_pc98_softc {
int type [NDOSPART];
struct dos_partition dp[NDOSPART];
u_int fwsectors, fwheads, sectorsize;
int type[NDOSPART];
u_char sec[8192];
};
static int
g_pc98_start(struct bio *bp)
{
struct g_provider *pp;
struct g_geom *gp;
struct g_pc98_softc *mp;
struct g_slicer *gsp;
int idx;
pp = bp->bio_to;
idx = pp->index;
gp = pp->geom;
gsp = gp->softc;
mp = gsp->softc;
if (bp->bio_cmd == BIO_GETATTR) {
if (g_handleattr_int(bp, "PC98::type", mp->type[idx]))
return (1);
if (g_handleattr_off_t(bp, "PC98::offset",
gsp->slices[idx].offset))
return (1);
}
return (0);
}
static void
g_pc98_dumpconf(struct sbuf *sb, const char *indent, struct g_geom *gp,
struct g_consumer *cp __unused, struct g_provider *pp)
{
struct g_pc98_softc *mp;
struct g_slicer *gsp;
char sname[17];
gsp = gp->softc;
mp = gsp->softc;
g_slice_dumpconf(sb, indent, gp, cp, pp);
if (pp != NULL) {
strncpy(sname, mp->dp[pp->index].dp_name, 16);
sname[16] = '\0';
if (indent == NULL) {
sbuf_printf(sb, " ty %d", mp->type[pp->index]);
sbuf_printf(sb, " sn %s", sname);
} else {
sbuf_printf(sb, "%s<type>%d</type>\n", indent,
mp->type[pp->index]);
sbuf_printf(sb, "%s<sname>%s</sname>\n", indent,
sname);
}
}
}
static void
g_pc98_print(int i, struct dos_partition *dp)
{
@ -147,17 +98,197 @@ g_pc98_print(int i, struct dos_partition *dp)
printf(" sname:%s\n", sname);
}
static int
g_pc98_modify(struct g_geom *gp, struct g_pc98_softc *ms, u_char *sec)
{
int i, error;
off_t s[NDOSPART], l[NDOSPART];
struct dos_partition dp[NDOSPART];
g_topology_assert();
if (sec[0x1fe] != 0x55 || sec[0x1ff] != 0xaa)
return (EBUSY);
#if 0
/*
* XXX: Some sources indicate this is a magic sequence, but appearantly
* XXX: it is not universal. Documentation would be wonderfule to have.
*/
if (sec[4] != 'I' || sec[5] != 'P' || sec[6] != 'L' || sec[7] != '1')
return (EBUSY);
#endif
for (i = 0; i < NDOSPART; i++)
g_dec_dos_partition(
sec + 512 + i * sizeof(struct dos_partition), &dp[i]);
for (i = 0; i < NDOSPART; i++) {
/* If start and end are identical it's bogus */
if (dp[i].dp_ssect == dp[i].dp_esect &&
dp[i].dp_shd == dp[i].dp_ehd &&
dp[i].dp_scyl == dp[i].dp_ecyl)
s[i] = l[i] = 0;
else if (dp[i].dp_ecyl == 0)
s[i] = l[i] = 0;
else {
s[i] = (off_t)dp[i].dp_scyl *
ms->fwsectors * ms->fwheads * ms->sectorsize;
l[i] = (off_t)(dp[i].dp_ecyl - dp[i].dp_scyl + 1) *
ms->fwsectors * ms->fwheads * ms->sectorsize;
}
if (bootverbose) {
printf("PC98 Slice %d on %s:\n", i + 1, gp->name);
g_pc98_print(i, dp + i);
}
error = g_slice_config(gp, i, G_SLICE_CONFIG_CHECK,
s[i], l[i], ms->sectorsize,
"%ss%d", gp->name, i + 1);
if (error)
return (error);
}
for (i = 0; i < NDOSPART; i++) {
ms->type[i] = (dp[i].dp_sid << 8) | dp[i].dp_mid;
g_slice_config(gp, i, G_SLICE_CONFIG_SET, s[i], l[i],
ms->sectorsize, "%ss%d", gp->name, i + 1);
}
bcopy(sec, ms->sec, sizeof (ms->sec));
return (0);
}
static void
g_pc98_ioctl(void *arg)
{
struct bio *bp;
struct g_geom *gp;
struct g_slicer *gsp;
struct g_pc98_softc *ms;
struct g_ioctl *gio;
struct g_consumer *cp;
u_char *sec;
int error;
/* Get hold of the interesting bits from the bio. */
bp = arg;
gp = bp->bio_to->geom;
gsp = gp->softc;
ms = gsp->softc;
gio = (struct g_ioctl *)bp->bio_data;
/* The disklabel to set is the ioctl argument. */
sec = gio->data;
error = g_pc98_modify(gp, ms, sec);
if (error) {
g_io_deliver(bp, error);
return;
}
cp = LIST_FIRST(&gp->consumer);
error = g_write_data(cp, 0, sec, 8192);
g_io_deliver(bp, error);
}
static int
g_pc98_start(struct bio *bp)
{
struct g_provider *pp;
struct g_geom *gp;
struct g_pc98_softc *mp;
struct g_slicer *gsp;
struct g_ioctl *gio;
int idx, error;
pp = bp->bio_to;
idx = pp->index;
gp = pp->geom;
gsp = gp->softc;
mp = gsp->softc;
if (bp->bio_cmd == BIO_GETATTR) {
if (g_handleattr_int(bp, "PC98::type", mp->type[idx]))
return (1);
if (g_handleattr_off_t(bp, "PC98::offset",
gsp->slices[idx].offset))
return (1);
}
/* We only handle ioctl(2) requests of the right format. */
if (strcmp(bp->bio_attribute, "GEOM::ioctl"))
return (0);
else if (bp->bio_length != sizeof(*gio))
return (0);
/* Get hold of the ioctl parameters. */
gio = (struct g_ioctl *)bp->bio_data;
switch (gio->cmd) {
case DIOCGPC98:
/* Return a copy of the disklabel to userland. */
bcopy(mp->sec, gio->data, 8192);
g_io_deliver(bp, 0);
return (1);
case DIOCSPC98:
/*
* These we cannot do without the topology lock and some
* some I/O requests. Ask the event-handler to schedule
* us in a less restricted environment.
*/
error = g_call_me(g_pc98_ioctl, bp);
if (error)
g_io_deliver(bp, error);
/*
* We must return non-zero to indicate that we will deal
* with this bio, even though we have not done so yet.
*/
return (1);
default:
return (0);
}
return (0);
}
static void
g_pc98_dumpconf(struct sbuf *sb, const char *indent, struct g_geom *gp,
struct g_consumer *cp __unused, struct g_provider *pp)
{
struct g_pc98_softc *mp;
struct g_slicer *gsp;
struct dos_partition dp;
char sname[17];
gsp = gp->softc;
mp = gsp->softc;
g_slice_dumpconf(sb, indent, gp, cp, pp);
if (pp != NULL) {
g_dec_dos_partition(
mp->sec + 512 +
pp->index * sizeof(struct dos_partition), &dp);
strncpy(sname, dp.dp_name, 16);
sname[16] = '\0';
if (indent == NULL) {
sbuf_printf(sb, " ty %d", mp->type[pp->index]);
sbuf_printf(sb, " sn %s", sname);
} else {
sbuf_printf(sb, "%s<type>%d</type>\n", indent,
mp->type[pp->index]);
sbuf_printf(sb, "%s<sname>%s</sname>\n", indent,
sname);
}
}
}
static struct g_geom *
g_pc98_taste(struct g_class *mp, struct g_provider *pp, int flags)
{
struct g_geom *gp;
struct g_consumer *cp;
int error, i, npart;
int error;
struct g_pc98_softc *ms;
struct g_slicer *gsp;
u_int fwsectors, fwheads, sectorsize;
u_char *buf;
off_t spercyl;
g_trace(G_T_TOPOLOGY, "g_pc98_taste(%s,%s)", mp->name, pp->name);
g_topology_assert();
@ -170,7 +301,6 @@ g_pc98_taste(struct g_class *mp, struct g_provider *pp, int flags)
gsp = gp->softc;
g_topology_unlock();
gp->dumpconf = g_pc98_dumpconf;
npart = 0;
while (1) { /* a trick to allow us to use break */
if (gp->rank != 2 && flags == G_TF_NORMAL)
break;
@ -190,56 +320,16 @@ g_pc98_taste(struct g_class *mp, struct g_provider *pp, int flags)
if (sectorsize < 512)
break;
gsp->frontstuff = sectorsize * fwsectors;
spercyl = (off_t)fwsectors * fwheads * sectorsize;
buf = g_read_data(cp, 0,
sectorsize < 1024 ? 1024 : sectorsize, &error);
buf = g_read_data(cp, 0, 8192, &error);
if (buf == NULL || error != 0)
break;
if (buf[0x1fe] != 0x55 || buf[0x1ff] != 0xaa) {
g_free(buf);
break;
}
#if 0
/*
* XXX: Some sources indicate this is a magic sequence, but appearantly
* XXX: it is not universal. Documentation would be wonderfule to have.
*/
if (buf[4] != 'I' || buf[5] != 'P' ||
buf[6] != 'L' || buf[7] != '1') {
g_free(buf);
break;
}
#endif
for (i = 0; i < NDOSPART; i++)
g_dec_dos_partition(
buf + 512 + i * sizeof(struct dos_partition),
ms->dp + i);
ms->fwsectors = fwsectors;
ms->fwheads = fwheads;
ms->sectorsize = sectorsize;
g_topology_lock();
g_pc98_modify(gp, ms, buf);
g_topology_unlock();
g_free(buf);
for (i = 0; i < NDOSPART; i++) {
/* If start and end are identical it's bogus */
if (ms->dp[i].dp_ssect == ms->dp[i].dp_esect &&
ms->dp[i].dp_shd == ms->dp[i].dp_ehd &&
ms->dp[i].dp_scyl == ms->dp[i].dp_ecyl)
continue;
if (ms->dp[i].dp_ecyl == 0)
continue;
if (bootverbose) {
printf("PC98 Slice %d on %s:\n",
i + 1, gp->name);
g_pc98_print(i, ms->dp + i);
}
npart++;
ms->type[i] = (ms->dp[i].dp_sid << 8) |
ms->dp[i].dp_mid;
g_topology_lock();
g_slice_config(gp, i, G_SLICE_CONFIG_SET,
ms->dp[i].dp_scyl * spercyl,
(ms->dp[i].dp_ecyl - ms->dp[i].dp_scyl + 1) *
spercyl,
sectorsize,
"%ss%d", gp->name, i + 1);
g_topology_unlock();
}
break;
}
g_topology_lock();

View File

@ -37,6 +37,8 @@
#ifndef _SYS_DISKPC98_H_
#define _SYS_DISKPC98_H_
#include <sys/ioccom.h>
#define DOSBBSECTOR 0 /* DOS boot block relative sector number */
#define DOSPARTOFF 0
#define NDOSPART 16
@ -64,4 +66,7 @@ struct dos_partition {
CTASSERT(sizeof (struct dos_partition) == 32);
#endif
#define DIOCGPC98 _IOR('M', 128, u_char[8192])
#define DIOCSPC98 _IOW('M', 129, u_char[8192])
#endif /* !_SYS_DISKPC98_H_ */