sysfs.h 3.6 KB
Newer Older
1 2 3 4 5 6 7
struct sysfs_dirent {
	atomic_t		s_count;
	struct list_head	s_sibling;
	struct list_head	s_children;
	void 			* s_element;
	int			s_type;
	umode_t			s_mode;
8
	ino_t			s_ino;
9 10 11 12
	struct dentry		* s_dentry;
	struct iattr		* s_iattr;
	atomic_t		s_event;
};
L
Linus Torvalds 已提交
13 14

extern struct vfsmount * sysfs_mount;
15
extern struct kmem_cache *sysfs_dir_cachep;
L
Linus Torvalds 已提交
16

17
extern void sysfs_delete_inode(struct inode *inode);
18
extern struct inode * sysfs_new_inode(mode_t mode, struct sysfs_dirent *);
L
Linus Torvalds 已提交
19 20
extern int sysfs_create(struct dentry *, int mode, int (*init)(struct inode *));

21
extern int sysfs_dirent_exist(struct sysfs_dirent *, const unsigned char *);
L
Linus Torvalds 已提交
22 23 24 25
extern int sysfs_make_dirent(struct sysfs_dirent *, struct dentry *, void *,
				umode_t, int);

extern int sysfs_add_file(struct dentry *, const struct attribute *, int);
26
extern int sysfs_hash_and_remove(struct dentry * dir, const char * name);
27
extern struct sysfs_dirent *sysfs_find(struct sysfs_dirent *dir, const char * name);
L
Linus Torvalds 已提交
28 29 30 31 32 33

extern int sysfs_create_subdir(struct kobject *, const char *, struct dentry **);
extern void sysfs_remove_subdir(struct dentry *);

extern const unsigned char * sysfs_get_name(struct sysfs_dirent *sd);
extern void sysfs_drop_dentry(struct sysfs_dirent *sd, struct dentry *parent);
34
extern int sysfs_setattr(struct dentry *dentry, struct iattr *iattr);
L
Linus Torvalds 已提交
35 36 37

extern struct rw_semaphore sysfs_rename_sem;
extern struct super_block * sysfs_sb;
38 39 40
extern const struct file_operations sysfs_dir_operations;
extern const struct file_operations sysfs_file_operations;
extern const struct file_operations bin_fops;
41 42
extern const struct inode_operations sysfs_dir_inode_operations;
extern const struct inode_operations sysfs_symlink_inode_operations;
L
Linus Torvalds 已提交
43 44 45 46 47 48

struct sysfs_symlink {
	char * link_name;
	struct kobject * target_kobj;
};

49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64
struct sysfs_buffer {
	struct list_head		associates;
	size_t				count;
	loff_t				pos;
	char				* page;
	struct sysfs_ops		* ops;
	struct semaphore		sem;
	int				orphaned;
	int				needs_read_fill;
	int				event;
};

struct sysfs_buffer_collection {
	struct list_head	associates;
};

L
Linus Torvalds 已提交
65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108
static inline struct kobject * to_kobj(struct dentry * dentry)
{
	struct sysfs_dirent * sd = dentry->d_fsdata;
	return ((struct kobject *) sd->s_element);
}

static inline struct attribute * to_attr(struct dentry * dentry)
{
	struct sysfs_dirent * sd = dentry->d_fsdata;
	return ((struct attribute *) sd->s_element);
}

static inline struct bin_attribute * to_bin_attr(struct dentry * dentry)
{
	struct sysfs_dirent * sd = dentry->d_fsdata;
	return ((struct bin_attribute *) sd->s_element);
}

static inline struct kobject *sysfs_get_kobject(struct dentry *dentry)
{
	struct kobject * kobj = NULL;

	spin_lock(&dcache_lock);
	if (!d_unhashed(dentry)) {
		struct sysfs_dirent * sd = dentry->d_fsdata;
		if (sd->s_type & SYSFS_KOBJ_LINK) {
			struct sysfs_symlink * sl = sd->s_element;
			kobj = kobject_get(sl->target_kobj);
		} else
			kobj = kobject_get(sd->s_element);
	}
	spin_unlock(&dcache_lock);

	return kobj;
}

static inline void release_sysfs_dirent(struct sysfs_dirent * sd)
{
	if (sd->s_type & SYSFS_KOBJ_LINK) {
		struct sysfs_symlink * sl = sd->s_element;
		kfree(sl->link_name);
		kobject_put(sl->target_kobj);
		kfree(sl);
	}
109
	kfree(sd->s_iattr);
L
Linus Torvalds 已提交
110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127
	kmem_cache_free(sysfs_dir_cachep, sd);
}

static inline struct sysfs_dirent * sysfs_get(struct sysfs_dirent * sd)
{
	if (sd) {
		WARN_ON(!atomic_read(&sd->s_count));
		atomic_inc(&sd->s_count);
	}
	return sd;
}

static inline void sysfs_put(struct sysfs_dirent * sd)
{
	if (atomic_dec_and_test(&sd->s_count))
		release_sysfs_dirent(sd);
}

128 129 130 131
static inline int sysfs_is_shadowed_inode(struct inode *inode)
{
	return S_ISDIR(inode->i_mode) && inode->i_op->follow_link;
}