diff options
author | smh <smh@FreeBSD.org> | 2015-11-17 20:55:50 +0000 |
---|---|---|
committer | smh <smh@FreeBSD.org> | 2015-11-17 20:55:50 +0000 |
commit | 3de9025c5c34fbd67d89ecee08190a4c4fa30fe5 (patch) | |
tree | c82e735c9835c3949b561a451bcb7e34da8a6f40 | |
parent | 195994e5fb55d95f6694729e2f17cee7e2313e3e (diff) | |
download | FreeBSD-src-3de9025c5c34fbd67d89ecee08190a4c4fa30fe5.zip FreeBSD-src-3de9025c5c34fbd67d89ecee08190a4c4fa30fe5.tar.gz |
Fix early kernel dump via dumpdev env
Setting the dumpdev via env e.g. loader.conf provides the ability to
configure the kernel dump device during early boot. When using this
g_io_getattr was returning EPERM due to cp->acr == 0.
Fix this by calling g_access to ensure we're a read consumer prior
to calling g_dev_setdumpdev.
MFC after: 2 weeks
Sponsored by: Multiplay
-rw-r--r-- | sys/geom/geom_dev.c | 27 |
1 files changed, 22 insertions, 5 deletions
diff --git a/sys/geom/geom_dev.c b/sys/geom/geom_dev.c index 6aecd78..3e3ae8f 100644 --- a/sys/geom/geom_dev.c +++ b/sys/geom/geom_dev.c @@ -150,24 +150,38 @@ g_dev_setdumpdev(struct cdev *dev, struct thread *td) return (error); } -static void +static int init_dumpdev(struct cdev *dev) { + struct g_consumer *cp; const char *devprefix = "/dev/", *devname; + int error; size_t len; if (dumpdev == NULL) - return; + return (0); + len = strlen(devprefix); devname = devtoname(dev); if (strcmp(devname, dumpdev) != 0 && (strncmp(dumpdev, devprefix, len) != 0 || strcmp(devname, dumpdev + len) != 0)) - return; - if (g_dev_setdumpdev(dev, curthread) == 0) { + return (0); + + cp = (struct g_consumer *)dev->si_drv2; + error = g_access(cp, 1, 0, 0); + if (error != 0) + return (error); + + error = g_dev_setdumpdev(dev, curthread); + if (error == 0) { freeenv(dumpdev); dumpdev = NULL; } + + (void)g_access(cp, -1, 0, 0); + + return (error); } static void @@ -312,7 +326,10 @@ g_dev_taste(struct g_class *mp, struct g_provider *pp, int insist __unused) dev->si_iosize_max = MAXPHYS; dev->si_drv2 = cp; - init_dumpdev(dev); + error = init_dumpdev(dev); + if (error != 0) + printf("%s: init_dumpdev() failed (gp->name=%s, error=%d)\n", + __func__, gp->name, error); g_dev_attrchanged(cp, "GEOM::physpath"); snprintf(buf, sizeof(buf), "cdev=%s", gp->name); |