xref: /freebsd/sys/fs/devfs/devfs_devs.c (revision 5e080af41fe6cd907de651434ce53de0c5e04114)
1d167cf6fSWarner Losh /*-
29d960907SPoul-Henning Kamp  * Copyright (c) 2000,2004
33f54a085SPoul-Henning Kamp  *	Poul-Henning Kamp.  All rights reserved.
43f54a085SPoul-Henning Kamp  *
53f54a085SPoul-Henning Kamp  * Redistribution and use in source and binary forms, with or without
63f54a085SPoul-Henning Kamp  * modification, are permitted provided that the following conditions
73f54a085SPoul-Henning Kamp  * are met:
83f54a085SPoul-Henning Kamp  * 1. Redistributions of source code must retain the above copyright
93f54a085SPoul-Henning Kamp  *    notice, this list of conditions and the following disclaimer.
103f54a085SPoul-Henning Kamp  * 2. Neither the name of the University nor the names of its contributors
113f54a085SPoul-Henning Kamp  *    may be used to endorse or promote products derived from this software
123f54a085SPoul-Henning Kamp  *    without specific prior written permission.
133f54a085SPoul-Henning Kamp  *
143f54a085SPoul-Henning Kamp  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
153f54a085SPoul-Henning Kamp  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
163f54a085SPoul-Henning Kamp  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
173f54a085SPoul-Henning Kamp  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
183f54a085SPoul-Henning Kamp  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
193f54a085SPoul-Henning Kamp  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
203f54a085SPoul-Henning Kamp  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
213f54a085SPoul-Henning Kamp  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
223f54a085SPoul-Henning Kamp  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
233f54a085SPoul-Henning Kamp  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
243f54a085SPoul-Henning Kamp  * SUCH DAMAGE.
253f54a085SPoul-Henning Kamp  *
263f54a085SPoul-Henning Kamp  * From: FreeBSD: src/sys/miscfs/kernfs/kernfs_vfsops.c 1.36
273f54a085SPoul-Henning Kamp  *
283f54a085SPoul-Henning Kamp  * $FreeBSD$
293f54a085SPoul-Henning Kamp  */
303f54a085SPoul-Henning Kamp 
3193bcdfe2SPoul-Henning Kamp #include "opt_devfs.h"
326742f328SRobert Watson #include "opt_mac.h"
3393bcdfe2SPoul-Henning Kamp 
343f54a085SPoul-Henning Kamp #include <sys/param.h>
353f54a085SPoul-Henning Kamp #include <sys/systm.h>
363f54a085SPoul-Henning Kamp #include <sys/conf.h>
37fb919e4dSMark Murray #include <sys/dirent.h>
3893bcdfe2SPoul-Henning Kamp #include <sys/kernel.h>
3993bcdfe2SPoul-Henning Kamp #include <sys/lock.h>
406742f328SRobert Watson #include <sys/mac.h>
41fb919e4dSMark Murray #include <sys/malloc.h>
42fb919e4dSMark Murray #include <sys/proc.h>
43fb919e4dSMark Murray #include <sys/sysctl.h>
44fb919e4dSMark Murray #include <sys/vnode.h>
453f54a085SPoul-Henning Kamp 
4693bcdfe2SPoul-Henning Kamp #include <machine/atomic.h>
4793bcdfe2SPoul-Henning Kamp 
483f54a085SPoul-Henning Kamp #include <fs/devfs/devfs.h>
499c0af131SPoul-Henning Kamp #include <fs/devfs/devfs_int.h>
503f54a085SPoul-Henning Kamp 
5189c9c53dSPoul-Henning Kamp static struct cdev *devfs_inot[NDEVFSINO];
5289c9c53dSPoul-Henning Kamp static struct cdev **devfs_overflow;
5393bcdfe2SPoul-Henning Kamp static int devfs_ref[NDEVFSINO];
5493bcdfe2SPoul-Henning Kamp static int *devfs_refoverflow;
5593bcdfe2SPoul-Henning Kamp static int devfs_nextino = 3;
5693bcdfe2SPoul-Henning Kamp static int devfs_numino;
5793bcdfe2SPoul-Henning Kamp static int devfs_topino;
5893bcdfe2SPoul-Henning Kamp static int devfs_noverflowwant = NDEVFSOVERFLOW;
5993bcdfe2SPoul-Henning Kamp static int devfs_noverflow;
6093bcdfe2SPoul-Henning Kamp static unsigned devfs_generation;
6193bcdfe2SPoul-Henning Kamp 
622a043678SPoul-Henning Kamp static struct devfs_dirent *devfs_find (struct devfs_dirent *dd, const char *name, int namelen);
632a043678SPoul-Henning Kamp 
645ece08f5SPoul-Henning Kamp static SYSCTL_NODE(_vfs, OID_AUTO, devfs, CTLFLAG_RW, 0, "DEVFS filesystem");
6593bcdfe2SPoul-Henning Kamp SYSCTL_UINT(_vfs_devfs, OID_AUTO, noverflow, CTLFLAG_RW,
6693bcdfe2SPoul-Henning Kamp 	&devfs_noverflowwant, 0, "Size of DEVFS overflow table");
6793bcdfe2SPoul-Henning Kamp SYSCTL_UINT(_vfs_devfs, OID_AUTO, generation, CTLFLAG_RD,
6893bcdfe2SPoul-Henning Kamp 	&devfs_generation, 0, "DEVFS generation number");
6993bcdfe2SPoul-Henning Kamp SYSCTL_UINT(_vfs_devfs, OID_AUTO, inodes, CTLFLAG_RD,
7093bcdfe2SPoul-Henning Kamp 	&devfs_numino, 0, "DEVFS inodes");
7193bcdfe2SPoul-Henning Kamp SYSCTL_UINT(_vfs_devfs, OID_AUTO, topinode, CTLFLAG_RD,
7293bcdfe2SPoul-Henning Kamp 	&devfs_topino, 0, "DEVFS highest inode#");
7393bcdfe2SPoul-Henning Kamp 
745e080af4SPoul-Henning Kamp unsigned devfs_rule_depth = 1;
755e080af4SPoul-Henning Kamp SYSCTL_UINT(_vfs_devfs, OID_AUTO, rule_depth, CTLFLAG_RW,
765e080af4SPoul-Henning Kamp     &devfs_rule_depth, 0, "Max depth of ruleset include");
775e080af4SPoul-Henning Kamp 
7831cc57cdSPoul-Henning Kamp /*
7931cc57cdSPoul-Henning Kamp  * Helper sysctl for devname(3).  We're given a struct cdev * and return
8031cc57cdSPoul-Henning Kamp  * the name, if any, registered by the device driver.
8131cc57cdSPoul-Henning Kamp  */
8231cc57cdSPoul-Henning Kamp static int
8331cc57cdSPoul-Henning Kamp sysctl_devname(SYSCTL_HANDLER_ARGS)
8431cc57cdSPoul-Henning Kamp {
8531cc57cdSPoul-Henning Kamp 	int error;
8631cc57cdSPoul-Henning Kamp 	dev_t ud;
8731cc57cdSPoul-Henning Kamp 	struct cdev *dev, **dp;
8831cc57cdSPoul-Henning Kamp 
8931cc57cdSPoul-Henning Kamp 	error = SYSCTL_IN(req, &ud, sizeof (ud));
9031cc57cdSPoul-Henning Kamp 	if (error)
9131cc57cdSPoul-Henning Kamp 		return (error);
9231cc57cdSPoul-Henning Kamp 	if (ud == NODEV)
9331cc57cdSPoul-Henning Kamp 		return(EINVAL);
9431cc57cdSPoul-Henning Kamp 	dp = devfs_itod(ud);
9531cc57cdSPoul-Henning Kamp 	if (dp == NULL)
9631cc57cdSPoul-Henning Kamp 		return(ENOENT);
9731cc57cdSPoul-Henning Kamp 	dev = *dp;
9831cc57cdSPoul-Henning Kamp 	if (dev == NULL)
9931cc57cdSPoul-Henning Kamp 		return(ENOENT);
10031cc57cdSPoul-Henning Kamp 	return(SYSCTL_OUT(req, dev->si_name, strlen(dev->si_name) + 1));
10131cc57cdSPoul-Henning Kamp 	return (error);
10231cc57cdSPoul-Henning Kamp }
10331cc57cdSPoul-Henning Kamp 
10431cc57cdSPoul-Henning Kamp SYSCTL_PROC(_kern, OID_AUTO, devname, CTLTYPE_OPAQUE|CTLFLAG_RW|CTLFLAG_ANYBODY,
10531cc57cdSPoul-Henning Kamp 	NULL, 0, sysctl_devname, "", "devname(3) handler");
10631cc57cdSPoul-Henning Kamp 
10731cc57cdSPoul-Henning Kamp SYSCTL_INT(_debug_sizeof, OID_AUTO, cdev, CTLFLAG_RD,
10831cc57cdSPoul-Henning Kamp     0, sizeof(struct cdev), "sizeof(struct cdev)");
10931cc57cdSPoul-Henning Kamp 
11093bcdfe2SPoul-Henning Kamp static int *
11193bcdfe2SPoul-Henning Kamp devfs_itor(int inode)
11293bcdfe2SPoul-Henning Kamp {
11393bcdfe2SPoul-Henning Kamp 	if (inode < NDEVFSINO)
11493bcdfe2SPoul-Henning Kamp 		return (&devfs_ref[inode]);
11593bcdfe2SPoul-Henning Kamp 	else if (inode < NDEVFSINO + devfs_noverflow)
11693bcdfe2SPoul-Henning Kamp 		return (&devfs_refoverflow[inode - NDEVFSINO]);
11793bcdfe2SPoul-Henning Kamp 	else
11893bcdfe2SPoul-Henning Kamp 		panic ("YRK!");
11993bcdfe2SPoul-Henning Kamp }
12093bcdfe2SPoul-Henning Kamp 
12193bcdfe2SPoul-Henning Kamp static void
12293bcdfe2SPoul-Henning Kamp devfs_dropref(int inode)
12393bcdfe2SPoul-Henning Kamp {
12493bcdfe2SPoul-Henning Kamp 	int *ip;
12593bcdfe2SPoul-Henning Kamp 
12693bcdfe2SPoul-Henning Kamp 	ip = devfs_itor(inode);
12793bcdfe2SPoul-Henning Kamp 	atomic_add_int(ip, -1);
12893bcdfe2SPoul-Henning Kamp }
12993bcdfe2SPoul-Henning Kamp 
13093bcdfe2SPoul-Henning Kamp static int
13193bcdfe2SPoul-Henning Kamp devfs_getref(int inode)
13293bcdfe2SPoul-Henning Kamp {
13393bcdfe2SPoul-Henning Kamp 	int *ip, i, j;
13489c9c53dSPoul-Henning Kamp 	struct cdev **dp;
13593bcdfe2SPoul-Henning Kamp 
13693bcdfe2SPoul-Henning Kamp 	ip = devfs_itor(inode);
13793bcdfe2SPoul-Henning Kamp 	dp = devfs_itod(inode);
13893bcdfe2SPoul-Henning Kamp 	for (;;) {
13993bcdfe2SPoul-Henning Kamp 		i = *ip;
14093bcdfe2SPoul-Henning Kamp 		j = i + 1;
14193bcdfe2SPoul-Henning Kamp 		if (!atomic_cmpset_int(ip, i, j))
14293bcdfe2SPoul-Henning Kamp 			continue;
14393bcdfe2SPoul-Henning Kamp 		if (*dp != NULL)
14493bcdfe2SPoul-Henning Kamp 			return (1);
14593bcdfe2SPoul-Henning Kamp 		atomic_add_int(ip, -1);
14693bcdfe2SPoul-Henning Kamp 		return(0);
14793bcdfe2SPoul-Henning Kamp 	}
14893bcdfe2SPoul-Henning Kamp }
14993bcdfe2SPoul-Henning Kamp 
15093bcdfe2SPoul-Henning Kamp struct devfs_dirent **
15193bcdfe2SPoul-Henning Kamp devfs_itode (struct devfs_mount *dm, int inode)
15293bcdfe2SPoul-Henning Kamp {
15393bcdfe2SPoul-Henning Kamp 
154b43ab0e3SPoul-Henning Kamp 	if (inode < 0)
155b43ab0e3SPoul-Henning Kamp 		return (NULL);
15693bcdfe2SPoul-Henning Kamp 	if (inode < NDEVFSINO)
15793bcdfe2SPoul-Henning Kamp 		return (&dm->dm_dirent[inode]);
15893bcdfe2SPoul-Henning Kamp 	if (devfs_overflow == NULL)
15993bcdfe2SPoul-Henning Kamp 		return (NULL);
16093bcdfe2SPoul-Henning Kamp 	if (inode < NDEVFSINO + devfs_noverflow)
16193bcdfe2SPoul-Henning Kamp 		return (&dm->dm_overflow[inode - NDEVFSINO]);
16293bcdfe2SPoul-Henning Kamp 	return (NULL);
16393bcdfe2SPoul-Henning Kamp }
16493bcdfe2SPoul-Henning Kamp 
16589c9c53dSPoul-Henning Kamp struct cdev **
16693bcdfe2SPoul-Henning Kamp devfs_itod (int inode)
16793bcdfe2SPoul-Henning Kamp {
16893bcdfe2SPoul-Henning Kamp 
169b43ab0e3SPoul-Henning Kamp 	if (inode < 0)
170b43ab0e3SPoul-Henning Kamp 		return (NULL);
17193bcdfe2SPoul-Henning Kamp 	if (inode < NDEVFSINO)
17293bcdfe2SPoul-Henning Kamp 		return (&devfs_inot[inode]);
17393bcdfe2SPoul-Henning Kamp 	if (devfs_overflow == NULL)
17493bcdfe2SPoul-Henning Kamp 		return (NULL);
17593bcdfe2SPoul-Henning Kamp 	if (inode < NDEVFSINO + devfs_noverflow)
17693bcdfe2SPoul-Henning Kamp 		return (&devfs_overflow[inode - NDEVFSINO]);
17793bcdfe2SPoul-Henning Kamp 	return (NULL);
17893bcdfe2SPoul-Henning Kamp }
17993bcdfe2SPoul-Henning Kamp 
1802a043678SPoul-Henning Kamp static struct devfs_dirent *
181c32d0a1dSPoul-Henning Kamp devfs_find(struct devfs_dirent *dd, const char *name, int namelen)
182c32d0a1dSPoul-Henning Kamp {
183c32d0a1dSPoul-Henning Kamp 	struct devfs_dirent *de;
184c32d0a1dSPoul-Henning Kamp 
185c32d0a1dSPoul-Henning Kamp 	TAILQ_FOREACH(de, &dd->de_dlist, de_list) {
186c32d0a1dSPoul-Henning Kamp 		if (namelen != de->de_dirent->d_namlen)
187c32d0a1dSPoul-Henning Kamp 			continue;
188c32d0a1dSPoul-Henning Kamp 		if (bcmp(name, de->de_dirent->d_name, namelen) != 0)
189c32d0a1dSPoul-Henning Kamp 			continue;
190c32d0a1dSPoul-Henning Kamp 		break;
191c32d0a1dSPoul-Henning Kamp 	}
192c32d0a1dSPoul-Henning Kamp 	return (de);
193c32d0a1dSPoul-Henning Kamp }
194c32d0a1dSPoul-Henning Kamp 
195c32d0a1dSPoul-Henning Kamp struct devfs_dirent *
1963f54a085SPoul-Henning Kamp devfs_newdirent(char *name, int namelen)
1973f54a085SPoul-Henning Kamp {
1983f54a085SPoul-Henning Kamp 	int i;
1993f54a085SPoul-Henning Kamp 	struct devfs_dirent *de;
2003f54a085SPoul-Henning Kamp 	struct dirent d;
2013f54a085SPoul-Henning Kamp 
2023f54a085SPoul-Henning Kamp 	d.d_namlen = namelen;
2033f54a085SPoul-Henning Kamp 	i = sizeof (*de) + GENERIC_DIRSIZ(&d);
204a163d034SWarner Losh 	MALLOC(de, struct devfs_dirent *, i, M_DEVFS, M_WAITOK | M_ZERO);
2053f54a085SPoul-Henning Kamp 	de->de_dirent = (struct dirent *)(de + 1);
2063f54a085SPoul-Henning Kamp 	de->de_dirent->d_namlen = namelen;
2073f54a085SPoul-Henning Kamp 	de->de_dirent->d_reclen = GENERIC_DIRSIZ(&d);
208aadf2655SPoul-Henning Kamp 	bcopy(name, de->de_dirent->d_name, namelen);
209aadf2655SPoul-Henning Kamp 	de->de_dirent->d_name[namelen] = '\0';
21093432a92SPoul-Henning Kamp 	vfs_timestamp(&de->de_ctime);
211a481b90bSPoul-Henning Kamp 	de->de_mtime = de->de_atime = de->de_ctime;
212a481b90bSPoul-Henning Kamp 	de->de_links = 1;
2136742f328SRobert Watson #ifdef MAC
2146742f328SRobert Watson 	mac_init_devfsdirent(de);
2156742f328SRobert Watson #endif
2163f54a085SPoul-Henning Kamp 	return (de);
2173f54a085SPoul-Henning Kamp }
2183f54a085SPoul-Henning Kamp 
219a481b90bSPoul-Henning Kamp struct devfs_dirent *
220a481b90bSPoul-Henning Kamp devfs_vmkdir(char *name, int namelen, struct devfs_dirent *dotdot)
2213f54a085SPoul-Henning Kamp {
222a481b90bSPoul-Henning Kamp 	struct devfs_dirent *dd;
2233f54a085SPoul-Henning Kamp 	struct devfs_dirent *de;
2243f54a085SPoul-Henning Kamp 
225a481b90bSPoul-Henning Kamp 	dd = devfs_newdirent(name, namelen);
226a481b90bSPoul-Henning Kamp 
227a481b90bSPoul-Henning Kamp 	TAILQ_INIT(&dd->de_dlist);
228a481b90bSPoul-Henning Kamp 
229a481b90bSPoul-Henning Kamp 	dd->de_dirent->d_type = DT_DIR;
23093432a92SPoul-Henning Kamp 	dd->de_mode = 0555;
231a481b90bSPoul-Henning Kamp 	dd->de_links = 2;
232a481b90bSPoul-Henning Kamp 	dd->de_dir = dd;
2333f54a085SPoul-Henning Kamp 
2343f54a085SPoul-Henning Kamp 	de = devfs_newdirent(".", 1);
2353f54a085SPoul-Henning Kamp 	de->de_dirent->d_type = DT_DIR;
236a481b90bSPoul-Henning Kamp 	de->de_dir = dd;
237a481b90bSPoul-Henning Kamp 	de->de_flags |= DE_DOT;
238a481b90bSPoul-Henning Kamp 	TAILQ_INSERT_TAIL(&dd->de_dlist, de, de_list);
239a481b90bSPoul-Henning Kamp 
240a481b90bSPoul-Henning Kamp 	de = devfs_newdirent("..", 2);
241a481b90bSPoul-Henning Kamp 	de->de_dirent->d_type = DT_DIR;
242a481b90bSPoul-Henning Kamp 	if (dotdot == NULL)
243a481b90bSPoul-Henning Kamp 		de->de_dir = dd;
244a481b90bSPoul-Henning Kamp 	else
245a481b90bSPoul-Henning Kamp 		de->de_dir = dotdot;
246a481b90bSPoul-Henning Kamp 	de->de_flags |= DE_DOTDOT;
247a481b90bSPoul-Henning Kamp 	TAILQ_INSERT_TAIL(&dd->de_dlist, de, de_list);
248a481b90bSPoul-Henning Kamp 
2493f54a085SPoul-Henning Kamp 	return (dd);
2503f54a085SPoul-Henning Kamp }
2513f54a085SPoul-Henning Kamp 
252a481b90bSPoul-Henning Kamp static void
253a481b90bSPoul-Henning Kamp devfs_delete(struct devfs_dirent *dd, struct devfs_dirent *de)
2543f54a085SPoul-Henning Kamp {
2553f54a085SPoul-Henning Kamp 
2563f54a085SPoul-Henning Kamp 	if (de->de_symlink) {
2573f54a085SPoul-Henning Kamp 		FREE(de->de_symlink, M_DEVFS);
2583f54a085SPoul-Henning Kamp 		de->de_symlink = NULL;
2593f54a085SPoul-Henning Kamp 	}
2608abea41dSPoul-Henning Kamp 	if (de->de_vnode)
2613f54a085SPoul-Henning Kamp 		de->de_vnode->v_data = NULL;
262a481b90bSPoul-Henning Kamp 	TAILQ_REMOVE(&dd->de_dlist, de, de_list);
2636742f328SRobert Watson #ifdef MAC
2646742f328SRobert Watson 	mac_destroy_devfsdirent(de);
2656742f328SRobert Watson #endif
2663f54a085SPoul-Henning Kamp 	FREE(de, M_DEVFS);
2673f54a085SPoul-Henning Kamp }
2683f54a085SPoul-Henning Kamp 
2693f54a085SPoul-Henning Kamp void
270a481b90bSPoul-Henning Kamp devfs_purge(struct devfs_dirent *dd)
2713f54a085SPoul-Henning Kamp {
2723f54a085SPoul-Henning Kamp 	struct devfs_dirent *de;
2733f54a085SPoul-Henning Kamp 
2743f54a085SPoul-Henning Kamp 	for (;;) {
275a481b90bSPoul-Henning Kamp 		de = TAILQ_FIRST(&dd->de_dlist);
2763f54a085SPoul-Henning Kamp 		if (de == NULL)
2773f54a085SPoul-Henning Kamp 			break;
2783f54a085SPoul-Henning Kamp 		devfs_delete(dd, de);
2793f54a085SPoul-Henning Kamp 	}
2803f54a085SPoul-Henning Kamp 	FREE(dd, M_DEVFS);
2813f54a085SPoul-Henning Kamp }
2823f54a085SPoul-Henning Kamp 
2833f54a085SPoul-Henning Kamp 
2843f54a085SPoul-Henning Kamp int
2853f54a085SPoul-Henning Kamp devfs_populate(struct devfs_mount *dm)
2863f54a085SPoul-Henning Kamp {
2873f54a085SPoul-Henning Kamp 	int i, j;
28889c9c53dSPoul-Henning Kamp 	struct cdev *dev, *pdev;
289a481b90bSPoul-Henning Kamp 	struct devfs_dirent *dd;
29093bcdfe2SPoul-Henning Kamp 	struct devfs_dirent *de, **dep;
2913f54a085SPoul-Henning Kamp 	char *q, *s;
2923f54a085SPoul-Henning Kamp 
29393bcdfe2SPoul-Henning Kamp 	if (dm->dm_generation == devfs_generation)
29493bcdfe2SPoul-Henning Kamp 		return (0);
295b40ce416SJulian Elischer 	lockmgr(&dm->dm_lock, LK_UPGRADE, 0, curthread);
29693bcdfe2SPoul-Henning Kamp 	if (devfs_noverflow && dm->dm_overflow == NULL) {
29793bcdfe2SPoul-Henning Kamp 		i = devfs_noverflow * sizeof (struct devfs_dirent *);
29893bcdfe2SPoul-Henning Kamp 		MALLOC(dm->dm_overflow, struct devfs_dirent **, i,
299a163d034SWarner Losh 			M_DEVFS, M_WAITOK | M_ZERO);
30093bcdfe2SPoul-Henning Kamp 	}
3013f54a085SPoul-Henning Kamp 	while (dm->dm_generation != devfs_generation) {
3023f54a085SPoul-Henning Kamp 		dm->dm_generation = devfs_generation;
30393bcdfe2SPoul-Henning Kamp 		for (i = 0; i <= devfs_topino; i++) {
30493bcdfe2SPoul-Henning Kamp 			dev = *devfs_itod(i);
30593bcdfe2SPoul-Henning Kamp 			dep = devfs_itode(dm, i);
30693bcdfe2SPoul-Henning Kamp 			de = *dep;
307a481b90bSPoul-Henning Kamp 			if (dev == NULL && de == DE_DELETED) {
30893bcdfe2SPoul-Henning Kamp 				*dep = NULL;
309a481b90bSPoul-Henning Kamp 				continue;
310a481b90bSPoul-Henning Kamp 			}
3113f54a085SPoul-Henning Kamp 			if (dev == NULL && de != NULL) {
3123f54a085SPoul-Henning Kamp 				dd = de->de_dir;
31393bcdfe2SPoul-Henning Kamp 				*dep = NULL;
314b43ab0e3SPoul-Henning Kamp 				devfs_delete(dd, de);
31593bcdfe2SPoul-Henning Kamp 				devfs_dropref(i);
3163f54a085SPoul-Henning Kamp 				continue;
3173f54a085SPoul-Henning Kamp 			}
3183f54a085SPoul-Henning Kamp 			if (dev == NULL)
3193f54a085SPoul-Henning Kamp 				continue;
3203f54a085SPoul-Henning Kamp 			if (de != NULL)
3213f54a085SPoul-Henning Kamp 				continue;
32293bcdfe2SPoul-Henning Kamp 			if (!devfs_getref(i))
32393bcdfe2SPoul-Henning Kamp 				continue;
324d785dfefSPoul-Henning Kamp 			dd = dm->dm_rootdir;
3253f54a085SPoul-Henning Kamp 			s = dev->si_name;
326c32d0a1dSPoul-Henning Kamp 			for (;;) {
3273f54a085SPoul-Henning Kamp 				for (q = s; *q != '/' && *q != '\0'; q++)
3283f54a085SPoul-Henning Kamp 					continue;
329c32d0a1dSPoul-Henning Kamp 				if (*q != '/')
330c32d0a1dSPoul-Henning Kamp 					break;
331c32d0a1dSPoul-Henning Kamp 				de = devfs_find(dd, s, q - s);
332c32d0a1dSPoul-Henning Kamp 				if (de == NULL) {
333a481b90bSPoul-Henning Kamp 					de = devfs_vmkdir(s, q - s, dd);
3346742f328SRobert Watson #ifdef MAC
335990b4b2dSRobert Watson 					mac_create_devfs_directory(
336990b4b2dSRobert Watson 					    dm->dm_mount, s, q - s, de);
3376742f328SRobert Watson #endif
338a481b90bSPoul-Henning Kamp 					de->de_inode = dm->dm_inode++;
339a481b90bSPoul-Henning Kamp 					TAILQ_INSERT_TAIL(&dd->de_dlist, de, de_list);
340a481b90bSPoul-Henning Kamp 					dd->de_links++;
341c32d0a1dSPoul-Henning Kamp 				}
342a481b90bSPoul-Henning Kamp 				s = q + 1;
343a481b90bSPoul-Henning Kamp 				dd = de;
3443f54a085SPoul-Henning Kamp 			}
3453f54a085SPoul-Henning Kamp 			de = devfs_newdirent(s, q - s);
3463f54a085SPoul-Henning Kamp 			if (dev->si_flags & SI_ALIAS) {
3473f54a085SPoul-Henning Kamp 				de->de_inode = dm->dm_inode++;
3483f54a085SPoul-Henning Kamp 				de->de_uid = 0;
3493f54a085SPoul-Henning Kamp 				de->de_gid = 0;
35093432a92SPoul-Henning Kamp 				de->de_mode = 0755;
3513f54a085SPoul-Henning Kamp 				de->de_dirent->d_type = DT_LNK;
3523344c5a1SPoul-Henning Kamp 				pdev = dev->si_parent;
3533f54a085SPoul-Henning Kamp 				j = strlen(pdev->si_name) + 1;
354a163d034SWarner Losh 				MALLOC(de->de_symlink, char *, j, M_DEVFS, M_WAITOK);
3553f54a085SPoul-Henning Kamp 				bcopy(pdev->si_name, de->de_symlink, j);
3563f54a085SPoul-Henning Kamp 			} else {
3573f54a085SPoul-Henning Kamp 				de->de_inode = i;
3589477d73eSPoul-Henning Kamp 				de->de_uid = dev->si_uid;
3599477d73eSPoul-Henning Kamp 				de->de_gid = dev->si_gid;
3609477d73eSPoul-Henning Kamp 				de->de_mode = dev->si_mode;
3613f54a085SPoul-Henning Kamp 				de->de_dirent->d_type = DT_CHR;
3623f54a085SPoul-Henning Kamp 			}
3636742f328SRobert Watson #ifdef MAC
364d26dd2d9SRobert Watson 			mac_create_devfs_device(dev->si_cred, dm->dm_mount,
365d26dd2d9SRobert Watson 			    dev, de);
3666742f328SRobert Watson #endif
36793bcdfe2SPoul-Henning Kamp 			*dep = de;
368c32d0a1dSPoul-Henning Kamp 			de->de_dir = dd;
369a1dc2096SDima Dorfman 			devfs_rules_apply(dm, de);
370a481b90bSPoul-Henning Kamp 			TAILQ_INSERT_TAIL(&dd->de_dlist, de, de_list);
3713f54a085SPoul-Henning Kamp 		}
3723f54a085SPoul-Henning Kamp 	}
373b40ce416SJulian Elischer 	lockmgr(&dm->dm_lock, LK_DOWNGRADE, 0, curthread);
3743f54a085SPoul-Henning Kamp 	return (0);
3753f54a085SPoul-Henning Kamp }
3763f54a085SPoul-Henning Kamp 
3779d960907SPoul-Henning Kamp /*
3789d960907SPoul-Henning Kamp  * devfs_create() and devfs_destroy() are called from kern_conf.c and
3799d960907SPoul-Henning Kamp  * in both cases the devlock() mutex is held, so no further locking
3809d960907SPoul-Henning Kamp  * is necesary and no sleeping allowed.
3819d960907SPoul-Henning Kamp  */
3829d960907SPoul-Henning Kamp 
3839285a87eSPoul-Henning Kamp void
38489c9c53dSPoul-Henning Kamp devfs_create(struct cdev *dev)
3853f54a085SPoul-Henning Kamp {
38693bcdfe2SPoul-Henning Kamp 	int ino, i, *ip;
38789c9c53dSPoul-Henning Kamp 	struct cdev **dp;
3889d960907SPoul-Henning Kamp 	struct cdev **ot;
3899d960907SPoul-Henning Kamp 	int *or;
3909d960907SPoul-Henning Kamp 	int n;
39193bcdfe2SPoul-Henning Kamp 
39293bcdfe2SPoul-Henning Kamp 	for (;;) {
39393bcdfe2SPoul-Henning Kamp 		/* Grab the next inode number */
39493bcdfe2SPoul-Henning Kamp 		ino = devfs_nextino;
39593bcdfe2SPoul-Henning Kamp 		i = ino + 1;
39693bcdfe2SPoul-Henning Kamp 		/* wrap around when we reach the end */
39793bcdfe2SPoul-Henning Kamp 		if (i >= NDEVFSINO + devfs_noverflow)
39893bcdfe2SPoul-Henning Kamp 			i = 3;
3999d960907SPoul-Henning Kamp 		devfs_nextino = i;
40093bcdfe2SPoul-Henning Kamp 
40193bcdfe2SPoul-Henning Kamp 		/* see if it was occupied */
40293bcdfe2SPoul-Henning Kamp 		dp = devfs_itod(ino);
4039d960907SPoul-Henning Kamp 		KASSERT(dp != NULL, ("DEVFS: No devptr inode %d", ino));
40493bcdfe2SPoul-Henning Kamp 		if (*dp != NULL)
40593bcdfe2SPoul-Henning Kamp 			continue;
40693bcdfe2SPoul-Henning Kamp 		ip = devfs_itor(ino);
4079d960907SPoul-Henning Kamp 		KASSERT(ip != NULL, ("DEVFS: No iptr inode %d", ino));
40893bcdfe2SPoul-Henning Kamp 		if (*ip != 0)
40993bcdfe2SPoul-Henning Kamp 			continue;
4109d960907SPoul-Henning Kamp 		break;
4119d960907SPoul-Henning Kamp 	}
41293bcdfe2SPoul-Henning Kamp 
4139d960907SPoul-Henning Kamp 	*dp = dev;
41493bcdfe2SPoul-Henning Kamp 	dev->si_inode = ino;
4159d960907SPoul-Henning Kamp 	if (i > devfs_topino)
4169d960907SPoul-Henning Kamp 		devfs_topino = i;
41793bcdfe2SPoul-Henning Kamp 
4189d960907SPoul-Henning Kamp 	devfs_numino++;
4199d960907SPoul-Henning Kamp 	devfs_generation++;
4209d960907SPoul-Henning Kamp 
4219d960907SPoul-Henning Kamp 	if (devfs_overflow != NULL || devfs_numino + 100 < NDEVFSINO)
4229d960907SPoul-Henning Kamp 		return;
4239d960907SPoul-Henning Kamp 
4249d960907SPoul-Henning Kamp 	/*
4259d960907SPoul-Henning Kamp 	 * Try to allocate overflow table
4269d960907SPoul-Henning Kamp 	 * XXX: we can probably be less panicy these days and a linked
4279d960907SPoul-Henning Kamp 	 * XXX: list of PAGESIZE/PTRSIZE entries might be a better idea.
4289d960907SPoul-Henning Kamp 	 *
4299d960907SPoul-Henning Kamp 	 * XXX: we may be into witness unlove here.
4309d960907SPoul-Henning Kamp 	 */
4319d960907SPoul-Henning Kamp 	n = devfs_noverflowwant;
4329d960907SPoul-Henning Kamp 	ot = malloc(sizeof(*ot) * n, M_DEVFS, M_NOWAIT | M_ZERO);
4339d960907SPoul-Henning Kamp 	if (ot == NULL)
4349d960907SPoul-Henning Kamp 		return;
4359d960907SPoul-Henning Kamp 	or = malloc(sizeof(*or) * n, M_DEVFS, M_NOWAIT | M_ZERO);
4369d960907SPoul-Henning Kamp 	if (or == NULL) {
4379d960907SPoul-Henning Kamp 		free(ot, M_DEVFS);
4389d960907SPoul-Henning Kamp 		return;
4399d960907SPoul-Henning Kamp 	}
4409d960907SPoul-Henning Kamp 	devfs_overflow = ot;
4419d960907SPoul-Henning Kamp 	devfs_refoverflow = or;
4429d960907SPoul-Henning Kamp 	devfs_noverflow = n;
4439d960907SPoul-Henning Kamp 	printf("DEVFS Overflow table with %d entries allocated\n", n);
4449d960907SPoul-Henning Kamp 	return;
4453f54a085SPoul-Henning Kamp }
4463f54a085SPoul-Henning Kamp 
4479285a87eSPoul-Henning Kamp void
44889c9c53dSPoul-Henning Kamp devfs_destroy(struct cdev *dev)
4493f54a085SPoul-Henning Kamp {
4509d960907SPoul-Henning Kamp 	int ino;
4519d960907SPoul-Henning Kamp 	struct cdev **dp;
45293bcdfe2SPoul-Henning Kamp 
45393bcdfe2SPoul-Henning Kamp 	ino = dev->si_inode;
45493bcdfe2SPoul-Henning Kamp 	dev->si_inode = 0;
45593bcdfe2SPoul-Henning Kamp 	if (ino == 0)
45693bcdfe2SPoul-Henning Kamp 		return;
4579d960907SPoul-Henning Kamp 	dp = devfs_itod(ino);
4589d960907SPoul-Henning Kamp 	KASSERT(*dp == dev,
4599d960907SPoul-Henning Kamp 	    ("DEVFS: destroying wrong cdev ino %d", ino));
4609d960907SPoul-Henning Kamp 	*dp = NULL;
4619d960907SPoul-Henning Kamp 	devfs_numino--;
4629d960907SPoul-Henning Kamp 	devfs_generation++;
4639d960907SPoul-Henning Kamp 	if (ino < devfs_nextino)
4649d960907SPoul-Henning Kamp 		devfs_nextino = ino;
4653f54a085SPoul-Henning Kamp }
466