xref: /freebsd/sys/fs/devfs/devfs_devs.c (revision 6742f32809defc591ee96910680ddf5b9f90709f)
13f54a085SPoul-Henning Kamp #define DEBUG 1
23f54a085SPoul-Henning Kamp /*
33f54a085SPoul-Henning Kamp  * Copyright (c) 2000
43f54a085SPoul-Henning Kamp  *	Poul-Henning Kamp.  All rights reserved.
53f54a085SPoul-Henning Kamp  *
63f54a085SPoul-Henning Kamp  * Redistribution and use in source and binary forms, with or without
73f54a085SPoul-Henning Kamp  * modification, are permitted provided that the following conditions
83f54a085SPoul-Henning Kamp  * are met:
93f54a085SPoul-Henning Kamp  * 1. Redistributions of source code must retain the above copyright
103f54a085SPoul-Henning Kamp  *    notice, this list of conditions and the following disclaimer.
113f54a085SPoul-Henning Kamp  * 2. Neither the name of the University nor the names of its contributors
123f54a085SPoul-Henning Kamp  *    may be used to endorse or promote products derived from this software
133f54a085SPoul-Henning Kamp  *    without specific prior written permission.
143f54a085SPoul-Henning Kamp  *
153f54a085SPoul-Henning Kamp  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
163f54a085SPoul-Henning Kamp  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
173f54a085SPoul-Henning Kamp  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
183f54a085SPoul-Henning Kamp  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
193f54a085SPoul-Henning Kamp  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
203f54a085SPoul-Henning Kamp  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
213f54a085SPoul-Henning Kamp  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
223f54a085SPoul-Henning Kamp  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
233f54a085SPoul-Henning Kamp  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
243f54a085SPoul-Henning Kamp  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
253f54a085SPoul-Henning Kamp  * SUCH DAMAGE.
263f54a085SPoul-Henning Kamp  *
273f54a085SPoul-Henning Kamp  * From: FreeBSD: src/sys/miscfs/kernfs/kernfs_vfsops.c 1.36
283f54a085SPoul-Henning Kamp  *
293f54a085SPoul-Henning Kamp  * $FreeBSD$
303f54a085SPoul-Henning Kamp  */
313f54a085SPoul-Henning Kamp 
3293bcdfe2SPoul-Henning Kamp #include "opt_devfs.h"
336742f328SRobert Watson #include "opt_mac.h"
34ab9f3b29SPoul-Henning Kamp #ifndef NODEVFS
3593bcdfe2SPoul-Henning Kamp 
363f54a085SPoul-Henning Kamp #include <sys/param.h>
373f54a085SPoul-Henning Kamp #include <sys/systm.h>
383f54a085SPoul-Henning Kamp #include <sys/conf.h>
39fb919e4dSMark Murray #include <sys/dirent.h>
4093bcdfe2SPoul-Henning Kamp #include <sys/kernel.h>
4193bcdfe2SPoul-Henning Kamp #include <sys/lock.h>
426742f328SRobert Watson #include <sys/mac.h>
43fb919e4dSMark Murray #include <sys/malloc.h>
44fb919e4dSMark Murray #include <sys/proc.h>
45fb919e4dSMark Murray #include <sys/sysctl.h>
46fb919e4dSMark Murray #include <sys/vnode.h>
473f54a085SPoul-Henning Kamp 
4893bcdfe2SPoul-Henning Kamp #include <machine/atomic.h>
4993bcdfe2SPoul-Henning Kamp 
503f54a085SPoul-Henning Kamp #include <fs/devfs/devfs.h>
513f54a085SPoul-Henning Kamp 
5293bcdfe2SPoul-Henning Kamp static dev_t devfs_inot[NDEVFSINO];
5393bcdfe2SPoul-Henning Kamp static dev_t *devfs_overflow;
5493bcdfe2SPoul-Henning Kamp static int devfs_ref[NDEVFSINO];
5593bcdfe2SPoul-Henning Kamp static int *devfs_refoverflow;
5693bcdfe2SPoul-Henning Kamp static int devfs_nextino = 3;
5793bcdfe2SPoul-Henning Kamp static int devfs_numino;
5893bcdfe2SPoul-Henning Kamp static int devfs_topino;
5993bcdfe2SPoul-Henning Kamp static int devfs_noverflowwant = NDEVFSOVERFLOW;
6093bcdfe2SPoul-Henning Kamp static int devfs_noverflow;
6193bcdfe2SPoul-Henning Kamp static unsigned devfs_generation;
6293bcdfe2SPoul-Henning Kamp 
632a043678SPoul-Henning Kamp static void devfs_attemptoverflow(int insist);
642a043678SPoul-Henning Kamp static struct devfs_dirent *devfs_find (struct devfs_dirent *dd, const char *name, int namelen);
652a043678SPoul-Henning Kamp 
6693bcdfe2SPoul-Henning Kamp SYSCTL_NODE(_vfs, OID_AUTO, devfs, CTLFLAG_RW, 0, "DEVFS filesystem");
6793bcdfe2SPoul-Henning Kamp SYSCTL_UINT(_vfs_devfs, OID_AUTO, noverflow, CTLFLAG_RW,
6893bcdfe2SPoul-Henning Kamp 	&devfs_noverflowwant, 0, "Size of DEVFS overflow table");
6993bcdfe2SPoul-Henning Kamp SYSCTL_UINT(_vfs_devfs, OID_AUTO, generation, CTLFLAG_RD,
7093bcdfe2SPoul-Henning Kamp 	&devfs_generation, 0, "DEVFS generation number");
7193bcdfe2SPoul-Henning Kamp SYSCTL_UINT(_vfs_devfs, OID_AUTO, inodes, CTLFLAG_RD,
7293bcdfe2SPoul-Henning Kamp 	&devfs_numino, 0, "DEVFS inodes");
7393bcdfe2SPoul-Henning Kamp SYSCTL_UINT(_vfs_devfs, OID_AUTO, topinode, CTLFLAG_RD,
7493bcdfe2SPoul-Henning Kamp 	&devfs_topino, 0, "DEVFS highest inode#");
7593bcdfe2SPoul-Henning Kamp 
7693bcdfe2SPoul-Henning Kamp static int *
7793bcdfe2SPoul-Henning Kamp devfs_itor(int inode)
7893bcdfe2SPoul-Henning Kamp {
7993bcdfe2SPoul-Henning Kamp 	if (inode < NDEVFSINO)
8093bcdfe2SPoul-Henning Kamp 		return (&devfs_ref[inode]);
8193bcdfe2SPoul-Henning Kamp 	else if (inode < NDEVFSINO + devfs_noverflow)
8293bcdfe2SPoul-Henning Kamp 		return (&devfs_refoverflow[inode - NDEVFSINO]);
8393bcdfe2SPoul-Henning Kamp 	else
8493bcdfe2SPoul-Henning Kamp 		panic ("YRK!");
8593bcdfe2SPoul-Henning Kamp }
8693bcdfe2SPoul-Henning Kamp 
8793bcdfe2SPoul-Henning Kamp static void
8893bcdfe2SPoul-Henning Kamp devfs_dropref(int inode)
8993bcdfe2SPoul-Henning Kamp {
9093bcdfe2SPoul-Henning Kamp 	int *ip;
9193bcdfe2SPoul-Henning Kamp 
9293bcdfe2SPoul-Henning Kamp 	ip = devfs_itor(inode);
9393bcdfe2SPoul-Henning Kamp 	atomic_add_int(ip, -1);
9493bcdfe2SPoul-Henning Kamp }
9593bcdfe2SPoul-Henning Kamp 
9693bcdfe2SPoul-Henning Kamp static int
9793bcdfe2SPoul-Henning Kamp devfs_getref(int inode)
9893bcdfe2SPoul-Henning Kamp {
9993bcdfe2SPoul-Henning Kamp 	int *ip, i, j;
10093bcdfe2SPoul-Henning Kamp 	dev_t *dp;
10193bcdfe2SPoul-Henning Kamp 
10293bcdfe2SPoul-Henning Kamp 	ip = devfs_itor(inode);
10393bcdfe2SPoul-Henning Kamp 	dp = devfs_itod(inode);
10493bcdfe2SPoul-Henning Kamp 	for (;;) {
10593bcdfe2SPoul-Henning Kamp 		i = *ip;
10693bcdfe2SPoul-Henning Kamp 		j = i + 1;
10793bcdfe2SPoul-Henning Kamp 		if (!atomic_cmpset_int(ip, i, j))
10893bcdfe2SPoul-Henning Kamp 			continue;
10993bcdfe2SPoul-Henning Kamp 		if (*dp != NULL)
11093bcdfe2SPoul-Henning Kamp 			return (1);
11193bcdfe2SPoul-Henning Kamp 		atomic_add_int(ip, -1);
11293bcdfe2SPoul-Henning Kamp 		return(0);
11393bcdfe2SPoul-Henning Kamp 	}
11493bcdfe2SPoul-Henning Kamp }
11593bcdfe2SPoul-Henning Kamp 
11693bcdfe2SPoul-Henning Kamp struct devfs_dirent **
11793bcdfe2SPoul-Henning Kamp devfs_itode (struct devfs_mount *dm, int inode)
11893bcdfe2SPoul-Henning Kamp {
11993bcdfe2SPoul-Henning Kamp 
12093bcdfe2SPoul-Henning Kamp 	if (inode < NDEVFSINO)
12193bcdfe2SPoul-Henning Kamp 		return (&dm->dm_dirent[inode]);
12293bcdfe2SPoul-Henning Kamp 	if (devfs_overflow == NULL)
12393bcdfe2SPoul-Henning Kamp 		return (NULL);
12493bcdfe2SPoul-Henning Kamp 	if (inode < NDEVFSINO + devfs_noverflow)
12593bcdfe2SPoul-Henning Kamp 		return (&dm->dm_overflow[inode - NDEVFSINO]);
12693bcdfe2SPoul-Henning Kamp 	return (NULL);
12793bcdfe2SPoul-Henning Kamp }
12893bcdfe2SPoul-Henning Kamp 
12993bcdfe2SPoul-Henning Kamp dev_t *
13093bcdfe2SPoul-Henning Kamp devfs_itod (int inode)
13193bcdfe2SPoul-Henning Kamp {
13293bcdfe2SPoul-Henning Kamp 
13393bcdfe2SPoul-Henning Kamp 	if (inode < NDEVFSINO)
13493bcdfe2SPoul-Henning Kamp 		return (&devfs_inot[inode]);
13593bcdfe2SPoul-Henning Kamp 	if (devfs_overflow == NULL)
13693bcdfe2SPoul-Henning Kamp 		return (NULL);
13793bcdfe2SPoul-Henning Kamp 	if (inode < NDEVFSINO + devfs_noverflow)
13893bcdfe2SPoul-Henning Kamp 		return (&devfs_overflow[inode - NDEVFSINO]);
13993bcdfe2SPoul-Henning Kamp 	return (NULL);
14093bcdfe2SPoul-Henning Kamp }
14193bcdfe2SPoul-Henning Kamp 
1422a043678SPoul-Henning Kamp static void
14393bcdfe2SPoul-Henning Kamp devfs_attemptoverflow(int insist)
14493bcdfe2SPoul-Henning Kamp {
14593bcdfe2SPoul-Henning Kamp 	dev_t **ot;
14693bcdfe2SPoul-Henning Kamp 	int *or;
14793bcdfe2SPoul-Henning Kamp 	int n, nb;
14893bcdfe2SPoul-Henning Kamp 
14993bcdfe2SPoul-Henning Kamp 	/* Check if somebody beat us to it */
15093bcdfe2SPoul-Henning Kamp 	if (devfs_overflow != NULL)
15193bcdfe2SPoul-Henning Kamp 		return;
15293bcdfe2SPoul-Henning Kamp 	ot = NULL;
15393bcdfe2SPoul-Henning Kamp 	or = NULL;
15493bcdfe2SPoul-Henning Kamp 	n = devfs_noverflowwant;
15593bcdfe2SPoul-Henning Kamp 	nb = sizeof (struct dev_t *) * n;
1567cc0979fSDavid Malone 	MALLOC(ot, dev_t **, nb, M_DEVFS, (insist ? M_WAITOK : M_NOWAIT) | M_ZERO);
15793bcdfe2SPoul-Henning Kamp 	if (ot == NULL)
15893bcdfe2SPoul-Henning Kamp 		goto bail;
15993bcdfe2SPoul-Henning Kamp 	nb = sizeof (int) * n;
1607cc0979fSDavid Malone 	MALLOC(or, int *, nb, M_DEVFS, (insist ? M_WAITOK : M_NOWAIT) | M_ZERO);
16193bcdfe2SPoul-Henning Kamp 	if (or == NULL)
16293bcdfe2SPoul-Henning Kamp 		goto bail;
16393bcdfe2SPoul-Henning Kamp 	if (!atomic_cmpset_ptr(&devfs_overflow, NULL, ot))
16493bcdfe2SPoul-Henning Kamp 		goto bail;
16593bcdfe2SPoul-Henning Kamp 	devfs_refoverflow = or;
16693bcdfe2SPoul-Henning Kamp 	devfs_noverflow = n;
16793bcdfe2SPoul-Henning Kamp 	printf("DEVFS Overflow table with %d entries allocated when %d in use\n", n, devfs_numino);
16893bcdfe2SPoul-Henning Kamp 	return;
16993bcdfe2SPoul-Henning Kamp 
17093bcdfe2SPoul-Henning Kamp bail:
17193bcdfe2SPoul-Henning Kamp 	/* Somebody beat us to it, or something went wrong. */
17293bcdfe2SPoul-Henning Kamp 	if (ot != NULL)
17393bcdfe2SPoul-Henning Kamp 		FREE(ot, M_DEVFS);
17493bcdfe2SPoul-Henning Kamp 	if (or != NULL)
17593bcdfe2SPoul-Henning Kamp 		FREE(or, M_DEVFS);
17693bcdfe2SPoul-Henning Kamp 	return;
17793bcdfe2SPoul-Henning Kamp }
17893bcdfe2SPoul-Henning Kamp 
1792a043678SPoul-Henning Kamp static struct devfs_dirent *
180c32d0a1dSPoul-Henning Kamp devfs_find(struct devfs_dirent *dd, const char *name, int namelen)
181c32d0a1dSPoul-Henning Kamp {
182c32d0a1dSPoul-Henning Kamp 	struct devfs_dirent *de;
183c32d0a1dSPoul-Henning Kamp 
184c32d0a1dSPoul-Henning Kamp 	TAILQ_FOREACH(de, &dd->de_dlist, de_list) {
185c32d0a1dSPoul-Henning Kamp 		if (namelen != de->de_dirent->d_namlen)
186c32d0a1dSPoul-Henning Kamp 			continue;
187c32d0a1dSPoul-Henning Kamp 		if (bcmp(name, de->de_dirent->d_name, namelen) != 0)
188c32d0a1dSPoul-Henning Kamp 			continue;
189c32d0a1dSPoul-Henning Kamp 		break;
190c32d0a1dSPoul-Henning Kamp 	}
191c32d0a1dSPoul-Henning Kamp 	return (de);
192c32d0a1dSPoul-Henning Kamp }
193c32d0a1dSPoul-Henning Kamp 
194c32d0a1dSPoul-Henning Kamp struct devfs_dirent *
1953f54a085SPoul-Henning Kamp devfs_newdirent(char *name, int namelen)
1963f54a085SPoul-Henning Kamp {
1973f54a085SPoul-Henning Kamp 	int i;
1983f54a085SPoul-Henning Kamp 	struct devfs_dirent *de;
1993f54a085SPoul-Henning Kamp 	struct dirent d;
2003f54a085SPoul-Henning Kamp 
2013f54a085SPoul-Henning Kamp 	d.d_namlen = namelen;
2023f54a085SPoul-Henning Kamp 	i = sizeof (*de) + GENERIC_DIRSIZ(&d);
2037cc0979fSDavid Malone 	MALLOC(de, struct devfs_dirent *, i, M_DEVFS, M_WAITOK | M_ZERO);
2043f54a085SPoul-Henning Kamp 	de->de_dirent = (struct dirent *)(de + 1);
2053f54a085SPoul-Henning Kamp 	de->de_dirent->d_namlen = namelen;
2063f54a085SPoul-Henning Kamp 	de->de_dirent->d_reclen = GENERIC_DIRSIZ(&d);
207aadf2655SPoul-Henning Kamp 	bcopy(name, de->de_dirent->d_name, namelen);
208aadf2655SPoul-Henning Kamp 	de->de_dirent->d_name[namelen] = '\0';
20993432a92SPoul-Henning Kamp 	vfs_timestamp(&de->de_ctime);
210a481b90bSPoul-Henning Kamp 	de->de_mtime = de->de_atime = de->de_ctime;
211a481b90bSPoul-Henning Kamp 	de->de_links = 1;
2126742f328SRobert Watson #ifdef MAC
2136742f328SRobert Watson 	mac_init_devfsdirent(de);
2146742f328SRobert Watson #endif
2153f54a085SPoul-Henning Kamp 	return (de);
2163f54a085SPoul-Henning Kamp }
2173f54a085SPoul-Henning Kamp 
218a481b90bSPoul-Henning Kamp struct devfs_dirent *
219a481b90bSPoul-Henning Kamp devfs_vmkdir(char *name, int namelen, struct devfs_dirent *dotdot)
2203f54a085SPoul-Henning Kamp {
221a481b90bSPoul-Henning Kamp 	struct devfs_dirent *dd;
2223f54a085SPoul-Henning Kamp 	struct devfs_dirent *de;
2233f54a085SPoul-Henning Kamp 
224a481b90bSPoul-Henning Kamp 	dd = devfs_newdirent(name, namelen);
225a481b90bSPoul-Henning Kamp 
226a481b90bSPoul-Henning Kamp 	TAILQ_INIT(&dd->de_dlist);
227a481b90bSPoul-Henning Kamp 
228a481b90bSPoul-Henning Kamp 	dd->de_dirent->d_type = DT_DIR;
22993432a92SPoul-Henning Kamp 	dd->de_mode = 0555;
230a481b90bSPoul-Henning Kamp 	dd->de_links = 2;
231a481b90bSPoul-Henning Kamp 	dd->de_dir = dd;
2323f54a085SPoul-Henning Kamp 
2333f54a085SPoul-Henning Kamp 	de = devfs_newdirent(".", 1);
2343f54a085SPoul-Henning Kamp 	de->de_dirent->d_type = DT_DIR;
235a481b90bSPoul-Henning Kamp 	de->de_dir = dd;
236a481b90bSPoul-Henning Kamp 	de->de_flags |= DE_DOT;
237a481b90bSPoul-Henning Kamp 	TAILQ_INSERT_TAIL(&dd->de_dlist, de, de_list);
238a481b90bSPoul-Henning Kamp 
239a481b90bSPoul-Henning Kamp 	de = devfs_newdirent("..", 2);
240a481b90bSPoul-Henning Kamp 	de->de_dirent->d_type = DT_DIR;
241a481b90bSPoul-Henning Kamp 	if (dotdot == NULL)
242a481b90bSPoul-Henning Kamp 		de->de_dir = dd;
243a481b90bSPoul-Henning Kamp 	else
244a481b90bSPoul-Henning Kamp 		de->de_dir = dotdot;
245a481b90bSPoul-Henning Kamp 	de->de_flags |= DE_DOTDOT;
246a481b90bSPoul-Henning Kamp 	TAILQ_INSERT_TAIL(&dd->de_dlist, de, de_list);
247a481b90bSPoul-Henning Kamp 
2483f54a085SPoul-Henning Kamp 	return (dd);
2493f54a085SPoul-Henning Kamp }
2503f54a085SPoul-Henning Kamp 
251a481b90bSPoul-Henning Kamp static void
252a481b90bSPoul-Henning Kamp devfs_delete(struct devfs_dirent *dd, struct devfs_dirent *de)
2533f54a085SPoul-Henning Kamp {
2543f54a085SPoul-Henning Kamp 
2553f54a085SPoul-Henning Kamp 	if (de->de_symlink) {
2563f54a085SPoul-Henning Kamp 		FREE(de->de_symlink, M_DEVFS);
2573f54a085SPoul-Henning Kamp 		de->de_symlink = NULL;
2583f54a085SPoul-Henning Kamp 	}
2598abea41dSPoul-Henning Kamp 	if (de->de_vnode)
2603f54a085SPoul-Henning Kamp 		de->de_vnode->v_data = NULL;
261a481b90bSPoul-Henning Kamp 	TAILQ_REMOVE(&dd->de_dlist, de, de_list);
2626742f328SRobert Watson #ifdef MAC
2636742f328SRobert Watson 	mac_destroy_devfsdirent(de);
2646742f328SRobert Watson #endif
2653f54a085SPoul-Henning Kamp 	FREE(de, M_DEVFS);
2663f54a085SPoul-Henning Kamp }
2673f54a085SPoul-Henning Kamp 
2683f54a085SPoul-Henning Kamp void
269a481b90bSPoul-Henning Kamp devfs_purge(struct devfs_dirent *dd)
2703f54a085SPoul-Henning Kamp {
2713f54a085SPoul-Henning Kamp 	struct devfs_dirent *de;
2723f54a085SPoul-Henning Kamp 
2733f54a085SPoul-Henning Kamp 	for (;;) {
274a481b90bSPoul-Henning Kamp 		de = TAILQ_FIRST(&dd->de_dlist);
2753f54a085SPoul-Henning Kamp 		if (de == NULL)
2763f54a085SPoul-Henning Kamp 			break;
2773f54a085SPoul-Henning Kamp 		devfs_delete(dd, de);
2783f54a085SPoul-Henning Kamp 	}
2793f54a085SPoul-Henning Kamp 	FREE(dd, M_DEVFS);
2803f54a085SPoul-Henning Kamp }
2813f54a085SPoul-Henning Kamp 
2823f54a085SPoul-Henning Kamp 
2833f54a085SPoul-Henning Kamp int
2843f54a085SPoul-Henning Kamp devfs_populate(struct devfs_mount *dm)
2853f54a085SPoul-Henning Kamp {
2863f54a085SPoul-Henning Kamp 	int i, j;
2873f54a085SPoul-Henning Kamp 	dev_t dev, pdev;
288a481b90bSPoul-Henning Kamp 	struct devfs_dirent *dd;
28993bcdfe2SPoul-Henning Kamp 	struct devfs_dirent *de, **dep;
2903f54a085SPoul-Henning Kamp 	char *q, *s;
2913f54a085SPoul-Henning Kamp 
29293bcdfe2SPoul-Henning Kamp 	if (dm->dm_generation == devfs_generation)
29393bcdfe2SPoul-Henning Kamp 		return (0);
294b40ce416SJulian Elischer 	lockmgr(&dm->dm_lock, LK_UPGRADE, 0, curthread);
29593bcdfe2SPoul-Henning Kamp 	if (devfs_noverflow && dm->dm_overflow == NULL) {
29693bcdfe2SPoul-Henning Kamp 		i = devfs_noverflow * sizeof (struct devfs_dirent *);
29793bcdfe2SPoul-Henning Kamp 		MALLOC(dm->dm_overflow, struct devfs_dirent **, i,
2987cc0979fSDavid Malone 			M_DEVFS, M_WAITOK | M_ZERO);
29993bcdfe2SPoul-Henning Kamp 	}
3003f54a085SPoul-Henning Kamp 	while (dm->dm_generation != devfs_generation) {
3013f54a085SPoul-Henning Kamp 		dm->dm_generation = devfs_generation;
30293bcdfe2SPoul-Henning Kamp 		for (i = 0; i <= devfs_topino; i++) {
30393bcdfe2SPoul-Henning Kamp 			dev = *devfs_itod(i);
30493bcdfe2SPoul-Henning Kamp 			dep = devfs_itode(dm, i);
30593bcdfe2SPoul-Henning Kamp 			de = *dep;
306a481b90bSPoul-Henning Kamp 			if (dev == NULL && de == DE_DELETED) {
30793bcdfe2SPoul-Henning Kamp 				*dep = NULL;
308a481b90bSPoul-Henning Kamp 				continue;
309a481b90bSPoul-Henning Kamp 			}
3103f54a085SPoul-Henning Kamp 			if (dev == NULL && de != NULL) {
3113f54a085SPoul-Henning Kamp 				dd = de->de_dir;
31293bcdfe2SPoul-Henning Kamp 				*dep = NULL;
313a481b90bSPoul-Henning Kamp 				TAILQ_REMOVE(&dd->de_dlist, de, de_list);
3148abea41dSPoul-Henning Kamp 				if (de->de_vnode)
3153f54a085SPoul-Henning Kamp 					de->de_vnode->v_data = NULL;
3163f54a085SPoul-Henning Kamp 				FREE(de, M_DEVFS);
31793bcdfe2SPoul-Henning Kamp 				devfs_dropref(i);
3183f54a085SPoul-Henning Kamp 				continue;
3193f54a085SPoul-Henning Kamp 			}
3203f54a085SPoul-Henning Kamp 			if (dev == NULL)
3213f54a085SPoul-Henning Kamp 				continue;
3223f54a085SPoul-Henning Kamp 			if (de != NULL)
3233f54a085SPoul-Henning Kamp 				continue;
32493bcdfe2SPoul-Henning Kamp 			if (!devfs_getref(i))
32593bcdfe2SPoul-Henning Kamp 				continue;
3263f54a085SPoul-Henning Kamp 			dd = dm->dm_basedir;
3273f54a085SPoul-Henning Kamp 			s = dev->si_name;
328c32d0a1dSPoul-Henning Kamp 			for (;;) {
3293f54a085SPoul-Henning Kamp 				for (q = s; *q != '/' && *q != '\0'; q++)
3303f54a085SPoul-Henning Kamp 					continue;
331c32d0a1dSPoul-Henning Kamp 				if (*q != '/')
332c32d0a1dSPoul-Henning Kamp 					break;
333c32d0a1dSPoul-Henning Kamp 				de = devfs_find(dd, s, q - s);
334c32d0a1dSPoul-Henning Kamp 				if (de == NULL) {
335a481b90bSPoul-Henning Kamp 					de = devfs_vmkdir(s, q - s, dd);
3366742f328SRobert Watson #ifdef MAC
3376742f328SRobert Watson 					mac_create_devfs_directory(s, q - s,
3386742f328SRobert Watson 					    de);
3396742f328SRobert Watson #endif
340a481b90bSPoul-Henning Kamp 					de->de_inode = dm->dm_inode++;
341a481b90bSPoul-Henning Kamp 					TAILQ_INSERT_TAIL(&dd->de_dlist, de, de_list);
342a481b90bSPoul-Henning Kamp 					dd->de_links++;
343c32d0a1dSPoul-Henning Kamp 				}
344a481b90bSPoul-Henning Kamp 				s = q + 1;
345a481b90bSPoul-Henning Kamp 				dd = de;
3463f54a085SPoul-Henning Kamp 			}
3473f54a085SPoul-Henning Kamp 			de = devfs_newdirent(s, q - s);
3483f54a085SPoul-Henning Kamp 			if (dev->si_flags & SI_ALIAS) {
3493f54a085SPoul-Henning Kamp 				de->de_inode = dm->dm_inode++;
3503f54a085SPoul-Henning Kamp 				de->de_uid = 0;
3513f54a085SPoul-Henning Kamp 				de->de_gid = 0;
35293432a92SPoul-Henning Kamp 				de->de_mode = 0755;
3533f54a085SPoul-Henning Kamp 				de->de_dirent->d_type = DT_LNK;
3543344c5a1SPoul-Henning Kamp 				pdev = dev->si_parent;
3553f54a085SPoul-Henning Kamp 				j = strlen(pdev->si_name) + 1;
3563f54a085SPoul-Henning Kamp 				MALLOC(de->de_symlink, char *, j, M_DEVFS, M_WAITOK);
3573f54a085SPoul-Henning Kamp 				bcopy(pdev->si_name, de->de_symlink, j);
3583f54a085SPoul-Henning Kamp 			} else {
3593f54a085SPoul-Henning Kamp 				de->de_inode = i;
3603f54a085SPoul-Henning Kamp 				de->de_uid = dev->si_uid;
3613f54a085SPoul-Henning Kamp 				de->de_gid = dev->si_gid;
3623f54a085SPoul-Henning Kamp 				de->de_mode = dev->si_mode;
3633f54a085SPoul-Henning Kamp 				de->de_dirent->d_type = DT_CHR;
3643f54a085SPoul-Henning Kamp 			}
3656742f328SRobert Watson #ifdef MAC
3666742f328SRobert Watson 			mac_create_devfs_device(dev, de);
3676742f328SRobert Watson #endif
36893bcdfe2SPoul-Henning Kamp 			*dep = de;
369c32d0a1dSPoul-Henning Kamp 			de->de_dir = dd;
370a1dc2096SDima Dorfman 			devfs_rules_apply(dm, de);
371a481b90bSPoul-Henning Kamp 			TAILQ_INSERT_TAIL(&dd->de_dlist, de, de_list);
3723f54a085SPoul-Henning Kamp #if 0
3733f54a085SPoul-Henning Kamp 			printf("Add ino%d %s\n", i, dev->si_name);
3743f54a085SPoul-Henning Kamp #endif
3753f54a085SPoul-Henning Kamp 		}
3763f54a085SPoul-Henning Kamp 	}
377b40ce416SJulian Elischer 	lockmgr(&dm->dm_lock, LK_DOWNGRADE, 0, curthread);
3783f54a085SPoul-Henning Kamp 	return (0);
3793f54a085SPoul-Henning Kamp }
3803f54a085SPoul-Henning Kamp 
3813f54a085SPoul-Henning Kamp static void
3823f54a085SPoul-Henning Kamp devfs_create(dev_t dev)
3833f54a085SPoul-Henning Kamp {
38493bcdfe2SPoul-Henning Kamp 	int ino, i, *ip;
38593bcdfe2SPoul-Henning Kamp 	dev_t *dp;
38693bcdfe2SPoul-Henning Kamp 
38793bcdfe2SPoul-Henning Kamp 	for (;;) {
38893bcdfe2SPoul-Henning Kamp 		/* Grab the next inode number */
38993bcdfe2SPoul-Henning Kamp 		ino = devfs_nextino;
39093bcdfe2SPoul-Henning Kamp 		i = ino + 1;
39193bcdfe2SPoul-Henning Kamp 		/* wrap around when we reach the end */
39293bcdfe2SPoul-Henning Kamp 		if (i >= NDEVFSINO + devfs_noverflow)
39393bcdfe2SPoul-Henning Kamp 			i = 3;
39493bcdfe2SPoul-Henning Kamp 		if (!atomic_cmpset_int(&devfs_nextino, ino, i))
39593bcdfe2SPoul-Henning Kamp 			continue;
39693bcdfe2SPoul-Henning Kamp 
39793bcdfe2SPoul-Henning Kamp 		/* see if it was occupied */
39893bcdfe2SPoul-Henning Kamp 		dp = devfs_itod(ino);
39993bcdfe2SPoul-Henning Kamp 		if (dp == NULL)
40093bcdfe2SPoul-Henning Kamp 			Debugger("dp == NULL\n");
40193bcdfe2SPoul-Henning Kamp 		if (*dp != NULL)
40293bcdfe2SPoul-Henning Kamp 			continue;
40393bcdfe2SPoul-Henning Kamp 		ip = devfs_itor(ino);
40493bcdfe2SPoul-Henning Kamp 		if (ip == NULL)
40593bcdfe2SPoul-Henning Kamp 			Debugger("ip == NULL\n");
40693bcdfe2SPoul-Henning Kamp 		if (*ip != 0)
40793bcdfe2SPoul-Henning Kamp 			continue;
40893bcdfe2SPoul-Henning Kamp 
40993bcdfe2SPoul-Henning Kamp 		if (!atomic_cmpset_ptr(dp, NULL, dev))
41093bcdfe2SPoul-Henning Kamp 			continue;
41193bcdfe2SPoul-Henning Kamp 
41293bcdfe2SPoul-Henning Kamp 		dev->si_inode = ino;
41393bcdfe2SPoul-Henning Kamp 		for (;;) {
41493bcdfe2SPoul-Henning Kamp 			i = devfs_topino;
41593bcdfe2SPoul-Henning Kamp 			if (i >= ino)
41693bcdfe2SPoul-Henning Kamp 				break;
41793bcdfe2SPoul-Henning Kamp 			if (atomic_cmpset_int(&devfs_topino, i, ino))
41893bcdfe2SPoul-Henning Kamp 				break;
41993bcdfe2SPoul-Henning Kamp 			printf("failed topino %d %d\n", i, ino);
4203f54a085SPoul-Henning Kamp 		}
42193bcdfe2SPoul-Henning Kamp 		break;
42293bcdfe2SPoul-Henning Kamp 	}
42393bcdfe2SPoul-Henning Kamp 
42493bcdfe2SPoul-Henning Kamp 	atomic_add_int(&devfs_numino, 1);
42593bcdfe2SPoul-Henning Kamp 	atomic_add_int(&devfs_generation, 1);
42693bcdfe2SPoul-Henning Kamp 	if (devfs_overflow == NULL && devfs_numino + 100 > NDEVFSINO)
42793bcdfe2SPoul-Henning Kamp 		devfs_attemptoverflow(0);
4283f54a085SPoul-Henning Kamp }
4293f54a085SPoul-Henning Kamp 
4303f54a085SPoul-Henning Kamp static void
431db901281SPoul-Henning Kamp devfs_destroy(dev_t dev)
4323f54a085SPoul-Henning Kamp {
43393bcdfe2SPoul-Henning Kamp 	int ino, i;
43493bcdfe2SPoul-Henning Kamp 
43593bcdfe2SPoul-Henning Kamp 	ino = dev->si_inode;
43693bcdfe2SPoul-Henning Kamp 	dev->si_inode = 0;
43793bcdfe2SPoul-Henning Kamp 	if (ino == 0)
43893bcdfe2SPoul-Henning Kamp 		return;
43993bcdfe2SPoul-Henning Kamp 	if (atomic_cmpset_ptr(devfs_itod(ino), dev, NULL)) {
44093bcdfe2SPoul-Henning Kamp 		atomic_add_int(&devfs_generation, 1);
44193bcdfe2SPoul-Henning Kamp 		atomic_add_int(&devfs_numino, -1);
44293bcdfe2SPoul-Henning Kamp 		i = devfs_nextino;
44393bcdfe2SPoul-Henning Kamp 		if (ino < i)
44493bcdfe2SPoul-Henning Kamp 			atomic_cmpset_int(&devfs_nextino, i, ino);
44593bcdfe2SPoul-Henning Kamp 	}
4463f54a085SPoul-Henning Kamp }
4473f54a085SPoul-Henning Kamp 
44892fef27dSMike Smith static void
44992fef27dSMike Smith devfs_init(void *junk)
45092fef27dSMike Smith {
45192fef27dSMike Smith 	devfs_create_hook = devfs_create;
45292fef27dSMike Smith 	devfs_destroy_hook = devfs_destroy;
45392fef27dSMike Smith 	devfs_present = 1;
454a1dc2096SDima Dorfman 	devfs_rules_init();
45592fef27dSMike Smith }
45692fef27dSMike Smith 
457a7489fe5SMike Smith SYSINIT(devfs, SI_SUB_DEVFS, SI_ORDER_FIRST, devfs_init, NULL);
45892fef27dSMike Smith 
459ab9f3b29SPoul-Henning Kamp #endif
460