summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorsmh <smh@FreeBSD.org>2015-11-17 20:55:50 +0000
committersmh <smh@FreeBSD.org>2015-11-17 20:55:50 +0000
commit3de9025c5c34fbd67d89ecee08190a4c4fa30fe5 (patch)
treec82e735c9835c3949b561a451bcb7e34da8a6f40
parent195994e5fb55d95f6694729e2f17cee7e2313e3e (diff)
downloadFreeBSD-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.c27
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);
OpenPOWER on IntegriCloud