/* devfs (Device FileSystem) driver. Copyright (C) 1998-2000 Richard Gooch This library is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with this library; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. Richard Gooch may be reached by email at rgooch@atnf.csiro.au The postal address is: Richard Gooch, c/o ATNF, P. O. Box 76, Epping, N.S.W., 2121, Australia. ChangeLog 19980110 Richard Gooch Original version. v0.1 19980111 Richard Gooch Created per-fs inode table rather than using inode->u.generic_ip v0.2 19980111 Richard Gooch Created .epoch inode which has a ctime of 0. Fixed loss of named pipes when dentries lost. Fixed loss of inode data when devfs_register() follows mknod(). v0.3 19980111 Richard Gooch Fix for when compiling with CONFIG_KERNELD. 19980112 Richard Gooch Fix for readdir() which sometimes didn't show entries. Added <> option to . v0.4 19980113 Richard Gooch Created function. v0.5 19980115 Richard Gooch Added subdirectory support. Major restructuring. 19980116 Richard Gooch Fixed to not search major=0,minor=0. Added symlink support. v0.6 19980120 Richard Gooch Created function and support directory unregister 19980120 Richard Gooch Auto-ownership uses real uid/gid rather than effective uid/gid. v0.7 19980121 Richard Gooch Supported creation of sockets. v0.8 19980122 Richard Gooch Added DEVFS_FL_HIDE_UNREG flag. Interface change to . Created to support symlink(2). v0.9 19980123 Richard Gooch Added check to to check inode is in devfs. Added optional traversal of symlinks. v0.10 19980124 Richard Gooch Created and . v0.11 19980125 C. Scott Ananian Created . 19980125 Richard Gooch Allow removal of symlinks. v0.12 19980125 Richard Gooch Created . 19980126 Richard Gooch Moved DEVFS_SUPER_MAGIC into header file. Added DEVFS_FL_HIDE flag. Created . Created . Fixed minor bug in . 19980127 Richard Gooch Changed interface to , , , and . Fixed inode times when symlink created with symlink(2). v0.13 19980129 C. Scott Ananian Exported , and . 19980129 Richard Gooch Created to support unlink(2). v0.14 19980129 Richard Gooch Fixed kerneld support for entries in devfs subdirectories. 19980130 Richard Gooch Bugfixes in . v0.15 19980207 Richard Gooch Call kerneld when looking up unregistered entries. v0.16 19980326 Richard Gooch Modified interface to for symlink traversal. v0.17 19980331 Richard Gooch Fixed persistence bug with device numbers for manually created device files. Fixed problem with recreating symlinks with different content. v0.18 19980401 Richard Gooch Changed to CONFIG_KMOD. Hide entries which are manually unlinked. Always invalidate devfs dentry cache when registering entries. Created to support rmdir(2). Ensure directories created by are visible. v0.19 19980402 Richard Gooch Invalidate devfs dentry cache when making directories. Invalidate devfs dentry cache when removing entries. Fixed persistence bug with fifos. v0.20 19980421 Richard Gooch Print process command when debugging kerneld/kmod. Added debugging for register/unregister/change operations. 19980422 Richard Gooch Added "devfs=" boot options. v0.21 19980426 Richard Gooch No longer lock/unlock superblock in . Drop negative dentries when they are released. Manage dcache more efficiently. v0.22 19980427 Richard Gooch Added DEVFS_FL_AUTO_DEVNUM flag. v0.23 19980430 Richard Gooch No longer set unnecessary methods. v0.24 19980504 Richard Gooch Added PID display to debugging message. Added "after" debugging message to . 19980519 Richard Gooch Added "diread" and "diwrite" boot options. 19980520 Richard Gooch Fixed persistence problem with permissions. v0.25 19980602 Richard Gooch Support legacy device nodes. Fixed bug where recreated inodes were hidden. v0.26 19980602 Richard Gooch Improved debugging in . 19980607 Richard Gooch No longer free old dentries in . Free all dentries for a given entry when deleting inodes. v0.27 19980627 Richard Gooch Limit auto-device numbering to majors 128 to 239. v0.28 19980629 Richard Gooch Fixed inode times persistence problem. v0.29 19980704 Richard Gooch Fixed spelling in debug. Fixed bug in parsing "dilookup". v0.30 19980705 Richard Gooch Fixed devfs inode leak when manually recreating inodes. Fixed permission persistence problem when recreating inodes. v0.31 19980727 Richard Gooch Removed harmless "unused variable" compiler warning. Fixed modes for manually recreated device nodes. v0.32 19980728 Richard Gooch Added NULL devfs inode warning in . Force all inode nlink values to 1. v0.33 19980730 Richard Gooch Added "dimknod" boot option. Set inode nlink to 0 when freeing dentries. Fixed modes for manually recreated symlinks. v0.34 19980802 Richard Gooch Fixed bugs in recreated directories and symlinks. v0.35 19980806 Richard Gooch Fixed bugs in recreated device nodes. 19980807 Richard Gooch Fixed bug in currently unused . Defined new type. Improved debugging when getting entries. Fixed bug where directories could be emptied. v0.36 19980809 Richard Gooch Replaced dummy .epoch inode with .devfsd character device. 19980810 Richard Gooch Implemented devfsd protocol revision 0. v0.37 19980819 Richard Gooch Added soothing message to warning in . v0.38 19980829 Richard Gooch Use GCC extensions for structure initialisations. Implemented async open notification. Incremented devfsd protocol revision to 1. v0.39 19980908 Richard Gooch Moved async open notification to end of . v0.40 19980910 Richard Gooch Prepended "/dev/" to module load request. Renamed to . v0.41 19980910 Richard Gooch Fixed typo "AYSNC" -> "ASYNC". v0.42 19980910 Richard Gooch Added open flag for files. v0.43 19980927 Richard Gooch Set i_blocks=0 and i_blksize=1024 in . v0.44 19981005 Richard Gooch Added test for empty <> in . Renamed to and published. v0.45 19981006 Richard Gooch Created . v0.46 19981007 Richard Gooch Limit auto-device numbering to majors 144 to 239. v0.47 19981010 Richard Gooch Updated for VFS change in 2.1.125. v0.48 19981022 Richard Gooch Created DEVFS_ FL_COMPAT flag. v0.49 19981023 Richard Gooch Created "nocompat" boot option. v0.50 19981025 Richard Gooch Replaced "mount" boot option with "nomount". v0.51 19981110 Richard Gooch Created "only" boot option. v0.52 19981112 Richard Gooch Added DEVFS_FL_REMOVABLE flag. v0.53 19981114 Richard Gooch Only call on first call to . v0.54 19981205 Richard Gooch Updated for VFS change in 2.1.131. v0.55 19981218 Richard Gooch Created . 19981220 Richard Gooch Check for partitions on removable media in . v0.56 19990118 Richard Gooch Added support for registering regular files. Created . Update devfs inodes from entries if not changed through FS. v0.57 19990124 Richard Gooch Fixed to only initialise temporary inodes. Trap for NULL fops in . Return -ENODEV in for non-driver inodes. v0.58 19990126 Richard Gooch Switched from PATH_MAX to DEVFS_PATHLEN. v0.59 19990127 Richard Gooch Created "nottycompat" boot option. v0.60 19990318 Richard Gooch Fixed to not overrun event buffer. v0.61 19990329 Richard Gooch Created . v0.62 19990330 Richard Gooch Don't return unregistred entries in . Panic in if entry unregistered. 19990401 Richard Gooch Don't panic in for duplicates. v0.63 19990402 Richard Gooch Don't unregister already unregistered entries in . v0.64 19990510 Richard Gooch Disable warning messages when unable to read partition table for removable media. v0.65 19990512 Richard Gooch Updated for VFS change in 2.3.1-pre1. Created "oops-on-panic" boot option. Improved debugging in and . v0.66 19990519 Richard Gooch Added documentation for some functions. 19990525 Richard Gooch Removed "oops-on-panic" boot option: now always Oops. v0.67 19990531 Richard Gooch Improved debugging in . v0.68 19990604 Richard Gooch Added "diunlink" and "nokmod" boot options. Removed superfluous warning message in . v0.69 19990611 Richard Gooch Took account of change to . v0.70 19990614 Richard Gooch Created separate event queue for each mounted devfs. Removed . Created new ioctl()s. Incremented devfsd protocol revision to 3. Fixed bug when re-creating directories: contents were lost. Block access to inodes until devfsd updates permissions. 19990615 Richard Gooch Support 2.2.x kernels. v0.71 19990623 Richard Gooch Switched to sending process uid/gid to devfsd. Renamed to . Added DEVFSD_NOTIFY_LOOKUP event. 19990624 Richard Gooch Added DEVFSD_NOTIFY_CHANGE event. Incremented devfsd protocol revision to 4. v0.72 19990713 Richard Gooch Return EISDIR rather than EINVAL for read(2) on directories. v0.73 19990809 Richard Gooch Changed to new __init scheme. v0.74 19990901 Richard Gooch Changed remaining function declarations to new __init scheme. v0.75 19991013 Richard Gooch Created , , and . Added <> parameter to , , and . Work sponsored by SGI. v0.76 19991017 Richard Gooch Allow multiple unregistrations. Work sponsored by SGI. v0.77 19991026 Richard Gooch Added major and minor number to devfsd protocol. Incremented devfsd protocol revision to 5. Work sponsored by SGI. v0.78 19991030 Richard Gooch Support info pointer for all devfs entry types. Added <> parameter to and . Work sponsored by SGI. v0.79 19991031 Richard Gooch Support "../" when searching devfs namespace. Work sponsored by SGI. v0.80 19991101 Richard Gooch Created . Work sponsored by SGI. v0.81 19991103 Richard Gooch Exported . Work sponsored by SGI. v0.82 19991104 Richard Gooch Removed unused . 19991105 Richard Gooch Do not hide entries from devfsd or children. Removed DEVFS_ FL_TTY_COMPAT flag. Removed "nottycompat" boot option. Removed . Work sponsored by SGI. v0.83 19991107 Richard Gooch Added DEVFS_ FL_WAIT flag. Work sponsored by SGI. v0.84 19991107 Richard Gooch Support new "disc" naming scheme in . Allow NULL fops in . Work sponsored by SGI. v0.85 19991110 Richard Gooch Fall back to major table if NULL fops given to . Work sponsored by SGI. v0.86 19991204 Richard Gooch Support fifos when unregistering. Work sponsored by SGI. v0.87 19991209 Richard Gooch Removed obsolete DEVFS_ FL_COMPAT and DEVFS_ FL_TOLERANT flags. Work sponsored by SGI. v0.88 19991214 Richard Gooch Removed kmod support. Work sponsored by SGI. v0.89 19991216 Richard Gooch Improved debugging in . Ensure dentries created by devfsd will be cleaned up. Work sponsored by SGI. v0.90 19991223 Richard Gooch Created . Work sponsored by SGI. v0.91 20000203 Richard Gooch Ported to kernel 2.3.42. Removed . Work sponsored by SGI. v0.92 */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define DEVFS_VERSION "0.92 (20000203)" #ifndef DEVFS_NAME # define DEVFS_NAME "devfs" #endif /* Compatibility for 2.2.x kernel series */ #if (LINUX_VERSION_CODE < KERNEL_VERSION(2,3,1)) # define init_waitqueue_head(p) init_waitqueue(p) # define DECLARE_WAITQUEUE(wait, p) struct wait_queue wait = {p, NULL} typedef struct wait_queue *wait_queue_head_t; #endif #if (LINUX_VERSION_CODE < KERNEL_VERSION(2,3,6)) # define D_ALLOC_ROOT(inode) d_alloc_root (inode, NULL) #else # define D_ALLOC_ROOT(inode) d_alloc_root (inode) #endif #if (LINUX_VERSION_CODE < KERNEL_VERSION(2,3,13)) # define SETUP_STATIC # define __setup(a,b) #else # define SETUP_STATIC static #endif #define INODE_TABLE_INC 250 #define FIRST_INODE 1 #define STRING_LENGTH 256 #define MIN_DEVNUM 36864 /* Use major numbers 144 */ #define MAX_DEVNUM 61439 /* through 239, inclusive */ #ifndef TRUE # define TRUE 1 # define FALSE 0 #endif #define IS_HIDDEN(de) (( ((de)->hide && !is_devfsd_or_child(fs_info)) || (!(de)->registered&& !(de)->show_unreg))) #define DEBUG_NONE 0x00000 #define DEBUG_MODULE_LOAD 0x00001 #define DEBUG_REGISTER 0x00002 #define DEBUG_UNREGISTER 0x00004 #define DEBUG_SET_FLAGS 0x00008 #define DEBUG_S_PUT 0x00010 #define DEBUG_I_LOOKUP 0x00020 #define DEBUG_I_CREATE 0x00040 #define DEBUG_I_READ 0x00080 #define DEBUG_I_WRITE 0x00100 #define DEBUG_I_UNLINK 0x00200 #define DEBUG_I_RLINK 0x00400 #define DEBUG_I_FLINK 0x00800 #define DEBUG_I_MKNOD 0x01000 #define DEBUG_F_READDIR 0x02000 #define DEBUG_D_DELETE 0x04000 #define DEBUG_D_RELEASE 0x08000 #define DEBUG_D_IPUT 0x10000 #define DEBUG_ALL (DEBUG_MODULE_LOAD | DEBUG_REGISTER | \ DEBUG_SET_FLAGS | DEBUG_I_LOOKUP | \ DEBUG_I_UNLINK | DEBUG_I_MKNOD | \ DEBUG_D_RELEASE | DEBUG_D_IPUT) #define DEBUG_DISABLED DEBUG_NONE #define OPTION_NONE 0x00 #define OPTION_SHOW 0x01 #define OPTION_NOMOUNT 0x02 #define OPTION_ONLY 0x04 #define OOPS(format, args...) {printk (format, ## args); \ printk ("Forcing Oops\n"); \ *(int *) 0 = 0;} struct directory_type { struct devfs_entry *first; struct devfs_entry *last; unsigned int num_removable; }; struct file_type { unsigned long size; }; struct device_type { unsigned short major; unsigned short minor; }; struct fcb_type /* File, char, block type */ { uid_t default_uid; gid_t default_gid; void *ops; union { struct file_type file; struct device_type device; } u; unsigned char auto_owner:1; unsigned char aopen_notify:1; unsigned char removable:1; /* Belongs in device_type, but save space */ unsigned char open:1; /* Not entirely correct */ }; struct symlink_type { unsigned int length; /* Not including the NULL-termimator */ char *linkname; /* This is NULL-terminated */ }; struct fifo_type { uid_t uid; gid_t gid; }; struct devfs_entry { void *info; union { struct directory_type dir; struct fcb_type fcb; struct symlink_type symlink; struct fifo_type fifo; } u; struct devfs_entry *prev; /* Previous entry in the parent directory */ struct devfs_entry *next; /* Next entry in the parent directory */ struct devfs_entry *parent; /* The parent directory */ struct devfs_entry *slave; /* Another entry to unregister */ struct devfs_inode *first_inode; struct devfs_inode *last_inode; umode_t mode; unsigned short namelen; /* I think 64k+ filenames are a way off... */ unsigned char registered:1; unsigned char show_unreg:1; unsigned char hide:1; char name[1]; /* This is just a dummy: the allocated array is bigger. This is NULL-terminated */ }; /* The root of the device tree */ static struct devfs_entry *root_entry = NULL; struct devfs_inode /* This structure is for "persistent" inode storage */ { time_t atime; time_t mtime; time_t ctime; unsigned int ino; /* Inode number as seen in the VFS */ struct devfs_entry *de; struct fs_info *fs_info; struct devfs_inode *prev; /* This pair are used to associate a list of */ struct devfs_inode *next; /* inodes (one per FS) for a devfs entry */ struct dentry *dentry; #ifdef CONFIG_DEVFS_TUNNEL struct dentry *covered; #endif umode_t mode; uid_t uid; gid_t gid; nlink_t nlink; }; struct devfsd_buf_entry { void *data; unsigned int type; umode_t mode; uid_t uid; gid_t gid; }; struct fs_info /* This structure is for each mounted devfs */ { unsigned int num_inodes; /* Number of inodes created */ unsigned int table_size; /* Size of the inode pointer table */ struct devfs_inode **table; struct super_block *sb; volatile struct devfsd_buf_entry *devfsd_buffer; volatile unsigned int devfsd_buf_in; volatile unsigned int devfsd_buf_out; volatile int devfsd_sleeping; volatile int devfsd_buffer_in_use; volatile struct task_struct *devfsd_task; volatile struct file *devfsd_file; volatile unsigned long devfsd_event_mask; atomic_t devfsd_overrun_count; wait_queue_head_t devfsd_wait_queue; wait_queue_head_t revalidate_wait_queue; struct fs_info *prev; struct fs_info *next; unsigned char require_explicit:1; }; static struct fs_info *first_fs = NULL; static struct fs_info *last_fs = NULL; static unsigned int next_devnum_char = MIN_DEVNUM; static unsigned int next_devnum_block = MIN_DEVNUM; static const int devfsd_buf_size = PAGE_SIZE / sizeof(struct devfsd_buf_entry); #ifdef CONFIG_DEVFS_DEBUG # ifdef MODULE unsigned int devfs_debug = DEBUG_NONE; # else static unsigned int devfs_debug_init __initdata = DEBUG_NONE; static unsigned int devfs_debug = DEBUG_NONE; # endif #endif static unsigned int boot_options = OPTION_NONE; /* Forward function declarations */ static struct devfs_entry *search_for_entry (struct devfs_entry *dir, const char *name, unsigned int namelen, int mkdir, int mkfile, int *is_new, int traverse_symlink); static ssize_t devfsd_read (struct file *file, char *buf, size_t len, loff_t *ppos); static int devfsd_ioctl (struct inode *inode, struct file *file, unsigned int cmd, unsigned long arg); static int devfsd_close (struct inode *inode, struct file *file); /* Devfs daemon file operations */ static struct file_operations devfsd_fops = { read: devfsd_read, ioctl: devfsd_ioctl, release: devfsd_close, }; /* Support functions follow */ static struct devfs_entry *search_for_entry_in_dir (struct devfs_entry *parent, const char *name, unsigned int namelen, int traverse_symlink) /* [SUMMARY] Search for a devfs entry inside another devfs entry. The parent devfs entry. The name of the entry. The number of characters in <>. If TRUE then the entry is traversed if it is a symlink. [RETURNS] A pointer to the entry on success, else NULL. */ { struct devfs_entry *curr; if ( !S_ISDIR (parent->mode) ) { printk ("%s: entry is not a directory\n", DEVFS_NAME); return NULL; } for (curr = parent->u.dir.first; curr != NULL; curr = curr->next) { if (curr->namelen != namelen) continue; if (memcmp (curr->name, name, namelen) == 0) break; /* Not found: try the next one */ } if (curr == NULL) return NULL; if (!S_ISLNK (curr->mode) || !traverse_symlink) return curr; /* Need to follow the link: this is a stack chomper */ return search_for_entry (parent, curr->u.symlink.linkname, curr->u.symlink.length, FALSE, FALSE, NULL, TRUE); } /* End Function search_for_entry_in_dir */ static struct devfs_entry *create_entry (struct devfs_entry *parent, const char *name,unsigned int namelen) { struct devfs_entry *new; if ( name && (namelen < 1) ) namelen = strlen (name); if ( ( new = kmalloc (sizeof *new + namelen, GFP_KERNEL) ) == NULL ) return NULL; memset (new, 0, sizeof *new + namelen); new->parent = parent; if (name) memcpy (new->name, name, namelen); new->namelen = namelen; if (parent == NULL) return new; new->prev = parent->u.dir.last; /* Insert into the parent directory's list of children */ if (parent->u.dir.first == NULL) parent->u.dir.first = new; else parent->u.dir.last->next = new; parent->u.dir.last = new; return new; } /* End Function create_entry */ static struct devfs_entry *get_root_entry (void) /* [SUMMARY] Get the root devfs entry. [RETURNS] The root devfs entry on success, else NULL. */ { struct devfs_entry *new; /* Always ensure the root is created */ if (root_entry != NULL) return root_entry; if ( ( root_entry = create_entry (NULL, NULL, 0) ) == NULL ) return NULL; root_entry->registered = TRUE; root_entry->mode = S_IFDIR; /* And create the entry for ".devfsd" */ if ( ( new = create_entry (root_entry, ".devfsd", 0) ) == NULL ) return NULL; new->registered = TRUE; new->u.fcb.u.device.major = next_devnum_char >> 8; new->u.fcb.u.device.minor = next_devnum_char & 0xff; ++next_devnum_char; new->mode = S_IFCHR | S_IRUSR | S_IWUSR; new->u.fcb.default_uid = 0; new->u.fcb.default_gid = 0; new->u.fcb.ops = &devfsd_fops; return root_entry; } /* End Function get_root_entry */ static struct devfs_entry *search_for_entry (struct devfs_entry *dir, const char *name, unsigned int namelen, int mkdir, int mkfile, int *is_new, int traverse_symlink) /* [SUMMARY] Search for an entry in the devfs tree. The parent directory to search from. If this is NULL the root is used The name of the entry. The number of characters in <>. If TRUE intermediate directories are created as needed. If TRUE the file entry is created if it doesn't exist. If the returned entry was newly made, TRUE is written here. If this is NULL nothing is written here. If TRUE then symbolic links are traversed. [NOTE] If the entry is created, then it will be in the unregistered state. [RETURNS] A pointer to the entry on success, else NULL. */ { int len; const char *subname, *stop, *ptr; struct devfs_entry *entry; if (is_new) *is_new = FALSE; if (dir == NULL) dir = get_root_entry (); if (dir == NULL) return NULL; /* Extract one filename component */ subname = name; stop = name + namelen; while (subname < stop) { /* Search for a possible '/' */ for (ptr = subname; (ptr < stop) && (*ptr != '/'); ++ptr); if (ptr >= stop) { /* Look for trailing component */ len = stop - subname; entry = search_for_entry_in_dir (dir, subname, len, traverse_symlink); if (entry != NULL) return entry; if (!mkfile) return NULL; entry = create_entry (dir, subname, len); if (entry && is_new) *is_new = TRUE; return entry; } /* Found '/': search for directory */ if (strncmp (subname, "../", 3) == 0) { /* Going up */ dir = dir->parent; if (dir == NULL) return NULL; /* Cannot escape from devfs */ subname += 3; continue; } len = ptr - subname; entry = search_for_entry_in_dir (dir, subname, len, traverse_symlink); if (!entry && !mkdir) return NULL; if (entry == NULL) { /* Make it */ if ( ( entry = create_entry (dir, subname, len) ) == NULL ) return NULL; entry->mode = S_IFDIR | S_IRUGO | S_IXUGO | S_IWUSR; if (is_new) *is_new = TRUE; } if ( !S_ISDIR (entry->mode) ) { printk ("%s: existing non-directory entry\n", DEVFS_NAME); return NULL; } /* Ensure an unregistered entry is re-registered and visible */ entry->registered = TRUE; entry->hide = FALSE; subname = ptr + 1; dir = entry; } return NULL; } /* End Function search_for_entry */ static struct devfs_entry *find_by_dev (struct devfs_entry *dir, unsigned int major, unsigned int minor, char type) /* [SUMMARY] Find a devfs entry in a directory. The major number to search for. The minor number to search for. The type of special file to search for. This may be either DEVFS_SPECIAL_CHR or DEVFS_SPECIAL_BLK. [RETURNS] The devfs_entry pointer on success, else NULL. */ { struct devfs_entry *entry, *de; if (dir == NULL) return NULL; if ( !S_ISDIR (dir->mode) ) { printk ("%s: find_by_dev(): not a directory\n", DEVFS_NAME); return NULL; } /* First search files in this directory */ for (entry = dir->u.dir.first; entry != NULL; entry = entry->next) { if ( !S_ISCHR (entry->mode) && !S_ISBLK (entry->mode) ) continue; if ( S_ISCHR (entry->mode) && (type != DEVFS_SPECIAL_CHR) ) continue; if ( S_ISBLK (entry->mode) && (type != DEVFS_SPECIAL_BLK) ) continue; if ( (entry->u.fcb.u.device.major == major) && (entry->u.fcb.u.device.minor == minor) ) return entry; /* Not found: try the next one */ } /* Now recursively search the subdirectories: this is a stack chomper */ for (entry = dir->u.dir.first; entry != NULL; entry = entry->next) { if ( !S_ISDIR (entry->mode) ) continue; de = find_by_dev (entry, major, minor, type); if (de) return de; } return NULL; } /* End Function find_by_dev */ static struct devfs_entry *find_entry (devfs_handle_t dir, const char *name, unsigned int namelen, unsigned int major, unsigned int minor, char type, int traverse_symlink) /* [SUMMARY] Find a devfs entry. The handle to the parent devfs directory entry. If this is NULL the name is relative to the root of the devfs. The name of the entry. This is ignored if <> is not NULL. The number of characters in <>, not including a NULL terminator. If this is 0, then <> must be NULL-terminated and the length is computed internally. The major number. This is used if <> and <> are NULL. The minor number. This is used if <> and <> are NULL. [NOTE] If <> and <> are both 0, searching by major and minor numbers is disabled. The type of special file to search for. This may be either DEVFS_SPECIAL_CHR or DEVFS_SPECIAL_BLK. If TRUE then symbolic links are traversed. [RETURNS] The devfs_entry pointer on success, else NULL. */ { struct devfs_entry *entry; if (name != NULL) { if (namelen < 1) namelen = strlen (name); if (name[0] == '/') { /* Skip leading pathname component */ if (namelen < 2) { printk ("%s: find_entry(%s): too short\n", DEVFS_NAME, name); return NULL; } for (++name, --namelen; (*name != '/') && (namelen > 0); ++name, --namelen); if (namelen < 2) { printk ("%s: find_entry(%s): too short\n", DEVFS_NAME, name); return NULL; } ++name; --namelen; } entry = search_for_entry (dir, name, namelen, TRUE, FALSE, NULL, traverse_symlink); if (entry != NULL) return entry; } /* Have to search by major and minor: slow */ if ( (major == 0) && (minor == 0) ) return NULL; return find_by_dev (root_entry, major, minor, type); } /* End Function find_entry */ static struct devfs_inode *get_devfs_inode_from_vfs_inode (struct inode *inode) { struct fs_info *fs_info; if (inode == NULL) return NULL; if (inode->i_ino < FIRST_INODE) return NULL; fs_info = inode->i_sb->u.generic_sbp; if (fs_info == NULL) return NULL; if (inode->i_ino - FIRST_INODE >= fs_info->num_inodes) return NULL; return fs_info->table[inode->i_ino - FIRST_INODE]; } /* End Function get_devfs_inode_from_vfs_inode */ static void free_dentries (struct devfs_entry *de) /* [SUMMARY] Free the dentries for a device entry and invalidate inodes. The entry. [RETURNS] Nothing. */ { struct devfs_inode *di; struct dentry *dentry; for (di = de->first_inode; di != NULL; di = di->next) { dentry = di->dentry; if (dentry != NULL) { dget (dentry); di->dentry = NULL; /* Forcefully remove the inode */ if (dentry->d_inode != NULL) dentry->d_inode->i_nlink = 0; d_drop (dentry); dput (dentry); } } } /* End Function free_dentries */ static int is_devfsd_or_child (struct fs_info *fs_info) /* [SUMMARY] Test if the current process is devfsd or one of its children. The filesystem information. [RETURNS] TRUE if devfsd or child, else FALSE. */ { struct task_struct *p; for (p = current; p != &init_task; p = p->p_opptr) { if (p == fs_info->devfsd_task) return (TRUE); } return (FALSE); } /* End Function is_devfsd_or_child */ static inline int devfsd_queue_empty (struct fs_info *fs_info) /* [SUMMARY] Test if devfsd has work pending in its event queue. The filesystem information. [RETURNS] TRUE if the queue is empty, else FALSE. */ { return (fs_info->devfsd_buf_out == fs_info->devfsd_buf_in) ? TRUE : FALSE; } /* End Function devfsd_queue_empty */ static int wait_for_devfsd_finished (struct fs_info *fs_info) /* [SUMMARY] Wait for devfsd to finish processing its event queue. The filesystem information. [RETURNS] TRUE if no more waiting will be required, else FALSE. */ { DECLARE_WAITQUEUE (wait, current); if (fs_info->devfsd_task == NULL) return (TRUE); if (devfsd_queue_empty (fs_info) && fs_info->devfsd_sleeping) return TRUE; if ( is_devfsd_or_child (fs_info) ) return (FALSE); add_wait_queue (&fs_info->revalidate_wait_queue, &wait); current->state = TASK_UNINTERRUPTIBLE; if (!devfsd_queue_empty (fs_info) || !fs_info->devfsd_sleeping) if (fs_info->devfsd_task) schedule(); remove_wait_queue (&fs_info->revalidate_wait_queue, &wait); current->state = TASK_RUNNING; return (TRUE); } /* End Function wait_for_devfsd_finished */ static int devfsd_notify_one (void *data, unsigned int type, umode_t mode, uid_t uid, gid_t gid, struct fs_info *fs_info) /* [SUMMARY] Notify a single devfsd daemon of a change. Data to be passed. The type of change. The mode of the entry. The user ID. The group ID. The filesystem info. [RETURNS] TRUE if an event was queued and devfsd woken up, else FALSE. */ { unsigned int next_pos; unsigned long flags; struct devfsd_buf_entry *entry; static spinlock_t lock = SPIN_LOCK_UNLOCKED; if ( !( fs_info->devfsd_event_mask & (1 << type) ) ) return (FALSE); next_pos = fs_info->devfsd_buf_in + 1; if (next_pos >= devfsd_buf_size) next_pos = 0; if (next_pos == fs_info->devfsd_buf_out) { /* Running up the arse of the reader: drop it */ atomic_inc (&fs_info->devfsd_overrun_count); return (FALSE); } spin_lock_irqsave (&lock, flags); fs_info->devfsd_buffer_in_use = TRUE; next_pos = fs_info->devfsd_buf_in + 1; if (next_pos >= devfsd_buf_size) next_pos = 0; entry = (struct devfsd_buf_entry *) fs_info->devfsd_buffer + fs_info->devfsd_buf_in; entry->data = data; entry->type = type; entry->mode = mode; entry->uid = uid; entry->gid = gid; fs_info->devfsd_buf_in = next_pos; fs_info->devfsd_buffer_in_use = FALSE; spin_unlock_irqrestore (&lock, flags); wake_up_interruptible (&fs_info->devfsd_wait_queue); return (TRUE); } /* End Function devfsd_notify_one */ static void devfsd_notify (struct devfs_entry *de, unsigned int type, int wait) /* [SUMMARY] Notify all devfsd daemons of a change. The devfs entry that has changed. The type of change event. If TRUE, the functions waits for all daemons to finish processing the event. [RETURNS] Nothing. */ { struct fs_info *fs_info; for (fs_info = first_fs; fs_info != NULL; fs_info = fs_info->next) { if (devfsd_notify_one (de, type, de->mode, current->euid, current->egid, fs_info) && wait) wait_for_devfsd_finished (fs_info); } } /* End Function devfsd_notify */ /*PUBLIC_FUNCTION*/ devfs_handle_t devfs_register (devfs_handle_t dir, const char *name, unsigned int namelen, unsigned int flags, unsigned int major, unsigned int minor, umode_t mode, uid_t uid, gid_t gid, void *ops, void *info) /* [SUMMARY] Register a device entry. The handle to the parent devfs directory entry. If this is NULL the new name is relative to the root of the devfs. The name of the entry. The number of characters in <>, not including a NULL terminator. If this is 0, then <> must be NULL-terminated and the length is computed internally. A set of bitwise-ORed flags (DEVFS_FL_*). The major number. Not needed for regular files. The minor number. Not needed for regular files. The default file mode. The default UID of the file. The default GID of the file. The <> or <> structure. This must not be externally deallocated. An arbitrary pointer which will be written to the <> field of the <> structure passed to the device driver. You can set this to whatever you like, and change it once the file is opened (the next file opened will not see this change). [RETURNS] A handle which may later be used in a call to []. On failure NULL is returned. */ { int is_new; struct devfs_entry *de; if (name == NULL) { printk ("%s: devfs_register(): NULL name pointer\n", DEVFS_NAME); return NULL; } if (ops == NULL) { if ( S_ISCHR (mode) ) ops = get_chrfops (major, 0); else if ( S_ISBLK (mode) ) ops = (void *) get_blkfops (major); if (ops == NULL) { printk ("%s: devfs_register(%s): NULL ops pointer\n", DEVFS_NAME, name); return NULL; } printk ("%s: devfs_register(%s): NULL ops, got %p from major table\n", DEVFS_NAME, name, ops); } if ( S_ISDIR (mode) ) { printk("%s: devfs_register(%s): creating directories is not allowed\n", DEVFS_NAME, name); return NULL; } if ( S_ISLNK (mode) ) { printk ("%s: devfs_register(%s): creating symlinks is not allowed\n", DEVFS_NAME, name); return NULL; } if (namelen < 1) namelen = strlen (name); if ( S_ISCHR (mode) && (flags & DEVFS_FL_AUTO_DEVNUM) ) { if (next_devnum_char >= MAX_DEVNUM) { printk ("%s: devfs_register(%s): exhausted char device numbers\n", DEVFS_NAME, name); return NULL; } major = next_devnum_char >> 8; minor = next_devnum_char & 0xff; ++next_devnum_char; } if ( S_ISBLK (mode) && (flags & DEVFS_FL_AUTO_DEVNUM) ) { if (next_devnum_block >= MAX_DEVNUM) { printk ("%s: devfs_register(%s): exhausted block device numbers\n", DEVFS_NAME, name); return NULL; } major = next_devnum_block >> 8; minor = next_devnum_block & 0xff; ++next_devnum_block; } de = search_for_entry (dir, name, namelen, TRUE, TRUE, &is_new, FALSE); if (de == NULL) { printk ("%s: devfs_register(): could not create entry: \"%s\"\n", DEVFS_NAME, name); return NULL; } #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_REGISTER) printk ("%s: devfs_register(%s): de: %p %s\n", DEVFS_NAME, name, de, is_new ? "new" : "existing"); #endif if (!is_new) { /* Existing entry */ if ( !S_ISCHR (de->mode) && !S_ISBLK (de->mode) && !S_ISREG (de->mode) ) { printk ("%s: devfs_register(): existing non-device/file entry: \"%s\"\n", DEVFS_NAME, name); return NULL; } if (de->registered) { printk("%s: devfs_register(): device already registered: \"%s\"\n", DEVFS_NAME, name); return NULL; } /* If entry already exists free any dentries associated with it */ if (de->registered) free_dentries (de); } de->registered = TRUE; if ( S_ISCHR (mode) || S_ISBLK (mode) ) { de->u.fcb.u.device.major = major; de->u.fcb.u.device.minor = minor; } else if ( S_ISREG (mode) ) de->u.fcb.u.file.size = 0; else { printk ("%s: devfs_register(): illegal mode: %x\n", DEVFS_NAME, mode); return (NULL); } de->info = info; de->mode = mode; de->u.fcb.default_uid = uid; de->u.fcb.default_gid = gid; de->registered = TRUE; de->u.fcb.ops = ops; de->u.fcb.auto_owner = (flags & DEVFS_FL_AUTO_OWNER) ? TRUE : FALSE; de->u.fcb.aopen_notify = (flags & DEVFS_FL_AOPEN_NOTIFY) ? TRUE : FALSE; if (flags & DEVFS_FL_REMOVABLE) { de->u.fcb.removable = TRUE; ++de->parent->u.dir.num_removable; } de->u.fcb.open = FALSE; de->show_unreg = ( (boot_options & OPTION_SHOW) || (flags & DEVFS_FL_SHOW_UNREG) ) ? TRUE : FALSE; de->hide = (flags & DEVFS_FL_HIDE) ? TRUE : FALSE; devfsd_notify (de, DEVFSD_NOTIFY_REGISTERED, flags & DEVFS_FL_WAIT); return de; } /* End Function devfs_register */ static void unregister (struct devfs_entry *de) /* [SUMMARY] Unregister a device entry. The entry to unregister. [RETURNS] Nothing. */ { struct devfs_entry *child; if ( (child = de->slave) != NULL ) { de->slave = NULL; /* Unhook first in case slave is parent directory */ unregister (child); } if (de->registered) { devfsd_notify (de, DEVFSD_NOTIFY_UNREGISTERED, 0); free_dentries (de); } de->info = NULL; if ( S_ISCHR (de->mode) || S_ISBLK (de->mode) || S_ISREG (de->mode) ) { de->registered = FALSE; de->u.fcb.ops = NULL; return; } if ( S_ISLNK (de->mode) ) { de->registered = FALSE; if (de->u.symlink.linkname != NULL) kfree (de->u.symlink.linkname); de->u.symlink.linkname = NULL; return; } if ( S_ISFIFO (de->mode) ) { de->registered = FALSE; return; } if (!de->registered) return; if ( !S_ISDIR (de->mode) ) { printk ("%s: unregister(): unsupported type\n", DEVFS_NAME); return; } de->registered = FALSE; /* Now recursively search the subdirectories: this is a stack chomper */ for (child = de->u.dir.first; child != NULL; child = child->next) { #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_UNREGISTER) printk ("%s: unregister(): child->name: \"%s\" child: %p\n", DEVFS_NAME, child->name, child); #endif unregister (child); } } /* End Function unregister */ /*PUBLIC_FUNCTION*/ void devfs_unregister (devfs_handle_t de) /* [SUMMARY] Unregister a device entry. A handle previously created by [] or returned from []. If this is NULL the routine does nothing. [RETURNS] Nothing. */ { if (de == NULL) return; #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_UNREGISTER) printk ("%s: devfs_unregister(): de->name: \"%s\" de: %p\n", DEVFS_NAME, de->name, de); #endif unregister (de); } /* End Function devfs_unregister */ /*PUBLIC_FUNCTION*/ int devfs_mk_symlink (devfs_handle_t dir, const char *name, unsigned int namelen, unsigned int flags, const char *link, unsigned int linklength, devfs_handle_t *handle, void *info) /* [SUMMARY] Create a symbolic link in the devfs namespace. The handle to the parent devfs directory entry. If this is NULL the new name is relative to the root of the devfs. The name of the entry. The number of characters in <>, not including a NULL terminator. If this is 0, then <> must be NULL-terminated and the length is computed internally. A set of bitwise-ORed flags (DEVFS_FL_*). The destination name. The number of characters in <>, not including a NULL terminator. If this is 0, then <> must be NULL-terminated and the length is computed internally. The handle to the symlink entry is written here. This may be NULL. An arbitrary pointer which will be associated with the entry. [RETURNS] 0 on success, else a negative error code is returned. */ { int is_new; char *newname; struct devfs_entry *de; if (handle != NULL) *handle = NULL; if (name == NULL) { printk ("%s: devfs_mk_symlink(): NULL name pointer\n", DEVFS_NAME); return -EINVAL; } #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_REGISTER) printk ("%s: devfs_mk_symlink(%s)\n", DEVFS_NAME, name); #endif if (namelen < 1) namelen = strlen (name); if (link == NULL) { printk ("%s: devfs_mk_symlink(): NULL link pointer\n", DEVFS_NAME); return -EINVAL; } if (linklength < 1) linklength = strlen (link); de = search_for_entry (dir, name, namelen, TRUE, TRUE, &is_new, FALSE); if (de == NULL) return -ENOMEM; if (!S_ISLNK (de->mode) && de->registered) { printk ("%s: devfs_mk_symlink(): non-link entry already exists\n", DEVFS_NAME); return -EEXIST; } if (handle != NULL) *handle = de; de->mode = S_IFLNK | S_IRUGO | S_IXUGO; de->info = info; de->show_unreg = ( (boot_options & OPTION_SHOW) || (flags & DEVFS_FL_SHOW_UNREG) ) ? TRUE : FALSE; de->hide = (flags & DEVFS_FL_HIDE) ? TRUE : FALSE; /* Note there is no need to fiddle the dentry cache if the symlink changes as the symlink follow method is called every time it's needed */ if ( de->registered && (linklength == de->u.symlink.length) ) { /* New link is same length as old link */ if (memcmp (link, de->u.symlink.linkname, linklength) == 0) return 0; return -EEXIST; /* Contents would change */ } /* Have to create/update */ if (de->registered) return -EEXIST; de->registered = TRUE; if ( ( newname = kmalloc (linklength + 1, GFP_KERNEL) ) == NULL ) { struct devfs_entry *parent = de->parent; if (!is_new) return -ENOMEM; /* Have to clean up */ if (de->prev == NULL) parent->u.dir.first = de->next; else de->prev->next = de->next; if (de->next == NULL) parent->u.dir.last = de->prev; else de->next->prev = de->prev; kfree (de); return -ENOMEM; } if (de->u.symlink.linkname != NULL) kfree (de->u.symlink.linkname); de->u.symlink.linkname = newname; memcpy (de->u.symlink.linkname, link, linklength); de->u.symlink.linkname[linklength] = '\0'; de->u.symlink.length = linklength; return 0; } /* End Function devfs_mk_symlink */ /*PUBLIC_FUNCTION*/ devfs_handle_t devfs_mk_dir (devfs_handle_t dir, const char *name, unsigned int namelen, void *info) /* [SUMMARY] Create a directory in the devfs namespace. The handle to the parent devfs directory entry. If this is NULL the new name is relative to the root of the devfs. The name of the entry. The number of characters in <>, not including a NULL terminator. If this is 0, then <> must be NULL-terminated and the length is computed internally. An arbitrary pointer which will be associated with the entry. [NOTE] Use of this function is optional. The [] function will automatically create intermediate directories as needed. This function is provided for efficiency reasons, as it provides a handle to a directory. [RETURNS] A handle which may later be used in a call to []. On failure NULL is returned. */ { int is_new; struct devfs_entry *de; if (name == NULL) { printk ("%s: devfs_mk_dir(): NULL name pointer\n", DEVFS_NAME); return NULL; } if (namelen < 1) namelen = strlen (name); de = search_for_entry (dir, name, namelen, TRUE, TRUE, &is_new, FALSE); if (de == NULL) { printk ("%s: devfs_mk_dir(): could not create entry: \"%s\"\n", DEVFS_NAME, name); return NULL; } if (!S_ISDIR (de->mode) && de->registered) { printk ("%s: devfs_mk_dir(): existing non-directory entry: \"%s\"\n", DEVFS_NAME, name); return NULL; } #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_REGISTER) printk ("%s: devfs_mk_dir(%s): de: %p %s\n", DEVFS_NAME, name, de, is_new ? "new" : "existing"); #endif if (!S_ISDIR (de->mode) && !is_new) { /* Transmogrifying an old entry */ de->u.dir.first = NULL; de->u.dir.last = NULL; } de->mode = S_IFDIR | S_IRUGO | S_IXUGO; de->info = info; if (!de->registered) de->u.dir.num_removable = 0; de->registered = TRUE; de->show_unreg = (boot_options & OPTION_SHOW) ? TRUE : FALSE; de->hide = FALSE; return de; } /* End Function devfs_mk_dir */ /*PUBLIC_FUNCTION*/ devfs_handle_t devfs_find_handle (devfs_handle_t dir, const char *name, unsigned int namelen, unsigned int major, unsigned int minor, char type, int traverse_symlinks) /* [SUMMARY] Find the handle of a devfs entry. The handle to the parent devfs directory entry. If this is NULL the name is relative to the root of the devfs. The name of the entry. The number of characters in <>, not including a NULL terminator. If this is 0, then <> must be NULL-terminated and the length is computed internally. The major number. This is used if <> is NULL. The minor number. This is used if <> is NULL. The type of special file to search for. This may be either DEVFS_SPECIAL_CHR or DEVFS_SPECIAL_BLK. If TRUE then symlink entries in the devfs namespace are traversed. Symlinks pointing out of the devfs namespace will cause a failure. Symlink traversal consumes stack space. [RETURNS] A handle which may later be used in a call to [], [], or []. On failure NULL is returned. */ { devfs_handle_t de; if ( (name != NULL) && (name[0] == '\0') ) name = NULL; de = find_entry (dir, name, namelen, major, minor, type, traverse_symlinks); if (de == NULL) return NULL; if (!de->registered) return NULL; return de; } /* End Function devfs_find_handle */ /*PUBLIC_FUNCTION*/ int devfs_get_flags (devfs_handle_t de, unsigned int *flags) /* [SUMMARY] Get the flags for a devfs entry. The handle to the device entry. The flags are written here. [RETURNS] 0 on success, else a negative error code. */ { unsigned int fl = 0; if (de == NULL) return -EINVAL; if (!de->registered) return -ENODEV; if (de->show_unreg) fl |= DEVFS_FL_SHOW_UNREG; if (de->hide) fl |= DEVFS_FL_HIDE; if ( S_ISCHR (de->mode) || S_ISBLK (de->mode) || S_ISREG (de->mode) ) { if (de->u.fcb.auto_owner) fl |= DEVFS_FL_AUTO_OWNER; if (de->u.fcb.aopen_notify) fl |= DEVFS_FL_AOPEN_NOTIFY; if (de->u.fcb.removable) fl |= DEVFS_FL_REMOVABLE; } *flags = fl; return 0; } /* End Function devfs_get_flags */ /*PUBLIC_FUNCTION*/ int devfs_set_flags (devfs_handle_t de, unsigned int flags) /* [SUMMARY] Set the flags for a devfs entry. The handle to the device entry. The flags to set. Unset flags are cleared. [RETURNS] 0 on success, else a negative error code. */ { if (de == NULL) return -EINVAL; if (!de->registered) return -ENODEV; #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_SET_FLAGS) printk ("%s: devfs_set_flags(): de->name: \"%s\"\n", DEVFS_NAME, de->name); #endif de->show_unreg = (flags & DEVFS_FL_SHOW_UNREG) ? TRUE : FALSE; de->hide = (flags & DEVFS_FL_HIDE) ? TRUE : FALSE; if ( S_ISCHR (de->mode) || S_ISBLK (de->mode) || S_ISREG (de->mode) ) { de->u.fcb.auto_owner = (flags & DEVFS_FL_AUTO_OWNER) ? TRUE : FALSE; de->u.fcb.aopen_notify = (flags & DEVFS_FL_AOPEN_NOTIFY) ? TRUE:FALSE; if ( de->u.fcb.removable && !(flags & DEVFS_FL_REMOVABLE) ) { de->u.fcb.removable = FALSE; --de->parent->u.dir.num_removable; } else if ( !de->u.fcb.removable && (flags & DEVFS_FL_REMOVABLE) ) { de->u.fcb.removable = TRUE; ++de->parent->u.dir.num_removable; } } return 0; } /* End Function devfs_set_flags */ /*PUBLIC_FUNCTION*/ int devfs_get_maj_min (devfs_handle_t de, unsigned int *major, unsigned int *minor) /* [SUMMARY] Get the major and minor numbers for a devfs entry. The handle to the device entry. The major number is written here. This may be NULL. The minor number is written here. This may be NULL. [RETURNS] 0 on success, else a negative error code. */ { if (de == NULL) return -EINVAL; if (!de->registered) return -ENODEV; if ( S_ISDIR (de->mode) ) return -EISDIR; if ( !S_ISCHR (de->mode) && !S_ISBLK (de->mode) ) return -EINVAL; if (major != NULL) *major = de->u.fcb.u.device.major; if (minor != NULL) *minor = de->u.fcb.u.device.minor; return 0; } /* End Function devfs_get_maj_min */ /*PUBLIC_FUNCTION*/ devfs_handle_t devfs_get_handle_from_inode (struct inode *inode) /* [SUMMARY] Get the devfs handle for a VFS inode. The VFS inode. [RETURNS] The devfs handle on success, else NULL. */ { struct devfs_inode *di; if (!inode || !inode->i_sb) return NULL; if (inode->i_sb->s_magic != DEVFS_SUPER_MAGIC) return NULL; di = get_devfs_inode_from_vfs_inode (inode); if (!di) return NULL; return di->de; } /* End Function devfs_get_handle_from_inode */ /*PUBLIC_FUNCTION*/ int devfs_generate_path (devfs_handle_t de, char *path, int buflen) /* [SUMMARY] Generate a pathname for an entry, relative to the devfs root. The devfs entry. The buffer to write the pathname to. The pathname and '\0' terminator will be written at the end of the buffer. The length of the buffer. [RETURNS] The offset in the buffer where the pathname starts on success, else a negative error code. */ { int pos; if (de == NULL) return -EINVAL; if (de->namelen >= buflen) return -ENAMETOOLONG; /* Must be first */ if (de->parent == NULL) return buflen; /* Don't prepend root */ pos = buflen - de->namelen - 1; memcpy (path + pos, de->name, de->namelen); path[buflen - 1] = '\0'; for (de = de->parent; de->parent != NULL; de = de->parent) { if (pos - de->namelen - 1 < 0) return -ENAMETOOLONG; path[--pos] = '/'; pos -= de->namelen; memcpy (path + pos, de->name, de->namelen); } return pos; } /* End Function devfs_generate_path */ /*PUBLIC_FUNCTION*/ void *devfs_get_ops (devfs_handle_t de) /* [SUMMARY] Get the device operations for a devfs entry. The handle to the device entry. [RETURNS] A pointer to the device operations on success, else NULL. */ { if (de == NULL) return NULL; if (!de->registered) return NULL; if ( S_ISCHR (de->mode) || S_ISBLK (de->mode) || S_ISREG (de->mode) ) return de->u.fcb.ops; return NULL; } /* End Function devfs_get_ops */ /*PUBLIC_FUNCTION*/ int devfs_set_file_size (devfs_handle_t de, unsigned long size) /* [SUMMARY] Set the file size for a devfs regular file. The handle to the device entry. The new file size. [RETURNS] 0 on success, else a negative error code. */ { struct devfs_inode *di; if (de == NULL) return -EINVAL; if (!de->registered) return -EINVAL; if ( !S_ISREG (de->mode) ) return -EINVAL; if (de->u.fcb.u.file.size == size) return 0; de->u.fcb.u.file.size = size; for (di = de->first_inode; di != NULL; di = di->next) { if (di->dentry == NULL) continue; if (di->dentry->d_inode == NULL) continue; di->dentry->d_inode->i_size = size; } return 0; } /* End Function devfs_set_file_size */ /*PUBLIC_FUNCTION*/ void *devfs_get_info (devfs_handle_t de) /* [SUMMARY] Get the info pointer written to <> upon open. The handle to the device entry. [RETURNS] The info pointer. */ { if (de == NULL) return NULL; if (!de->registered) return NULL; return de->info; } /* End Function devfs_get_info */ /*PUBLIC_FUNCTION*/ int devfs_set_info (devfs_handle_t de, void *info) /* [SUMMARY] Set the info pointer written to <> upon open. The handle to the device entry. [RETURNS] 0 on success, else a negative error code. */ { if (de == NULL) return -EINVAL; if (!de->registered) return -EINVAL; de->info = info; return 0; } /* End Function devfs_set_info */ /*PUBLIC_FUNCTION*/ devfs_handle_t devfs_get_parent (devfs_handle_t de) /* [SUMMARY] Get the parent device entry. The handle to the device entry. [RETURNS] The parent device entry if it exists, else NULL. */ { if (de == NULL) return NULL; if (!de->registered) return NULL; return de->parent; } /* End Function devfs_get_parent */ /*PUBLIC_FUNCTION*/ devfs_handle_t devfs_get_first_child (devfs_handle_t de) /* [SUMMARY] Get the first leaf node in a directory. The handle to the device entry. [RETURNS] The leaf node device entry if it exists, else NULL. */ { if (de == NULL) return NULL; if (!de->registered) return NULL; if ( !S_ISDIR (de->mode) ) return NULL; return de->u.dir.first; } /* End Function devfs_get_first_child */ /*PUBLIC_FUNCTION*/ devfs_handle_t devfs_get_next_sibling (devfs_handle_t de) /* [SUMMARY] Get the next sibling leaf node. for a device entry. The handle to the device entry. [RETURNS] The leaf node device entry if it exists, else NULL. */ { if (de == NULL) return NULL; if (!de->registered) return NULL; return de->next; } /* End Function devfs_get_next_sibling */ /*PUBLIC_FUNCTION*/ void devfs_auto_unregister (devfs_handle_t master, devfs_handle_t slave) /* [SUMMARY] Configure a devfs entry to be automatically unregistered. The master devfs entry. Only one slave may be registered. The devfs entry which will be automatically unregistered when the master entry is unregistered. It is illegal to call [] on this entry. [RETURNS] Nothing. */ { if (master == NULL) return; if (master->slave != NULL) { /* Because of the dumbness of the layers above, ignore duplicates */ if (master->slave == slave) return; printk ("%s: devfs_auto_unregister(): only one slave allowed\n", DEVFS_NAME); OOPS (" master: \"%s\" old slave: \"%s\" new slave: \"%s\"\n", master->name, master->slave->name, slave->name); } master->slave = slave; } /* End Function devfs_auto_unregister */ /*PUBLIC_FUNCTION*/ devfs_handle_t devfs_get_unregister_slave (devfs_handle_t master) /* [SUMMARY] Get the slave entry which will be automatically unregistered. The master devfs entry. [RETURNS] The slave which will be unregistered when <> is unregistered. */ { if (master == NULL) return NULL; return master->slave; } /* End Function devfs_get_unregister_slave */ /*PUBLIC_FUNCTION*/ const char *devfs_get_name (devfs_handle_t de, unsigned int *namelen) /* [SUMMARY] Get the name for a device entry in its parent directory. The handle to the device entry. The length of the name is written here. This may be NULL. [RETURNS] The name on success, else NULL. */ { if (de == NULL) return NULL; if (!de->registered) return NULL; if (namelen != NULL) *namelen = de->namelen; return de->name; } /* End Function devfs_get_name */ /*PUBLIC_FUNCTION*/ int devfs_register_chrdev (unsigned int major, const char *name, struct file_operations *fops) /* [SUMMARY] Optionally register a conventional character driver. [PURPOSE] This function will register a character driver provided the "devfs=only" option was not provided at boot time. The major number for the driver. The name of the driver (as seen in /proc/devices). The file_operations structure pointer. [RETURNS] 0 on success, else a negative error code on failure. */ { if (boot_options & OPTION_ONLY) return 0; return register_chrdev (major, name, fops); } /* End Function devfs_register_chrdev */ /*PUBLIC_FUNCTION*/ int devfs_register_blkdev (unsigned int major, const char *name, struct block_device_operations *bdops) /* [SUMMARY] Optionally register a conventional block driver. [PURPOSE] This function will register a block driver provided the "devfs=only" option was not provided at boot time. The major number for the driver. The name of the driver (as seen in /proc/devices). The block_device_operations structure pointer. [RETURNS] 0 on success, else a negative error code on failure. */ { if (boot_options & OPTION_ONLY) return 0; return register_blkdev (major, name, bdops); } /* End Function devfs_register_blkdev */ /*PUBLIC_FUNCTION*/ int devfs_unregister_chrdev (unsigned int major, const char *name) /* [SUMMARY] Optionally unregister a conventional character driver. [PURPOSE] This function will unregister a character driver provided the "devfs=only" option was not provided at boot time. The major number for the driver. The name of the driver (as seen in /proc/devices). [RETURNS] 0 on success, else a negative error code on failure. */ { if (boot_options & OPTION_ONLY) return 0; return unregister_chrdev (major, name); } /* End Function devfs_unregister_chrdev */ /*PUBLIC_FUNCTION*/ int devfs_unregister_blkdev (unsigned int major, const char *name) /* [SUMMARY] Optionally unregister a conventional block driver. [PURPOSE] This function will unregister a block driver provided the "devfs=only" option was not provided at boot time. The major number for the driver. The name of the driver (as seen in /proc/devices). [RETURNS] 0 on success, else a negative error code on failure. */ { if (boot_options & OPTION_ONLY) return 0; return unregister_blkdev (major, name); } /* End Function devfs_unregister_blkdev */ #ifndef MODULE /*UNPUBLISHED_FUNCTION*/ SETUP_STATIC int __init devfs_setup (char *str) /* [SUMMARY] Process kernel boot options. The boot options after the "devfs=". Unused. [RETURNS] Nothing. */ { while ( (*str != '\0') && !isspace (*str) ) { # ifdef CONFIG_DEVFS_DEBUG if (strncmp (str, "dall", 4) == 0) { devfs_debug_init |= DEBUG_ALL; str += 4; } else if (strncmp (str, "dmod", 4) == 0) { devfs_debug_init |= DEBUG_MODULE_LOAD; str += 4; } else if (strncmp (str, "dreg", 4) == 0) { devfs_debug_init |= DEBUG_REGISTER; str += 4; } else if (strncmp (str, "dunreg", 6) == 0) { devfs_debug_init |= DEBUG_UNREGISTER; str += 6; } else if (strncmp (str, "diread", 6) == 0) { devfs_debug_init |= DEBUG_I_READ; str += 6; } else if (strncmp (str, "dchange", 7) == 0) { devfs_debug_init |= DEBUG_SET_FLAGS; str += 7; } else if (strncmp (str, "diwrite", 7) == 0) { devfs_debug_init |= DEBUG_I_WRITE; str += 7; } else if (strncmp (str, "dimknod", 7) == 0) { devfs_debug_init |= DEBUG_I_MKNOD; str += 7; } else if (strncmp (str, "dilookup", 8) == 0) { devfs_debug_init |= DEBUG_I_LOOKUP; str += 8; } else if (strncmp (str, "diunlink", 8) == 0) { devfs_debug_init |= DEBUG_I_UNLINK; str += 8; } else # endif /* CONFIG_DEVFS_DEBUG */ if (strncmp (str, "show", 4) == 0) { boot_options |= OPTION_SHOW; str += 4; } else if (strncmp (str, "only", 4) == 0) { boot_options |= OPTION_ONLY; str += 4; } else if (strncmp (str, "nomount", 7) == 0) { boot_options |= OPTION_NOMOUNT; str += 7; } else return 0; if (*str != ',') return 0; ++str; } return 1; } /* End Function devfs_setup */ __setup("devfs=", devfs_setup); #endif /* !MODULE */ EXPORT_SYMBOL(devfs_register); EXPORT_SYMBOL(devfs_unregister); EXPORT_SYMBOL(devfs_mk_symlink); EXPORT_SYMBOL(devfs_mk_dir); EXPORT_SYMBOL(devfs_find_handle); EXPORT_SYMBOL(devfs_get_flags); EXPORT_SYMBOL(devfs_set_flags); EXPORT_SYMBOL(devfs_get_maj_min); EXPORT_SYMBOL(devfs_get_handle_from_inode); EXPORT_SYMBOL(devfs_generate_path); EXPORT_SYMBOL(devfs_get_ops); EXPORT_SYMBOL(devfs_set_file_size); EXPORT_SYMBOL(devfs_get_info); EXPORT_SYMBOL(devfs_set_info); EXPORT_SYMBOL(devfs_get_parent); EXPORT_SYMBOL(devfs_get_first_child); EXPORT_SYMBOL(devfs_get_next_sibling); EXPORT_SYMBOL(devfs_auto_unregister); EXPORT_SYMBOL(devfs_get_unregister_slave); EXPORT_SYMBOL(devfs_register_chrdev); EXPORT_SYMBOL(devfs_register_blkdev); EXPORT_SYMBOL(devfs_unregister_chrdev); EXPORT_SYMBOL(devfs_unregister_blkdev); #ifdef CONFIG_DEVFS_DEBUG MODULE_PARM(devfs_debug, "i"); #endif static void update_devfs_inode_from_entry (struct devfs_inode *di) { if (di == NULL) return; if (di->de == NULL) { printk ("%s: update_devfs_inode_from_entry(): NULL entry\n", DEVFS_NAME); return; } if ( S_ISDIR (di->de->mode) ) { di->mode = S_IFDIR | S_IRWXU | S_IRUGO | S_IXUGO; di->uid = 0; di->gid = 0; } else if ( S_ISLNK (di->de->mode) ) { di->mode = S_IFLNK | S_IRUGO | S_IXUGO; di->uid = 0; di->gid = 0; } else if ( S_ISFIFO (di->de->mode) ) { di->mode = di->de->mode; di->uid = di->de->u.fifo.uid; di->gid = di->de->u.fifo.gid; } else { if (di->de->u.fcb.auto_owner) { mode_t mode = di->de->mode; di->mode = (mode & ~S_IALLUGO) | S_IRUGO | S_IWUGO; } else { di->mode = di->de->mode; } di->uid = di->de->u.fcb.default_uid; di->gid = di->de->u.fcb.default_gid; } } /* End Function update_devfs_inode_from_entry */ static struct devfs_inode *create_devfs_inode (struct devfs_entry *entry, struct fs_info *fs_info) /* [SUMMARY] Create a devfs inode entry. The devfs entry to associate the new inode with. The FS info. [RETURNS] A pointer to the devfs inode on success, else NULL. */ { struct devfs_inode *di, **table; /* First ensure table size is enough */ if (fs_info->num_inodes >= fs_info->table_size) { if ( ( table = kmalloc (sizeof *table * (fs_info->table_size + INODE_TABLE_INC), GFP_KERNEL) ) == NULL ) return NULL; fs_info->table_size += INODE_TABLE_INC; #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_I_CREATE) printk ("%s: create_devfs_inode(): grew inode table to: %u entries\n", DEVFS_NAME, fs_info->table_size); #endif if (fs_info->table) { memcpy (table, fs_info->table, sizeof *table *fs_info->num_inodes); kfree (fs_info->table); } fs_info->table = table; } if ( ( di = kmalloc (sizeof *di, GFP_KERNEL) ) == NULL ) return NULL; memset (di, 0, sizeof *di); di->ino = fs_info->num_inodes + FIRST_INODE; di->nlink = 1; fs_info->table[fs_info->num_inodes] = di; ++fs_info->num_inodes; di->de = entry; di->fs_info = fs_info; di->prev = entry->last_inode; if (entry->first_inode == NULL) entry->first_inode = di; else entry->last_inode->next = di; entry->last_inode = di; update_devfs_inode_from_entry (di); #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_I_CREATE) printk ("%s: create_devfs_inode(): new di(%u): %p\n", DEVFS_NAME, di->ino, di); #endif return di; } /* End Function create_devfs_inode */ static int try_modload (struct devfs_entry *parent, struct fs_info *fs_info, const char *name, unsigned namelen, char buf[STRING_LENGTH]) /* [SUMMARY] Notify devfsd of an inode lookup. The parent devfs entry. The filesystem info. The device name. The number of characters in <>. A working area that will be used. This must not go out of scope until devfsd is idle again. [RETURNS] 0 on success, else a negative error code. */ { int pos; if ( !( fs_info->devfsd_event_mask & (1 << DEVFSD_NOTIFY_LOOKUP) ) ) return -ENOENT; if ( is_devfsd_or_child (fs_info) ) return -ENOENT; if (namelen >= STRING_LENGTH) return -ENAMETOOLONG; memcpy (buf + STRING_LENGTH - namelen - 1, name, namelen); buf[STRING_LENGTH - 1] = '\0'; pos = devfs_generate_path (parent, buf, STRING_LENGTH - namelen - 1); if (pos < 0) return pos; buf[STRING_LENGTH - namelen - 2] = '/'; if ( !devfsd_notify_one (buf + pos, DEVFSD_NOTIFY_LOOKUP, 0, current->euid, current->egid, fs_info) ) return -ENOENT; /* Possible success */ return 0; } /* End Function try_modload */ static void delete_fs (struct fs_info *fs_info) { unsigned int count; struct devfs_inode *di; struct devfs_entry *de; if (fs_info == NULL) return; for (count = 0; count < fs_info->num_inodes; ++count) { /* Unhook this inode from the devfs tree */ di = fs_info->table[count]; de = di->de; if (di->prev == NULL) de->first_inode = di->next; else di->prev->next = di->next; if (di->next == NULL) de->last_inode = di->prev; else di->next->prev = di->prev; memset (di, 0, sizeof *di); kfree (di); } if (fs_info->table) kfree (fs_info->table); if (fs_info->prev == NULL) first_fs = fs_info->next; else fs_info->prev->next = fs_info->next; if (fs_info->next == NULL) last_fs = fs_info->prev; else fs_info->next->prev = fs_info->prev; memset (fs_info, 0, sizeof *fs_info); kfree (fs_info); } /* End Function delete_fs */ static int check_disc_changed (struct devfs_entry *de) /* [SUMMARY] Check if a removable disc was changed. The device. [RETURNS] 1 if the media was changed, else 0. */ { int tmp; kdev_t dev = MKDEV (de->u.fcb.u.device.major, de->u.fcb.u.device.minor); struct block_device_operations *bdops = de->u.fcb.ops; struct super_block * sb; extern int warn_no_part; if ( !S_ISBLK (de->mode) ) return 0; if (bdops == NULL) return 0; if (bdops->check_media_change == NULL) return 0; if ( !bdops->check_media_change (dev) ) return 0; printk ( KERN_DEBUG "VFS: Disk change detected on device %s\n", kdevname (dev) ); sb = get_super (dev); if ( sb && invalidate_inodes (sb) ) printk("VFS: busy inodes on changed media..\n"); invalidate_buffers (dev); /* Ugly hack to disable messages about unable to read partition table */ tmp = warn_no_part; warn_no_part = 0; if (bdops->revalidate) bdops->revalidate (dev); warn_no_part = tmp; return 1; } /* End Function check_disc_changed */ static void scan_dir_for_removable (struct devfs_entry *dir) /* [SUMMARY] Scan a directory for removable media devices and check media. The directory. [RETURNS] Nothing. */ { struct devfs_entry *de; if (dir->u.dir.num_removable < 1) return; for (de = dir->u.dir.first; de != NULL; de = de->next) { if (!de->registered) continue; if ( !S_ISBLK (de->mode) ) continue; if (!de->u.fcb.removable) continue; check_disc_changed (de); } } /* End Function scan_dir_for_removable */ static int get_removable_partition (struct devfs_entry *dir, const char *name, unsigned int namelen) /* [SUMMARY] Get removable media partition. The parent directory. The name of the entry. The number of characters in <>. [RETURNS] 1 if the media was changed, else 0. */ { struct devfs_entry *de; for (de = dir->u.dir.first; de != NULL; de = de->next) { if (!de->registered) continue; if ( !S_ISBLK (de->mode) ) continue; if (!de->u.fcb.removable) continue; if (strcmp (de->name, "disc") == 0) return check_disc_changed (de); /* Support for names where the partition is appended to the disc name */ if (de->namelen >= namelen) continue; if (strncmp (de->name, name, de->namelen) != 0) continue; return check_disc_changed (de); } return 0; } /* End Function get_removable_partition */ /* Superblock operations follow */ extern struct inode_operations devfs_iops; static struct file_operations devfs_fops; static void devfs_read_inode (struct inode *inode) { struct devfs_inode *di; di = get_devfs_inode_from_vfs_inode (inode); if (di == NULL) { printk ("%s: read_inode(%d): VFS inode: %p NO devfs_inode\n", DEVFS_NAME, (int) inode->i_ino, inode); return; } #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_I_READ) printk ("%s: read_inode(%d): VFS inode: %p devfs_inode: %p\n", DEVFS_NAME, (int) inode->i_ino, inode, di); #endif inode->i_size = 0; inode->i_blocks = 0; inode->i_blksize = 1024; inode->i_op = &devfs_iops; inode->i_fop = &devfs_fops; inode->i_rdev = NODEV; if ( S_ISCHR (di->mode) ) inode->i_rdev = MKDEV (di->de->u.fcb.u.device.major, di->de->u.fcb.u.device.minor); else if ( S_ISBLK (di->mode) ) { inode->i_rdev = MKDEV (di->de->u.fcb.u.device.major, di->de->u.fcb.u.device.minor); inode->i_bdev = bdget (inode->i_rdev); if (inode->i_bdev) inode->i_bdev->bd_op = di->de->u.fcb.ops; else printk ("%s: read_inode(%d): no block device from bdget()\n", DEVFS_NAME, (int) inode->i_ino); } else if ( S_ISFIFO (di->mode) ) inode->i_fop = &def_fifo_fops; else if ( S_ISREG (di->mode) ) inode->i_size = di->de->u.fcb.u.file.size; inode->i_mode = di->mode; inode->i_uid = di->uid; inode->i_gid = di->gid; inode->i_atime = di->atime; inode->i_mtime = di->mtime; inode->i_ctime = di->ctime; inode->i_nlink = di->nlink; #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_I_READ) printk ("%s: mode: 0%o uid: %d gid: %d\n", DEVFS_NAME, (int) inode->i_mode, (int) inode->i_uid, (int) inode->i_gid); #endif } /* End Function devfs_read_inode */ static void devfs_write_inode (struct inode *inode) { int index; struct devfs_inode *di; struct fs_info *fs_info = inode->i_sb->u.generic_sbp; if (inode->i_ino < FIRST_INODE) return; index = inode->i_ino - FIRST_INODE; if (index >= fs_info->num_inodes) { printk ("%s: writing inode: %lu for which there is no entry!\n", DEVFS_NAME, inode->i_ino); return; } di = fs_info->table[index]; #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_I_WRITE) { printk ("%s: write_inode(%d): VFS inode: %p devfs_inode: %p\n", DEVFS_NAME, (int) inode->i_ino, inode, di); printk ("%s: mode: 0%o uid: %d gid: %d\n", DEVFS_NAME, (int) inode->i_mode, (int) inode->i_uid, (int) inode->i_gid); } #endif di->mode = inode->i_mode; di->uid = inode->i_uid; di->gid = inode->i_gid; di->atime = inode->i_atime; di->mtime = inode->i_mtime; di->ctime = inode->i_ctime; } /* End Function devfs_write_inode */ static int devfs_notify_change (struct dentry *dentry, struct iattr *iattr) { int retval; struct devfs_inode *di; struct inode *inode = dentry->d_inode; struct fs_info *fs_info = inode->i_sb->u.generic_sbp; di = get_devfs_inode_from_vfs_inode (inode); if (di == NULL) return -ENODEV; retval = inode_change_ok (inode, iattr); if (retval != 0) return retval; inode_setattr (inode, iattr); if ( iattr->ia_valid & (ATTR_MODE | ATTR_UID | ATTR_GID) ) devfsd_notify_one (di->de, DEVFSD_NOTIFY_CHANGE, inode->i_mode, inode->i_uid, inode->i_gid, fs_info); return 0; } /* End Function devfs_notify_change */ static void devfs_put_super (struct super_block *sb) { struct fs_info *fs_info = sb->u.generic_sbp; #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_S_PUT) printk ("%s: put_super(): devfs ptr: %p\n", DEVFS_NAME, fs_info); #endif sb->s_dev = 0; #ifdef CONFIG_DEVFS_TUNNEL dput (fs_info->table[0]->covered); #endif delete_fs (fs_info); MOD_DEC_USE_COUNT; } /* End Function devfs_put_super */ static int devfs_statfs (struct super_block *sb, struct statfs *buf,int bufsiz) { struct statfs tmp; tmp.f_type = DEVFS_SUPER_MAGIC; tmp.f_bsize = PAGE_SIZE / sizeof (long); tmp.f_blocks = 0; tmp.f_bfree = 0; tmp.f_bavail = 0; tmp.f_files = 0; tmp.f_ffree = 0; tmp.f_namelen = NAME_MAX; return copy_to_user (buf, &tmp, bufsiz) ? -EFAULT : 0; } /* End Function devfs_statfs */ static struct super_operations devfs_sops = { read_inode: devfs_read_inode, write_inode: devfs_write_inode, put_super: devfs_put_super, statfs: devfs_statfs, }; static struct inode *get_vfs_inode (struct super_block *sb, struct devfs_inode *di, struct dentry *dentry) /* [SUMMARY] Get a VFS inode. The super block. The devfs inode. The dentry to register with the devfs inode. [RETURNS] The inode on success, else NULL. */ { struct inode *inode; if (di->dentry != NULL) { printk ("%s: get_vfs_inode(%u): old di->dentry: %p \"%s\" new dentry: %p \"%s\"\n", DEVFS_NAME, di->ino, di->dentry, di->dentry->d_name.name, dentry, dentry->d_name.name); printk (" old inode: %p\n", di->dentry->d_inode); return NULL; } if ( ( inode = iget (sb, di->ino) ) == NULL ) return NULL; di->dentry = dentry; return inode; } /* End Function get_vfs_inode */ /* File operations for device entries follow */ static int devfs_read (struct file *file, char *buf, size_t len, loff_t *ppos) { if ( S_ISDIR (file->f_dentry->d_inode->i_mode) ) return -EISDIR; return -EINVAL; } /* End Function devfs_read */ static int devfs_readdir (struct file *file, void *dirent, filldir_t filldir) { int err, count; int stored = 0; struct fs_info *fs_info; struct devfs_inode *di; struct devfs_entry *parent, *de; struct inode *inode = file->f_dentry->d_inode; if (inode == NULL) { printk ("%s: readdir(): NULL inode\n", DEVFS_NAME); return -EBADF; } if ( !S_ISDIR (inode->i_mode) ) { printk ("%s: readdir(): inode is not a directory\n", DEVFS_NAME); return -ENOTDIR; } fs_info = inode->i_sb->u.generic_sbp; di = get_devfs_inode_from_vfs_inode (file->f_dentry->d_inode); parent = di->de; if ( (long) file->f_pos < 0 ) return -EINVAL; #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_F_READDIR) printk ("%s: readdir(): fs_info: %p pos: %ld\n", DEVFS_NAME, fs_info, (long) file->f_pos); #endif switch ( (long) file->f_pos ) { case 0: scan_dir_for_removable (parent); err = (*filldir) (dirent, "..", 2, file->f_pos, file->f_dentry->d_parent->d_inode->i_ino); if (err == -EINVAL) break; if (err < 0) return err; file->f_pos++; ++stored; /* Fall through */ case 1: err = (*filldir) (dirent, ".", 1, file->f_pos, inode->i_ino); if (err == -EINVAL) break; if (err < 0) return err; file->f_pos++; ++stored; /* Fall through */ default: /* Skip entries */ count = file->f_pos - 2; for (de = parent->u.dir.first; (de != NULL) && (count > 0); de = de->next) { if ( IS_HIDDEN (de) ) continue; if (!fs_info->require_explicit) { --count; continue; } /* Must search for an inode for this FS */ for (di = de->first_inode; di != NULL; di = di->next) { if (fs_info == di->fs_info) break; } if (di != NULL) --count; } /* Now add all remaining entries */ for (; de != NULL; de = de->next) { if ( IS_HIDDEN (de) ) continue; /* Must search for an inode for this FS */ for (di = de->first_inode; di != NULL; di = di->next) { if (fs_info == di->fs_info) break; } if (di == NULL) { if (fs_info->require_explicit) continue; /* Have to create the inode right now */ di = create_devfs_inode (de, fs_info); if (di == NULL) return -ENOMEM; } else if (di->ctime == 0) update_devfs_inode_from_entry (di); err = (*filldir) (dirent, de->name, de->namelen, file->f_pos, di->ino); if (err == -EINVAL) break; if (err < 0) return err; file->f_pos++; ++stored; } break; } return stored; } /* End Function devfs_readdir */ static int devfs_open (struct inode *inode, struct file *file) { int err; struct fcb_type *df; struct devfs_inode *di; struct dentry *dentry = file->f_dentry; struct fs_info *fs_info = inode->i_sb->u.generic_sbp; di = get_devfs_inode_from_vfs_inode (inode); if (di == NULL) return -ENODEV; if ( S_ISDIR (di->de->mode) ) return 0; df = &di->de->u.fcb; if (!di->de->registered) return -ENODEV; file->private_data = di->de->info; if ( S_ISBLK (inode->i_mode) ) { file->f_op = &def_blk_fops; if (df->ops) inode->i_bdev->bd_op = df->ops; } else file->f_op = df->ops; if (file->f_op) err = file->f_op->open ? (*file->f_op->open) (inode, file) : 0; else { /* Fallback to legacy scheme */ if ( S_ISCHR (inode->i_mode) ) err = chrdev_open (inode, file); else err = -ENODEV; } if (err < 0) return err; /* Open was successful */ df->open = TRUE; if (dentry->d_count != 1) return 0; /* No fancy operations */ /* This is the first open */ if (df->auto_owner) { /* Change the ownership/protection */ di->mode = (di->mode & ~S_IALLUGO) | (di->de->mode & S_IRWXUGO); di->uid = current->euid; di->gid = current->egid; inode->i_mode = di->mode; inode->i_uid = di->uid; inode->i_gid = di->gid; } if (df->aopen_notify) devfsd_notify_one (di->de, DEVFSD_NOTIFY_ASYNC_OPEN, inode->i_mode, current->euid, current->egid, fs_info); return 0; } /* End Function devfs_open */ static struct file_operations devfs_fops = { read: devfs_read, readdir: devfs_readdir, open: devfs_open, }; /* Dentry operations for device entries follow */ static void devfs_d_release (struct dentry *dentry) /* [SUMMARY] Callback for when a dentry is freed. */ { #ifdef CONFIG_DEVFS_DEBUG struct inode *inode = dentry->d_inode; if (devfs_debug & DEBUG_D_RELEASE) printk ("%s: d_release(): dentry: %p inode: %p\n", DEVFS_NAME, dentry, inode); #endif } /* End Function devfs_d_release */ static void devfs_d_iput (struct dentry *dentry, struct inode *inode) /* [SUMMARY] Callback for when a dentry loses its inode. */ { struct devfs_inode *di; di = get_devfs_inode_from_vfs_inode (inode); #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_D_IPUT) printk ("%s: d_iput(): dentry: %p inode: %p di: %p di->dentry: %p\n", DEVFS_NAME, dentry, inode, di, di->dentry); #endif if (di->dentry == dentry) { di->dentry = NULL; #ifdef CONFIG_DEVFS_TUNNEL dput (di->covered); di->covered = NULL; #endif } iput (inode); } /* End Function devfs_d_iput */ static void devfs_d_delete (struct dentry *dentry); static struct dentry_operations devfs_dops = { d_delete: devfs_d_delete, d_release: devfs_d_release, d_iput: devfs_d_iput, }; static int devfs_d_revalidate_wait (struct dentry *dentry, int flags); static struct dentry_operations devfs_wait_dops = { d_delete: devfs_d_delete, d_release: devfs_d_release, d_iput: devfs_d_iput, d_revalidate: devfs_d_revalidate_wait, }; static void devfs_d_delete (struct dentry *dentry) /* [SUMMARY] Callback for when all files for a dentry are closed. */ { struct inode *inode = dentry->d_inode; struct devfs_inode *di; struct fs_info *fs_info; if (dentry->d_op == &devfs_wait_dops) dentry->d_op = &devfs_dops; /* Unhash dentry if negative (has no inode) */ if (inode == NULL) { #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_D_DELETE) printk ("%s: d_delete(): dropping negative dentry: %p\n", DEVFS_NAME, dentry); #endif d_drop (dentry); return; } fs_info = inode->i_sb->u.generic_sbp; di = get_devfs_inode_from_vfs_inode (inode); #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_D_DELETE) printk ("%s: d_delete(): dentry: %p inode: %p devfs_inode: %p\n", DEVFS_NAME, dentry, inode, di); #endif if (di == NULL) return; if (di->de == NULL) return; if ( !S_ISCHR (di->mode) && !S_ISBLK (di->mode) && !S_ISREG (di->mode) ) return; if (!di->de->u.fcb.open) return; di->de->u.fcb.open = FALSE; if (di->de->u.fcb.aopen_notify) devfsd_notify_one (di->de, DEVFSD_NOTIFY_CLOSE, inode->i_mode, current->euid, current->egid, fs_info); if (!di->de->u.fcb.auto_owner) return; /* Change the ownership/protection back */ di->mode = (di->mode & ~S_IALLUGO) | S_IRUGO | S_IWUGO; di->uid = di->de->u.fcb.default_uid; di->gid = di->de->u.fcb.default_gid; inode->i_mode = di->mode; inode->i_uid = di->uid; inode->i_gid = di->gid; } /* End Function devfs_d_delete */ static int devfs_d_revalidate_wait (struct dentry *dentry, int flags) { devfs_handle_t de = dentry->d_fsdata; struct inode *dir = dentry->d_parent->d_inode; struct fs_info *fs_info = dir->i_sb->u.generic_sbp; if (!de || de->registered) { if ( !dentry->d_inode && is_devfsd_or_child (fs_info) ) { struct devfs_inode *di = NULL; struct inode *inode; #ifdef CONFIG_DEVFS_DEBUG char txt[STRING_LENGTH]; memset (txt, 0, STRING_LENGTH); memcpy (txt, dentry->d_name.name, (dentry->d_name.len >= STRING_LENGTH) ? (STRING_LENGTH - 1) : dentry->d_name.len); if (devfs_debug & DEBUG_I_LOOKUP) printk ("%s: d_revalidate(): dentry: %p name: \"%s\" by: \"%s\"\n", DEVFS_NAME, dentry, txt, current->comm); #endif if (de) { /* Search for an inode for this FS */ for (di = de->first_inode; di != NULL; di = di->next) if (di->fs_info == fs_info) break; } if (de == NULL) { devfs_handle_t parent; struct devfs_inode *pi; pi = get_devfs_inode_from_vfs_inode (dir); parent = pi->de; de = search_for_entry_in_dir (parent, dentry->d_name.name, dentry->d_name.len, FALSE); } if (de == NULL) return 1; /* Create an inode, now that the driver information is available */ if (di == NULL) di = create_devfs_inode (de, fs_info); else if (di->ctime == 0) update_devfs_inode_from_entry (di); else di->mode = (de->mode & ~S_IALLUGO) | (di->mode & S_IALLUGO); if (di == NULL) return 1; if ( ( inode = get_vfs_inode (dir->i_sb, di, dentry) ) == NULL ) return 1; #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_I_LOOKUP) printk ("%s: d_revalidate(): new VFS inode(%u): %p devfs_inode: %p\n", DEVFS_NAME, di->ino, inode, di); #endif d_instantiate (dentry, inode); return 1; } } if ( wait_for_devfsd_finished (fs_info) ) dentry->d_op = &devfs_dops; return 1; } /* End Function devfs_d_revalidate_wait */ /* Inode operations for device entries follow */ static struct dentry *devfs_lookup (struct inode *dir, struct dentry *dentry) { struct fs_info *fs_info; struct devfs_inode *di = NULL; struct devfs_inode *pi; struct devfs_entry *parent, *de; struct inode *inode; char txt[STRING_LENGTH]; /* Set up the dentry operations before anything else, to ensure cleaning up on any error */ dentry->d_op = &devfs_dops; if (dir == NULL) { printk ("%s: lookup(): NULL directory inode\n", DEVFS_NAME); return ERR_PTR (-ENOTDIR); } if ( !S_ISDIR (dir->i_mode) ) return ERR_PTR (-ENOTDIR); memset (txt, 0, STRING_LENGTH); memcpy (txt, dentry->d_name.name, (dentry->d_name.len >= STRING_LENGTH) ? (STRING_LENGTH - 1) : dentry->d_name.len); #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_I_LOOKUP) printk ("%s: lookup(%s): dentry: %p by: \"%s\"\n", DEVFS_NAME, txt, dentry, current->comm); #endif fs_info = dir->i_sb->u.generic_sbp; /* First try to get the devfs entry for this directory */ pi = get_devfs_inode_from_vfs_inode (dir); if (pi == NULL) return ERR_PTR (-EINVAL); parent = pi->de; if (!parent->registered) return ERR_PTR (-ENOENT); /* Try to reclaim an existing devfs entry */ de = search_for_entry_in_dir (parent, dentry->d_name.name, dentry->d_name.len, FALSE); if (de) { /* Search for an inode for this FS */ for (di = de->first_inode; di != NULL; di = di->next) if (di->fs_info == fs_info) break; } if (fs_info->require_explicit) { if (di == NULL) { /* Make the dentry negative so a subsequent operation can deal with it (for the benefit of mknod()). Leaving the dentry unhashed will cause to fail which in turns causes to fail */ d_add (dentry, NULL); return NULL; } } if ( ( (de == NULL) || !de->registered ) && (parent->u.dir.num_removable > 0) && get_removable_partition (parent, dentry->d_name.name, dentry->d_name.len) ) { if (de == NULL) de = search_for_entry_in_dir (parent, dentry->d_name.name, dentry->d_name.len, FALSE); } if ( (de == NULL) || (!de->registered) ) { /* Try with devfsd. For any kind of failure, leave a negative dentry so someone else can deal with it (in the case where the sysadmin does a mknod()). It's important to do this before hashing the dentry, so that the devfsd queue is filled before revalidates can start */ if (try_modload (parent, fs_info, dentry->d_name.name, dentry->d_name.len, txt) < 0) { d_add (dentry, NULL); return NULL; } /* devfsd claimed success */ dentry->d_op = &devfs_wait_dops; dentry->d_fsdata = de; d_add (dentry, NULL); /* Open the floodgates */ /* Unlock directory semaphore, which will release any waiters. They will get the hashed dentry, and may be forced to wait for revalidation */ up (&dir->i_sem); devfs_d_revalidate_wait (dentry, 0); /* I might have to wait too */ down (&dir->i_sem); /* Grab it again because them's the rules */ /* If someone else has been so kind as to make the inode, we go home early */ if (dentry->d_inode) return NULL; if (de && !de->registered) return NULL; if (de == NULL) de = search_for_entry_in_dir (parent, dentry->d_name.name, dentry->d_name.len, FALSE); if (de == NULL) return NULL; /* OK, there's an entry now, but no VFS inode yet */ } else { dentry->d_op = &devfs_wait_dops; d_add (dentry, NULL); /* Open the floodgates */ } /* Create an inode, now that the driver information is available */ if (di == NULL) di = create_devfs_inode (de, fs_info); else if (di->ctime == 0) update_devfs_inode_from_entry (di); else di->mode = (de->mode & ~S_IALLUGO) | (di->mode & S_IALLUGO); if (di == NULL) return ERR_PTR (-ENOMEM); if ( ( inode = get_vfs_inode (dir->i_sb, di, dentry) ) == NULL ) return ERR_PTR (-ENOMEM); #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_I_LOOKUP) printk ("%s: lookup(): new VFS inode(%u): %p devfs_inode: %p\n", DEVFS_NAME, di->ino, inode, di); #endif d_instantiate (dentry, inode); /* Unlock directory semaphore, which will release any waiters. They will get the hashed dentry, and may be forced to wait for revalidation */ up (&dir->i_sem); if (dentry->d_op == &devfs_wait_dops) devfs_d_revalidate_wait (dentry, 0); /* I might have to wait too */ down (&dir->i_sem); /* Grab it again because them's the rules */ return NULL; } /* End Function devfs_lookup */ static int devfs_link (struct dentry *old_dentry, struct inode *dir, struct dentry *dentry) { /*struct inode *inode = old_dentry->d_inode;*/ char txt[STRING_LENGTH]; memset (txt, 0, STRING_LENGTH); memcpy (txt, old_dentry->d_name.name, old_dentry->d_name.len); txt[STRING_LENGTH - 1] = '\0'; printk ("%s: link of \"%s\"\n", DEVFS_NAME, txt); return -EPERM; } /* End Function devfs_link */ static int devfs_unlink (struct inode *dir, struct dentry *dentry) { struct devfs_inode *di; #ifdef CONFIG_DEVFS_DEBUG char txt[STRING_LENGTH]; if (devfs_debug & DEBUG_I_UNLINK) { memset (txt, 0, STRING_LENGTH); memcpy (txt, dentry->d_name.name, dentry->d_name.len); txt[STRING_LENGTH - 1] = '\0'; printk ("%s: unlink(%s)\n", DEVFS_NAME, txt); } #endif if ( !dir || !S_ISDIR (dir->i_mode) ) return -ENOTDIR; if (!dentry || !dentry->d_inode) return -ENOENT; di = get_devfs_inode_from_vfs_inode (dentry->d_inode); if (di == NULL) return -ENOENT; if (!di->de->registered) return -ENOENT; di->de->registered = FALSE; di->de->hide = TRUE; free_dentries (di->de); return 0; } /* End Function devfs_unlink */ static int devfs_symlink (struct inode *dir, struct dentry *dentry, const char *symname) { int err; struct fs_info *fs_info; struct devfs_inode *pi; struct devfs_inode *di = NULL; struct devfs_entry *parent, *de; struct inode *inode; if ( !dir || !S_ISDIR (dir->i_mode) ) return -ENOTDIR; fs_info = dir->i_sb->u.generic_sbp; /* First try to get the devfs entry for this directory */ pi = get_devfs_inode_from_vfs_inode (dir); if (pi == NULL) return -EINVAL; parent = pi->de; if (!parent->registered) return -ENOENT; err = devfs_mk_symlink (parent, dentry->d_name.name, dentry->d_name.len, DEVFS_FL_NONE, symname, 0, &de, NULL); #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_DISABLED) printk ("%s: symlink(): errcode from : %d\n", DEVFS_NAME, err); #endif if (err < 0) return err; /* Search for an inode for this FS */ for (di = de->first_inode; di != NULL; di = di->next) { if (di->fs_info == fs_info) break; } if (di == NULL) di = create_devfs_inode (de, fs_info); if (di == NULL) return -ENOMEM; di->mode = de->mode; di->atime = CURRENT_TIME; di->mtime = CURRENT_TIME; di->ctime = CURRENT_TIME; if ( ( inode = get_vfs_inode (dir->i_sb, di, dentry) ) == NULL ) return -ENOMEM; #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_DISABLED) printk ("%s: symlink(): new VFS inode(%u): %p dentry: %p\n", DEVFS_NAME, di->ino, inode, dentry); #endif de->hide = FALSE; d_instantiate (dentry, inode); devfsd_notify_one (di->de, DEVFSD_NOTIFY_CREATE, inode->i_mode, inode->i_uid, inode->i_gid, fs_info); return 0; } /* End Function devfs_symlink */ static int devfs_mkdir (struct inode *dir, struct dentry *dentry, int mode) { int is_new; struct fs_info *fs_info; struct devfs_inode *di = NULL; struct devfs_inode *pi; struct devfs_entry *parent, *de; struct inode *inode; mode = (mode & ~S_IFMT) | S_IFDIR; if ( !dir || !S_ISDIR (dir->i_mode) ) return -ENOTDIR; fs_info = dir->i_sb->u.generic_sbp; /* We are allowed to create the directory */ /* First try to get the devfs entry for this directory */ pi = get_devfs_inode_from_vfs_inode (dir); if (pi == NULL) return -EINVAL; parent = pi->de; if (!parent->registered) return -ENOENT; /* Try to reclaim an existing devfs entry, create if there isn't one */ de = search_for_entry (parent, dentry->d_name.name, dentry->d_name.len, FALSE, TRUE, &is_new, FALSE); if (de == NULL) return -ENOMEM; if (de->registered) { printk ("%s: mkdir(): existing entry\n", DEVFS_NAME); return -EEXIST; } de->registered = TRUE; de->hide = FALSE; if (!S_ISDIR (de->mode) && !is_new) { /* Transmogrifying an old entry */ de->u.dir.first = NULL; de->u.dir.last = NULL; } de->mode = mode; de->u.dir.num_removable = 0; /* Search for an inode for this FS */ for (di = de->first_inode; di != NULL; di = di->next) { if (di->fs_info == fs_info) break; } if (di == NULL) di = create_devfs_inode (de, fs_info); if (di == NULL) return -ENOMEM; di->mode = mode; di->uid = current->euid; di->gid = current->egid; di->atime = CURRENT_TIME; di->mtime = CURRENT_TIME; di->ctime = CURRENT_TIME; if ( ( inode = get_vfs_inode (dir->i_sb, di, dentry) ) == NULL ) return -ENOMEM; #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_DISABLED) printk ("%s: mkdir(): new VFS inode(%u): %p dentry: %p\n", DEVFS_NAME, di->ino, inode, dentry); #endif d_instantiate (dentry, inode); devfsd_notify_one (di->de, DEVFSD_NOTIFY_CREATE, inode->i_mode, inode->i_uid, inode->i_gid, fs_info); return 0; } /* End Function devfs_mkdir */ static int devfs_rmdir (struct inode *dir, struct dentry *dentry) { int has_children = FALSE; struct fs_info *fs_info; struct devfs_inode *di = NULL; struct devfs_entry *de, *child; struct inode *inode = dentry->d_inode; if ( !dir || !S_ISDIR (dir->i_mode) ) return -ENOTDIR; if (dir->i_sb->u.generic_sbp != inode->i_sb->u.generic_sbp) return -EINVAL; if (inode == dir) return -EPERM; fs_info = dir->i_sb->u.generic_sbp; di = get_devfs_inode_from_vfs_inode (inode); if (di == NULL) return -ENOENT; de = di->de; if (!de->registered) return -ENOENT; if ( !S_ISDIR (de->mode) ) return -ENOTDIR; for (child = de->u.dir.first; child != NULL; child = child->next) { if (child->registered) { has_children = TRUE; break; } } if (has_children) return -ENOTEMPTY; de->registered = FALSE; de->hide = TRUE; free_dentries (de); return 0; } /* End Function devfs_rmdir */ static int devfs_mknod (struct inode *dir, struct dentry *dentry, int mode, int rdev) { int is_new; struct fs_info *fs_info; struct devfs_inode *di = NULL; struct devfs_inode *pi; struct devfs_entry *parent, *de; struct inode *inode; #ifdef CONFIG_DEVFS_DEBUG char txt[STRING_LENGTH]; if (devfs_debug & DEBUG_I_MKNOD) { memset (txt, 0, STRING_LENGTH); memcpy (txt, dentry->d_name.name, dentry->d_name.len); txt[STRING_LENGTH - 1] = '\0'; printk ("%s: mknod(%s): mode: 0%o dev: %d\n", DEVFS_NAME, txt, mode, rdev); } #endif if ( !dir || !S_ISDIR (dir->i_mode) ) return -ENOTDIR; fs_info = dir->i_sb->u.generic_sbp; if ( !S_ISBLK (mode) && !S_ISCHR (mode) && !S_ISFIFO (mode) && !S_ISSOCK (mode) ) return -EPERM; /* We are allowed to create the node */ /* First try to get the devfs entry for this directory */ pi = get_devfs_inode_from_vfs_inode (dir); if (pi == NULL) return -EINVAL; parent = pi->de; if (!parent->registered) return -ENOENT; /* Try to reclaim an existing devfs entry, create if there isn't one */ de = search_for_entry (parent, dentry->d_name.name, dentry->d_name.len, FALSE, TRUE, &is_new, FALSE); if (de == NULL) return -ENOMEM; if (!de->registered) { /* Since we created the devfs entry we get to choose things */ de->info = NULL; de->mode = mode; if ( S_ISBLK (mode) || S_ISCHR (mode) ) { de->u.fcb.u.device.major = MAJOR (rdev); de->u.fcb.u.device.minor = MINOR (rdev); de->u.fcb.default_uid = current->euid; de->u.fcb.default_gid = current->egid; de->u.fcb.ops = NULL; de->u.fcb.auto_owner = FALSE; de->u.fcb.aopen_notify = FALSE; de->u.fcb.open = FALSE; } else if ( S_ISFIFO (mode) ) { de->u.fifo.uid = current->euid; de->u.fifo.gid = current->egid; } } de->registered = TRUE; de->show_unreg = FALSE; de->hide = FALSE; /* Search for an inode for this FS */ for (di = de->first_inode; di != NULL; di = di->next) { if (di->fs_info == fs_info) break; } if (di == NULL) di = create_devfs_inode (de, fs_info); if (di == NULL) return -ENOMEM; di->mode = mode; di->uid = current->euid; di->gid = current->egid; di->atime = CURRENT_TIME; di->mtime = CURRENT_TIME; di->ctime = CURRENT_TIME; if ( ( inode = get_vfs_inode (dir->i_sb, di, dentry) ) == NULL ) return -ENOMEM; #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_I_MKNOD) printk ("%s: new VFS inode(%u): %p dentry: %p\n", DEVFS_NAME, di->ino, inode, dentry); #endif d_instantiate (dentry, inode); devfsd_notify_one (di->de, DEVFSD_NOTIFY_CREATE, inode->i_mode, inode->i_uid, inode->i_gid, fs_info); return 0; } /* End Function devfs_mknod */ static int devfs_readlink (struct dentry *dentry, char *buffer, int buflen) { struct inode *inode = dentry->d_inode; struct devfs_inode *di; if ( !inode || !S_ISLNK (inode->i_mode) ) return -EINVAL; di = get_devfs_inode_from_vfs_inode (inode); if (di == NULL) return -ENOENT; if (!di->de->registered) return -ENOENT; #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_I_RLINK) printk ("%s: readlink(): dentry: %p\n", DEVFS_NAME, dentry); #endif if (buflen > di->de->u.symlink.length + 1) buflen = di->de->u.symlink.length + 1; if (copy_to_user (buffer, di->de->u.symlink.linkname, buflen) == 0) return buflen; return -EFAULT; } /* End Function devfs_readlink */ static struct dentry *devfs_follow_link (struct dentry *dentry, struct dentry *base, unsigned int follow) { struct inode *inode = dentry->d_inode; struct devfs_inode *di; if ( !inode || !S_ISLNK (inode->i_mode) ) { dget (dentry); dput (base); return dentry; } #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_I_FLINK) printk ("%s: follow_link(): dentry: %p\n", DEVFS_NAME, dentry); #endif di = get_devfs_inode_from_vfs_inode (inode); if ( (di == NULL) || !di->de->registered ) { dput (base); return ERR_PTR (-ENOENT); } base = lookup_dentry (di->de->u.symlink.linkname, base, follow); return base; } /* End Function devfs_follow_link */ static struct inode_operations devfs_iops = { lookup: devfs_lookup, link: devfs_link, unlink: devfs_unlink, symlink: devfs_symlink, mkdir: devfs_mkdir, rmdir: devfs_rmdir, mknod: devfs_mknod, readlink: devfs_readlink, follow_link: devfs_follow_link, setattr: devfs_notify_change, }; static struct super_block *devfs_read_super (struct super_block *sb, void *data, int silent) { char *aopt = data; struct fs_info *fs_info = NULL; struct devfs_inode *di; struct inode *root_inode = NULL; if (get_root_entry () == NULL) goto out_no_root; if ( ( fs_info = kmalloc (sizeof *fs_info, GFP_KERNEL) ) == NULL ) return NULL; memset (fs_info, 0, sizeof *fs_info); atomic_set (&fs_info->devfsd_overrun_count, 0); init_waitqueue_head (&fs_info->devfsd_wait_queue); init_waitqueue_head (&fs_info->revalidate_wait_queue); fs_info->prev = last_fs; if (first_fs == NULL) first_fs = fs_info; else last_fs->next = fs_info; last_fs = fs_info; fs_info->sb = sb; if (aopt) { if (strcmp (aopt, "explicit") == 0) fs_info->require_explicit = TRUE; } lock_super (sb); sb->u.generic_sbp = fs_info; sb->s_blocksize = 1024; sb->s_blocksize_bits = 10; sb->s_magic = DEVFS_SUPER_MAGIC; sb->s_op = &devfs_sops; di = create_devfs_inode (root_entry, fs_info); if (di == NULL) goto out_no_root; if (di->ino != 1) { printk ("%s: read_super: root inode number is: %d!\n", DEVFS_NAME, di->ino); goto out_no_root; } if ( ( root_inode = get_vfs_inode (sb, di, NULL) ) == NULL ) goto out_no_root; sb->s_root = D_ALLOC_ROOT (root_inode); if (!sb->s_root) goto out_no_root; #ifdef CONFIG_DEVFS_TUNNEL di->covered = dget (sb->s_root->d_covered); #endif unlock_super (sb); #ifdef CONFIG_DEVFS_DEBUG if (devfs_debug & DEBUG_DISABLED) printk ("%s: read super, made devfs ptr: %p\n", DEVFS_NAME, sb->u.generic_sbp); #endif MOD_INC_USE_COUNT; return sb; out_no_root: printk ("devfs_read_super: get root inode failed\n"); delete_fs (fs_info); if (root_inode) iput (root_inode); sb->s_dev = 0; unlock_super (sb); return NULL; } /* End Function devfs_read_super */ static struct file_system_type devfs_fs_type = { DEVFS_NAME, 0, devfs_read_super, NULL, }; /* File operations for devfsd follow */ static ssize_t devfsd_read (struct file *file, char *buf, size_t len, loff_t *ppos) { int done = FALSE; int ival; loff_t pos, devname_offset, tlen, rpos; struct devfsd_notify_struct info; struct devfsd_buf_entry *entry; struct fs_info *fs_info = file->f_dentry->d_inode->i_sb->u.generic_sbp; DECLARE_WAITQUEUE (wait, current); /* Can't seek (pread) on this device */ if (ppos != &file->f_pos) return -ESPIPE; /* Verify the task has grabbed the queue */ if (fs_info->devfsd_task != current) return -EPERM; info.major = 0; info.minor = 0; /* Block for a new entry */ add_wait_queue (&fs_info->devfsd_wait_queue, &wait); current->state = TASK_INTERRUPTIBLE; while ( devfsd_queue_empty (fs_info) ) { fs_info->devfsd_sleeping = TRUE; wake_up (&fs_info->revalidate_wait_queue); schedule (); fs_info->devfsd_sleeping = FALSE; if ( signal_pending (current) ) { remove_wait_queue (&fs_info->devfsd_wait_queue, &wait); current->state = TASK_RUNNING; return -EINTR; } } remove_wait_queue (&fs_info->devfsd_wait_queue, &wait); current->state = TASK_RUNNING; /* Now play with the data */ ival = atomic_read (&fs_info->devfsd_overrun_count); if (ival > 0) atomic_sub (ival, &fs_info->devfsd_overrun_count); info.overrun_count = ival; entry = (struct devfsd_buf_entry *) fs_info->devfsd_buffer + fs_info->devfsd_buf_out; info.type = entry->type; info.mode = entry->mode; info.uid = entry->uid; info.gid = entry->gid; if (entry->type == DEVFSD_NOTIFY_LOOKUP) { info.namelen = strlen (entry->data); pos = 0; memcpy (info.devname, entry->data, info.namelen + 1); } else { devfs_handle_t de = entry->data; if ( S_ISCHR (de->mode) || S_ISBLK (de->mode) || S_ISREG (de->mode) ) { info.major = de->u.fcb.u.device.major; info.minor = de->u.fcb.u.device.minor; } pos = devfs_generate_path (de, info.devname, DEVFS_PATHLEN); if (pos < 0) return pos; info.namelen = DEVFS_PATHLEN - pos - 1; if (info.mode == 0) info.mode = de->mode; } devname_offset = info.devname - (char *) &info; rpos = *ppos; if (rpos < devname_offset) { /* Copy parts of the header */ tlen = devname_offset - rpos; if (tlen > len) tlen = len; if ( copy_to_user (buf, (char *) &info + rpos, tlen) ) { return -EFAULT; } rpos += tlen; buf += tlen; len -= tlen; } if ( (rpos >= devname_offset) && (len > 0) ) { /* Copy the name */ tlen = info.namelen + 1; if (tlen > len) tlen = len; else done = TRUE; if ( copy_to_user (buf, info.devname + pos + rpos - devname_offset, tlen) ) { return -EFAULT; } rpos += tlen; } tlen = rpos - *ppos; if (done) { unsigned int next_pos = fs_info->devfsd_buf_out + 1; if (next_pos >= devfsd_buf_size) next_pos = 0; fs_info->devfsd_buf_out = next_pos; *ppos = 0; } else *ppos = rpos; return tlen; } /* End Function devfsd_read */ static int devfsd_ioctl (struct inode *inode, struct file *file, unsigned int cmd, unsigned long arg) { int ival; struct fs_info *fs_info = inode->i_sb->u.generic_sbp; switch (cmd) { case DEVFSDIOC_GET_PROTO_REV: ival = DEVFSD_PROTOCOL_REVISION_KERNEL; if ( copy_to_user ( (void *)arg, &ival, sizeof ival ) ) return -EFAULT; break; case DEVFSDIOC_SET_EVENT_MASK: /* Ensure only one reader has access to the queue. This scheme will work even if the global kernel lock were to be removed, because it doesn't matter who gets in first, as long as only one gets it */ if (fs_info->devfsd_task == NULL) { #ifdef __SMP__ /* Looks like no-one has it: check again and grab, with interrupts disabled */ __cli (); if (fs_info->devfsd_task == NULL) #endif { fs_info->devfsd_event_mask = 0; /* Temporary disable */ fs_info->devfsd_task = current; } #ifdef __SMP__ __sti (); #endif } /* Verify the task has grabbed the queue */ if (fs_info->devfsd_task != current) return -EBUSY; fs_info->devfsd_file = file; fs_info->devfsd_buffer = (void *) __get_free_page (GFP_KERNEL); if (fs_info->devfsd_buffer == NULL) { devfsd_close (inode, file); return -ENOMEM; } fs_info->devfsd_buf_out = fs_info->devfsd_buf_in; fs_info->devfsd_event_mask = arg; /* Let the masses come forth */ break; case DEVFSDIOC_RELEASE_EVENT_QUEUE: if (fs_info->devfsd_file != file) return -EPERM; return devfsd_close (inode, file); /*break;*/ #ifdef CONFIG_DEVFS_DEBUG case DEVFSDIOC_SET_DEBUG_MASK: if ( copy_from_user (&ival, (void *) arg, sizeof ival) )return -EFAULT; devfs_debug = ival; break; #endif default: return -ENOIOCTLCMD; } return 0; } /* End Function devfsd_ioctl */ static int devfsd_close (struct inode *inode, struct file *file) { struct fs_info *fs_info = inode->i_sb->u.generic_sbp; if (fs_info->devfsd_file != file) return 0; fs_info->devfsd_event_mask = 0; fs_info->devfsd_file = NULL; if (fs_info->devfsd_buffer) { while (fs_info->devfsd_buffer_in_use) schedule (); free_page ( (unsigned long) fs_info->devfsd_buffer ); } fs_info->devfsd_buffer = NULL; fs_info->devfsd_task = NULL; wake_up (&fs_info->revalidate_wait_queue); return 0; } /* End Function devfsd_close */ #ifdef MODULE int init_module (void) #else int __init init_devfs_fs (void) #endif { printk ("%s: v%s Richard Gooch (rgooch@atnf.csiro.au)\n", DEVFS_NAME, DEVFS_VERSION); #if defined(CONFIG_DEVFS_DEBUG) && !defined(MODULE) devfs_debug = devfs_debug_init; printk ("%s: devfs_debug: 0x%0x\n", DEVFS_NAME, devfs_debug); #endif #if !defined(MODULE) printk ("%s: boot_options: 0x%0x\n", DEVFS_NAME, boot_options); #endif return register_filesystem (&devfs_fs_type); } #ifndef MODULE void __init mount_devfs_fs (void) { int err; extern int do_mount (struct block_device *bdev, const char *dev_name, const char *dir_name, const char * type, int flags, void * data); if ( (boot_options & OPTION_NOMOUNT) ) return; err = do_mount (NULL, "none", "/dev", "devfs", 0, ""); if (err == 0) printk ("Mounted devfs on /dev\n"); else printk ("Warning: unable to mount devfs, err: %d\n", err); } /* End Function mount_devfs_fs */ #endif #ifdef MODULE static void free_entry (struct devfs_entry *parent) { struct devfs_entry *de, *next; if (parent == NULL) return; for (de = parent->u.dir.first; de != NULL; de = next) { next = de->next; if (de->first_inode != NULL) { printk ("%s: free_entry(): unfreed inodes!\n", DEVFS_NAME); } if ( S_ISDIR (de->mode) ) { /* Recursively free the subdirectories: this is a stack chomper */ free_entry (de); } else kfree (de); } kfree (parent); } /* End Function free_entry */ void cleanup_module (void) { unregister_filesystem (&devfs_fs_type); if (first_fs != NULL) { printk ("%s: cleanup_module(): still mounted mounted filesystems!\n", DEVFS_NAME); } free_entry (root_entry); } #endif /* MODULE */