2005-02-27 00:34:55 +03:00
|
|
|
/* $NetBSD: subr_autoconf.c,v 1.93 2005/02/26 21:34:55 perry Exp $ */
|
2000-06-04 23:14:14 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (c) 1996, 2000 Christopher G. Demetriou
|
|
|
|
* All rights reserved.
|
2005-02-27 00:34:55 +03:00
|
|
|
*
|
2000-06-04 23:14:14 +04:00
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
|
|
* must display the following acknowledgement:
|
2000-06-14 02:36:16 +04:00
|
|
|
* This product includes software developed for the
|
2003-11-17 13:07:58 +03:00
|
|
|
* NetBSD Project. See http://www.NetBSD.org/ for
|
2000-06-14 02:36:16 +04:00
|
|
|
* information about NetBSD.
|
2000-06-04 23:14:14 +04:00
|
|
|
* 4. The name of the author may not be used to endorse or promote products
|
2000-06-14 02:36:16 +04:00
|
|
|
* derived from this software without specific prior written permission.
|
2005-02-27 00:34:55 +03:00
|
|
|
*
|
2000-06-04 23:14:14 +04:00
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
2005-02-27 00:34:55 +03:00
|
|
|
*
|
2000-06-14 02:36:16 +04:00
|
|
|
* --(license Id: LICENSE.proto,v 1.1 2000/06/13 21:40:26 cgd Exp )--
|
2000-06-04 23:14:14 +04:00
|
|
|
*/
|
1994-06-29 10:29:24 +04:00
|
|
|
|
1993-08-13 17:19:41 +04:00
|
|
|
/*
|
1994-05-20 08:15:04 +04:00
|
|
|
* Copyright (c) 1992, 1993
|
|
|
|
* The Regents of the University of California. All rights reserved.
|
1993-08-13 17:19:41 +04:00
|
|
|
*
|
|
|
|
* This software was developed by the Computer Systems Engineering group
|
|
|
|
* at Lawrence Berkeley Laboratory under DARPA contract BG 91-66 and
|
|
|
|
* contributed to Berkeley.
|
|
|
|
*
|
|
|
|
* All advertising materials mentioning features or use of this software
|
|
|
|
* must display the following acknowledgement:
|
|
|
|
* This product includes software developed by the University of
|
|
|
|
* California, Lawrence Berkeley Laboratories.
|
|
|
|
*
|
1994-05-20 08:15:04 +04:00
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
2003-08-07 20:26:28 +04:00
|
|
|
* 3. Neither the name of the University nor the names of its contributors
|
1994-05-20 08:15:04 +04:00
|
|
|
* may be used to endorse or promote products derived from this software
|
|
|
|
* without specific prior written permission.
|
1993-08-13 17:19:41 +04:00
|
|
|
*
|
1994-05-20 08:15:04 +04:00
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
|
|
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
|
|
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
* SUCH DAMAGE.
|
1993-08-13 17:19:41 +04:00
|
|
|
*
|
1994-05-20 08:15:04 +04:00
|
|
|
* from: Header: subr_autoconf.c,v 1.12 93/02/01 19:31:48 torek Exp (LBL)
|
1994-06-29 10:29:24 +04:00
|
|
|
*
|
1998-03-01 05:20:01 +03:00
|
|
|
* @(#)subr_autoconf.c 8.3 (Berkeley) 5/17/94
|
1993-08-13 17:19:41 +04:00
|
|
|
*/
|
|
|
|
|
2000-06-02 05:31:52 +04:00
|
|
|
#include <sys/cdefs.h>
|
2005-02-27 00:34:55 +03:00
|
|
|
__KERNEL_RCSID(0, "$NetBSD: subr_autoconf.c,v 1.93 2005/02/26 21:34:55 perry Exp $");
|
2002-02-15 14:18:26 +03:00
|
|
|
|
|
|
|
#include "opt_ddb.h"
|
2000-06-02 05:31:52 +04:00
|
|
|
|
1993-12-18 06:59:02 +03:00
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/device.h>
|
|
|
|
#include <sys/malloc.h>
|
1996-02-04 05:15:01 +03:00
|
|
|
#include <sys/systm.h>
|
1999-09-18 00:11:56 +04:00
|
|
|
#include <sys/kernel.h>
|
1998-11-17 11:38:07 +03:00
|
|
|
#include <sys/errno.h>
|
2000-01-24 21:03:19 +03:00
|
|
|
#include <sys/proc.h>
|
2003-01-02 03:12:16 +03:00
|
|
|
#include <sys/reboot.h>
|
1994-11-04 09:40:11 +03:00
|
|
|
#include <machine/limits.h>
|
1993-08-13 17:19:41 +04:00
|
|
|
|
2001-07-01 06:56:20 +04:00
|
|
|
#include "opt_userconf.h"
|
|
|
|
#ifdef USERCONF
|
|
|
|
#include <sys/userconf.h>
|
|
|
|
#endif
|
|
|
|
|
1993-08-13 17:19:41 +04:00
|
|
|
/*
|
|
|
|
* Autoconfiguration subroutines.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* ioconf.c exports exactly two names: cfdata and cfroots. All system
|
|
|
|
* devices and drivers are found via these tables.
|
|
|
|
*/
|
|
|
|
extern struct cfdata cfdata[];
|
2003-03-16 11:09:58 +03:00
|
|
|
extern const short cfroots[];
|
1993-08-13 17:19:41 +04:00
|
|
|
|
2002-09-27 06:24:06 +04:00
|
|
|
/*
|
|
|
|
* List of all cfdriver structures. We use this to detect duplicates
|
|
|
|
* when other cfdrivers are loaded.
|
|
|
|
*/
|
2002-09-27 09:45:03 +04:00
|
|
|
struct cfdriverlist allcfdrivers = LIST_HEAD_INITIALIZER(&allcfdrivers);
|
|
|
|
extern struct cfdriver * const cfdriver_list_initial[];
|
2002-09-27 06:24:06 +04:00
|
|
|
|
2002-10-04 05:50:53 +04:00
|
|
|
/*
|
|
|
|
* Initial list of cfattach's.
|
|
|
|
*/
|
|
|
|
extern const struct cfattachinit cfattachinit[];
|
|
|
|
|
2002-09-24 03:16:06 +04:00
|
|
|
/*
|
|
|
|
* List of cfdata tables. We always have one such list -- the one
|
|
|
|
* built statically when the kernel was configured.
|
|
|
|
*/
|
|
|
|
struct cftablelist allcftables;
|
|
|
|
static struct cftable initcftable;
|
|
|
|
|
2003-07-04 04:24:29 +04:00
|
|
|
/*
|
|
|
|
* Database of device properties.
|
|
|
|
*/
|
|
|
|
propdb_t dev_propdb;
|
|
|
|
|
1993-08-13 17:19:41 +04:00
|
|
|
#define ROOT ((struct device *)NULL)
|
|
|
|
|
1994-11-04 09:40:11 +03:00
|
|
|
struct matchinfo {
|
|
|
|
cfmatch_t fn;
|
2004-08-18 02:13:18 +04:00
|
|
|
cfmatch_loc_t fn_loc;
|
1994-11-04 09:40:11 +03:00
|
|
|
struct device *parent;
|
2004-08-18 02:13:18 +04:00
|
|
|
const locdesc_t *ldesc;
|
1996-12-05 03:09:10 +03:00
|
|
|
void *aux;
|
|
|
|
struct cfdata *match;
|
|
|
|
int pri;
|
1994-11-04 09:40:11 +03:00
|
|
|
};
|
|
|
|
|
2000-06-02 05:31:52 +04:00
|
|
|
static char *number(char *, int);
|
|
|
|
static void mapply(struct matchinfo *, struct cfdata *);
|
1996-02-04 05:15:01 +03:00
|
|
|
|
1998-06-09 22:46:12 +04:00
|
|
|
struct deferred_config {
|
|
|
|
TAILQ_ENTRY(deferred_config) dc_queue;
|
|
|
|
struct device *dc_dev;
|
2000-06-02 05:31:52 +04:00
|
|
|
void (*dc_func)(struct device *);
|
1998-06-09 22:46:12 +04:00
|
|
|
};
|
|
|
|
|
1999-09-15 23:37:08 +04:00
|
|
|
TAILQ_HEAD(deferred_config_head, deferred_config);
|
1998-06-09 22:46:12 +04:00
|
|
|
|
1999-09-15 23:37:08 +04:00
|
|
|
struct deferred_config_head deferred_config_queue;
|
|
|
|
struct deferred_config_head interrupt_config_queue;
|
|
|
|
|
2000-06-02 05:31:52 +04:00
|
|
|
static void config_process_deferred(struct deferred_config_head *,
|
|
|
|
struct device *);
|
1998-06-09 22:46:12 +04:00
|
|
|
|
2002-10-01 22:11:57 +04:00
|
|
|
/* Hooks to finalize configuration once all real devices have been found. */
|
|
|
|
struct finalize_hook {
|
|
|
|
TAILQ_ENTRY(finalize_hook) f_list;
|
|
|
|
int (*f_func)(struct device *);
|
|
|
|
struct device *f_dev;
|
|
|
|
};
|
|
|
|
static TAILQ_HEAD(, finalize_hook) config_finalize_list;
|
|
|
|
static int config_finalize_done;
|
|
|
|
|
2001-05-28 20:40:31 +04:00
|
|
|
/* list of all devices */
|
|
|
|
struct devicelist alldevs;
|
|
|
|
|
2000-01-24 21:03:19 +03:00
|
|
|
__volatile int config_pending; /* semaphore for mountroot */
|
|
|
|
|
2002-09-27 06:24:06 +04:00
|
|
|
#define STREQ(s1, s2) \
|
2002-09-27 10:12:55 +04:00
|
|
|
(*(s1) == *(s2) && strcmp((s1), (s2)) == 0)
|
2002-09-27 06:24:06 +04:00
|
|
|
|
2002-09-30 21:36:31 +04:00
|
|
|
static int config_initialized; /* config_init() has been called. */
|
|
|
|
|
2003-01-01 02:59:11 +03:00
|
|
|
static int config_do_twiddle;
|
|
|
|
|
1996-04-04 04:25:44 +04:00
|
|
|
/*
|
2002-09-30 21:36:31 +04:00
|
|
|
* Initialize the autoconfiguration data structures. Normally this
|
|
|
|
* is done by configure(), but some platforms need to do this very
|
|
|
|
* early (to e.g. initialize the console).
|
1996-04-04 04:25:44 +04:00
|
|
|
*/
|
|
|
|
void
|
2002-09-30 21:36:31 +04:00
|
|
|
config_init(void)
|
1996-04-04 04:25:44 +04:00
|
|
|
{
|
2002-10-04 05:50:53 +04:00
|
|
|
const struct cfattachinit *cfai;
|
|
|
|
int i, j;
|
1996-04-04 04:25:44 +04:00
|
|
|
|
2002-09-30 21:36:31 +04:00
|
|
|
if (config_initialized)
|
|
|
|
return;
|
|
|
|
|
2002-09-27 09:45:03 +04:00
|
|
|
/* allcfdrivers is statically initialized. */
|
2002-10-04 05:50:53 +04:00
|
|
|
for (i = 0; cfdriver_list_initial[i] != NULL; i++) {
|
2002-09-27 06:24:06 +04:00
|
|
|
if (config_cfdriver_attach(cfdriver_list_initial[i]) != 0)
|
|
|
|
panic("configure: duplicate `%s' drivers",
|
|
|
|
cfdriver_list_initial[i]->cd_name);
|
2002-10-04 05:50:53 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
for (cfai = &cfattachinit[0]; cfai->cfai_name != NULL; cfai++) {
|
|
|
|
for (j = 0; cfai->cfai_list[j] != NULL; j++) {
|
|
|
|
if (config_cfattach_attach(cfai->cfai_name,
|
|
|
|
cfai->cfai_list[j]) != 0)
|
|
|
|
panic("configure: duplicate `%s' attachment "
|
|
|
|
"of `%s' driver",
|
|
|
|
cfai->cfai_list[j]->ca_name,
|
|
|
|
cfai->cfai_name);
|
|
|
|
}
|
|
|
|
}
|
2002-09-24 03:16:06 +04:00
|
|
|
|
2002-09-27 06:24:06 +04:00
|
|
|
TAILQ_INIT(&allcftables);
|
2002-09-24 03:16:06 +04:00
|
|
|
initcftable.ct_cfdata = cfdata;
|
|
|
|
TAILQ_INSERT_TAIL(&allcftables, &initcftable, ct_list);
|
|
|
|
|
1998-06-09 22:46:12 +04:00
|
|
|
TAILQ_INIT(&deferred_config_queue);
|
1999-09-15 23:37:08 +04:00
|
|
|
TAILQ_INIT(&interrupt_config_queue);
|
2002-10-01 22:11:57 +04:00
|
|
|
TAILQ_INIT(&config_finalize_list);
|
2005-02-27 00:34:55 +03:00
|
|
|
TAILQ_INIT(&alldevs);
|
1999-09-15 22:10:33 +04:00
|
|
|
|
2002-09-30 21:36:31 +04:00
|
|
|
config_initialized = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Configure the system's hardware.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
configure(void)
|
|
|
|
{
|
2003-01-01 02:59:11 +03:00
|
|
|
int errcnt;
|
2002-09-30 21:36:31 +04:00
|
|
|
|
|
|
|
/* Initialize data structures. */
|
|
|
|
config_init();
|
|
|
|
|
2003-07-04 04:24:29 +04:00
|
|
|
/* Initialize the device property database. */
|
|
|
|
dev_propdb = propdb_create("device properties");
|
|
|
|
if (dev_propdb == NULL)
|
|
|
|
panic("unable to create device property database");
|
|
|
|
|
2001-07-01 06:56:20 +04:00
|
|
|
#ifdef USERCONF
|
|
|
|
if (boothowto & RB_USERCONF)
|
|
|
|
user_config();
|
|
|
|
#endif
|
|
|
|
|
2003-01-01 02:59:11 +03:00
|
|
|
if ((boothowto & (AB_SILENT|AB_VERBOSE)) == AB_SILENT) {
|
|
|
|
config_do_twiddle = 1;
|
|
|
|
printf_nolog("Detecting hardware...");
|
|
|
|
}
|
|
|
|
|
1999-09-15 22:10:33 +04:00
|
|
|
/*
|
|
|
|
* Do the machine-dependent portion of autoconfiguration. This
|
|
|
|
* sets the configuration machinery here in motion by "finding"
|
|
|
|
* the root bus. When this function returns, we expect interrupts
|
|
|
|
* to be enabled.
|
|
|
|
*/
|
|
|
|
cpu_configure();
|
1999-09-18 00:11:56 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Now that we've found all the hardware, start the real time
|
|
|
|
* and statistics clocks.
|
|
|
|
*/
|
|
|
|
initclocks();
|
|
|
|
|
|
|
|
cold = 0; /* clocks are running, we're warm now! */
|
1999-09-15 23:37:08 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Now callback to finish configuration for devices which want
|
|
|
|
* to do this once interrupts are enabled.
|
|
|
|
*/
|
|
|
|
config_process_deferred(&interrupt_config_queue, NULL);
|
2003-01-01 02:59:11 +03:00
|
|
|
|
|
|
|
errcnt = aprint_get_error_count();
|
|
|
|
if ((boothowto & (AB_QUIET|AB_SILENT)) != 0 &&
|
|
|
|
(boothowto & AB_VERBOSE) == 0) {
|
|
|
|
if (config_do_twiddle) {
|
|
|
|
config_do_twiddle = 0;
|
|
|
|
printf_nolog("done.\n");
|
|
|
|
}
|
|
|
|
if (errcnt != 0) {
|
|
|
|
printf("WARNING: %d error%s while detecting hardware; "
|
|
|
|
"check system log.\n", errcnt,
|
|
|
|
errcnt == 1 ? "" : "s");
|
|
|
|
}
|
|
|
|
}
|
1996-04-04 04:25:44 +04:00
|
|
|
}
|
|
|
|
|
2002-09-27 06:24:06 +04:00
|
|
|
/*
|
|
|
|
* Add a cfdriver to the system.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
config_cfdriver_attach(struct cfdriver *cd)
|
|
|
|
{
|
|
|
|
struct cfdriver *lcd;
|
|
|
|
|
|
|
|
/* Make sure this driver isn't already in the system. */
|
|
|
|
LIST_FOREACH(lcd, &allcfdrivers, cd_list) {
|
|
|
|
if (STREQ(lcd->cd_name, cd->cd_name))
|
|
|
|
return (EEXIST);
|
|
|
|
}
|
|
|
|
|
2002-10-04 05:50:53 +04:00
|
|
|
LIST_INIT(&cd->cd_attach);
|
2002-09-27 06:24:06 +04:00
|
|
|
LIST_INSERT_HEAD(&allcfdrivers, cd, cd_list);
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Remove a cfdriver from the system.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
config_cfdriver_detach(struct cfdriver *cd)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* Make sure there are no active instances. */
|
|
|
|
for (i = 0; i < cd->cd_ndevs; i++) {
|
|
|
|
if (cd->cd_devs[i] != NULL)
|
|
|
|
return (EBUSY);
|
|
|
|
}
|
|
|
|
|
2002-10-04 05:50:53 +04:00
|
|
|
/* ...and no attachments loaded. */
|
|
|
|
if (LIST_EMPTY(&cd->cd_attach) == 0)
|
|
|
|
return (EBUSY);
|
|
|
|
|
2002-09-27 06:24:06 +04:00
|
|
|
LIST_REMOVE(cd, cd_list);
|
|
|
|
|
|
|
|
KASSERT(cd->cd_devs == NULL);
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Look up a cfdriver by name.
|
|
|
|
*/
|
2002-10-20 06:26:59 +04:00
|
|
|
struct cfdriver *
|
2002-09-27 06:24:06 +04:00
|
|
|
config_cfdriver_lookup(const char *name)
|
|
|
|
{
|
|
|
|
struct cfdriver *cd;
|
|
|
|
|
|
|
|
LIST_FOREACH(cd, &allcfdrivers, cd_list) {
|
|
|
|
if (STREQ(cd->cd_name, name))
|
|
|
|
return (cd);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (NULL);
|
|
|
|
}
|
|
|
|
|
2002-10-04 05:50:53 +04:00
|
|
|
/*
|
|
|
|
* Add a cfattach to the specified driver.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
config_cfattach_attach(const char *driver, struct cfattach *ca)
|
|
|
|
{
|
|
|
|
struct cfattach *lca;
|
|
|
|
struct cfdriver *cd;
|
|
|
|
|
|
|
|
cd = config_cfdriver_lookup(driver);
|
|
|
|
if (cd == NULL)
|
|
|
|
return (ESRCH);
|
|
|
|
|
|
|
|
/* Make sure this attachment isn't already on this driver. */
|
|
|
|
LIST_FOREACH(lca, &cd->cd_attach, ca_list) {
|
|
|
|
if (STREQ(lca->ca_name, ca->ca_name))
|
|
|
|
return (EEXIST);
|
|
|
|
}
|
|
|
|
|
|
|
|
LIST_INSERT_HEAD(&cd->cd_attach, ca, ca_list);
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Remove a cfattach from the specified driver.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
config_cfattach_detach(const char *driver, struct cfattach *ca)
|
|
|
|
{
|
|
|
|
struct cfdriver *cd;
|
|
|
|
struct device *dev;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
cd = config_cfdriver_lookup(driver);
|
|
|
|
if (cd == NULL)
|
|
|
|
return (ESRCH);
|
|
|
|
|
|
|
|
/* Make sure there are no active instances. */
|
|
|
|
for (i = 0; i < cd->cd_ndevs; i++) {
|
|
|
|
if ((dev = cd->cd_devs[i]) == NULL)
|
|
|
|
continue;
|
2002-10-09 06:59:55 +04:00
|
|
|
if (dev->dv_cfattach == ca)
|
2002-10-04 05:50:53 +04:00
|
|
|
return (EBUSY);
|
|
|
|
}
|
|
|
|
|
|
|
|
LIST_REMOVE(ca, ca_list);
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Look up a cfattach by name.
|
|
|
|
*/
|
|
|
|
static struct cfattach *
|
|
|
|
config_cfattach_lookup_cd(struct cfdriver *cd, const char *atname)
|
|
|
|
{
|
|
|
|
struct cfattach *ca;
|
|
|
|
|
|
|
|
LIST_FOREACH(ca, &cd->cd_attach, ca_list) {
|
|
|
|
if (STREQ(ca->ca_name, atname))
|
|
|
|
return (ca);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Look up a cfattach by driver/attachment name.
|
|
|
|
*/
|
|
|
|
struct cfattach *
|
|
|
|
config_cfattach_lookup(const char *name, const char *atname)
|
|
|
|
{
|
|
|
|
struct cfdriver *cd;
|
|
|
|
|
|
|
|
cd = config_cfdriver_lookup(name);
|
|
|
|
if (cd == NULL)
|
|
|
|
return (NULL);
|
|
|
|
|
|
|
|
return (config_cfattach_lookup_cd(cd, atname));
|
|
|
|
}
|
|
|
|
|
1993-08-13 17:19:41 +04:00
|
|
|
/*
|
|
|
|
* Apply the matching function and choose the best. This is used
|
|
|
|
* a few times and we want to keep the code small.
|
|
|
|
*/
|
1994-11-04 09:40:11 +03:00
|
|
|
static void
|
2000-06-02 05:31:52 +04:00
|
|
|
mapply(struct matchinfo *m, struct cfdata *cf)
|
1993-08-13 17:19:41 +04:00
|
|
|
{
|
2000-03-28 21:30:10 +04:00
|
|
|
int pri;
|
1993-08-13 17:19:41 +04:00
|
|
|
|
2004-08-18 02:13:18 +04:00
|
|
|
if (m->fn != NULL) {
|
|
|
|
KASSERT(m->fn_loc == NULL);
|
1996-12-05 03:09:10 +03:00
|
|
|
pri = (*m->fn)(m->parent, cf, m->aux);
|
2004-08-18 02:13:18 +04:00
|
|
|
} else if (m->fn_loc != NULL) {
|
|
|
|
pri = (*m->fn_loc)(m->parent, cf, m->ldesc, m->aux);
|
|
|
|
} else {
|
2002-10-04 05:50:53 +04:00
|
|
|
struct cfattach *ca;
|
|
|
|
|
|
|
|
ca = config_cfattach_lookup(cf->cf_name, cf->cf_atname);
|
|
|
|
if (ca == NULL) {
|
|
|
|
/* No attachment for this entry, oh well. */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (ca->ca_match == NULL) {
|
|
|
|
panic("mapply: no match function for '%s' attachment "
|
|
|
|
"of '%s'", cf->cf_atname, cf->cf_name);
|
1993-08-16 03:01:58 +04:00
|
|
|
}
|
2002-10-04 05:50:53 +04:00
|
|
|
pri = (*ca->ca_match)(m->parent, cf, m->aux);
|
1993-08-16 03:01:58 +04:00
|
|
|
}
|
1993-08-13 17:19:41 +04:00
|
|
|
if (pri > m->pri) {
|
1996-12-05 03:09:10 +03:00
|
|
|
m->match = cf;
|
1993-08-13 17:19:41 +04:00
|
|
|
m->pri = pri;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-08-30 13:48:05 +04:00
|
|
|
/*
|
|
|
|
* Helper function: check whether the driver supports the interface attribute.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
cfdriver_has_iattr(const struct cfdriver *cd, const char *ia)
|
|
|
|
{
|
|
|
|
const char * const *cpp;
|
|
|
|
|
|
|
|
if (cd->cd_attrs == NULL)
|
|
|
|
return (0);
|
|
|
|
|
|
|
|
for (cpp = cd->cd_attrs; *cpp; cpp++) {
|
|
|
|
if (STREQ(*cpp, ia)) {
|
|
|
|
/* Match. */
|
|
|
|
return (1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
2002-09-26 08:07:35 +04:00
|
|
|
/*
|
|
|
|
* Determine if `parent' is a potential parent for a device spec based
|
|
|
|
* on `cfp'.
|
|
|
|
*/
|
|
|
|
static int
|
2004-08-18 02:13:18 +04:00
|
|
|
cfparent_match(const struct device *parent, const struct cfparent *cfp)
|
2002-09-26 08:07:35 +04:00
|
|
|
{
|
2002-09-27 06:24:06 +04:00
|
|
|
struct cfdriver *pcd;
|
2002-09-26 08:07:35 +04:00
|
|
|
|
2002-09-27 10:12:55 +04:00
|
|
|
/* We don't match root nodes here. */
|
|
|
|
if (cfp == NULL)
|
|
|
|
return (0);
|
|
|
|
|
2002-10-09 06:59:55 +04:00
|
|
|
pcd = parent->dv_cfdriver;
|
2002-09-27 06:24:06 +04:00
|
|
|
KASSERT(pcd != NULL);
|
|
|
|
|
2002-09-26 08:07:35 +04:00
|
|
|
/*
|
|
|
|
* First, ensure this parent has the correct interface
|
|
|
|
* attribute.
|
|
|
|
*/
|
2004-08-30 13:48:05 +04:00
|
|
|
if (!cfdriver_has_iattr(pcd, cfp->cfp_iattr))
|
|
|
|
return (0);
|
2002-09-26 08:07:35 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* If no specific parent device instance was specified (i.e.
|
|
|
|
* we're attaching to the attribute only), we're done!
|
|
|
|
*/
|
|
|
|
if (cfp->cfp_parent == NULL)
|
|
|
|
return (1);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check the parent device's name.
|
|
|
|
*/
|
2002-09-27 10:30:05 +04:00
|
|
|
if (STREQ(pcd->cd_name, cfp->cfp_parent) == 0)
|
2002-09-26 08:07:35 +04:00
|
|
|
return (0); /* not the same parent */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Make sure the unit number matches.
|
|
|
|
*/
|
2002-10-09 06:59:55 +04:00
|
|
|
if (cfp->cfp_unit == DVUNIT_ANY || /* wildcard */
|
2002-09-26 08:07:35 +04:00
|
|
|
cfp->cfp_unit == parent->dv_unit)
|
|
|
|
return (1);
|
|
|
|
|
|
|
|
/* Unit numbers don't match. */
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
2004-08-18 02:13:18 +04:00
|
|
|
/*
|
|
|
|
* Helper for config_cfdata_attach(): check all devices whether it could be
|
|
|
|
* parent any attachment in the config data table passed, and rescan.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
rescan_with_cfdata(const struct cfdata *cf)
|
|
|
|
{
|
|
|
|
struct device *d;
|
|
|
|
const struct cfdata *cf1;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* "alldevs" is likely longer than an LKM's cfdata, so make it
|
|
|
|
* the outer loop.
|
|
|
|
*/
|
|
|
|
TAILQ_FOREACH(d, &alldevs, dv_list) {
|
|
|
|
|
|
|
|
if (!(d->dv_cfattach->ca_rescan))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
for (cf1 = cf; cf1->cf_name; cf1++) {
|
|
|
|
|
|
|
|
if (!cfparent_match(d, cf1->cf_pspec))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
(*d->dv_cfattach->ca_rescan)(d,
|
|
|
|
cf1->cf_pspec->cfp_iattr, cf1->cf_loc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Attach a supplemental config data table and rescan potential
|
|
|
|
* parent devices if required.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
config_cfdata_attach(struct cfdata *cf, int scannow)
|
|
|
|
{
|
|
|
|
struct cftable *ct;
|
|
|
|
|
|
|
|
ct = malloc(sizeof(struct cftable), M_DEVBUF, M_WAITOK);
|
|
|
|
ct->ct_cfdata = cf;
|
|
|
|
TAILQ_INSERT_TAIL(&allcftables, ct, ct_list);
|
|
|
|
|
|
|
|
if (scannow)
|
|
|
|
rescan_with_cfdata(cf);
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Helper for config_cfdata_detach: check whether a device is
|
|
|
|
* found through any attachment in the config data table.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
dev_in_cfdata(const struct device *d, const struct cfdata *cf)
|
|
|
|
{
|
|
|
|
const struct cfdata *cf1;
|
|
|
|
|
|
|
|
for (cf1 = cf; cf1->cf_name; cf1++)
|
|
|
|
if (d->dv_cfdata == cf1)
|
|
|
|
return (1);
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Detach a supplemental config data table. Detach all devices found
|
|
|
|
* through that table (and thus keeping references to it) before.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
config_cfdata_detach(struct cfdata *cf)
|
|
|
|
{
|
|
|
|
struct device *d;
|
|
|
|
int error;
|
|
|
|
struct cftable *ct;
|
|
|
|
|
|
|
|
again:
|
|
|
|
TAILQ_FOREACH(d, &alldevs, dv_list) {
|
|
|
|
if (dev_in_cfdata(d, cf)) {
|
|
|
|
error = config_detach(d, 0);
|
|
|
|
if (error) {
|
|
|
|
aprint_error("%s: unable to detach instance\n",
|
|
|
|
d->dv_xname);
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
goto again;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TAILQ_FOREACH(ct, &allcftables, ct_list) {
|
|
|
|
if (ct->ct_cfdata == cf) {
|
|
|
|
TAILQ_REMOVE(&allcftables, ct, ct_list);
|
|
|
|
free(ct, M_DEVBUF);
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* not found -- shouldn't happen */
|
|
|
|
return (EINVAL);
|
|
|
|
}
|
|
|
|
|
2002-09-27 07:17:40 +04:00
|
|
|
/*
|
|
|
|
* Invoke the "match" routine for a cfdata entry on behalf of
|
|
|
|
* an external caller, usually a "submatch" routine.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
config_match(struct device *parent, struct cfdata *cf, void *aux)
|
|
|
|
{
|
2002-10-04 05:50:53 +04:00
|
|
|
struct cfattach *ca;
|
|
|
|
|
|
|
|
ca = config_cfattach_lookup(cf->cf_name, cf->cf_atname);
|
|
|
|
if (ca == NULL) {
|
|
|
|
/* No attachment for this entry, oh well. */
|
|
|
|
return (0);
|
|
|
|
}
|
2002-09-27 07:17:40 +04:00
|
|
|
|
2002-10-04 05:50:53 +04:00
|
|
|
return ((*ca->ca_match)(parent, cf, aux));
|
2002-09-27 07:17:40 +04:00
|
|
|
}
|
|
|
|
|
1993-08-13 17:19:41 +04:00
|
|
|
/*
|
|
|
|
* Iterate over all potential children of some device, calling the given
|
|
|
|
* function (default being the child's match function) for each one.
|
|
|
|
* Nonzero returns are matches; the highest value returned is considered
|
|
|
|
* the best match. Return the `found child' if we got a match, or NULL
|
|
|
|
* otherwise. The `aux' pointer is simply passed on through.
|
|
|
|
*
|
|
|
|
* Note that this function is designed so that it can be used to apply
|
|
|
|
* an arbitrary function to all potential children (its return value
|
|
|
|
* can be ignored).
|
|
|
|
*/
|
1996-12-05 03:09:10 +03:00
|
|
|
struct cfdata *
|
2000-06-02 05:31:52 +04:00
|
|
|
config_search(cfmatch_t fn, struct device *parent, void *aux)
|
1993-08-13 17:19:41 +04:00
|
|
|
{
|
2002-09-24 03:16:06 +04:00
|
|
|
struct cftable *ct;
|
2000-03-28 21:30:10 +04:00
|
|
|
struct cfdata *cf;
|
1993-08-13 17:19:41 +04:00
|
|
|
struct matchinfo m;
|
|
|
|
|
2002-09-30 21:36:31 +04:00
|
|
|
KASSERT(config_initialized);
|
|
|
|
|
1993-08-13 17:19:41 +04:00
|
|
|
m.fn = fn;
|
2004-08-18 02:13:18 +04:00
|
|
|
m.fn_loc = NULL;
|
|
|
|
m.parent = parent;
|
|
|
|
m.aux = aux;
|
|
|
|
m.match = NULL;
|
|
|
|
m.pri = 0;
|
|
|
|
|
|
|
|
TAILQ_FOREACH(ct, &allcftables, ct_list) {
|
|
|
|
for (cf = ct->ct_cfdata; cf->cf_name; cf++) {
|
|
|
|
/*
|
|
|
|
* Skip cf if no longer eligible, otherwise scan
|
|
|
|
* through parents for one matching `parent', and
|
|
|
|
* try match function.
|
|
|
|
*/
|
|
|
|
if (cf->cf_fstate == FSTATE_FOUND)
|
|
|
|
continue;
|
|
|
|
if (cf->cf_fstate == FSTATE_DNOTFOUND ||
|
|
|
|
cf->cf_fstate == FSTATE_DSTAR)
|
|
|
|
continue;
|
|
|
|
if (cfparent_match(parent, cf->cf_pspec))
|
|
|
|
mapply(&m, cf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (m.match);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* same as above, with real locators passed */
|
|
|
|
struct cfdata *
|
|
|
|
config_search_loc(cfmatch_loc_t fn, struct device *parent,
|
|
|
|
const char *ifattr, const locdesc_t *ldesc, void *aux)
|
|
|
|
{
|
|
|
|
struct cftable *ct;
|
|
|
|
struct cfdata *cf;
|
|
|
|
struct matchinfo m;
|
|
|
|
|
|
|
|
KASSERT(config_initialized);
|
2004-08-30 13:48:05 +04:00
|
|
|
KASSERT(!ifattr || cfdriver_has_iattr(parent->dv_cfdriver, ifattr));
|
2004-08-18 02:13:18 +04:00
|
|
|
|
|
|
|
m.fn = NULL;
|
|
|
|
m.fn_loc = fn;
|
1993-08-13 17:19:41 +04:00
|
|
|
m.parent = parent;
|
2004-08-18 02:13:18 +04:00
|
|
|
m.ldesc = ldesc;
|
1994-11-04 06:07:17 +03:00
|
|
|
m.aux = aux;
|
1996-12-05 03:09:10 +03:00
|
|
|
m.match = NULL;
|
1993-08-13 17:19:41 +04:00
|
|
|
m.pri = 0;
|
2002-09-24 03:16:06 +04:00
|
|
|
|
|
|
|
TAILQ_FOREACH(ct, &allcftables, ct_list) {
|
2002-09-27 06:24:06 +04:00
|
|
|
for (cf = ct->ct_cfdata; cf->cf_name; cf++) {
|
2004-08-18 02:13:18 +04:00
|
|
|
|
|
|
|
/* We don't match root nodes here. */
|
|
|
|
if (!cf->cf_pspec)
|
|
|
|
continue;
|
|
|
|
|
2002-09-24 03:16:06 +04:00
|
|
|
/*
|
|
|
|
* Skip cf if no longer eligible, otherwise scan
|
|
|
|
* through parents for one matching `parent', and
|
|
|
|
* try match function.
|
|
|
|
*/
|
|
|
|
if (cf->cf_fstate == FSTATE_FOUND)
|
|
|
|
continue;
|
|
|
|
if (cf->cf_fstate == FSTATE_DNOTFOUND ||
|
|
|
|
cf->cf_fstate == FSTATE_DSTAR)
|
|
|
|
continue;
|
2004-08-18 02:13:18 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* If an interface attribute was specified,
|
|
|
|
* consider only children which attach to
|
|
|
|
* that attribute.
|
|
|
|
*/
|
|
|
|
if (ifattr && !STREQ(ifattr, cf->cf_pspec->cfp_iattr))
|
|
|
|
continue;
|
|
|
|
|
2002-09-26 08:07:35 +04:00
|
|
|
if (cfparent_match(parent, cf->cf_pspec))
|
|
|
|
mapply(&m, cf);
|
2002-09-24 03:16:06 +04:00
|
|
|
}
|
1993-08-13 17:19:41 +04:00
|
|
|
}
|
|
|
|
return (m.match);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Find the given root device.
|
|
|
|
* This is much like config_search, but there is no parent.
|
2002-09-24 03:16:06 +04:00
|
|
|
* Don't bother with multiple cfdata tables; the root node
|
|
|
|
* must always be in the initial table.
|
1993-08-13 17:19:41 +04:00
|
|
|
*/
|
1996-12-05 03:09:10 +03:00
|
|
|
struct cfdata *
|
2000-06-02 05:48:50 +04:00
|
|
|
config_rootsearch(cfmatch_t fn, const char *rootname, void *aux)
|
1993-08-13 17:19:41 +04:00
|
|
|
{
|
2000-03-28 21:30:10 +04:00
|
|
|
struct cfdata *cf;
|
2003-03-16 11:09:58 +03:00
|
|
|
const short *p;
|
1993-08-13 17:19:41 +04:00
|
|
|
struct matchinfo m;
|
|
|
|
|
|
|
|
m.fn = fn;
|
2004-08-18 02:13:18 +04:00
|
|
|
m.fn_loc = NULL;
|
1993-08-13 17:19:41 +04:00
|
|
|
m.parent = ROOT;
|
1994-11-04 06:07:17 +03:00
|
|
|
m.aux = aux;
|
1996-12-05 03:09:10 +03:00
|
|
|
m.match = NULL;
|
1993-08-13 17:19:41 +04:00
|
|
|
m.pri = 0;
|
|
|
|
/*
|
|
|
|
* Look at root entries for matching name. We do not bother
|
|
|
|
* with found-state here since only one root should ever be
|
|
|
|
* searched (and it must be done first).
|
|
|
|
*/
|
|
|
|
for (p = cfroots; *p >= 0; p++) {
|
|
|
|
cf = &cfdata[*p];
|
2002-09-27 06:24:06 +04:00
|
|
|
if (strcmp(cf->cf_name, rootname) == 0)
|
1994-11-04 09:40:11 +03:00
|
|
|
mapply(&m, cf);
|
1993-08-13 17:19:41 +04:00
|
|
|
}
|
|
|
|
return (m.match);
|
|
|
|
}
|
|
|
|
|
2003-02-09 12:14:58 +03:00
|
|
|
static const char * const msgs[3] = { "", " not configured\n", " unsupported\n" };
|
1993-08-13 17:19:41 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The given `aux' argument describes a device that has been found
|
|
|
|
* on the given parent, but not necessarily configured. Locate the
|
1996-02-28 00:45:46 +03:00
|
|
|
* configuration data for that device (using the submatch function
|
|
|
|
* provided, or using candidates' cd_match configuration driver
|
|
|
|
* functions) and attach it, and return true. If the device was
|
1993-08-13 17:19:41 +04:00
|
|
|
* not configured, call the given `print' function and return 0.
|
|
|
|
*/
|
1996-04-04 10:06:18 +04:00
|
|
|
struct device *
|
2000-06-02 05:31:52 +04:00
|
|
|
config_found_sm(struct device *parent, void *aux, cfprint_t print,
|
|
|
|
cfmatch_t submatch)
|
1993-08-13 17:19:41 +04:00
|
|
|
{
|
1996-12-05 03:09:10 +03:00
|
|
|
struct cfdata *cf;
|
|
|
|
|
|
|
|
if ((cf = config_search(submatch, parent, aux)) != NULL)
|
|
|
|
return (config_attach(parent, cf, aux, print));
|
2003-01-01 02:59:11 +03:00
|
|
|
if (print) {
|
|
|
|
if (config_do_twiddle)
|
|
|
|
twiddle();
|
|
|
|
aprint_normal("%s", msgs[(*print)(aux, parent->dv_xname)]);
|
|
|
|
}
|
1996-04-04 10:06:18 +04:00
|
|
|
return (NULL);
|
1993-08-13 17:19:41 +04:00
|
|
|
}
|
|
|
|
|
2004-08-18 02:13:18 +04:00
|
|
|
/* same as above, with real locators passed */
|
|
|
|
struct device *
|
|
|
|
config_found_sm_loc(struct device *parent,
|
|
|
|
const char *ifattr, const locdesc_t *ldesc, void *aux,
|
|
|
|
cfprint_t print, cfmatch_loc_t submatch)
|
|
|
|
{
|
|
|
|
struct cfdata *cf;
|
|
|
|
|
|
|
|
if ((cf = config_search_loc(submatch, parent, ifattr, ldesc, aux)))
|
|
|
|
return(config_attach_loc(parent, cf, ldesc, aux, print));
|
|
|
|
if (print) {
|
|
|
|
if (config_do_twiddle)
|
|
|
|
twiddle();
|
|
|
|
aprint_normal("%s", msgs[(*print)(aux, parent->dv_xname)]);
|
|
|
|
}
|
|
|
|
return (NULL);
|
|
|
|
}
|
|
|
|
|
1993-08-13 17:19:41 +04:00
|
|
|
/*
|
|
|
|
* As above, but for root devices.
|
|
|
|
*/
|
1996-04-04 10:06:18 +04:00
|
|
|
struct device *
|
2000-06-02 05:48:50 +04:00
|
|
|
config_rootfound(const char *rootname, void *aux)
|
1993-08-13 17:19:41 +04:00
|
|
|
{
|
1996-12-05 03:09:10 +03:00
|
|
|
struct cfdata *cf;
|
|
|
|
|
|
|
|
if ((cf = config_rootsearch((cfmatch_t)NULL, rootname, aux)) != NULL)
|
|
|
|
return (config_attach(ROOT, cf, aux, (cfprint_t)NULL));
|
2003-01-01 02:59:11 +03:00
|
|
|
aprint_error("root device %s not configured\n", rootname);
|
1996-04-04 10:06:18 +04:00
|
|
|
return (NULL);
|
1993-08-13 17:19:41 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* just like sprintf(buf, "%d") except that it works from the end */
|
|
|
|
static char *
|
2000-06-02 05:31:52 +04:00
|
|
|
number(char *ep, int n)
|
1993-08-13 17:19:41 +04:00
|
|
|
{
|
|
|
|
|
|
|
|
*--ep = 0;
|
|
|
|
while (n >= 10) {
|
|
|
|
*--ep = (n % 10) + '0';
|
|
|
|
n /= 10;
|
|
|
|
}
|
|
|
|
*--ep = n + '0';
|
|
|
|
return (ep);
|
|
|
|
}
|
|
|
|
|
2001-12-02 05:40:57 +03:00
|
|
|
/*
|
|
|
|
* Expand the size of the cd_devs array if necessary.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
config_makeroom(int n, struct cfdriver *cd)
|
|
|
|
{
|
|
|
|
int old, new;
|
|
|
|
void **nsp;
|
|
|
|
|
|
|
|
if (n < cd->cd_ndevs)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Need to expand the array.
|
|
|
|
*/
|
|
|
|
old = cd->cd_ndevs;
|
2001-12-10 03:33:04 +03:00
|
|
|
if (old == 0)
|
2001-12-02 05:40:57 +03:00
|
|
|
new = MINALLOCSIZE / sizeof(void *);
|
|
|
|
else
|
|
|
|
new = old * 2;
|
|
|
|
while (new <= n)
|
|
|
|
new *= 2;
|
|
|
|
cd->cd_ndevs = new;
|
|
|
|
nsp = malloc(new * sizeof(void *), M_DEVBUF,
|
2005-02-27 00:34:55 +03:00
|
|
|
cold ? M_NOWAIT : M_WAITOK);
|
2001-12-05 19:12:42 +03:00
|
|
|
if (nsp == NULL)
|
2001-12-02 05:40:57 +03:00
|
|
|
panic("config_attach: %sing dev array",
|
|
|
|
old != 0 ? "expand" : "creat");
|
|
|
|
memset(nsp + old, 0, (new - old) * sizeof(void *));
|
2001-12-10 03:33:04 +03:00
|
|
|
if (old != 0) {
|
2001-12-02 05:40:57 +03:00
|
|
|
memcpy(nsp, cd->cd_devs, old * sizeof(void *));
|
|
|
|
free(cd->cd_devs, M_DEVBUF);
|
|
|
|
}
|
|
|
|
cd->cd_devs = nsp;
|
|
|
|
}
|
|
|
|
|
1993-08-13 17:19:41 +04:00
|
|
|
/*
|
|
|
|
* Attach a found device. Allocates memory for device variables.
|
|
|
|
*/
|
1996-12-05 03:09:10 +03:00
|
|
|
struct device *
|
2004-08-18 02:13:18 +04:00
|
|
|
config_attach_loc(struct device *parent, struct cfdata *cf,
|
|
|
|
const locdesc_t *ldesc, void *aux, cfprint_t print)
|
1996-12-05 03:09:10 +03:00
|
|
|
{
|
2000-03-28 21:30:10 +04:00
|
|
|
struct device *dev;
|
2002-09-24 03:16:06 +04:00
|
|
|
struct cftable *ct;
|
2000-03-28 21:30:10 +04:00
|
|
|
struct cfdriver *cd;
|
2002-10-04 05:50:53 +04:00
|
|
|
struct cfattach *ca;
|
2000-03-28 21:30:10 +04:00
|
|
|
size_t lname, lunit;
|
2000-06-02 05:48:50 +04:00
|
|
|
const char *xunit;
|
1996-12-05 03:09:10 +03:00
|
|
|
int myunit;
|
|
|
|
char num[10];
|
|
|
|
|
2002-09-27 06:24:06 +04:00
|
|
|
cd = config_cfdriver_lookup(cf->cf_name);
|
|
|
|
KASSERT(cd != NULL);
|
2002-10-04 05:50:53 +04:00
|
|
|
|
|
|
|
ca = config_cfattach_lookup_cd(cd, cf->cf_atname);
|
|
|
|
KASSERT(ca != NULL);
|
|
|
|
|
1996-12-05 03:09:10 +03:00
|
|
|
if (ca->ca_devsize < sizeof(struct device))
|
|
|
|
panic("config_attach");
|
2002-09-26 08:07:35 +04:00
|
|
|
|
2000-01-18 10:45:04 +03:00
|
|
|
#ifndef __BROKEN_CONFIG_UNIT_USAGE
|
avoid 'marching unit numbers' for cloning devices. (e.g., previously,
if you com* at pcmcia?, and com3 and com4 as pcmcia cards, and removed
and reinserted the card that was com3, it would become com5. if you then
removed and reinserted com4, it would become com6. etc.) Now, instead
of incrementing FSTATE_STAR configuration entries for a driver when
a cloning instance is attached, leave it alone, and scan the device softc
array (starting at the first cloning unit number) for units which are
available for use. This wastes a tiny bit of time (can require a linear
scan of the softc table for the device), but device attachment should be
relatively infrequent and the number of units of each type of device
is never particularly large anyway.
1999-12-30 04:03:43 +03:00
|
|
|
if (cf->cf_fstate == FSTATE_STAR) {
|
|
|
|
for (myunit = cf->cf_unit; myunit < cd->cd_ndevs; myunit++)
|
|
|
|
if (cd->cd_devs[myunit] == NULL)
|
|
|
|
break;
|
|
|
|
/*
|
|
|
|
* myunit is now the unit of the first NULL device pointer,
|
|
|
|
* or max(cd->cd_ndevs,cf->cf_unit).
|
|
|
|
*/
|
|
|
|
} else {
|
|
|
|
myunit = cf->cf_unit;
|
2002-09-26 08:07:35 +04:00
|
|
|
KASSERT(cf->cf_fstate == FSTATE_NOTFOUND);
|
|
|
|
cf->cf_fstate = FSTATE_FOUND;
|
|
|
|
}
|
|
|
|
#else
|
2000-01-18 10:45:04 +03:00
|
|
|
myunit = cf->cf_unit;
|
|
|
|
if (cf->cf_fstate == FSTATE_STAR)
|
|
|
|
cf->cf_unit++;
|
|
|
|
else {
|
1996-12-05 03:09:10 +03:00
|
|
|
KASSERT(cf->cf_fstate == FSTATE_NOTFOUND);
|
|
|
|
cf->cf_fstate = FSTATE_FOUND;
|
|
|
|
}
|
2002-09-26 08:07:35 +04:00
|
|
|
#endif /* ! __BROKEN_CONFIG_UNIT_USAGE */
|
1996-12-05 03:09:10 +03:00
|
|
|
|
|
|
|
/* compute length of name and decimal expansion of unit number */
|
|
|
|
lname = strlen(cd->cd_name);
|
1998-08-01 02:50:48 +04:00
|
|
|
xunit = number(&num[sizeof(num)], myunit);
|
|
|
|
lunit = &num[sizeof(num)] - xunit;
|
2002-07-10 23:04:09 +04:00
|
|
|
if (lname + lunit > sizeof(dev->dv_xname))
|
1996-12-05 03:09:10 +03:00
|
|
|
panic("config_attach: device name too long");
|
|
|
|
|
|
|
|
/* get memory for all device vars */
|
1999-09-18 00:11:56 +04:00
|
|
|
dev = (struct device *)malloc(ca->ca_devsize, M_DEVBUF,
|
|
|
|
cold ? M_NOWAIT : M_WAITOK);
|
1996-12-05 03:09:10 +03:00
|
|
|
if (!dev)
|
|
|
|
panic("config_attach: memory allocation for device softc failed");
|
Abolition of bcopy, ovbcopy, bcmp, and bzero, phase one.
bcopy(x, y, z) -> memcpy(y, x, z)
ovbcopy(x, y, z) -> memmove(y, x, z)
bcmp(x, y, z) -> memcmp(x, y, z)
bzero(x, y) -> memset(x, 0, y)
1998-08-04 08:03:10 +04:00
|
|
|
memset(dev, 0, ca->ca_devsize);
|
1996-12-05 03:09:10 +03:00
|
|
|
TAILQ_INSERT_TAIL(&alldevs, dev, dv_list); /* link up */
|
|
|
|
dev->dv_class = cd->cd_class;
|
|
|
|
dev->dv_cfdata = cf;
|
2002-10-04 05:50:53 +04:00
|
|
|
dev->dv_cfdriver = cd;
|
|
|
|
dev->dv_cfattach = ca;
|
1996-12-05 03:09:10 +03:00
|
|
|
dev->dv_unit = myunit;
|
Abolition of bcopy, ovbcopy, bcmp, and bzero, phase one.
bcopy(x, y, z) -> memcpy(y, x, z)
ovbcopy(x, y, z) -> memmove(y, x, z)
bcmp(x, y, z) -> memcmp(x, y, z)
bzero(x, y) -> memset(x, 0, y)
1998-08-04 08:03:10 +04:00
|
|
|
memcpy(dev->dv_xname, cd->cd_name, lname);
|
|
|
|
memcpy(dev->dv_xname + lname, xunit, lunit);
|
1996-12-05 03:09:10 +03:00
|
|
|
dev->dv_parent = parent;
|
1998-11-17 11:38:07 +03:00
|
|
|
dev->dv_flags = DVF_ACTIVE; /* always initially active */
|
2004-08-18 02:13:18 +04:00
|
|
|
if (ldesc) {
|
|
|
|
dev->dv_locators = malloc(ldesc->len * sizeof(int),
|
|
|
|
M_DEVBUF, cold ? M_NOWAIT : M_WAITOK);
|
|
|
|
memcpy(dev->dv_locators, ldesc->locs, ldesc->len * sizeof(int));
|
|
|
|
}
|
1998-06-09 22:46:12 +04:00
|
|
|
|
2003-01-01 02:59:11 +03:00
|
|
|
if (config_do_twiddle)
|
|
|
|
twiddle();
|
|
|
|
else
|
|
|
|
aprint_naive("Found ");
|
|
|
|
/*
|
|
|
|
* We want the next two printfs for normal, verbose, and quiet,
|
|
|
|
* but not silent (in which case, we're twiddling, instead).
|
|
|
|
*/
|
|
|
|
if (parent == ROOT) {
|
2003-04-29 04:56:52 +04:00
|
|
|
aprint_naive("%s (root)", dev->dv_xname);
|
|
|
|
aprint_normal("%s (root)", dev->dv_xname);
|
2003-01-01 02:59:11 +03:00
|
|
|
} else {
|
2003-04-29 04:56:52 +04:00
|
|
|
aprint_naive("%s at %s", dev->dv_xname, parent->dv_xname);
|
|
|
|
aprint_normal("%s at %s", dev->dv_xname, parent->dv_xname);
|
1996-12-05 03:09:10 +03:00
|
|
|
if (print)
|
2000-06-02 05:48:50 +04:00
|
|
|
(void) (*print)(aux, NULL);
|
1996-12-05 03:09:10 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* put this device in the devices array */
|
2001-12-02 05:40:57 +03:00
|
|
|
config_makeroom(dev->dv_unit, cd);
|
1996-12-05 03:09:10 +03:00
|
|
|
if (cd->cd_devs[dev->dv_unit])
|
|
|
|
panic("config_attach: duplicate %s", dev->dv_xname);
|
|
|
|
cd->cd_devs[dev->dv_unit] = dev;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Before attaching, clobber any unfound devices that are
|
avoid 'marching unit numbers' for cloning devices. (e.g., previously,
if you com* at pcmcia?, and com3 and com4 as pcmcia cards, and removed
and reinserted the card that was com3, it would become com5. if you then
removed and reinserted com4, it would become com6. etc.) Now, instead
of incrementing FSTATE_STAR configuration entries for a driver when
a cloning instance is attached, leave it alone, and scan the device softc
array (starting at the first cloning unit number) for units which are
available for use. This wastes a tiny bit of time (can require a linear
scan of the softc table for the device), but device attachment should be
relatively infrequent and the number of units of each type of device
is never particularly large anyway.
1999-12-30 04:03:43 +03:00
|
|
|
* otherwise identical.
|
1996-12-05 03:09:10 +03:00
|
|
|
*/
|
2002-09-24 03:16:06 +04:00
|
|
|
TAILQ_FOREACH(ct, &allcftables, ct_list) {
|
2002-09-27 06:24:06 +04:00
|
|
|
for (cf = ct->ct_cfdata; cf->cf_name; cf++) {
|
|
|
|
if (STREQ(cf->cf_name, cd->cd_name) &&
|
2002-09-24 03:16:06 +04:00
|
|
|
cf->cf_unit == dev->dv_unit) {
|
|
|
|
if (cf->cf_fstate == FSTATE_NOTFOUND)
|
|
|
|
cf->cf_fstate = FSTATE_FOUND;
|
2000-01-18 10:45:04 +03:00
|
|
|
#ifdef __BROKEN_CONFIG_UNIT_USAGE
|
2002-09-26 08:07:35 +04:00
|
|
|
/*
|
|
|
|
* Bump the unit number on all starred cfdata
|
|
|
|
* entries for this device.
|
|
|
|
*/
|
2002-09-24 03:16:06 +04:00
|
|
|
if (cf->cf_fstate == FSTATE_STAR)
|
|
|
|
cf->cf_unit++;
|
2000-01-18 10:45:04 +03:00
|
|
|
#endif /* __BROKEN_CONFIG_UNIT_USAGE */
|
2002-09-24 03:16:06 +04:00
|
|
|
}
|
1996-12-05 03:09:10 +03:00
|
|
|
}
|
2002-09-24 03:16:06 +04:00
|
|
|
}
|
2000-02-01 07:01:19 +03:00
|
|
|
#ifdef __HAVE_DEVICE_REGISTER
|
1996-12-05 03:09:10 +03:00
|
|
|
device_register(dev, aux);
|
|
|
|
#endif
|
|
|
|
(*ca->ca_attach)(parent, dev, aux);
|
1999-09-15 23:37:08 +04:00
|
|
|
config_process_deferred(&deferred_config_queue, dev);
|
1996-12-05 03:09:10 +03:00
|
|
|
return (dev);
|
|
|
|
}
|
1993-08-13 17:19:41 +04:00
|
|
|
|
2002-10-09 06:59:55 +04:00
|
|
|
/*
|
|
|
|
* As above, but for pseudo-devices. Pseudo-devices attached in this
|
|
|
|
* way are silently inserted into the device tree, and their children
|
|
|
|
* attached.
|
|
|
|
*
|
|
|
|
* Note that because pseudo-devices are attached silently, any information
|
|
|
|
* the attach routine wishes to print should be prefixed with the device
|
|
|
|
* name by the attach routine.
|
|
|
|
*/
|
|
|
|
struct device *
|
2004-10-15 08:38:36 +04:00
|
|
|
config_attach_pseudo(struct cfdata *cf)
|
2002-10-09 06:59:55 +04:00
|
|
|
{
|
|
|
|
struct device *dev;
|
|
|
|
struct cfdriver *cd;
|
|
|
|
struct cfattach *ca;
|
|
|
|
size_t lname, lunit;
|
|
|
|
const char *xunit;
|
|
|
|
int myunit;
|
|
|
|
char num[10];
|
|
|
|
|
2004-10-15 08:38:36 +04:00
|
|
|
cd = config_cfdriver_lookup(cf->cf_name);
|
2002-10-09 06:59:55 +04:00
|
|
|
if (cd == NULL)
|
|
|
|
return (NULL);
|
|
|
|
|
2004-10-15 08:38:36 +04:00
|
|
|
ca = config_cfattach_lookup_cd(cd, cf->cf_atname);
|
2002-10-09 06:59:55 +04:00
|
|
|
if (ca == NULL)
|
|
|
|
return (NULL);
|
|
|
|
|
|
|
|
if (ca->ca_devsize < sizeof(struct device))
|
|
|
|
panic("config_attach_pseudo");
|
|
|
|
|
2004-10-15 08:38:36 +04:00
|
|
|
/*
|
|
|
|
* We just ignore cf_fstate, instead doing everything with
|
|
|
|
* cf_unit.
|
|
|
|
*
|
|
|
|
* XXX Should we change this and use FSTATE_NOTFOUND and
|
|
|
|
* XXX FSTATE_STAR?
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (cf->cf_unit == DVUNIT_ANY) {
|
2002-10-09 06:59:55 +04:00
|
|
|
for (myunit = 0; myunit < cd->cd_ndevs; myunit++)
|
|
|
|
if (cd->cd_devs[myunit] == NULL)
|
|
|
|
break;
|
|
|
|
/*
|
|
|
|
* myunit is now the unit of the first NULL device pointer.
|
|
|
|
*/
|
|
|
|
} else {
|
2004-10-15 08:38:36 +04:00
|
|
|
myunit = cf->cf_unit;
|
2002-10-09 06:59:55 +04:00
|
|
|
if (myunit < cd->cd_ndevs && cd->cd_devs[myunit] != NULL)
|
|
|
|
return (NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* compute length of name and decimal expansion of unit number */
|
|
|
|
lname = strlen(cd->cd_name);
|
|
|
|
xunit = number(&num[sizeof(num)], myunit);
|
|
|
|
lunit = &num[sizeof(num)] - xunit;
|
|
|
|
if (lname + lunit > sizeof(dev->dv_xname))
|
|
|
|
panic("config_attach_pseudo: device name too long");
|
|
|
|
|
|
|
|
/* get memory for all device vars */
|
|
|
|
dev = (struct device *)malloc(ca->ca_devsize, M_DEVBUF,
|
|
|
|
cold ? M_NOWAIT : M_WAITOK);
|
|
|
|
if (!dev)
|
|
|
|
panic("config_attach_pseudo: memory allocation for device "
|
|
|
|
"softc failed");
|
|
|
|
memset(dev, 0, ca->ca_devsize);
|
|
|
|
TAILQ_INSERT_TAIL(&alldevs, dev, dv_list); /* link up */
|
|
|
|
dev->dv_class = cd->cd_class;
|
2004-10-15 08:38:36 +04:00
|
|
|
dev->dv_cfdata = cf;
|
2002-10-09 06:59:55 +04:00
|
|
|
dev->dv_cfdriver = cd;
|
|
|
|
dev->dv_cfattach = ca;
|
|
|
|
dev->dv_unit = myunit;
|
|
|
|
memcpy(dev->dv_xname, cd->cd_name, lname);
|
|
|
|
memcpy(dev->dv_xname + lname, xunit, lunit);
|
|
|
|
dev->dv_parent = ROOT;
|
|
|
|
dev->dv_flags = DVF_ACTIVE; /* always initially active */
|
|
|
|
|
|
|
|
/* put this device in the devices array */
|
|
|
|
config_makeroom(dev->dv_unit, cd);
|
|
|
|
if (cd->cd_devs[dev->dv_unit])
|
|
|
|
panic("config_attach_pseudo: duplicate %s", dev->dv_xname);
|
|
|
|
cd->cd_devs[dev->dv_unit] = dev;
|
|
|
|
|
|
|
|
#if 0 /* XXXJRT not yet */
|
|
|
|
#ifdef __HAVE_DEVICE_REGISTER
|
|
|
|
device_register(dev, NULL); /* like a root node */
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
(*ca->ca_attach)(ROOT, dev, NULL);
|
|
|
|
config_process_deferred(&deferred_config_queue, dev);
|
|
|
|
return (dev);
|
|
|
|
}
|
|
|
|
|
1998-11-17 11:38:07 +03:00
|
|
|
/*
|
|
|
|
* Detach a device. Optionally forced (e.g. because of hardware
|
|
|
|
* removal) and quiet. Returns zero if successful, non-zero
|
|
|
|
* (an error code) otherwise.
|
|
|
|
*
|
|
|
|
* Note that this code wants to be run from a process context, so
|
|
|
|
* that the detach can sleep to allow processes which have a device
|
|
|
|
* open to run and unwind their stacks.
|
|
|
|
*/
|
|
|
|
int
|
2000-06-02 05:31:52 +04:00
|
|
|
config_detach(struct device *dev, int flags)
|
1998-11-17 11:38:07 +03:00
|
|
|
{
|
2002-09-24 03:16:06 +04:00
|
|
|
struct cftable *ct;
|
1998-11-17 11:38:07 +03:00
|
|
|
struct cfdata *cf;
|
2002-09-28 00:41:46 +04:00
|
|
|
const struct cfattach *ca;
|
1998-11-17 11:38:07 +03:00
|
|
|
struct cfdriver *cd;
|
|
|
|
#ifdef DIAGNOSTIC
|
|
|
|
struct device *d;
|
|
|
|
#endif
|
1998-11-18 21:40:54 +03:00
|
|
|
int rv = 0, i;
|
1998-11-17 11:38:07 +03:00
|
|
|
|
|
|
|
#ifdef DIAGNOSTIC
|
2002-10-09 06:59:55 +04:00
|
|
|
if (dev->dv_cfdata != NULL &&
|
|
|
|
dev->dv_cfdata->cf_fstate != FSTATE_FOUND &&
|
|
|
|
dev->dv_cfdata->cf_fstate != FSTATE_STAR)
|
1998-11-17 11:38:07 +03:00
|
|
|
panic("config_detach: bad device fstate");
|
|
|
|
#endif
|
2002-10-09 06:59:55 +04:00
|
|
|
cd = dev->dv_cfdriver;
|
2002-09-27 06:24:06 +04:00
|
|
|
KASSERT(cd != NULL);
|
2002-10-04 05:50:53 +04:00
|
|
|
|
2002-10-09 06:59:55 +04:00
|
|
|
ca = dev->dv_cfattach;
|
2002-10-04 05:50:53 +04:00
|
|
|
KASSERT(ca != NULL);
|
1998-11-17 11:38:07 +03:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Ensure the device is deactivated. If the device doesn't
|
|
|
|
* have an activation entry point, we allow DVF_ACTIVE to
|
|
|
|
* remain set. Otherwise, if DVF_ACTIVE is still set, the
|
|
|
|
* device is busy, and the detach fails.
|
|
|
|
*/
|
1998-11-18 21:40:08 +03:00
|
|
|
if (ca->ca_activate != NULL)
|
|
|
|
rv = config_deactivate(dev);
|
1998-11-17 11:38:07 +03:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Try to detach the device. If that's not possible, then
|
|
|
|
* we either panic() (for the forced but failed case), or
|
|
|
|
* return an error.
|
|
|
|
*/
|
|
|
|
if (rv == 0) {
|
|
|
|
if (ca->ca_detach != NULL)
|
|
|
|
rv = (*ca->ca_detach)(dev, flags);
|
|
|
|
else
|
|
|
|
rv = EOPNOTSUPP;
|
|
|
|
}
|
|
|
|
if (rv != 0) {
|
|
|
|
if ((flags & DETACH_FORCE) == 0)
|
|
|
|
return (rv);
|
|
|
|
else
|
|
|
|
panic("config_detach: forced detach of %s failed (%d)",
|
|
|
|
dev->dv_xname, rv);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The device has now been successfully detached.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef DIAGNOSTIC
|
|
|
|
/*
|
|
|
|
* Sanity: If you're successfully detached, you should have no
|
|
|
|
* children. (Note that because children must be attached
|
|
|
|
* after parents, we only need to search the latter part of
|
|
|
|
* the list.)
|
|
|
|
*/
|
|
|
|
for (d = TAILQ_NEXT(dev, dv_list); d != NULL;
|
2000-01-25 16:23:26 +03:00
|
|
|
d = TAILQ_NEXT(d, dv_list)) {
|
|
|
|
if (d->dv_parent == dev) {
|
|
|
|
printf("config_detach: detached device %s"
|
|
|
|
" has children %s\n", dev->dv_xname, d->dv_xname);
|
|
|
|
panic("config_detach");
|
|
|
|
}
|
1998-11-17 11:38:07 +03:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2004-08-18 02:13:18 +04:00
|
|
|
/* notify the parent that the child is gone */
|
|
|
|
if (dev->dv_parent) {
|
|
|
|
struct device *p = dev->dv_parent;
|
|
|
|
if (p->dv_cfattach->ca_childdetached)
|
|
|
|
(*p->dv_cfattach->ca_childdetached)(p, dev);
|
|
|
|
}
|
|
|
|
|
1998-11-17 11:38:07 +03:00
|
|
|
/*
|
|
|
|
* Mark cfdata to show that the unit can be reused, if possible.
|
|
|
|
*/
|
2002-09-24 03:16:06 +04:00
|
|
|
TAILQ_FOREACH(ct, &allcftables, ct_list) {
|
2002-09-27 06:24:06 +04:00
|
|
|
for (cf = ct->ct_cfdata; cf->cf_name; cf++) {
|
|
|
|
if (STREQ(cf->cf_name, cd->cd_name)) {
|
2002-09-24 03:16:06 +04:00
|
|
|
if (cf->cf_fstate == FSTATE_FOUND &&
|
|
|
|
cf->cf_unit == dev->dv_unit)
|
|
|
|
cf->cf_fstate = FSTATE_NOTFOUND;
|
2000-01-18 10:45:04 +03:00
|
|
|
#ifdef __BROKEN_CONFIG_UNIT_USAGE
|
2002-09-26 08:07:35 +04:00
|
|
|
/*
|
|
|
|
* Note that we can only re-use a starred
|
|
|
|
* unit number if the unit being detached
|
|
|
|
* had the last assigned unit number.
|
|
|
|
*/
|
2002-09-24 03:16:06 +04:00
|
|
|
if (cf->cf_fstate == FSTATE_STAR &&
|
|
|
|
cf->cf_unit == dev->dv_unit + 1)
|
|
|
|
cf->cf_unit--;
|
2000-01-18 10:45:04 +03:00
|
|
|
#endif /* __BROKEN_CONFIG_UNIT_USAGE */
|
2002-09-24 03:16:06 +04:00
|
|
|
}
|
1998-11-17 11:38:07 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Unlink from device list.
|
|
|
|
*/
|
|
|
|
TAILQ_REMOVE(&alldevs, dev, dv_list);
|
|
|
|
|
|
|
|
/*
|
2002-10-09 06:59:55 +04:00
|
|
|
* Remove from cfdriver's array, tell the world (unless it was
|
|
|
|
* a pseudo-device), and free softc.
|
1998-11-17 11:38:07 +03:00
|
|
|
*/
|
|
|
|
cd->cd_devs[dev->dv_unit] = NULL;
|
2002-10-09 06:59:55 +04:00
|
|
|
if (dev->dv_cfdata != NULL && (flags & DETACH_QUIET) == 0)
|
2003-01-01 02:59:11 +03:00
|
|
|
aprint_normal("%s detached\n", dev->dv_xname);
|
2004-08-18 02:13:18 +04:00
|
|
|
if (dev->dv_locators)
|
|
|
|
free(dev->dv_locators, M_DEVBUF);
|
1998-11-17 11:38:07 +03:00
|
|
|
free(dev, M_DEVBUF);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the device now has no units in use, deallocate its softc array.
|
|
|
|
*/
|
|
|
|
for (i = 0; i < cd->cd_ndevs; i++)
|
|
|
|
if (cd->cd_devs[i] != NULL)
|
|
|
|
break;
|
|
|
|
if (i == cd->cd_ndevs) { /* nothing found; deallocate */
|
|
|
|
free(cd->cd_devs, M_DEVBUF);
|
|
|
|
cd->cd_devs = NULL;
|
|
|
|
cd->cd_ndevs = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return success.
|
|
|
|
*/
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2000-06-02 05:31:52 +04:00
|
|
|
config_activate(struct device *dev)
|
1998-11-17 11:38:07 +03:00
|
|
|
{
|
2002-10-04 05:50:53 +04:00
|
|
|
const struct cfattach *ca = dev->dv_cfattach;
|
1998-11-18 21:38:07 +03:00
|
|
|
int rv = 0, oflags = dev->dv_flags;
|
1998-11-17 11:38:07 +03:00
|
|
|
|
|
|
|
if (ca->ca_activate == NULL)
|
|
|
|
return (EOPNOTSUPP);
|
|
|
|
|
|
|
|
if ((dev->dv_flags & DVF_ACTIVE) == 0) {
|
|
|
|
dev->dv_flags |= DVF_ACTIVE;
|
|
|
|
rv = (*ca->ca_activate)(dev, DVACT_ACTIVATE);
|
1998-11-18 21:38:07 +03:00
|
|
|
if (rv)
|
|
|
|
dev->dv_flags = oflags;
|
1998-11-17 11:38:07 +03:00
|
|
|
}
|
|
|
|
return (rv);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2000-06-02 05:31:52 +04:00
|
|
|
config_deactivate(struct device *dev)
|
1998-11-17 11:38:07 +03:00
|
|
|
{
|
2002-10-04 05:50:53 +04:00
|
|
|
const struct cfattach *ca = dev->dv_cfattach;
|
1998-11-18 21:38:07 +03:00
|
|
|
int rv = 0, oflags = dev->dv_flags;
|
1998-11-17 11:38:07 +03:00
|
|
|
|
|
|
|
if (ca->ca_activate == NULL)
|
|
|
|
return (EOPNOTSUPP);
|
|
|
|
|
|
|
|
if (dev->dv_flags & DVF_ACTIVE) {
|
|
|
|
dev->dv_flags &= ~DVF_ACTIVE;
|
|
|
|
rv = (*ca->ca_activate)(dev, DVACT_DEACTIVATE);
|
1998-11-18 21:38:07 +03:00
|
|
|
if (rv)
|
|
|
|
dev->dv_flags = oflags;
|
1998-11-17 11:38:07 +03:00
|
|
|
}
|
|
|
|
return (rv);
|
|
|
|
}
|
|
|
|
|
1998-06-09 22:46:12 +04:00
|
|
|
/*
|
|
|
|
* Defer the configuration of the specified device until all
|
|
|
|
* of its parent's devices have been attached.
|
|
|
|
*/
|
|
|
|
void
|
2000-06-02 05:31:52 +04:00
|
|
|
config_defer(struct device *dev, void (*func)(struct device *))
|
1998-06-09 22:46:12 +04:00
|
|
|
{
|
|
|
|
struct deferred_config *dc;
|
|
|
|
|
|
|
|
if (dev->dv_parent == NULL)
|
|
|
|
panic("config_defer: can't defer config of a root device");
|
|
|
|
|
|
|
|
#ifdef DIAGNOSTIC
|
|
|
|
for (dc = TAILQ_FIRST(&deferred_config_queue); dc != NULL;
|
|
|
|
dc = TAILQ_NEXT(dc, dc_queue)) {
|
|
|
|
if (dc->dc_dev == dev)
|
|
|
|
panic("config_defer: deferred twice");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
1999-09-18 00:11:56 +04:00
|
|
|
dc = malloc(sizeof(*dc), M_DEVBUF, cold ? M_NOWAIT : M_WAITOK);
|
|
|
|
if (dc == NULL)
|
|
|
|
panic("config_defer: unable to allocate callback");
|
1998-06-09 22:46:12 +04:00
|
|
|
|
|
|
|
dc->dc_dev = dev;
|
|
|
|
dc->dc_func = func;
|
|
|
|
TAILQ_INSERT_TAIL(&deferred_config_queue, dc, dc_queue);
|
2000-01-24 21:03:19 +03:00
|
|
|
config_pending_incr();
|
1998-06-09 22:46:12 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
1999-09-15 23:37:08 +04:00
|
|
|
* Defer some autoconfiguration for a device until after interrupts
|
|
|
|
* are enabled.
|
|
|
|
*/
|
|
|
|
void
|
2000-06-02 05:31:52 +04:00
|
|
|
config_interrupts(struct device *dev, void (*func)(struct device *))
|
1999-09-15 23:37:08 +04:00
|
|
|
{
|
|
|
|
struct deferred_config *dc;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If interrupts are enabled, callback now.
|
|
|
|
*/
|
1999-09-18 00:11:56 +04:00
|
|
|
if (cold == 0) {
|
1999-09-15 23:37:08 +04:00
|
|
|
(*func)(dev);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DIAGNOSTIC
|
|
|
|
for (dc = TAILQ_FIRST(&interrupt_config_queue); dc != NULL;
|
|
|
|
dc = TAILQ_NEXT(dc, dc_queue)) {
|
|
|
|
if (dc->dc_dev == dev)
|
|
|
|
panic("config_interrupts: deferred twice");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
1999-09-18 00:11:56 +04:00
|
|
|
dc = malloc(sizeof(*dc), M_DEVBUF, cold ? M_NOWAIT : M_WAITOK);
|
|
|
|
if (dc == NULL)
|
|
|
|
panic("config_interrupts: unable to allocate callback");
|
1999-09-15 23:37:08 +04:00
|
|
|
|
|
|
|
dc->dc_dev = dev;
|
|
|
|
dc->dc_func = func;
|
|
|
|
TAILQ_INSERT_TAIL(&interrupt_config_queue, dc, dc_queue);
|
2000-01-24 21:03:19 +03:00
|
|
|
config_pending_incr();
|
1999-09-15 23:37:08 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Process a deferred configuration queue.
|
1998-06-09 22:46:12 +04:00
|
|
|
*/
|
|
|
|
static void
|
2000-06-02 05:31:52 +04:00
|
|
|
config_process_deferred(struct deferred_config_head *queue,
|
|
|
|
struct device *parent)
|
1998-06-09 22:46:12 +04:00
|
|
|
{
|
|
|
|
struct deferred_config *dc, *ndc;
|
|
|
|
|
1999-09-15 23:37:08 +04:00
|
|
|
for (dc = TAILQ_FIRST(queue); dc != NULL; dc = ndc) {
|
1998-06-09 22:46:12 +04:00
|
|
|
ndc = TAILQ_NEXT(dc, dc_queue);
|
1999-09-15 23:37:08 +04:00
|
|
|
if (parent == NULL || dc->dc_dev->dv_parent == parent) {
|
|
|
|
TAILQ_REMOVE(queue, dc, dc_queue);
|
1998-06-09 22:46:12 +04:00
|
|
|
(*dc->dc_func)(dc->dc_dev);
|
|
|
|
free(dc, M_DEVBUF);
|
2000-01-24 21:03:19 +03:00
|
|
|
config_pending_decr();
|
1998-06-09 22:46:12 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2000-01-24 21:03:19 +03:00
|
|
|
/*
|
|
|
|
* Manipulate the config_pending semaphore.
|
|
|
|
*/
|
|
|
|
void
|
2000-06-02 05:31:52 +04:00
|
|
|
config_pending_incr(void)
|
2000-01-24 21:03:19 +03:00
|
|
|
{
|
|
|
|
|
|
|
|
config_pending++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2000-06-02 05:31:52 +04:00
|
|
|
config_pending_decr(void)
|
2000-01-24 21:03:19 +03:00
|
|
|
{
|
|
|
|
|
|
|
|
#ifdef DIAGNOSTIC
|
|
|
|
if (config_pending == 0)
|
|
|
|
panic("config_pending_decr: config_pending == 0");
|
|
|
|
#endif
|
|
|
|
config_pending--;
|
|
|
|
if (config_pending == 0)
|
|
|
|
wakeup((void *)&config_pending);
|
|
|
|
}
|
|
|
|
|
2002-10-01 22:11:57 +04:00
|
|
|
/*
|
|
|
|
* Register a "finalization" routine. Finalization routines are
|
|
|
|
* called iteratively once all real devices have been found during
|
|
|
|
* autoconfiguration, for as long as any one finalizer has done
|
|
|
|
* any work.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
config_finalize_register(struct device *dev, int (*fn)(struct device *))
|
|
|
|
{
|
|
|
|
struct finalize_hook *f;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If finalization has already been done, invoke the
|
|
|
|
* callback function now.
|
|
|
|
*/
|
|
|
|
if (config_finalize_done) {
|
|
|
|
while ((*fn)(dev) != 0)
|
|
|
|
/* loop */ ;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Ensure this isn't already on the list. */
|
|
|
|
TAILQ_FOREACH(f, &config_finalize_list, f_list) {
|
|
|
|
if (f->f_func == fn && f->f_dev == dev)
|
|
|
|
return (EEXIST);
|
|
|
|
}
|
|
|
|
|
|
|
|
f = malloc(sizeof(*f), M_TEMP, M_WAITOK);
|
|
|
|
f->f_func = fn;
|
|
|
|
f->f_dev = dev;
|
|
|
|
TAILQ_INSERT_TAIL(&config_finalize_list, f, f_list);
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
config_finalize(void)
|
|
|
|
{
|
|
|
|
struct finalize_hook *f;
|
|
|
|
int rv;
|
|
|
|
|
|
|
|
/* Run the hooks until none of them does any work. */
|
|
|
|
do {
|
|
|
|
rv = 0;
|
|
|
|
TAILQ_FOREACH(f, &config_finalize_list, f_list)
|
|
|
|
rv |= (*f->f_func)(f->f_dev);
|
|
|
|
} while (rv != 0);
|
|
|
|
|
|
|
|
config_finalize_done = 1;
|
|
|
|
|
|
|
|
/* Now free all the hooks. */
|
|
|
|
while ((f = TAILQ_FIRST(&config_finalize_list)) != NULL) {
|
|
|
|
TAILQ_REMOVE(&config_finalize_list, f, f_list);
|
|
|
|
free(f, M_TEMP);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|