From: Maneesh Soni => changes in version 0.3 o Nil, just re-diffed => changes in version 0.2 o Nil, just re-diffed => Changes: o Removed the extra kobject_get from sysfs_release() ======================================================= o sysfs_create_file() will just link a new sysfs_dirent() structure representing the attribute file to the kobject's s_children list. o in sysfs_create() we take extra ref. only for dentries corresponding to non-regular files or in other words pin only non-leaf dentries. fs/sysfs/file.c | 63 +++++++++++++++++++++++++------------------------------ fs/sysfs/inode.c | 14 +++++++++--- 2 files changed, 40 insertions(+), 37 deletions(-) diff -upN reference/fs/sysfs/file.c current/fs/sysfs/file.c --- reference/fs/sysfs/file.c 2003-10-01 11:48:19.000000000 -0700 +++ current/fs/sysfs/file.c 2004-03-30 21:42:13.000000000 -0800 @@ -9,14 +9,6 @@ #include "sysfs.h" -static struct file_operations sysfs_file_operations; - -static int init_file(struct inode * inode) -{ - inode->i_size = PAGE_SIZE; - inode->i_fop = &sysfs_file_operations; - return 0; -} #define to_subsys(k) container_of(k,struct subsystem,kset.kobj) #define to_sattr(a) container_of(a,struct subsys_attribute,attr) @@ -77,8 +69,10 @@ struct sysfs_buffer { */ static int fill_read_buffer(struct file * file, struct sysfs_buffer * buffer) { - struct attribute * attr = file->f_dentry->d_fsdata; - struct kobject * kobj = file->f_dentry->d_parent->d_fsdata; + struct sysfs_dirent * sd_attr = file->f_dentry->d_fsdata; + struct attribute * attr = sd_attr->s_element; + struct sysfs_dirent * sd_kobj = file->f_dentry->d_parent->d_fsdata; + struct kobject * kobj = sd_kobj->s_element; struct sysfs_ops * ops = buffer->ops; int ret = 0; ssize_t count; @@ -198,8 +192,10 @@ fill_write_buffer(struct sysfs_buffer * static int flush_write_buffer(struct file * file, struct sysfs_buffer * buffer, size_t count) { - struct attribute * attr = file->f_dentry->d_fsdata; - struct kobject * kobj = file->f_dentry->d_parent->d_fsdata; + struct sysfs_dirent * sd_attr = file->f_dentry->d_fsdata; + struct attribute * attr = sd_attr->s_element; + struct sysfs_dirent * sd_kobj = file->f_dentry->d_parent->d_fsdata; + struct kobject * kobj = sd_kobj->s_element; struct sysfs_ops * ops = buffer->ops; return ops->store(kobj,attr,buffer->page,count); @@ -238,8 +234,10 @@ sysfs_write_file(struct file *file, cons static int check_perm(struct inode * inode, struct file * file) { - struct kobject * kobj = kobject_get(file->f_dentry->d_parent->d_fsdata); - struct attribute * attr = file->f_dentry->d_fsdata; + struct sysfs_dirent * sd_attr = file->f_dentry->d_fsdata; + struct attribute * attr = sd_attr->s_element; + struct sysfs_dirent * sd_kobj = file->f_dentry->d_parent->d_fsdata; + struct kobject * kobj = kobject_get(sd_kobj->s_element); struct sysfs_buffer * buffer; struct sysfs_ops * ops = NULL; int error = 0; @@ -320,8 +318,10 @@ static int sysfs_open_file(struct inode static int sysfs_release(struct inode * inode, struct file * filp) { - struct kobject * kobj = filp->f_dentry->d_parent->d_fsdata; - struct attribute * attr = filp->f_dentry->d_fsdata; + struct sysfs_dirent * sd_attr = filp->f_dentry->d_fsdata; + struct attribute * attr = sd_attr->s_element; + struct sysfs_dirent * sd_kobj = filp->f_dentry->d_parent->d_fsdata; + struct kobject * kobj = sd_kobj->s_element; struct sysfs_buffer * buffer = filp->private_data; if (kobj) @@ -336,7 +336,7 @@ static int sysfs_release(struct inode * return 0; } -static struct file_operations sysfs_file_operations = { +struct file_operations sysfs_file_operations = { .read = sysfs_read_file, .write = sysfs_write_file, .llseek = generic_file_llseek, @@ -345,23 +345,18 @@ static struct file_operations sysfs_file }; -int sysfs_add_file(struct dentry * dir, const struct attribute * attr) +int sysfs_add_file(struct dentry * parent, const struct attribute * attr, int t) { - struct dentry * dentry; - int error; + struct sysfs_dirent * sd; + struct sysfs_dirent * parent_sd = parent->d_fsdata; + int error = 0; - down(&dir->d_inode->i_sem); - dentry = sysfs_get_dentry(dir,attr->name); - if (!IS_ERR(dentry)) { - error = sysfs_create(dentry, - (attr->mode & S_IALLUGO) | S_IFREG, - init_file); - if (!error) - dentry->d_fsdata = (void *)attr; - dput(dentry); - } else - error = PTR_ERR(dentry); - up(&dir->d_inode->i_sem); + down(&parent->d_inode->i_sem); + sd = sysfs_new_dirent(parent_sd, (void *) attr, t); + if (!sd) + error = -ENOMEM; + up(&parent->d_inode->i_sem); + return error; } @@ -374,8 +369,8 @@ int sysfs_add_file(struct dentry * dir, int sysfs_create_file(struct kobject * kobj, const struct attribute * attr) { - if (kobj && attr) - return sysfs_add_file(kobj->dentry,attr); + if (kobj && kobj->dentry && attr) + return sysfs_add_file(kobj->dentry, attr, SYSFS_KOBJ_ATTR); return -EINVAL; } diff -upN reference/fs/sysfs/inode.c current/fs/sysfs/inode.c --- reference/fs/sysfs/inode.c 2003-10-01 11:47:05.000000000 -0700 +++ current/fs/sysfs/inode.c 2004-03-30 21:42:13.000000000 -0800 @@ -11,6 +11,8 @@ #include #include #include +#include "sysfs.h" + extern struct super_block * sysfs_sb; static struct address_space_operations sysfs_aops = { @@ -61,7 +63,8 @@ int sysfs_create(struct dentry * dentry, error = init(inode); if (!error) { d_instantiate(dentry, inode); - dget(dentry); /* Extra count - pin the dentry in core */ + if (S_ISDIR(mode)) + dget(dentry); /* pin only directory dentry in core */ } else iput(inode); Done: @@ -96,14 +99,19 @@ void sysfs_hash_and_remove(struct dentry pr_debug("sysfs: Removing %s (%d)\n", victim->d_name.name, atomic_read(&victim->d_count)); - d_delete(victim); - simple_unlink(dir->d_inode,victim); + if (S_ISDIR(victim->d_inode->i_mode)) { + d_delete(victim); + simple_unlink(dir->d_inode,victim); + } + else + d_drop(victim); } /* * Drop reference from sysfs_get_dentry() above. */ dput(victim); } + sysfs_remove_dirent(dir->d_fsdata, name); up(&dir->d_inode->i_sem); }