2005-04-17 02:20:36 +04:00
/*
* dir . c - Operations for sysfs directories .
*/
# undef DEBUG
# include <linux/fs.h>
# include <linux/mount.h>
# include <linux/module.h>
# include <linux/kobject.h>
2005-06-23 11:09:12 +04:00
# include <linux/namei.h>
2006-12-20 12:52:44 +03:00
# include <asm/semaphore.h>
2005-04-17 02:20:36 +04:00
# include "sysfs.h"
DECLARE_RWSEM ( sysfs_rename_sem ) ;
static void sysfs_d_iput ( struct dentry * dentry , struct inode * inode )
{
struct sysfs_dirent * sd = dentry - > d_fsdata ;
if ( sd ) {
BUG_ON ( sd - > s_dentry ! = dentry ) ;
sd - > s_dentry = NULL ;
sysfs_put ( sd ) ;
}
iput ( inode ) ;
}
static struct dentry_operations sysfs_dentry_ops = {
. d_iput = sysfs_d_iput ,
} ;
/*
* Allocates a new sysfs_dirent and links it to the parent sysfs_dirent
*/
2007-01-24 22:35:52 +03:00
static struct sysfs_dirent * __sysfs_new_dirent ( void * element )
2005-04-17 02:20:36 +04:00
{
struct sysfs_dirent * sd ;
2007-02-10 12:45:03 +03:00
sd = kmem_cache_zalloc ( sysfs_dir_cachep , GFP_KERNEL ) ;
2005-04-17 02:20:36 +04:00
if ( ! sd )
return NULL ;
atomic_set ( & sd - > s_count , 1 ) ;
2006-08-03 20:06:25 +04:00
atomic_set ( & sd - > s_event , 1 ) ;
2005-04-17 02:20:36 +04:00
INIT_LIST_HEAD ( & sd - > s_children ) ;
2007-01-24 22:35:52 +03:00
INIT_LIST_HEAD ( & sd - > s_sibling ) ;
2005-04-17 02:20:36 +04:00
sd - > s_element = element ;
return sd ;
}
2007-01-24 22:35:52 +03:00
static void __sysfs_list_dirent ( struct sysfs_dirent * parent_sd ,
struct sysfs_dirent * sd )
{
if ( sd )
list_add ( & sd - > s_sibling , & parent_sd - > s_children ) ;
}
static struct sysfs_dirent * sysfs_new_dirent ( struct sysfs_dirent * parent_sd ,
void * element )
{
struct sysfs_dirent * sd ;
sd = __sysfs_new_dirent ( element ) ;
__sysfs_list_dirent ( parent_sd , sd ) ;
return sd ;
}
2006-04-02 15:59:55 +04:00
/*
2006-03-09 17:10:14 +03:00
*
* Return - EEXIST if there is already a sysfs element with the same name for
* the same parent .
*
* called with parent inode ' s i_mutex held
*/
int sysfs_dirent_exist ( struct sysfs_dirent * parent_sd ,
const unsigned char * new )
{
struct sysfs_dirent * sd ;
list_for_each_entry ( sd , & parent_sd - > s_children , s_sibling ) {
if ( sd - > s_element ) {
const unsigned char * existing = sysfs_get_name ( sd ) ;
if ( strcmp ( existing , new ) )
continue ;
else
return - EEXIST ;
}
}
return 0 ;
}
2007-01-24 22:35:52 +03:00
static struct sysfs_dirent *
__sysfs_make_dirent ( struct dentry * dentry , void * element , mode_t mode , int type )
2005-04-17 02:20:36 +04:00
{
struct sysfs_dirent * sd ;
2007-01-24 22:35:52 +03:00
sd = __sysfs_new_dirent ( element ) ;
2005-04-17 02:20:36 +04:00
if ( ! sd )
2007-01-24 22:35:52 +03:00
goto out ;
2005-04-17 02:20:36 +04:00
sd - > s_mode = mode ;
sd - > s_type = type ;
sd - > s_dentry = dentry ;
if ( dentry ) {
dentry - > d_fsdata = sysfs_get ( sd ) ;
dentry - > d_op = & sysfs_dentry_ops ;
}
2007-01-24 22:35:52 +03:00
out :
return sd ;
}
int sysfs_make_dirent ( struct sysfs_dirent * parent_sd , struct dentry * dentry ,
void * element , umode_t mode , int type )
{
struct sysfs_dirent * sd ;
sd = __sysfs_make_dirent ( dentry , element , mode , type ) ;
__sysfs_list_dirent ( parent_sd , sd ) ;
return sd ? 0 : - ENOMEM ;
2005-04-17 02:20:36 +04:00
}
static int init_dir ( struct inode * inode )
{
inode - > i_op = & sysfs_dir_inode_operations ;
inode - > i_fop = & sysfs_dir_operations ;
/* directory inodes start off with i_nlink == 2 (for "." entry) */
2006-10-01 10:29:04 +04:00
inc_nlink ( inode ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
static int init_file ( struct inode * inode )
{
inode - > i_size = PAGE_SIZE ;
inode - > i_fop = & sysfs_file_operations ;
return 0 ;
}
static int init_symlink ( struct inode * inode )
{
inode - > i_op = & sysfs_symlink_inode_operations ;
return 0 ;
}
static int create_dir ( struct kobject * k , struct dentry * p ,
const char * n , struct dentry * * d )
{
int error ;
umode_t mode = S_IFDIR | S_IRWXU | S_IRUGO | S_IXUGO ;
2006-01-10 02:59:24 +03:00
mutex_lock ( & p - > d_inode - > i_mutex ) ;
2005-06-23 11:09:12 +04:00
* d = lookup_one_len ( n , p , strlen ( n ) ) ;
2005-04-17 02:20:36 +04:00
if ( ! IS_ERR ( * d ) ) {
2006-03-09 17:10:14 +03:00
if ( sysfs_dirent_exist ( p - > d_fsdata , n ) )
error = - EEXIST ;
else
error = sysfs_make_dirent ( p - > d_fsdata , * d , k , mode ,
SYSFS_DIR ) ;
2005-04-17 02:20:36 +04:00
if ( ! error ) {
2005-05-31 09:08:12 +04:00
error = sysfs_create ( * d , mode , init_dir ) ;
2005-04-17 02:20:36 +04:00
if ( ! error ) {
2006-10-01 10:29:04 +04:00
inc_nlink ( p - > d_inode ) ;
2005-04-17 02:20:36 +04:00
( * d ) - > d_op = & sysfs_dentry_ops ;
d_rehash ( * d ) ;
}
}
2005-05-31 09:08:12 +04:00
if ( error & & ( error ! = - EEXIST ) ) {
2005-11-23 17:15:44 +03:00
struct sysfs_dirent * sd = ( * d ) - > d_fsdata ;
if ( sd ) {
list_del_init ( & sd - > s_sibling ) ;
sysfs_put ( sd ) ;
}
2005-04-17 02:20:36 +04:00
d_drop ( * d ) ;
2005-05-31 09:08:12 +04:00
}
2005-04-17 02:20:36 +04:00
dput ( * d ) ;
} else
error = PTR_ERR ( * d ) ;
2006-01-10 02:59:24 +03:00
mutex_unlock ( & p - > d_inode - > i_mutex ) ;
2005-04-17 02:20:36 +04:00
return error ;
}
int sysfs_create_subdir ( struct kobject * k , const char * n , struct dentry * * d )
{
return create_dir ( k , k - > dentry , n , d ) ;
}
/**
* sysfs_create_dir - create a directory for an object .
* @ kobj : object we ' re creating directory for .
2007-01-24 22:35:52 +03:00
* @ shadow_parent : parent parent object .
2005-04-17 02:20:36 +04:00
*/
2007-01-24 22:35:52 +03:00
int sysfs_create_dir ( struct kobject * kobj , struct dentry * shadow_parent )
2005-04-17 02:20:36 +04:00
{
struct dentry * dentry = NULL ;
struct dentry * parent ;
int error = 0 ;
BUG_ON ( ! kobj ) ;
2007-01-24 22:35:52 +03:00
if ( shadow_parent )
parent = shadow_parent ;
else if ( kobj - > parent )
2005-04-17 02:20:36 +04:00
parent = kobj - > parent - > dentry ;
else if ( sysfs_mount & & sysfs_mount - > mnt_sb )
parent = sysfs_mount - > mnt_sb - > s_root ;
else
return - EFAULT ;
error = create_dir ( kobj , parent , kobject_name ( kobj ) , & dentry ) ;
if ( ! error )
kobj - > dentry = dentry ;
return error ;
}
/* attaches attribute's sysfs_dirent to the dentry corresponding to the
* attribute file
*/
static int sysfs_attach_attr ( struct sysfs_dirent * sd , struct dentry * dentry )
{
struct attribute * attr = NULL ;
struct bin_attribute * bin_attr = NULL ;
int ( * init ) ( struct inode * ) = NULL ;
int error = 0 ;
if ( sd - > s_type & SYSFS_KOBJ_BIN_ATTR ) {
bin_attr = sd - > s_element ;
attr = & bin_attr - > attr ;
} else {
attr = sd - > s_element ;
init = init_file ;
}
2005-05-31 09:08:12 +04:00
dentry - > d_fsdata = sysfs_get ( sd ) ;
sd - > s_dentry = dentry ;
2005-04-17 02:20:36 +04:00
error = sysfs_create ( dentry , ( attr - > mode & S_IALLUGO ) | S_IFREG , init ) ;
2005-05-31 09:08:12 +04:00
if ( error ) {
sysfs_put ( sd ) ;
2005-04-17 02:20:36 +04:00
return error ;
2005-05-31 09:08:12 +04:00
}
2005-04-17 02:20:36 +04:00
if ( bin_attr ) {
dentry - > d_inode - > i_size = bin_attr - > size ;
dentry - > d_inode - > i_fop = & bin_fops ;
}
dentry - > d_op = & sysfs_dentry_ops ;
d_rehash ( dentry ) ;
return 0 ;
}
static int sysfs_attach_link ( struct sysfs_dirent * sd , struct dentry * dentry )
{
int err = 0 ;
2005-05-31 09:08:12 +04:00
dentry - > d_fsdata = sysfs_get ( sd ) ;
sd - > s_dentry = dentry ;
2005-04-17 02:20:36 +04:00
err = sysfs_create ( dentry , S_IFLNK | S_IRWXUGO , init_symlink ) ;
if ( ! err ) {
dentry - > d_op = & sysfs_dentry_ops ;
d_rehash ( dentry ) ;
2005-05-31 09:08:12 +04:00
} else
sysfs_put ( sd ) ;
2005-04-17 02:20:36 +04:00
return err ;
}
static struct dentry * sysfs_lookup ( struct inode * dir , struct dentry * dentry ,
struct nameidata * nd )
{
struct sysfs_dirent * parent_sd = dentry - > d_parent - > d_fsdata ;
struct sysfs_dirent * sd ;
int err = 0 ;
list_for_each_entry ( sd , & parent_sd - > s_children , s_sibling ) {
if ( sd - > s_type & SYSFS_NOT_PINNED ) {
const unsigned char * name = sysfs_get_name ( sd ) ;
if ( strcmp ( name , dentry - > d_name . name ) )
continue ;
if ( sd - > s_type & SYSFS_KOBJ_LINK )
err = sysfs_attach_link ( sd , dentry ) ;
else
err = sysfs_attach_attr ( sd , dentry ) ;
break ;
}
}
return ERR_PTR ( err ) ;
}
2007-02-12 11:55:40 +03:00
const struct inode_operations sysfs_dir_inode_operations = {
2005-04-17 02:20:36 +04:00
. lookup = sysfs_lookup ,
2005-05-31 09:09:14 +04:00
. setattr = sysfs_setattr ,
2005-04-17 02:20:36 +04:00
} ;
static void remove_dir ( struct dentry * d )
{
struct dentry * parent = dget ( d - > d_parent ) ;
struct sysfs_dirent * sd ;
2006-01-10 02:59:24 +03:00
mutex_lock ( & parent - > d_inode - > i_mutex ) ;
2005-04-17 02:20:36 +04:00
d_delete ( d ) ;
sd = d - > d_fsdata ;
list_del_init ( & sd - > s_sibling ) ;
sysfs_put ( sd ) ;
if ( d - > d_inode )
simple_rmdir ( parent - > d_inode , d ) ;
pr_debug ( " o %s removing done (%d) \n " , d - > d_name . name ,
atomic_read ( & d - > d_count ) ) ;
2006-01-10 02:59:24 +03:00
mutex_unlock ( & parent - > d_inode - > i_mutex ) ;
2005-04-17 02:20:36 +04:00
dput ( parent ) ;
}
void sysfs_remove_subdir ( struct dentry * d )
{
remove_dir ( d ) ;
}
2007-01-24 22:35:52 +03:00
static void __sysfs_remove_dir ( struct dentry * dentry )
2005-04-17 02:20:36 +04:00
{
struct sysfs_dirent * parent_sd ;
struct sysfs_dirent * sd , * tmp ;
2007-01-24 22:35:52 +03:00
dget ( dentry ) ;
2005-04-17 02:20:36 +04:00
if ( ! dentry )
return ;
pr_debug ( " sysfs %s: removing dir \n " , dentry - > d_name . name ) ;
2006-01-10 02:59:24 +03:00
mutex_lock ( & dentry - > d_inode - > i_mutex ) ;
2005-04-17 02:20:36 +04:00
parent_sd = dentry - > d_fsdata ;
list_for_each_entry_safe ( sd , tmp , & parent_sd - > s_children , s_sibling ) {
if ( ! sd - > s_element | | ! ( sd - > s_type & SYSFS_NOT_PINNED ) )
continue ;
list_del_init ( & sd - > s_sibling ) ;
sysfs_drop_dentry ( sd , dentry ) ;
sysfs_put ( sd ) ;
}
2006-01-10 02:59:24 +03:00
mutex_unlock ( & dentry - > d_inode - > i_mutex ) ;
2005-04-17 02:20:36 +04:00
remove_dir ( dentry ) ;
/**
* Drop reference from dget ( ) on entrance .
*/
dput ( dentry ) ;
2007-01-24 22:35:52 +03:00
}
/**
* sysfs_remove_dir - remove an object ' s directory .
* @ kobj : object .
*
* The only thing special about this is that we remove any files in
* the directory before we remove the directory , and we ' ve inlined
* what used to be sysfs_rmdir ( ) below , instead of calling separately .
*/
void sysfs_remove_dir ( struct kobject * kobj )
{
__sysfs_remove_dir ( kobj - > dentry ) ;
2006-03-17 02:44:26 +03:00
kobj - > dentry = NULL ;
2005-04-17 02:20:36 +04:00
}
2007-01-24 22:35:52 +03:00
int sysfs_rename_dir ( struct kobject * kobj , struct dentry * new_parent ,
const char * new_name )
2005-04-17 02:20:36 +04:00
{
int error = 0 ;
2007-01-24 22:35:52 +03:00
struct dentry * new_dentry ;
2005-04-17 02:20:36 +04:00
2007-01-24 22:35:52 +03:00
if ( ! new_parent )
return - EFAULT ;
2005-04-17 02:20:36 +04:00
down_write ( & sysfs_rename_sem ) ;
2007-01-24 22:35:52 +03:00
mutex_lock ( & new_parent - > d_inode - > i_mutex ) ;
2005-04-17 02:20:36 +04:00
2007-01-24 22:35:52 +03:00
new_dentry = lookup_one_len ( new_name , new_parent , strlen ( new_name ) ) ;
2005-04-17 02:20:36 +04:00
if ( ! IS_ERR ( new_dentry ) ) {
2007-01-24 22:35:52 +03:00
/* By allowing two different directories with the
* same d_parent we allow this routine to move
* between different shadows of the same directory
*/
if ( kobj - > dentry - > d_parent - > d_inode ! = new_parent - > d_inode )
return - EINVAL ;
else if ( new_dentry - > d_parent - > d_inode ! = new_parent - > d_inode )
error = - EINVAL ;
else if ( new_dentry = = kobj - > dentry )
error = - EINVAL ;
else if ( ! new_dentry - > d_inode ) {
2005-04-17 02:20:36 +04:00
error = kobject_set_name ( kobj , " %s " , new_name ) ;
if ( ! error ) {
2007-01-24 22:35:52 +03:00
struct sysfs_dirent * sd , * parent_sd ;
2005-04-17 02:20:36 +04:00
d_add ( new_dentry , NULL ) ;
d_move ( kobj - > dentry , new_dentry ) ;
2007-01-24 22:35:52 +03:00
sd = kobj - > dentry - > d_fsdata ;
parent_sd = new_parent - > d_fsdata ;
list_del_init ( & sd - > s_sibling ) ;
list_add ( & sd - > s_sibling , & parent_sd - > s_children ) ;
2005-04-17 02:20:36 +04:00
}
else
d_drop ( new_dentry ) ;
} else
error = - EEXIST ;
dput ( new_dentry ) ;
}
2007-01-24 22:35:52 +03:00
mutex_unlock ( & new_parent - > d_inode - > i_mutex ) ;
2005-04-17 02:20:36 +04:00
up_write ( & sysfs_rename_sem ) ;
return error ;
}
2006-11-20 19:07:51 +03:00
int sysfs_move_dir ( struct kobject * kobj , struct kobject * new_parent )
{
struct dentry * old_parent_dentry , * new_parent_dentry , * new_dentry ;
struct sysfs_dirent * new_parent_sd , * sd ;
int error ;
old_parent_dentry = kobj - > parent ?
kobj - > parent - > dentry : sysfs_mount - > mnt_sb - > s_root ;
2007-01-08 22:16:44 +03:00
new_parent_dentry = new_parent ?
new_parent - > dentry : sysfs_mount - > mnt_sb - > s_root ;
2006-11-20 19:07:51 +03:00
again :
mutex_lock ( & old_parent_dentry - > d_inode - > i_mutex ) ;
if ( ! mutex_trylock ( & new_parent_dentry - > d_inode - > i_mutex ) ) {
mutex_unlock ( & old_parent_dentry - > d_inode - > i_mutex ) ;
goto again ;
}
new_parent_sd = new_parent_dentry - > d_fsdata ;
sd = kobj - > dentry - > d_fsdata ;
new_dentry = lookup_one_len ( kobj - > name , new_parent_dentry ,
strlen ( kobj - > name ) ) ;
if ( IS_ERR ( new_dentry ) ) {
error = PTR_ERR ( new_dentry ) ;
goto out ;
} else
error = 0 ;
d_add ( new_dentry , NULL ) ;
d_move ( kobj - > dentry , new_dentry ) ;
dput ( new_dentry ) ;
/* Remove from old parent's list and insert into new parent's list. */
list_del_init ( & sd - > s_sibling ) ;
list_add ( & sd - > s_sibling , & new_parent_sd - > s_children ) ;
out :
mutex_unlock ( & new_parent_dentry - > d_inode - > i_mutex ) ;
mutex_unlock ( & old_parent_dentry - > d_inode - > i_mutex ) ;
return error ;
}
2005-04-17 02:20:36 +04:00
static int sysfs_dir_open ( struct inode * inode , struct file * file )
{
2006-12-08 13:36:36 +03:00
struct dentry * dentry = file - > f_path . dentry ;
2005-04-17 02:20:36 +04:00
struct sysfs_dirent * parent_sd = dentry - > d_fsdata ;
2006-01-10 02:59:24 +03:00
mutex_lock ( & dentry - > d_inode - > i_mutex ) ;
2005-04-17 02:20:36 +04:00
file - > private_data = sysfs_new_dirent ( parent_sd , NULL ) ;
2006-01-10 02:59:24 +03:00
mutex_unlock ( & dentry - > d_inode - > i_mutex ) ;
2005-04-17 02:20:36 +04:00
return file - > private_data ? 0 : - ENOMEM ;
}
static int sysfs_dir_close ( struct inode * inode , struct file * file )
{
2006-12-08 13:36:36 +03:00
struct dentry * dentry = file - > f_path . dentry ;
2005-04-17 02:20:36 +04:00
struct sysfs_dirent * cursor = file - > private_data ;
2006-01-10 02:59:24 +03:00
mutex_lock ( & dentry - > d_inode - > i_mutex ) ;
2005-04-17 02:20:36 +04:00
list_del_init ( & cursor - > s_sibling ) ;
2006-01-10 02:59:24 +03:00
mutex_unlock ( & dentry - > d_inode - > i_mutex ) ;
2005-04-17 02:20:36 +04:00
release_sysfs_dirent ( cursor ) ;
return 0 ;
}
/* Relationship between s_mode and the DT_xxx types */
static inline unsigned char dt_type ( struct sysfs_dirent * sd )
{
return ( sd - > s_mode > > 12 ) & 15 ;
}
static int sysfs_readdir ( struct file * filp , void * dirent , filldir_t filldir )
{
2006-12-08 13:36:36 +03:00
struct dentry * dentry = filp - > f_path . dentry ;
2005-04-17 02:20:36 +04:00
struct sysfs_dirent * parent_sd = dentry - > d_fsdata ;
struct sysfs_dirent * cursor = filp - > private_data ;
struct list_head * p , * q = & cursor - > s_sibling ;
ino_t ino ;
int i = filp - > f_pos ;
switch ( i ) {
case 0 :
ino = dentry - > d_inode - > i_ino ;
if ( filldir ( dirent , " . " , 1 , i , ino , DT_DIR ) < 0 )
break ;
filp - > f_pos + + ;
i + + ;
/* fallthrough */
case 1 :
ino = parent_ino ( dentry ) ;
if ( filldir ( dirent , " .. " , 2 , i , ino , DT_DIR ) < 0 )
break ;
filp - > f_pos + + ;
i + + ;
/* fallthrough */
default :
2006-06-26 11:24:40 +04:00
if ( filp - > f_pos = = 2 )
list_move ( q , & parent_sd - > s_children ) ;
2005-04-17 02:20:36 +04:00
for ( p = q - > next ; p ! = & parent_sd - > s_children ; p = p - > next ) {
struct sysfs_dirent * next ;
const char * name ;
int len ;
next = list_entry ( p , struct sysfs_dirent ,
s_sibling ) ;
if ( ! next - > s_element )
continue ;
name = sysfs_get_name ( next ) ;
len = strlen ( name ) ;
if ( next - > s_dentry )
ino = next - > s_dentry - > d_inode - > i_ino ;
else
ino = iunique ( sysfs_sb , 2 ) ;
if ( filldir ( dirent , name , len , filp - > f_pos , ino ,
dt_type ( next ) ) < 0 )
return 0 ;
2006-06-26 11:24:40 +04:00
list_move ( q , p ) ;
2005-04-17 02:20:36 +04:00
p = q ;
filp - > f_pos + + ;
}
}
return 0 ;
}
static loff_t sysfs_dir_lseek ( struct file * file , loff_t offset , int origin )
{
2006-12-08 13:36:36 +03:00
struct dentry * dentry = file - > f_path . dentry ;
2005-04-17 02:20:36 +04:00
2006-01-10 02:59:24 +03:00
mutex_lock ( & dentry - > d_inode - > i_mutex ) ;
2005-04-17 02:20:36 +04:00
switch ( origin ) {
case 1 :
offset + = file - > f_pos ;
case 0 :
if ( offset > = 0 )
break ;
default :
2006-12-08 13:36:36 +03:00
mutex_unlock ( & file - > f_path . dentry - > d_inode - > i_mutex ) ;
2005-04-17 02:20:36 +04:00
return - EINVAL ;
}
if ( offset ! = file - > f_pos ) {
file - > f_pos = offset ;
if ( file - > f_pos > = 2 ) {
struct sysfs_dirent * sd = dentry - > d_fsdata ;
struct sysfs_dirent * cursor = file - > private_data ;
struct list_head * p ;
loff_t n = file - > f_pos - 2 ;
list_del ( & cursor - > s_sibling ) ;
p = sd - > s_children . next ;
while ( n & & p ! = & sd - > s_children ) {
struct sysfs_dirent * next ;
next = list_entry ( p , struct sysfs_dirent ,
s_sibling ) ;
if ( next - > s_element )
n - - ;
p = p - > next ;
}
list_add_tail ( & cursor - > s_sibling , p ) ;
}
}
2006-01-10 02:59:24 +03:00
mutex_unlock ( & dentry - > d_inode - > i_mutex ) ;
2005-04-17 02:20:36 +04:00
return offset ;
}
2007-01-24 22:35:52 +03:00
/**
* sysfs_make_shadowed_dir - Setup so a directory can be shadowed
* @ kobj : object we ' re creating shadow of .
*/
int sysfs_make_shadowed_dir ( struct kobject * kobj ,
void * ( * follow_link ) ( struct dentry * , struct nameidata * ) )
{
struct inode * inode ;
struct inode_operations * i_op ;
inode = kobj - > dentry - > d_inode ;
if ( inode - > i_op ! = & sysfs_dir_inode_operations )
return - EINVAL ;
i_op = kmalloc ( sizeof ( * i_op ) , GFP_KERNEL ) ;
if ( ! i_op )
return - ENOMEM ;
memcpy ( i_op , & sysfs_dir_inode_operations , sizeof ( * i_op ) ) ;
i_op - > follow_link = follow_link ;
/* Locking of inode->i_op?
* Since setting i_op is a single word write and they
* are atomic we should be ok here .
*/
inode - > i_op = i_op ;
return 0 ;
}
/**
* sysfs_create_shadow_dir - create a shadow directory for an object .
* @ kobj : object we ' re creating directory for .
*
* sysfs_make_shadowed_dir must already have been called on this
* directory .
*/
struct dentry * sysfs_create_shadow_dir ( struct kobject * kobj )
{
struct sysfs_dirent * sd ;
struct dentry * parent , * dir , * shadow ;
struct inode * inode ;
dir = kobj - > dentry ;
inode = dir - > d_inode ;
parent = dir - > d_parent ;
shadow = ERR_PTR ( - EINVAL ) ;
if ( ! sysfs_is_shadowed_inode ( inode ) )
goto out ;
shadow = d_alloc ( parent , & dir - > d_name ) ;
if ( ! shadow )
goto nomem ;
sd = __sysfs_make_dirent ( shadow , kobj , inode - > i_mode , SYSFS_DIR ) ;
if ( ! sd )
goto nomem ;
d_instantiate ( shadow , igrab ( inode ) ) ;
inc_nlink ( inode ) ;
inc_nlink ( parent - > d_inode ) ;
shadow - > d_op = & sysfs_dentry_ops ;
dget ( shadow ) ; /* Extra count - pin the dentry in core */
out :
return shadow ;
nomem :
dput ( shadow ) ;
shadow = ERR_PTR ( - ENOMEM ) ;
goto out ;
}
/**
* sysfs_remove_shadow_dir - remove an object ' s directory .
* @ shadow : dentry of shadow directory
*
* The only thing special about this is that we remove any files in
* the directory before we remove the directory , and we ' ve inlined
* what used to be sysfs_rmdir ( ) below , instead of calling separately .
*/
void sysfs_remove_shadow_dir ( struct dentry * shadow )
{
__sysfs_remove_dir ( shadow ) ;
}
2006-03-28 13:56:42 +04:00
const struct file_operations sysfs_dir_operations = {
2005-04-17 02:20:36 +04:00
. open = sysfs_dir_open ,
. release = sysfs_dir_close ,
. llseek = sysfs_dir_lseek ,
. read = generic_read_dir ,
. readdir = sysfs_readdir ,
} ;