summaryrefslogtreecommitdiffstats
path: root/sys/geom/geom_pc98.c
diff options
context:
space:
mode:
authornyan <nyan@FreeBSD.org>2003-01-03 07:13:36 +0000
committernyan <nyan@FreeBSD.org>2003-01-03 07:13:36 +0000
commitfac5fc844d9bad89c2653c1f4d2fcd054adabfe0 (patch)
tree29151d0659d6b9e88647e9f61e7977cfcd9fe4ab /sys/geom/geom_pc98.c
parente3fdcf114e265cd8f1fd17cf2302c66b0234dc3a (diff)
downloadFreeBSD-src-fac5fc844d9bad89c2653c1f4d2fcd054adabfe0.zip
FreeBSD-src-fac5fc844d9bad89c2653c1f4d2fcd054adabfe0.tar.gz
MFMBR: Add ioctls for writing an IPL and a boot menu.
Diffstat (limited to 'sys/geom/geom_pc98.c')
-rw-r--r--sys/geom/geom_pc98.c230
1 files changed, 160 insertions, 70 deletions
diff --git a/sys/geom/geom_pc98.c b/sys/geom/geom_pc98.c
index 19fa752..90f51ed 100644
--- a/sys/geom/geom_pc98.c
+++ b/sys/geom/geom_pc98.c
@@ -77,10 +77,120 @@ 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 void
+g_pc98_print(int i, struct dos_partition *dp)
+{
+ char sname[17];
+
+ strncpy(sname, dp->dp_name, 16);
+ sname[16] = '\0';
+
+ g_hexdump(dp, sizeof(dp[0]));
+ printf("[%d] mid:%d(0x%x) sid:%d(0x%x)",
+ i, dp->dp_mid, dp->dp_mid, dp->dp_sid, dp->dp_sid);
+ printf(" s:%d/%d/%d", dp->dp_scyl, dp->dp_shd, dp->dp_ssect);
+ printf(" e:%d/%d/%d", dp->dp_ecyl, dp->dp_ehd, dp->dp_esect);
+ 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)
{
@@ -88,7 +198,8 @@ g_pc98_start(struct bio *bp)
struct g_geom *gp;
struct g_pc98_softc *mp;
struct g_slicer *gsp;
- int idx;
+ struct g_ioctl *gio;
+ int idx, error;
pp = bp->bio_to;
idx = pp->index;
@@ -102,6 +213,39 @@ g_pc98_start(struct bio *bp)
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);
}
@@ -111,13 +255,17 @@ g_pc98_dumpconf(struct sbuf *sb, const char *indent, struct g_geom *gp,
{
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) {
- strncpy(sname, mp->dp[pp->index].dp_name, 16);
+ 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]);
@@ -131,33 +279,16 @@ g_pc98_dumpconf(struct sbuf *sb, const char *indent, struct g_geom *gp,
}
}
-static void
-g_pc98_print(int i, struct dos_partition *dp)
-{
- char sname[17];
-
- strncpy(sname, dp->dp_name, 16);
- sname[16] = '\0';
-
- g_hexdump(dp, sizeof(dp[0]));
- printf("[%d] mid:%d(0x%x) sid:%d(0x%x)",
- i, dp->dp_mid, dp->dp_mid, dp->dp_sid, dp->dp_sid);
- printf(" s:%d/%d/%d", dp->dp_scyl, dp->dp_shd, dp->dp_ssect);
- printf(" e:%d/%d/%d", dp->dp_ecyl, dp->dp_ehd, dp->dp_esect);
- printf(" sname:%s\n", 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();
OpenPOWER on IntegriCloud