2007-09-20 12:31:38 +04:00
/*
* fs / sysfs / sysfs . h - sysfs internal header file
*
* Copyright ( c ) 2001 - 3 Patrick Mochel
* Copyright ( c ) 2007 SUSE Linux Products GmbH
* Copyright ( c ) 2007 Tejun Heo < teheo @ suse . de >
*
* This file is released under the GPLv2 .
*/
2010-01-03 00:37:12 +03:00
# include <linux/lockdep.h>
2011-01-10 09:18:25 +03:00
# include <linux/kobject_ns.h>
2009-09-09 22:25:37 +04:00
# include <linux/fs.h>
2011-07-26 01:55:57 +04:00
# include <linux/rbtree.h>
2009-09-09 22:25:37 +04:00
2007-09-20 11:05:12 +04:00
struct sysfs_open_dirent ;
2007-09-20 11:05:10 +04:00
/* type-specific structures for sysfs_dirent->s_* union members */
2007-06-13 22:45:15 +04:00
struct sysfs_elem_dir {
2007-09-20 11:05:10 +04:00
struct kobject * kobj ;
2011-07-22 03:59:22 +04:00
2012-03-09 01:03:10 +04:00
unsigned long subdirs ;
2011-12-19 08:05:43 +04:00
/* children rbtree starts here and goes through sd->s_rb */
struct rb_root children ;
2007-06-13 22:45:15 +04:00
} ;
struct sysfs_elem_symlink {
2007-09-20 11:05:10 +04:00
struct sysfs_dirent * target_sd ;
2007-06-13 22:45:15 +04:00
} ;
struct sysfs_elem_attr {
2007-09-20 11:05:10 +04:00
struct attribute * attr ;
2007-09-20 11:05:12 +04:00
struct sysfs_open_dirent * open ;
2007-06-13 22:45:15 +04:00
} ;
struct sysfs_elem_bin_attr {
2007-09-20 11:05:10 +04:00
struct bin_attribute * bin_attr ;
2009-03-04 22:57:20 +03:00
struct hlist_head buffers ;
2007-06-13 22:45:15 +04:00
} ;
2009-09-09 22:25:37 +04:00
struct sysfs_inode_attrs {
struct iattr ia_iattr ;
void * ia_secdata ;
u32 ia_secdata_len ;
} ;
2007-06-13 22:45:16 +04:00
/*
2007-09-20 11:05:10 +04:00
* sysfs_dirent - the building block of sysfs hierarchy . Each and
* every sysfs node is represented by single sysfs_dirent .
*
2007-06-13 22:45:16 +04:00
* As long as s_count reference is held , the sysfs_dirent itself is
* accessible . Dereferencing s_elem or any other outer entity
* requires s_active reference .
*/
2007-02-16 16:35:25 +03:00
struct sysfs_dirent {
atomic_t s_count ;
2007-06-13 22:45:18 +04:00
atomic_t s_active ;
2010-01-03 00:37:12 +03:00
# ifdef CONFIG_DEBUG_LOCK_ALLOC
struct lockdep_map dep_map ;
# endif
2007-09-20 11:05:10 +04:00
struct sysfs_dirent * s_parent ;
const char * s_name ;
2007-06-13 22:45:15 +04:00
2011-12-19 08:05:43 +04:00
struct rb_node s_rb ;
2011-07-26 01:55:57 +04:00
2011-07-22 04:01:12 +04:00
union {
struct completion * completion ;
struct sysfs_dirent * removed_list ;
} u ;
2010-05-04 01:23:15 +04:00
const void * s_ns ; /* namespace tag */
2011-12-19 08:05:43 +04:00
unsigned int s_hash ; /* ns + name hash */
2007-06-13 22:45:15 +04:00
union {
2007-09-20 11:05:11 +04:00
struct sysfs_elem_dir s_dir ;
struct sysfs_elem_symlink s_symlink ;
struct sysfs_elem_attr s_attr ;
struct sysfs_elem_bin_attr s_bin_attr ;
} ;
2007-06-13 22:45:15 +04:00
2011-12-19 08:07:23 +04:00
unsigned short s_flags ;
2011-07-25 07:44:53 +04:00
umode_t s_mode ;
2011-12-19 08:08:16 +04:00
unsigned int s_ino ;
2009-09-09 22:25:37 +04:00
struct sysfs_inode_attrs * s_iattr ;
2007-02-16 16:35:25 +03:00
} ;
2005-04-17 02:20:36 +04:00
2007-09-20 11:05:10 +04:00
# define SD_DEACTIVATED_BIAS INT_MIN
2007-06-13 22:45:16 +04:00
2007-09-20 11:05:10 +04:00
# define SYSFS_TYPE_MASK 0x00ff
2007-09-20 11:05:12 +04:00
# define SYSFS_DIR 0x0001
# define SYSFS_KOBJ_ATTR 0x0002
# define SYSFS_KOBJ_BIN_ATTR 0x0004
# define SYSFS_KOBJ_LINK 0x0008
2007-09-20 11:05:10 +04:00
# define SYSFS_COPY_NAME (SYSFS_DIR | SYSFS_KOBJ_LINK)
2010-02-12 02:20:00 +03:00
# define SYSFS_ACTIVE_REF (SYSFS_KOBJ_ATTR | SYSFS_KOBJ_BIN_ATTR)
2007-09-20 11:05:10 +04:00
2010-05-04 01:23:15 +04:00
/* identify any namespace tag on sysfs_dirents */
2011-12-19 08:07:23 +04:00
# define SYSFS_NS_TYPE_MASK 0xf00
2010-03-30 22:31:26 +04:00
# define SYSFS_NS_TYPE_SHIFT 8
# define SYSFS_FLAG_MASK ~(SYSFS_NS_TYPE_MASK|SYSFS_TYPE_MASK)
2011-12-19 08:07:23 +04:00
# define SYSFS_FLAG_REMOVED 0x02000
2007-09-20 11:05:10 +04:00
static inline unsigned int sysfs_type ( struct sysfs_dirent * sd )
{
return sd - > s_flags & SYSFS_TYPE_MASK ;
}
2010-05-04 01:23:15 +04:00
/*
* Return any namespace tags on this dirent .
* enum kobj_ns_type is defined in linux / kobject . h
*/
2010-03-30 22:31:26 +04:00
static inline enum kobj_ns_type sysfs_ns_type ( struct sysfs_dirent * sd )
{
return ( sd - > s_flags & SYSFS_NS_TYPE_MASK ) > > SYSFS_NS_TYPE_SHIFT ;
}
2010-01-03 00:37:12 +03:00
# ifdef CONFIG_DEBUG_LOCK_ALLOC
# define sysfs_dirent_init_lockdep(sd) \
do { \
2010-02-12 02:21:53 +03:00
struct attribute * attr = sd - > s_attr . attr ; \
struct lock_class_key * key = attr - > key ; \
if ( ! key ) \
key = & attr - > skey ; \
2010-01-03 00:37:12 +03:00
\
2010-02-12 02:21:53 +03:00
lockdep_init_map ( & sd - > dep_map , " s_active " , key , 0 ) ; \
2010-01-03 00:37:12 +03:00
} while ( 0 )
# else
# define sysfs_dirent_init_lockdep(sd) do {} while(0)
# endif
2007-09-20 11:05:10 +04:00
/*
* Context structure to be used while adding / removing nodes .
*/
2007-06-13 23:27:24 +04:00
struct sysfs_addrm_cxt {
struct sysfs_dirent * parent_sd ;
struct sysfs_dirent * removed ;
} ;
2007-09-20 11:05:10 +04:00
/*
* mount . c
*/
2010-05-04 01:23:15 +04:00
/*
* Each sb is associated with a set of namespace tags ( i . e .
* the network namespace of the task which mounted this sysfs
* instance ) .
*/
2010-03-30 22:31:24 +04:00
struct sysfs_super_info {
2011-06-09 05:13:01 +04:00
void * ns [ KOBJ_NS_TYPES ] ;
2010-03-30 22:31:24 +04:00
} ;
# define sysfs_info(SB) ((struct sysfs_super_info *)(SB->s_fs_info))
2007-06-13 23:27:25 +04:00
extern struct sysfs_dirent sysfs_root ;
2006-12-07 07:33:20 +03:00
extern struct kmem_cache * sysfs_dir_cachep ;
2005-04-17 02:20:36 +04:00
2007-09-20 11:05:10 +04:00
/*
* dir . c
*/
2007-06-13 23:27:23 +04:00
extern struct mutex sysfs_mutex ;
2007-09-20 11:05:10 +04:00
extern spinlock_t sysfs_assoc_lock ;
2012-06-08 04:51:39 +04:00
extern const struct dentry_operations sysfs_dentry_ops ;
2007-09-20 11:05:10 +04:00
2006-03-28 13:56:42 +04:00
extern const struct file_operations sysfs_dir_operations ;
2007-02-12 11:55:40 +03:00
extern const struct inode_operations sysfs_dir_inode_operations ;
2005-04-17 02:20:36 +04:00
2007-09-20 11:05:10 +04:00
struct dentry * sysfs_get_dentry ( struct sysfs_dirent * sd ) ;
2010-02-12 02:18:38 +03:00
struct sysfs_dirent * sysfs_get_active ( struct sysfs_dirent * sd ) ;
void sysfs_put_active ( struct sysfs_dirent * sd ) ;
2007-09-20 11:05:10 +04:00
void sysfs_addrm_start ( struct sysfs_addrm_cxt * acxt ,
struct sysfs_dirent * parent_sd ) ;
2008-06-10 13:09:08 +04:00
int __sysfs_add_one ( struct sysfs_addrm_cxt * acxt , struct sysfs_dirent * sd ) ;
2007-09-20 11:05:10 +04:00
int sysfs_add_one ( struct sysfs_addrm_cxt * acxt , struct sysfs_dirent * sd ) ;
void sysfs_remove_one ( struct sysfs_addrm_cxt * acxt , struct sysfs_dirent * sd ) ;
void sysfs_addrm_finish ( struct sysfs_addrm_cxt * acxt ) ;
struct sysfs_dirent * sysfs_find_dirent ( struct sysfs_dirent * parent_sd ,
2010-03-30 22:31:26 +04:00
const void * ns ,
2007-09-20 11:05:10 +04:00
const unsigned char * name ) ;
struct sysfs_dirent * sysfs_get_dirent ( struct sysfs_dirent * parent_sd ,
2010-03-30 22:31:26 +04:00
const void * ns ,
2007-09-20 11:05:10 +04:00
const unsigned char * name ) ;
struct sysfs_dirent * sysfs_new_dirent ( const char * name , umode_t mode , int type ) ;
void release_sysfs_dirent ( struct sysfs_dirent * sd ) ;
int sysfs_create_subdir ( struct kobject * kobj , const char * name ,
struct sysfs_dirent * * p_sd ) ;
void sysfs_remove_subdir ( struct sysfs_dirent * sd ) ;
2009-11-21 03:08:57 +03:00
int sysfs_rename ( struct sysfs_dirent * sd ,
2010-03-30 22:31:26 +04:00
struct sysfs_dirent * new_parent_sd , const void * ns , const char * new_name ) ;
2009-11-21 03:08:57 +03:00
2008-07-16 02:58:04 +04:00
static inline struct sysfs_dirent * __sysfs_get ( struct sysfs_dirent * sd )
2005-04-17 02:20:36 +04:00
{
2007-06-13 22:45:16 +04:00
if ( sd ) {
WARN_ON ( ! atomic_read ( & sd - > s_count ) ) ;
atomic_inc ( & sd - > s_count ) ;
}
return sd ;
2005-04-17 02:20:36 +04:00
}
2008-07-16 02:58:04 +04:00
# define sysfs_get(sd) __sysfs_get(sd)
2005-04-17 02:20:36 +04:00
2008-07-16 02:58:04 +04:00
static inline void __sysfs_put ( struct sysfs_dirent * sd )
2005-04-17 02:20:36 +04:00
{
2007-06-13 22:45:16 +04:00
if ( sd & & atomic_dec_and_test ( & sd - > s_count ) )
release_sysfs_dirent ( sd ) ;
}
2008-07-16 02:58:04 +04:00
# define sysfs_put(sd) __sysfs_put(sd)
2007-09-20 11:05:10 +04:00
/*
* inode . c
*/
2010-02-13 06:22:27 +03:00
struct inode * sysfs_get_inode ( struct super_block * sb , struct sysfs_dirent * sd ) ;
2010-06-05 06:21:54 +04:00
void sysfs_evict_inode ( struct inode * inode ) ;
2009-11-21 03:08:50 +03:00
int sysfs_sd_setattr ( struct sysfs_dirent * sd , struct iattr * iattr ) ;
2011-06-21 03:28:19 +04:00
int sysfs_permission ( struct inode * inode , int mask ) ;
2007-09-20 11:05:10 +04:00
int sysfs_setattr ( struct dentry * dentry , struct iattr * iattr ) ;
2009-11-21 03:08:53 +03:00
int sysfs_getattr ( struct vfsmount * mnt , struct dentry * dentry , struct kstat * stat ) ;
2009-09-09 22:25:37 +04:00
int sysfs_setxattr ( struct dentry * dentry , const char * name , const void * value ,
size_t size , int flags ) ;
2010-03-30 22:31:26 +04:00
int sysfs_hash_and_remove ( struct sysfs_dirent * dir_sd , const void * ns , const char * name ) ;
2007-10-17 10:25:46 +04:00
int sysfs_inode_init ( void ) ;
2007-09-20 11:05:10 +04:00
/*
* file . c
*/
extern const struct file_operations sysfs_file_operations ;
int sysfs_add_file ( struct sysfs_dirent * dir_sd ,
const struct attribute * attr , int type ) ;
2008-03-21 04:47:52 +03:00
int sysfs_add_file_mode ( struct sysfs_dirent * dir_sd ,
2011-07-25 07:44:53 +04:00
const struct attribute * attr , int type , umode_t amode ) ;
2007-09-20 11:05:10 +04:00
/*
* bin . c
*/
extern const struct file_operations bin_fops ;
2009-03-04 22:57:20 +03:00
void unmap_bin_file ( struct sysfs_dirent * attr_sd ) ;
2007-09-20 11:05:10 +04:00
/*
* symlink . c
*/
extern const struct inode_operations sysfs_symlink_inode_operations ;
2013-01-26 00:51:13 +04:00
int sysfs_create_link_sd ( struct sysfs_dirent * sd , struct kobject * target ,
const char * name ) ;