2 * Copyright (c) 2000-2014 Apple Inc. All rights reserved.
4 * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
6 * This file contains Original Code and/or Modifications of Original Code
7 * as defined in and that are subject to the Apple Public Source License
8 * Version 2.0 (the 'License'). You may not use this file except in
9 * compliance with the License. The rights granted to you under the License
10 * may not be used to create, or enable the creation or redistribution of,
11 * unlawful or unlicensed copies of an Apple operating system, or to
12 * circumvent, violate, or enable the circumvention or violation of, any
13 * terms of an Apple operating system software license agreement.
15 * Please obtain a copy of the License at
16 * http://www.opensource.apple.com/apsl/ and read it before using this file.
18 * The Original Code and all software distributed under the License are
19 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23 * Please see the License for the specific language governing rights and
24 * limitations under the License.
26 * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
30 * Copyright 1997,1998 Julian Elischer. All rights reserved.
33 * Redistribution and use in source and binary forms, with or without
34 * modification, are permitted provided that the following conditions are
36 * 1. Redistributions of source code must retain the above copyright
37 * notice, this list of conditions and the following disclaimer.
38 * 2. Redistributions in binary form must reproduce the above copyright notice,
39 * this list of conditions and the following disclaimer in the documentation
40 * and/or other materials provided with the distribution.
42 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER ``AS IS'' AND ANY EXPRESS
43 * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
44 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
45 * DISCLAIMED. IN NO EVENT SHALL THE HOLDER OR CONTRIBUTORS BE LIABLE FOR
46 * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
47 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
48 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
49 * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
50 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
51 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
57 * NOTICE: This file was modified by SPARTA, Inc. in 2005 to introduce
58 * support for mandatory and extensible security protections. This notice
59 * is included in support of clause 2.2 (b) of the Apple Public License,
65 * Dieter Siegmund (dieter@apple.com) Thu Apr 8 14:08:19 PDT 1999
66 * - removed mounting of "hidden" mountpoint
67 * - fixed problem in which devnode->dn_vn pointer was not
68 * updated with the vnode returned from checkalias()
69 * - replaced devfs_vntodn() with a macro VTODN()
70 * - rewrote dev_finddir() to not use recursion
71 * - added locking to avoid data structure corruption (DEVFS_(UN)LOCK())
72 * Dieter Siegmund (dieter@apple.com) Wed Jul 14 13:37:59 PDT 1999
73 * - fixed problem with devfs_dntovn() checking the v_id against the
74 * value cached in the device node; a union mount on top of us causes
75 * the v_id to get incremented thus, we would end up returning a new
76 * vnode instead of the existing one that has the mounted_here
77 * field filled in; the net effect was that the filesystem mounted
78 * on top of us would never show up
79 * - added devfs_stats to store how many data structures are actually
83 /* SPLIT_DEVS means each devfs uses a different devnode for the same device */
84 /* Otherwise the same device always ends up at the same vnode even if */
85 /* reached througgh a different devfs instance. The practical difference */
86 /* is that with the same vnode, chmods and chowns show up on all instances of */
89 #define SPLIT_DEVS 1 /* maybe make this an option */
90 /*#define SPLIT_DEVS 1*/
92 #include <sys/param.h>
93 #include <sys/systm.h>
94 #include <sys/kernel.h>
96 #include <sys/malloc.h>
97 #include <sys/mount_internal.h>
99 #include <sys/vnode_internal.h>
101 #include <libkern/OSAtomic.h>
102 #include <os/refcnt.h>
103 #define BSD_KERNEL_PRIVATE 1 /* devfs_make_link() prototype */
105 #include "devfsdefs.h"
108 #include <security/mac_framework.h>
115 typedef struct devfs_vnode_event
{
119 } *devfs_vnode_event_t
;
122 * Size of stack buffer (fast path) for notifications. If
123 * the number of mounts is small, no need to malloc a buffer.
125 #define NUM_STACK_ENTRIES 5
127 typedef struct devfs_event_log
{
130 devfs_vnode_event_t del_entries
;
131 } *devfs_event_log_t
;
134 static void dev_free_hier(devdirent_t
*);
135 static int devfs_propogate(devdirent_t
*, devdirent_t
*, devfs_event_log_t
);
136 static int dev_finddir(const char *, devnode_t
*, int, devnode_t
**, devfs_event_log_t
);
137 static int dev_dup_entry(devnode_t
*, devdirent_t
*, devdirent_t
**, struct devfsmount
*);
138 void devfs_ref_node(devnode_t
*);
139 void devfs_rele_node(devnode_t
*);
140 static void devfs_record_event(devfs_event_log_t
, devnode_t
*, uint32_t);
141 static int devfs_init_event_log(devfs_event_log_t
, uint32_t, devfs_vnode_event_t
);
142 static void devfs_release_event_log(devfs_event_log_t
, int);
143 static void devfs_bulk_notify(devfs_event_log_t
);
144 static devdirent_t
*devfs_make_node_internal(dev_t
, devfstype_t type
, uid_t
, gid_t
, int,
145 int (*clone
)(dev_t dev
, int action
), const char *fmt
, va_list ap
);
148 lck_grp_t
* devfs_lck_grp
;
149 lck_grp_attr_t
* devfs_lck_grp_attr
;
150 lck_attr_t
* devfs_lck_attr
;
151 lck_mtx_t devfs_mutex
;
152 lck_mtx_t devfs_attr_mutex
;
154 os_refgrp_decl(static, devfs_refgrp
, "devfs", NULL
);
156 devdirent_t
* dev_root
= NULL
; /* root of backing tree */
157 struct devfs_stats devfs_stats
; /* hold stats */
159 static ino_t devfs_unique_fileno
= 0;
161 #ifdef HIDDEN_MOUNTPOINT
162 static struct mount
*devfs_hidden_mount
;
163 #endif /* HIDDEN_MOINTPOINT */
165 static int devfs_ready
= 0;
166 static uint32_t devfs_nmountplanes
= 0; /* The first plane is not used for a mount */
168 #define DEVFS_NOCREATE FALSE
169 #define DEVFS_CREATE TRUE
172 * Set up the root directory node in the backing plane
173 * This is happenning before the vfs system has been
174 * set up yet, so be careful about what we reference..
175 * Notice that the ops are by indirection.. as they haven't
177 * DEVFS has a hidden mountpoint that is used as the anchor point
178 * for the internal 'blueprint' version of the dev filesystem tree.
186 devfs_lck_grp_attr
= lck_grp_attr_alloc_init();
187 devfs_lck_grp
= lck_grp_alloc_init("devfs_lock", devfs_lck_grp_attr
);
189 devfs_lck_attr
= lck_attr_alloc_init();
191 lck_mtx_init(&devfs_mutex
, devfs_lck_grp
, devfs_lck_attr
);
192 lck_mtx_init(&devfs_attr_mutex
, devfs_lck_grp
, devfs_lck_attr
);
195 error
= dev_add_entry("root", NULL
, DEV_DIR
, NULL
, NULL
, NULL
, &dev_root
);
199 printf("devfs_sinit: dev_add_entry failed ");
202 #ifdef HIDDEN_MOUNTPOINT
203 MALLOC(devfs_hidden_mount
, struct mount
*, sizeof(struct mount
),
205 bzero(devfs_hidden_mount
, sizeof(struct mount
));
206 mount_lock_init(devfs_hidden_mount
);
207 TAILQ_INIT(&devfs_hidden_mount
->mnt_vnodelist
);
208 TAILQ_INIT(&devfs_hidden_mount
->mnt_workerqueue
);
209 TAILQ_INIT(&devfs_hidden_mount
->mnt_newvnodes
);
211 mac_mount_label_init(devfs_hidden_mount
);
212 mac_mount_label_associate(vfs_context_kernel(), devfs_hidden_mount
);
215 /* Initialize the default IO constraints */
216 mp
->mnt_maxreadcnt
= mp
->mnt_maxwritecnt
= MAXPHYS
;
217 mp
->mnt_segreadcnt
= mp
->mnt_segwritecnt
= 32;
219 mp
->mnt_realrootvp
= NULLVP
;
220 mp
->mnt_authcache_ttl
= CACHED_LOOKUP_RIGHT_TTL
;
222 devfs_mount(devfs_hidden_mount
, "dummy", NULL
, NULL
, NULL
);
223 dev_root
->de_dnp
->dn_dvm
224 = (struct devfsmount
*)devfs_hidden_mount
->mnt_data
;
225 #endif /* HIDDEN_MOUNTPOINT */
227 mac_devfs_label_associate_directory("/", strlen("/"),
228 dev_root
->de_dnp
, "/");
234 /***********************************************************************\
235 *************************************************************************
236 * Routines used to find our way to a point in the tree *
237 *************************************************************************
238 \***********************************************************************/
242 /***************************************************************
243 * Search down the linked list off a dir to find "name"
244 * return the devnode_t * for that node.
246 * called with DEVFS_LOCK held
247 ***************************************************************/
249 dev_findname(devnode_t
* dir
, const char *name
)
252 if (dir
->dn_type
!= DEV_DIR
) {
253 return 0; /*XXX*/ /* printf?*/
255 if (name
[0] == '.') {
257 return dir
->dn_typeinfo
.Dir
.myname
;
259 if ((name
[1] == '.') && (name
[2] == 0)) {
260 /* for root, .. == . */
261 return dir
->dn_typeinfo
.Dir
.parent
->dn_typeinfo
.Dir
.myname
;
264 newfp
= dir
->dn_typeinfo
.Dir
.dirlist
;
267 if (!(strncmp(name
, newfp
->de_name
, sizeof(newfp
->de_name
)))) {
270 newfp
= newfp
->de_next
;
275 /***********************************************************************
276 * Given a starting node (0 for root) and a pathname, return the node
277 * for the end item on the path. It MUST BE A DIRECTORY. If the 'DEVFS_CREATE'
278 * option is true, then create any missing nodes in the path and create
279 * and return the final node as well.
280 * This is used to set up a directory, before making nodes in it..
282 * called with DEVFS_LOCK held
283 ***********************************************************************/
285 dev_finddir(const char * path
,
289 devfs_event_log_t delp
)
291 devnode_t
* dnp
= NULL
;
295 char fullpath
[DEVMAXPATHSIZE
];
299 if (!dirnode
) { /* dirnode == NULL means start at root */
300 dirnode
= dev_root
->de_dnp
;
303 if (dirnode
->dn_type
!= DEV_DIR
) {
307 if (strlen(path
) > (DEVMAXPATHSIZE
- 1)) {
312 strlcpy(fullpath
, path
, DEVMAXPATHSIZE
);
316 while (*scan
== '/') {
323 char component
[DEVMAXPATHSIZE
];
324 devdirent_t
* dirent_p
;
328 /* we hit the end of the string, we're done */
333 while (*scan
!= '/' && *scan
) {
337 strlcpy(component
, start
, (scan
- start
) + 1);
342 dirent_p
= dev_findname(dirnode
, component
);
344 dnp
= dirent_p
->de_dnp
;
345 if (dnp
->dn_type
!= DEV_DIR
) {
354 error
= dev_add_entry(component
, dirnode
,
355 DEV_DIR
, NULL
, NULL
, NULL
, &dirent_p
);
359 dnp
= dirent_p
->de_dnp
;
361 mac_devfs_label_associate_directory(
362 dirnode
->dn_typeinfo
.Dir
.myname
->de_name
,
363 strlen(dirnode
->dn_typeinfo
.Dir
.myname
->de_name
),
366 devfs_propogate(dirnode
->dn_typeinfo
.Dir
.myname
, dirent_p
, delp
);
368 dirnode
= dnp
; /* continue relative to this directory */
374 /***********************************************************************
375 * Add a new NAME element to the devfs
376 * If we're creating a root node, then dirname is NULL
377 * Basically this creates a new namespace entry for the device node
379 * Creates a name node, and links it to the supplied node
381 * called with DEVFS_LOCK held
382 ***********************************************************************/
384 dev_add_name(const char * name
, devnode_t
* dirnode
, __unused devdirent_t
* back
,
385 devnode_t
* dnp
, devdirent_t
* *dirent_pp
)
387 devdirent_t
* dirent_p
= NULL
;
389 if (dirnode
!= NULL
) {
390 if (dirnode
->dn_type
!= DEV_DIR
) {
394 if (dev_findname(dirnode
, name
)) {
399 * make sure the name is legal
400 * slightly misleading in the case of NULL
402 if (!name
|| (strlen(name
) > (DEVMAXNAMESIZE
- 1))) {
407 * Allocate and fill out a new directory entry
409 MALLOC(dirent_p
, devdirent_t
*, sizeof(devdirent_t
),
410 M_DEVFSNAME
, M_WAITOK
);
414 bzero(dirent_p
, sizeof(devdirent_t
));
416 /* inherrit our parent's mount info */ /*XXX*/
417 /* a kludge but.... */
418 if (dirnode
&& (dnp
->dn_dvm
== NULL
)) {
419 dnp
->dn_dvm
= dirnode
->dn_dvm
;
420 /* if(!dnp->dn_dvm) printf("parent had null dvm "); */
424 * Link the two together
425 * include the implicit link in the count of links to the devnode..
426 * this stops it from being accidentally freed later.
428 dirent_p
->de_dnp
= dnp
;
429 dnp
->dn_links
++; /* implicit from our own name-node */
432 * Make sure that we can find all the links that reference a node
433 * so that we can get them all if we need to zap the node.
435 if (dnp
->dn_linklist
) {
436 dirent_p
->de_nextlink
= dnp
->dn_linklist
;
437 dirent_p
->de_prevlinkp
= dirent_p
->de_nextlink
->de_prevlinkp
;
438 dirent_p
->de_nextlink
->de_prevlinkp
= &(dirent_p
->de_nextlink
);
439 *dirent_p
->de_prevlinkp
= dirent_p
;
441 dirent_p
->de_nextlink
= dirent_p
;
442 dirent_p
->de_prevlinkp
= &(dirent_p
->de_nextlink
);
444 dnp
->dn_linklist
= dirent_p
;
447 * If the node is a directory, then we need to handle the
448 * creation of the .. link.
449 * A NULL dirnode indicates a root node, so point to ourself.
451 if (dnp
->dn_type
== DEV_DIR
) {
452 dnp
->dn_typeinfo
.Dir
.myname
= dirent_p
;
454 * If we are unlinking from an old dir, decrement its links
455 * as we point our '..' elsewhere
456 * Note: it's up to the calling code to remove the
457 * us from the original directory's list
459 if (dnp
->dn_typeinfo
.Dir
.parent
) {
460 dnp
->dn_typeinfo
.Dir
.parent
->dn_links
--;
463 dnp
->dn_typeinfo
.Dir
.parent
= dirnode
;
465 dnp
->dn_typeinfo
.Dir
.parent
= dnp
;
467 dnp
->dn_typeinfo
.Dir
.parent
->dn_links
++; /* account for the new '..' */
471 * put the name into the directory entry.
473 strlcpy(dirent_p
->de_name
, name
, DEVMAXNAMESIZE
);
477 * Check if we are not making a root node..
482 * Put it on the END of the linked list of directory entries
484 dirent_p
->de_parent
= dirnode
; /* null for root */
485 dirent_p
->de_prevp
= dirnode
->dn_typeinfo
.Dir
.dirlast
;
486 dirent_p
->de_next
= *(dirent_p
->de_prevp
); /* should be NULL */
488 *(dirent_p
->de_prevp
) = dirent_p
;
489 dirnode
->dn_typeinfo
.Dir
.dirlast
= &(dirent_p
->de_next
);
490 dirnode
->dn_typeinfo
.Dir
.entrycount
++;
491 dirnode
->dn_len
+= strlen(name
) + 8;/*ok, ok?*/
494 *dirent_pp
= dirent_p
;
495 DEVFS_INCR_ENTRIES();
500 /***********************************************************************
501 * Add a new element to the devfs plane.
503 * Creates a new dev_node to go with it if the prototype should not be
504 * reused. (Is a DIR, or we select SPLIT_DEVS at compile time)
505 * typeinfo gives us info to make our node if we don't have a prototype.
506 * If typeinfo is null and proto exists, then the typeinfo field of
507 * the proto is used intead in the DEVFS_CREATE case.
508 * note the 'links' count is 0 (except if a dir)
509 * but it is only cleared on a transition
510 * so this is ok till we link it to something
511 * Even in SPLIT_DEVS mode,
512 * if the node already exists on the wanted plane, just return it
514 * called with DEVFS_LOCK held
515 ***********************************************************************/
517 dev_add_node(int entrytype
, devnode_type_t
* typeinfo
, devnode_t
* proto
,
518 devnode_t
* *dn_pp
, struct devfsmount
*dvm
)
520 devnode_t
* dnp
= NULL
;
523 #if defined SPLIT_DEVS
525 * If we have a prototype, then check if there is already a sibling
526 * on the mount plane we are looking at, if so, just return it.
529 dnp
= proto
->dn_nextsibling
;
530 while (dnp
!= proto
) {
531 if (dnp
->dn_dvm
== dvm
) {
535 dnp
= dnp
->dn_nextsibling
;
537 if (typeinfo
== NULL
) {
538 typeinfo
= &(proto
->dn_typeinfo
);
541 #else /* SPLIT_DEVS */
543 switch (proto
->type
) {
550 #endif /* SPLIT_DEVS */
551 MALLOC(dnp
, devnode_t
*, sizeof(devnode_t
), M_DEVFSNODE
, M_WAITOK
);
557 * If we have a proto, that means that we are duplicating some
558 * other device, which can only happen if we are not at the back plane
561 bcopy(proto
, dnp
, sizeof(devnode_t
));
563 dnp
->dn_linklist
= NULL
;
566 /* add to END of siblings list */
567 dnp
->dn_prevsiblingp
= proto
->dn_prevsiblingp
;
568 *(dnp
->dn_prevsiblingp
) = dnp
;
569 dnp
->dn_nextsibling
= proto
;
570 proto
->dn_prevsiblingp
= &(dnp
->dn_nextsibling
);
572 mac_devfs_label_init(dnp
);
573 mac_devfs_label_copy(proto
->dn_label
, dnp
->dn_label
);
579 * We have no prototype, so start off with a clean slate
582 bzero(dnp
, sizeof(devnode_t
));
583 dnp
->dn_type
= entrytype
;
584 dnp
->dn_nextsibling
= dnp
;
585 dnp
->dn_prevsiblingp
= &(dnp
->dn_nextsibling
);
586 dnp
->dn_atime
.tv_sec
= tv
.tv_sec
;
587 dnp
->dn_mtime
.tv_sec
= tv
.tv_sec
;
588 dnp
->dn_ctime
.tv_sec
= tv
.tv_sec
;
590 mac_devfs_label_init(dnp
);
595 /* Note: this inits the reference count to 1, this is considered unreferenced */
596 os_ref_init_raw(&dnp
->dn_refcount
, &devfs_refgrp
);
597 dnp
->dn_ino
= devfs_unique_fileno
;
598 devfs_unique_fileno
++;
601 * fill out the dev node according to type
606 * As it's a directory, make sure
607 * it has a null entries list
609 dnp
->dn_typeinfo
.Dir
.dirlast
= &(dnp
->dn_typeinfo
.Dir
.dirlist
);
610 dnp
->dn_typeinfo
.Dir
.dirlist
= (devdirent_t
*)0;
611 dnp
->dn_typeinfo
.Dir
.entrycount
= 0;
612 /* until we know better, it has a null parent pointer*/
613 dnp
->dn_typeinfo
.Dir
.parent
= NULL
;
614 dnp
->dn_links
++; /* for .*/
615 dnp
->dn_typeinfo
.Dir
.myname
= NULL
;
617 * make sure that the ops associated with it are the ops
618 * that we use (by default) for directories
620 dnp
->dn_ops
= &devfs_vnodeop_p
;
621 dnp
->dn_mode
|= 0555; /* default perms */
625 * As it's a symlink allocate and store the link info
626 * Symlinks should only ever be created by the user,
627 * so they are not on the back plane and should not be
628 * propogated forward.. a bit like directories in that way..
629 * A symlink only exists on one plane and has its own
630 * node.. therefore we might be on any random plane.
632 MALLOC(dnp
->dn_typeinfo
.Slnk
.name
, char *,
633 typeinfo
->Slnk
.namelen
+ 1,
634 M_DEVFSNODE
, M_WAITOK
);
635 if (!dnp
->dn_typeinfo
.Slnk
.name
) {
639 strlcpy(dnp
->dn_typeinfo
.Slnk
.name
, typeinfo
->Slnk
.name
,
640 typeinfo
->Slnk
.namelen
+ 1);
641 dnp
->dn_typeinfo
.Slnk
.namelen
= typeinfo
->Slnk
.namelen
;
642 DEVFS_INCR_STRINGSPACE(dnp
->dn_typeinfo
.Slnk
.namelen
+ 1);
643 dnp
->dn_ops
= &devfs_vnodeop_p
;
644 dnp
->dn_mode
|= 0555; /* default perms */
649 * Make sure it has DEVICE type ops
650 * and device specific fields are correct
652 dnp
->dn_ops
= &devfs_spec_vnodeop_p
;
653 dnp
->dn_typeinfo
.dev
= typeinfo
->dev
;
657 /* /dev/fd is special */
659 dnp
->dn_ops
= &devfs_devfd_vnodeop_p
;
660 dnp
->dn_mode
|= 0555; /* default perms */
669 FREE(dnp
, M_DEVFSNODE
);
679 /***********************************************************************
680 * called with DEVFS_LOCK held
681 **********************************************************************/
683 devnode_free(devnode_t
* dnp
)
686 mac_devfs_label_destroy(dnp
);
688 if (dnp
->dn_type
== DEV_SLNK
) {
689 DEVFS_DECR_STRINGSPACE(dnp
->dn_typeinfo
.Slnk
.namelen
+ 1);
690 FREE(dnp
->dn_typeinfo
.Slnk
.name
, M_DEVFSNODE
);
693 FREE(dnp
, M_DEVFSNODE
);
697 /***********************************************************************
698 * called with DEVFS_LOCK held
699 **********************************************************************/
701 devfs_dn_free(devnode_t
* dnp
)
703 if (--dnp
->dn_links
<= 0) { /* can be -1 for initial free, on error */
704 /*probably need to do other cleanups XXX */
705 if (dnp
->dn_nextsibling
!= dnp
) {
706 devnode_t
* * prevp
= dnp
->dn_prevsiblingp
;
707 *prevp
= dnp
->dn_nextsibling
;
708 dnp
->dn_nextsibling
->dn_prevsiblingp
= prevp
;
711 /* Can only free if there are no references; otherwise, wait for last vnode to be reclaimed */
712 os_ref_count_t rc
= os_ref_get_count_raw(&dnp
->dn_refcount
);
714 /* release final reference from dev_add_node */
715 (void) os_ref_release_locked_raw(&dnp
->dn_refcount
, &devfs_refgrp
);
718 dnp
->dn_lflags
|= DN_DELETE
;
723 /***********************************************************************\
724 * Front Node Operations *
725 * Add or delete a chain of front nodes *
726 \***********************************************************************/
729 /***********************************************************************
730 * Given a directory backing node, and a child backing node, add the
731 * appropriate front nodes to the front nodes of the directory to
732 * represent the child node to the user
734 * on failure, front nodes will either be correct or not exist for each
735 * front dir, however dirs completed will not be stripped of completed
736 * frontnodes on failure of a later frontnode
738 * This allows a new node to be propogated through all mounted planes
740 * called with DEVFS_LOCK held
741 ***********************************************************************/
743 devfs_propogate(devdirent_t
* parent
, devdirent_t
* child
, devfs_event_log_t delp
)
746 devdirent_t
* newnmp
;
747 devnode_t
* dnp
= child
->de_dnp
;
748 devnode_t
* pdnp
= parent
->de_dnp
;
749 devnode_t
* adnp
= parent
->de_dnp
;
750 int type
= child
->de_dnp
->dn_type
;
753 events
= (dnp
->dn_type
== DEV_DIR
? VNODE_EVENT_DIR_CREATED
: VNODE_EVENT_FILE_CREATED
);
755 devfs_record_event(delp
, pdnp
, events
);
758 /***********************************************
759 * Find the other instances of the parent node
760 ***********************************************/
761 for (adnp
= pdnp
->dn_nextsibling
;
763 adnp
= adnp
->dn_nextsibling
) {
765 * Make the node, using the original as a prototype)
766 * if the node already exists on that plane it won't be
769 if ((error
= dev_add_entry(child
->de_name
, adnp
, type
,
770 NULL
, dnp
, adnp
->dn_dvm
,
772 printf("duplicating %s failed\n", child
->de_name
);
775 devfs_record_event(delp
, adnp
, events
);
778 * Slightly subtle. We're guaranteed that there will
779 * only be a vnode hooked into this devnode if we're creating
780 * a new link to an existing node; otherwise, the devnode is new
781 * and no one can have looked it up yet. If we're making a link,
782 * then the buffer is large enough for two nodes in each
783 * plane; otherwise, there's no vnode and this call will
786 devfs_record_event(delp
, newnmp
->de_dnp
, VNODE_EVENT_LINK
);
790 return 0; /* for now always succeed */
794 remove_notify_count(devnode_t
*dnp
)
796 uint32_t notify_count
= 0;
800 * Could need to notify for one removed node on each mount and
801 * one parent for each such node.
803 notify_count
= devfs_nmountplanes
;
804 notify_count
+= dnp
->dn_links
;
805 for (dnp2
= dnp
->dn_nextsibling
; dnp2
!= dnp
; dnp2
= dnp2
->dn_nextsibling
) {
806 notify_count
+= dnp2
->dn_links
;
812 /***********************************************************************
813 * remove all instances of this devicename [for backing nodes..]
814 * note.. if there is another link to the node (non dir nodes only)
815 * then the devfs_node will still exist as the ref count will be non-0
816 * removing a directory node will remove all sup-nodes on all planes (ZAP)
818 * Used by device drivers to remove nodes that are no longer relevant
819 * The argument is the 'cookie' they were given when they created the node
820 * this function is exported.. see devfs.h
821 ***********************************************************************/
823 devfs_remove(void *dirent_p
)
825 devnode_t
* dnp
= ((devdirent_t
*)dirent_p
)->de_dnp
;
828 struct devfs_event_log event_log
;
829 uint32_t log_count
= 0;
832 struct devfs_vnode_event stackbuf
[NUM_STACK_ENTRIES
];
837 printf("devfs_remove: not ready for devices!\n");
841 log_count
= remove_notify_count(dnp
);
843 if (log_count
> NUM_STACK_ENTRIES
) {
847 if (devfs_init_event_log(&event_log
, log_count
, NULL
) == 0) {
853 new_count
= remove_notify_count(dnp
);
854 if (need_free
&& (new_count
> log_count
)) {
855 devfs_release_event_log(&event_log
, 1);
858 log_count
= log_count
* 2;
862 if (devfs_init_event_log(&event_log
, NUM_STACK_ENTRIES
, &stackbuf
[0]) == 0) {
867 /* This file has been deleted */
868 if (do_notify
!= 0) {
869 devfs_record_event(&event_log
, dnp
, VNODE_EVENT_DELETE
);
872 /* keep removing the next sibling till only we exist. */
873 while ((dnp2
= dnp
->dn_nextsibling
) != dnp
) {
875 * Keep removing the next front node till no more exist
877 dnp
->dn_nextsibling
= dnp2
->dn_nextsibling
;
878 dnp
->dn_nextsibling
->dn_prevsiblingp
= &(dnp
->dn_nextsibling
);
879 dnp2
->dn_nextsibling
= dnp2
;
880 dnp2
->dn_prevsiblingp
= &(dnp2
->dn_nextsibling
);
882 /* This file has been deleted in this plane */
883 if (do_notify
!= 0) {
884 devfs_record_event(&event_log
, dnp2
, VNODE_EVENT_DELETE
);
887 if (dnp2
->dn_linklist
) {
889 lastlink
= (1 == dnp2
->dn_links
);
890 /* Each parent of a link to this file has lost a child in this plane */
891 if (do_notify
!= 0) {
892 devfs_record_event(&event_log
, dnp2
->dn_linklist
->de_parent
, VNODE_EVENT_FILE_REMOVED
);
894 dev_free_name(dnp2
->dn_linklist
);
900 * then free the main node
901 * If we are not running in SPLIT_DEVS mode, then
902 * THIS is what gets rid of the propogated nodes.
904 if (dnp
->dn_linklist
) {
906 lastlink
= (1 == dnp
->dn_links
);
907 /* Each parent of a link to this file has lost a child */
908 if (do_notify
!= 0) {
909 devfs_record_event(&event_log
, dnp
->dn_linklist
->de_parent
, VNODE_EVENT_FILE_REMOVED
);
911 dev_free_name(dnp
->dn_linklist
);
916 if (do_notify
!= 0) {
917 devfs_bulk_notify(&event_log
);
918 devfs_release_event_log(&event_log
, need_free
);
926 /***************************************************************
927 * duplicate the backing tree into a tree of nodes hung off the
928 * mount point given as the argument. Do this by
929 * calling dev_dup_entry which recurses all the way
932 * called with DEVFS_LOCK held
933 **************************************************************/
935 dev_dup_plane(struct devfsmount
*devfs_mp_p
)
940 if ((error
= dev_dup_entry(NULL
, dev_root
, &new, devfs_mp_p
))) {
943 devfs_mp_p
->plane_root
= new;
944 devfs_nmountplanes
++;
950 /***************************************************************
953 * called with DEVFS_LOCK held
954 ***************************************************************/
956 devfs_free_plane(struct devfsmount
*devfs_mp_p
)
958 devdirent_t
* dirent_p
;
960 dirent_p
= devfs_mp_p
->plane_root
;
962 dev_free_hier(dirent_p
);
963 dev_free_name(dirent_p
);
965 devfs_mp_p
->plane_root
= NULL
;
966 devfs_nmountplanes
--;
968 if (devfs_nmountplanes
> (devfs_nmountplanes
+ 1)) {
969 panic("plane count wrapped around.\n");
974 /***************************************************************
975 * Create and link in a new front element..
976 * Parent can be 0 for a root node
977 * Not presently usable to make a symlink XXX
978 * (Ok, symlinks don't propogate)
979 * recursively will create subnodes corresponding to equivalent
980 * child nodes in the base level
982 * called with DEVFS_LOCK held
983 ***************************************************************/
985 dev_dup_entry(devnode_t
* parent
, devdirent_t
* back
, devdirent_t
* *dnm_pp
,
986 struct devfsmount
*dvm
)
988 devdirent_t
* entry_p
= NULL
;
989 devdirent_t
* newback
;
990 devdirent_t
* newfront
;
992 devnode_t
* dnp
= back
->de_dnp
;
993 int type
= dnp
->dn_type
;
996 * go get the node made (if we need to)
997 * use the back one as a prototype
999 error
= dev_add_entry(back
->de_name
, parent
, type
, NULL
, dnp
,
1000 parent
?parent
->dn_dvm
:dvm
, &entry_p
);
1001 if (!error
&& (entry_p
== NULL
)) {
1002 error
= ENOMEM
; /* Really can't happen, but make static analyzer happy */
1005 printf("duplicating %s failed\n", back
->de_name
);
1010 * If we have just made the root, then insert the pointer to the
1014 entry_p
->de_dnp
->dn_dvm
= dvm
;
1018 * If it is a directory, then recurse down all the other
1019 * subnodes in it....
1020 * note that this time we don't pass on the mount info..
1022 if (type
== DEV_DIR
) {
1023 for (newback
= back
->de_dnp
->dn_typeinfo
.Dir
.dirlist
;
1024 newback
; newback
= newback
->de_next
) {
1025 if ((error
= dev_dup_entry(entry_p
->de_dnp
,
1026 newback
, &newfront
, NULL
)) != 0) {
1027 break; /* back out with an error */
1037 /***************************************************************
1039 * remember that if there are other names pointing to the
1040 * dev_node then it may not get freed yet
1041 * can handle if there is no dnp
1043 * called with DEVFS_LOCK held
1044 ***************************************************************/
1047 dev_free_name(devdirent_t
* dirent_p
)
1049 devnode_t
* parent
= dirent_p
->de_parent
;
1050 devnode_t
* dnp
= dirent_p
->de_dnp
;
1053 if (dnp
->dn_type
== DEV_DIR
) {
1056 if (dnp
->dn_typeinfo
.Dir
.dirlist
) {
1059 p
= dnp
->dn_typeinfo
.Dir
.parent
;
1060 devfs_dn_free(dnp
); /* account for '.' */
1061 devfs_dn_free(p
); /* '..' */
1064 * unlink us from the list of links for this node
1065 * If we are the only link, it's easy!
1066 * if we are a DIR of course there should not be any
1069 if (dirent_p
->de_nextlink
== dirent_p
) {
1070 dnp
->dn_linklist
= NULL
;
1072 if (dnp
->dn_linklist
== dirent_p
) {
1073 dnp
->dn_linklist
= dirent_p
->de_nextlink
;
1079 dirent_p
->de_nextlink
->de_prevlinkp
= dirent_p
->de_prevlinkp
;
1080 *(dirent_p
->de_prevlinkp
) = dirent_p
->de_nextlink
;
1083 * unlink ourselves from the directory on this plane
1085 if (parent
) { /* if not fs root */
1086 if ((*dirent_p
->de_prevp
= dirent_p
->de_next
)) {/* yes, assign */
1087 dirent_p
->de_next
->de_prevp
= dirent_p
->de_prevp
;
1089 parent
->dn_typeinfo
.Dir
.dirlast
1090 = dirent_p
->de_prevp
;
1092 parent
->dn_typeinfo
.Dir
.entrycount
--;
1093 parent
->dn_len
-= strlen(dirent_p
->de_name
) + 8;
1096 DEVFS_DECR_ENTRIES();
1097 FREE(dirent_p
, M_DEVFSNAME
);
1102 /***************************************************************
1103 * Free a hierarchy starting at a directory node name
1104 * remember that if there are other names pointing to the
1105 * dev_node then it may not get freed yet
1106 * can handle if there is no dnp
1107 * leave the node itself allocated.
1109 * called with DEVFS_LOCK held
1110 ***************************************************************/
1113 dev_free_hier(devdirent_t
* dirent_p
)
1115 devnode_t
* dnp
= dirent_p
->de_dnp
;
1118 if (dnp
->dn_type
== DEV_DIR
) {
1119 while (dnp
->dn_typeinfo
.Dir
.dirlist
) {
1120 dev_free_hier(dnp
->dn_typeinfo
.Dir
.dirlist
);
1121 dev_free_name(dnp
->dn_typeinfo
.Dir
.dirlist
);
1128 /***************************************************************
1129 * given a dev_node, find the appropriate vnode if one is already
1130 * associated, or get a new one and associate it with the dev_node
1132 * called with DEVFS_LOCK held
1134 * If an error is returned, then the dnp may have been freed (we
1135 * raced with a delete and lost). A devnode should not be accessed
1136 * after devfs_dntovn() fails.
1137 ****************************************************************/
1139 devfs_dntovn(devnode_t
* dnp
, struct vnode
**vn_pp
, __unused
struct proc
* p
)
1143 struct vnode_fsparam vfsp
;
1144 enum vtype vtype
= 0;
1147 int n_minor
= DEVFS_CLONE_ALLOC
; /* new minor number for clone device */
1150 * We should never come in and find that our devnode has been marked for delete.
1151 * The lookup should have held the lock from entry until now; it should not have
1152 * been able to find a removed entry. Any other pathway would have just created
1153 * the devnode and come here without dropping the devfs lock, so no one would
1154 * have a chance to delete.
1156 if (dnp
->dn_lflags
& DN_DELETE
) {
1157 panic("devfs_dntovn: DN_DELETE set on a devnode upon entry.");
1160 devfs_ref_node(dnp
);
1166 if (vn_p
) { /* already has a vnode */
1169 vid
= vnode_vid(vn_p
);
1174 * We want to use the drainok variant of vnode_getwithvid
1175 * because we _don't_ want to get an iocount if the vnode is
1176 * is blocked in vnode_drain as it can cause infinite
1177 * loops in vn_open_auth. While in use vnodes are typically
1178 * only reclaimed on forced unmounts, In use devfs tty vnodes
1179 * can be quite frequently reclaimed by revoke(2) or by the
1180 * exit of a controlling process.
1182 error
= vnode_getwithvid_drainok(vn_p
, vid
);
1186 if (dnp
->dn_lflags
& DN_DELETE
) {
1188 * our BUSY node got marked for
1189 * deletion while the DEVFS lock
1194 * vnode_getwithvid returned a valid ref
1195 * which we need to drop
1201 * This entry is no longer in the namespace. This is only
1202 * possible for lookup: no other path would not find an existing
1203 * vnode. Therefore, ENOENT is a valid result.
1206 } else if (error
== ENODEV
) {
1208 * The Filesystem is getting unmounted.
1211 } else if (error
&& (nretries
< DEV_MAX_VNODE_RETRY
)) {
1213 * If we got an error from vnode_getwithvid, it means
1214 * we raced with a recycle and lost i.e. we asked for
1215 * an iocount only after vnode_drain had been entered
1216 * for the vnode and returned with an error only after
1217 * devfs_reclaim was called on the vnode. devfs_reclaim
1218 * sets dn_vn to NULL but while we were waiting to
1219 * reacquire DEVFS_LOCK, another vnode might have gotten
1220 * associated with the dnp. In either case, we need to
1221 * retry otherwise we will end up returning an ENOENT
1222 * for this lookup but the next lookup will succeed
1223 * because it creates a new vnode (or a racing lookup
1224 * created a new vnode already).
1238 * If we get here, then we've beaten any deletes;
1239 * if someone sets DN_DELETE during a subsequent drop
1240 * of the devfs lock, we'll still vend a vnode.
1243 if (dnp
->dn_lflags
& DN_CREATE
) {
1244 dnp
->dn_lflags
|= DN_CREATEWAIT
;
1245 msleep(&dnp
->dn_lflags
, &devfs_mutex
, PRIBIO
, 0, 0);
1249 dnp
->dn_lflags
|= DN_CREATE
;
1251 switch (dnp
->dn_type
) {
1256 if (dnp
->dn_typeinfo
.Dir
.parent
== dnp
) {
1263 vtype
= (dnp
->dn_type
== DEV_BDEV
) ? VBLK
: VCHR
;
1271 vfsp
.vnfs_mp
= dnp
->dn_dvm
->mount
;
1272 vfsp
.vnfs_vtype
= vtype
;
1273 vfsp
.vnfs_str
= "devfs";
1275 vfsp
.vnfs_fsnode
= dnp
;
1277 vfsp
.vnfs_vops
= *(dnp
->dn_ops
);
1279 if (vtype
== VBLK
|| vtype
== VCHR
) {
1281 * Ask the clone minor number function for a new minor number
1282 * to use for the next device instance. If an administative
1283 * limit has been reached, this function will return -1.
1285 if (dnp
->dn_clone
!= NULL
) {
1286 int n_major
= major(dnp
->dn_typeinfo
.dev
);
1288 n_minor
= (*dnp
->dn_clone
)(dnp
->dn_typeinfo
.dev
, DEVFS_CLONE_ALLOC
);
1289 if (n_minor
== -1) {
1294 vfsp
.vnfs_rdev
= makedev(n_major
, n_minor
);;
1296 vfsp
.vnfs_rdev
= dnp
->dn_typeinfo
.dev
;
1301 vfsp
.vnfs_filesize
= 0;
1302 vfsp
.vnfs_flags
= VNFS_NOCACHE
| VNFS_CANTCACHE
;
1303 /* Tag system files */
1304 vfsp
.vnfs_marksystem
= 0;
1305 vfsp
.vnfs_markroot
= markroot
;
1309 error
= vnode_create(VNCREATE_FLAVOR
, VCREATESIZE
, &vfsp
, &vn_p
);
1311 /* Do this before grabbing the lock */
1313 vnode_setneedinactive(vn_p
);
1319 vnode_settag(vn_p
, VT_DEVFS
);
1321 if ((dnp
->dn_clone
!= NULL
) && (dnp
->dn_vn
!= NULLVP
)) {
1322 panic("devfs_dntovn: cloning device with a vnode?\n");
1328 * Another vnode that has this devnode as its v_data.
1329 * This reference, unlike the one taken at the start
1330 * of the function, persists until a VNOP_RECLAIM
1331 * comes through for this vnode.
1333 devfs_ref_node(dnp
);
1336 * A cloned vnode is not hooked into the devnode; every lookup
1339 if (dnp
->dn_clone
== NULL
) {
1342 } else if (n_minor
!= DEVFS_CLONE_ALLOC
) {
1344 * If we failed the create, we need to release the cloned minor
1345 * back to the free list. In general, this is only useful if
1346 * the clone function results in a state change in the cloned
1347 * device for which the minor number was obtained. If we get
1348 * past this point withouth falling into this case, it's
1349 * assumed that any state to be released will be released when
1350 * the vnode is dropped, instead.
1352 (void)(*dnp
->dn_clone
)(dnp
->dn_typeinfo
.dev
, DEVFS_CLONE_FREE
);
1355 dnp
->dn_lflags
&= ~DN_CREATE
;
1356 if (dnp
->dn_lflags
& DN_CREATEWAIT
) {
1357 dnp
->dn_lflags
&= ~DN_CREATEWAIT
;
1358 wakeup(&dnp
->dn_lflags
);
1363 * Release the reference we took to prevent deletion while we weren't holding the lock.
1364 * If not returning success, then dropping this reference could delete the devnode;
1365 * no one should access a devnode after a call to devfs_dntovn fails.
1367 devfs_rele_node(dnp
);
1373 * Increment refcount on a devnode; prevents free of the node
1374 * while the devfs lock is not held.
1377 devfs_ref_node(devnode_t
*dnp
)
1379 os_ref_retain_locked_raw(&dnp
->dn_refcount
, &devfs_refgrp
);
1383 * Release a reference on a devnode. If the devnode is marked for
1384 * free and the refcount is dropped to one, do the free.
1387 devfs_rele_node(devnode_t
*dnp
)
1389 os_ref_count_t rc
= os_ref_release_locked_raw(&dnp
->dn_refcount
, &devfs_refgrp
);
1391 panic("devfs_rele_node: devnode without a refcount!\n");
1392 } else if ((rc
== 1) && (dnp
->dn_lflags
& DN_DELETE
)) {
1393 /* release final reference from dev_add_node */
1394 (void) os_ref_release_locked_raw(&dnp
->dn_refcount
, &devfs_refgrp
);
1399 /***********************************************************************
1400 * add a whole device, with no prototype.. make name element and node
1401 * Used for adding the original device entries
1403 * called with DEVFS_LOCK held
1404 ***********************************************************************/
1406 dev_add_entry(const char *name
, devnode_t
* parent
, int type
, devnode_type_t
* typeinfo
,
1407 devnode_t
* proto
, struct devfsmount
*dvm
, devdirent_t
* *nm_pp
)
1412 if ((error
= dev_add_node(type
, typeinfo
, proto
, &dnp
,
1413 (parent
?parent
->dn_dvm
:dvm
))) != 0) {
1414 printf("devfs: %s: base node allocation failed (Errno=%d)\n",
1418 if ((error
= dev_add_name(name
, parent
, NULL
, dnp
, nm_pp
)) != 0) {
1419 devfs_dn_free(dnp
); /* 1->0 for dir, 0->(-1) for other */
1420 printf("devfs: %s: name slot allocation failed (Errno=%d)\n",
1427 devfs_bulk_notify(devfs_event_log_t delp
)
1430 for (i
= 0; i
< delp
->del_used
; i
++) {
1431 devfs_vnode_event_t dvep
= &delp
->del_entries
[i
];
1432 if (vnode_getwithvid(dvep
->dve_vp
, dvep
->dve_vid
) == 0) {
1433 vnode_notify(dvep
->dve_vp
, dvep
->dve_events
, NULL
);
1434 vnode_put(dvep
->dve_vp
);
1440 devfs_record_event(devfs_event_log_t delp
, devnode_t
*dnp
, uint32_t events
)
1442 if (delp
->del_used
>= delp
->del_max
) {
1443 panic("devfs event log overflowed.\n");
1446 /* Can only notify for nodes that have an associated vnode */
1447 if (dnp
->dn_vn
!= NULLVP
&& vnode_ismonitored(dnp
->dn_vn
)) {
1448 devfs_vnode_event_t dvep
= &delp
->del_entries
[delp
->del_used
];
1449 dvep
->dve_vp
= dnp
->dn_vn
;
1450 dvep
->dve_vid
= vnode_vid(dnp
->dn_vn
);
1451 dvep
->dve_events
= events
;
1457 devfs_init_event_log(devfs_event_log_t delp
, uint32_t count
, devfs_vnode_event_t buf
)
1459 devfs_vnode_event_t dvearr
;
1462 MALLOC(dvearr
, devfs_vnode_event_t
, count
* sizeof(struct devfs_vnode_event
), M_TEMP
, M_WAITOK
| M_ZERO
);
1463 if (dvearr
== NULL
) {
1470 delp
->del_max
= count
;
1472 delp
->del_entries
= dvearr
;
1477 devfs_release_event_log(devfs_event_log_t delp
, int need_free
)
1479 if (delp
->del_entries
== NULL
) {
1480 panic("Free of devfs notify info that has not been intialized.\n");
1484 FREE(delp
->del_entries
, M_TEMP
);
1487 delp
->del_entries
= NULL
;
1491 * Function: devfs_make_node
1494 * Create a device node with the given pathname in the devfs namespace.
1497 * dev - the dev_t value to associate
1498 * chrblk - block or character device (DEVFS_CHAR or DEVFS_BLOCK)
1499 * uid, gid - ownership
1500 * perms - permissions
1501 * clone - minor number cloning function
1502 * fmt, ... - path format string with printf args to format the path name
1504 * A handle to a device node if successful, NULL otherwise.
1507 devfs_make_node_clone(dev_t dev
, int chrblk
, uid_t uid
,
1508 gid_t gid
, int perms
, int (*clone
)(dev_t dev
, int action
),
1509 const char *fmt
, ...)
1511 devdirent_t
* new_dev
= NULL
;
1527 new_dev
= devfs_make_node_internal(dev
, type
, uid
, gid
, perms
, clone
, fmt
, ap
);
1535 * Function: devfs_make_node
1538 * Create a device node with the given pathname in the devfs namespace.
1541 * dev - the dev_t value to associate
1542 * chrblk - block or character device (DEVFS_CHAR or DEVFS_BLOCK)
1543 * uid, gid - ownership
1544 * perms - permissions
1545 * fmt, ... - path format string with printf args to format the path name
1547 * A handle to a device node if successful, NULL otherwise.
1550 devfs_make_node(dev_t dev
, int chrblk
, uid_t uid
,
1551 gid_t gid
, int perms
, const char *fmt
, ...)
1553 devdirent_t
* new_dev
= NULL
;
1557 if (chrblk
!= DEVFS_CHAR
&& chrblk
!= DEVFS_BLOCK
) {
1561 type
= (chrblk
== DEVFS_BLOCK
? DEV_BDEV
: DEV_CDEV
);
1564 new_dev
= devfs_make_node_internal(dev
, type
, uid
, gid
, perms
, NULL
, fmt
, ap
);
1571 static devdirent_t
*
1572 devfs_make_node_internal(dev_t dev
, devfstype_t type
, uid_t uid
,
1573 gid_t gid
, int perms
, int (*clone
)(dev_t dev
, int action
), const char *fmt
, va_list ap
)
1575 devdirent_t
* new_dev
= NULL
;
1577 devnode_type_t typeinfo
;
1579 char *name
, buf
[256]; /* XXX */
1582 char buff
[sizeof(buf
)];
1586 struct devfs_event_log event_log
;
1587 struct devfs_vnode_event stackbuf
[NUM_STACK_ENTRIES
];
1590 vsnprintf(buf
, sizeof(buf
), fmt
, ap
);
1593 bcopy(buf
, buff
, sizeof(buff
));
1594 buff
[sizeof(buff
) - 1] = 0;
1598 for (i
= strlen(buf
); i
> 0; i
--) {
1599 if (buf
[i
] == '/') {
1614 log_count
= devfs_nmountplanes
;
1615 if (log_count
> NUM_STACK_ENTRIES
) {
1618 if (devfs_init_event_log(&event_log
, log_count
, NULL
) != 0) {
1623 log_count
= NUM_STACK_ENTRIES
;
1624 if (devfs_init_event_log(&event_log
, log_count
, &stackbuf
[0]) != 0) {
1630 if (log_count
< devfs_nmountplanes
) {
1632 devfs_release_event_log(&event_log
, need_free
);
1633 log_count
= log_count
* 2;
1638 printf("devfs_make_node: not ready for devices!\n");
1642 /* find/create directory path ie. mkdir -p */
1643 if (dev_finddir(path
, NULL
, DEVFS_CREATE
, &dnp
, &event_log
) == 0) {
1645 if (dev_add_entry(name
, dnp
, type
, &typeinfo
, NULL
, NULL
, &new_dev
) == 0) {
1646 new_dev
->de_dnp
->dn_gid
= gid
;
1647 new_dev
->de_dnp
->dn_uid
= uid
;
1648 new_dev
->de_dnp
->dn_mode
|= perms
;
1649 new_dev
->de_dnp
->dn_clone
= clone
;
1651 mac_devfs_label_associate_device(dev
, new_dev
->de_dnp
, buff
);
1653 devfs_propogate(dnp
->dn_typeinfo
.Dir
.myname
, new_dev
, &event_log
);
1660 devfs_bulk_notify(&event_log
);
1661 devfs_release_event_log(&event_log
, need_free
);
1666 * Function: devfs_make_link
1669 * Create a link to a previously created device node.
1672 * 0 if successful, -1 if failed
1675 devfs_make_link(void *original
, char *fmt
, ...)
1677 devdirent_t
* new_dev
= NULL
;
1678 devdirent_t
* orig
= (devdirent_t
*) original
;
1679 devnode_t
* dirnode
; /* devnode for parent directory */
1680 struct devfs_event_log event_log
;
1684 char *p
, buf
[256]; /* XXX */
1691 printf("devfs_make_link: not ready for devices!\n");
1697 vsnprintf(buf
, sizeof(buf
), fmt
, ap
);
1702 for (i
= strlen(buf
); i
> 0; i
--) {
1703 if (buf
[i
] == '/') {
1711 * One slot for each directory, one for each devnode
1712 * whose link count changes
1714 log_count
= devfs_nmountplanes
* 2;
1716 if (devfs_init_event_log(&event_log
, log_count
, NULL
) != 0) {
1717 /* No lock held, no allocations done, can just return */
1723 if (log_count
< devfs_nmountplanes
) {
1725 devfs_release_event_log(&event_log
, 1);
1726 log_count
= log_count
* 2;
1733 if (dev_finddir(buf
, NULL
, DEVFS_CREATE
, &dirnode
, &event_log
)
1734 || dev_add_name(p
, dirnode
, NULL
, orig
->de_dnp
, &new_dev
)) {
1738 if (dev_finddir("", NULL
, DEVFS_CREATE
, &dirnode
, &event_log
)
1739 || dev_add_name(buf
, dirnode
, NULL
, orig
->de_dnp
, &new_dev
)) {
1743 devfs_propogate(dirnode
->dn_typeinfo
.Dir
.myname
, new_dev
, &event_log
);
1746 devfs_bulk_notify(&event_log
);
1747 devfs_release_event_log(&event_log
, 1);
1749 return (new_dev
!= NULL
) ? 0 : -1;