diff options
author | Matthew Dempsky <matthew@cvs.openbsd.org> | 2011-06-01 03:25:02 +0000 |
---|---|---|
committer | Matthew Dempsky <matthew@cvs.openbsd.org> | 2011-06-01 03:25:02 +0000 |
commit | 14a238cd2fb6586628725e50ef1d3ef4ad01c771 (patch) | |
tree | 699ad2f7ca5bc885bc29c86ec295c69a58d0800a | |
parent | b446932b07f5b00e5606436e3455e1c58082b71c (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.c | 16 |
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 /* |