summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMatthew Dempsky <matthew@cvs.openbsd.org>2011-06-01 03:25:02 +0000
committerMatthew Dempsky <matthew@cvs.openbsd.org>2011-06-01 03:25:02 +0000
commit14a238cd2fb6586628725e50ef1d3ef4ad01c771 (patch)
tree699ad2f7ca5bc885bc29c86ec295c69a58d0800a
parentb446932b07f5b00e5606436e3455e1c58082b71c (diff)
Add a few KASSERTs to config_attach() for sanity to make sure we don't
try to reuse device unit numbers and to check that the device pointer array is allocated and large enough. Also, improve the panic message generated by config_detach() when we detect that we're detaching a device that still has children. Discussed with deraadt@ while trying to brainstorm ways that interleaving config_attach and config_detach could blow up.
-rw-r--r--sys/kern/subr_autoconf.c16
1 files changed, 13 insertions, 3 deletions
diff --git a/sys/kern/subr_autoconf.c b/sys/kern/subr_autoconf.c
index 4d9fcbe1bf4..8f641ee706d 100644
--- a/sys/kern/subr_autoconf.c
+++ b/sys/kern/subr_autoconf.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: subr_autoconf.c,v 1.63 2010/09/08 15:50:11 deraadt Exp $ */
+/* $OpenBSD: subr_autoconf.c,v 1.64 2011/06/01 03:25:01 matthew Exp $ */
/* $NetBSD: subr_autoconf.c,v 1.21 1996/04/04 06:06:18 cgd Exp $ */
/*
@@ -350,6 +350,9 @@ config_attach(struct device *parent, void *match, void *aux, cfprint_t print)
cd = cf->cf_driver;
ca = cf->cf_attach;
+ KASSERT(cd->cd_devs != NULL);
+ KASSERT(dev->dv_unit < cd->cd_ndevs);
+ KASSERT(cd->cd_devs[dev->dv_unit] == NULL);
cd->cd_devs[dev->dv_unit] = dev;
/*
@@ -543,11 +546,18 @@ config_detach(struct device *dev, int flags)
* after parents, we only need to search the latter part of
* the list.)
*/
+ i = 0;
for (d = TAILQ_NEXT(dev, dv_list); d != NULL;
d = TAILQ_NEXT(d, dv_list)) {
- if (d->dv_parent == dev)
- panic("config_detach: detached device has children");
+ if (d->dv_parent == dev) {
+ printf("config_detach: %s attached at %s\n",
+ d->dv_xname, dev->dv_xname);
+ i = 1;
+ }
}
+ if (i != 0)
+ panic("config_detach: detached device (%s) has children",
+ dev->dv_xname);
#endif
/*