转自:http://blog.csdn.net/npy_lp/article/details/78933292

内核源码:linux-2.6.38.8.tar.bz2

目标平台:ARM体系结构

sysfs是基于内存的文件系统,用于向用户空间导出内核对象并且能对其进行读写。

1、sysfs文件系统不支持特殊文件,只支持目录、普通文件(文本或二进制文件)和符号链接文件等三种类型,在内核中都使用struct  sysfs_dirent结构体来表示,相当于其他文件系统在硬盘或flash里的数据。源代码如下所示:

  1. /* fs/sysfs/sysfs.h */
  2. struct sysfs_dirent {
  3. atomic_t        s_count;    //struct sysfs_dirent结构体实例自身的引用计数
  4. atomic_t        s_active;   //struct sysfs_elem_*所涉及的外部对象的引用计数
  5. #ifdef CONFIG_DEBUG_LOCK_ALLOC
  6. struct lockdep_map  dep_map; //死锁检测模块,针对s_attr.attr中的key或skey
  7. #endif
  8. struct sysfs_dirent *s_parent;  //指向父节点
  9. struct sysfs_dirent *s_sibling; //同级节点链表,插入到父节点的s_dir.children链表中
  10. const char      *s_name;    //文件名
  11. const void      *s_ns;      //命名空间
  12. union {
  13. struct sysfs_elem_dir       s_dir;      //目录
  14. struct sysfs_elem_symlink   s_symlink;  //符号链接文件
  15. struct sysfs_elem_attr      s_attr;     //文本文件
  16. struct sysfs_elem_bin_attr  s_bin_attr; //二进制文件
  17. };
  18. unsigned int        s_flags;    //标志,表示struct sysfs_dirent类型、命名空间类型等信息
  19. unsigned short      s_mode;     //文件访问权限,包含文件类型信息
  20. ino_t           s_ino;          //对应于i节点号
  21. struct sysfs_inode_attrs *s_iattr;  //文件属性
  22. };
  23. struct sysfs_inode_attrs {
  24. struct iattr    ia_iattr;   //文件属性
  25. void        *ia_secdata;    //安全检测模块所用数据
  26. u32     ia_secdata_len;     //ia_secdata所指数据的长度
  27. };
  28. /* include/linux/fs.h */
  29. struct iattr {
  30. unsigned int    ia_valid;   //文件属性标志
  31. umode_t     ia_mode;        //文件类型及其访问权限
  32. uid_t       ia_uid;         //用户ID
  33. gid_t       ia_gid;         //组ID
  34. loff_t      ia_size;        //文件大小
  35. struct timespec ia_atime;   //访问时间
  36. struct timespec ia_mtime;   //数据修改时间
  37. struct timespec ia_ctime;   //元数据修改时间
  38. struct file *ia_file;   //辅助信息,用于想实现ftruncate等方法的文件系统
  39. };

struct  sysfs_dirent分为四种类型,如下所示:

  1. /* fs/sysfs/sysfs.h */
  2. struct sysfs_elem_dir {
  3. struct kobject      *kobj;  //指向内核对象
  4. //子节点链表,只有目录才有可能包含文件或子目录
  5. //子节点通过s_sibling成员以s_ino成员升序的方式插入到该链表
  6. struct sysfs_dirent *children;
  7. };
  8. struct sysfs_elem_symlink {
  9. struct sysfs_dirent *target_sd; //指向所链接的节点
  10. };
  11. struct sysfs_elem_attr {
  12. struct attribute    *attr;  //对象属性
  13. struct sysfs_open_dirent *open; //文件open信息,其中buffers成员是struct sysfs_buffer的链表
  14. };
  15. struct sysfs_elem_bin_attr {
  16. struct bin_attribute    *bin_attr;  //二进制的对象属性
  17. struct hlist_head   buffers; //struct bin_buffer的链表
  18. };

struct  sysfs_open_dirent等结构体的详细信息后文说明。

2、sysfs文件系统的初始化是由sysfs_init函数来完成的(该函数由vfs_caches_init函数所调用的mnt_init函数调用),执行文件系统的注册和挂载,与rootfs根文件系统的相关操作类似。源代码如下所示:

  1. /* fs/sysfs/mount.c */
  2. static struct vfsmount *sysfs_mnt;
  3. struct kmem_cache *sysfs_dir_cachep;
  4. int __init sysfs_init(void)
  5. {
  6. int err = -ENOMEM;
  7. sysfs_dir_cachep = kmem_cache_create("sysfs_dir_cache",
  8. sizeof(struct sysfs_dirent),
  9. 0, 0, NULL);  //创建用于struct sysfs_dirent的高速缓存
  10. if (!sysfs_dir_cachep)
  11. goto out;
  12. //初始化后备存储介质相关结构体struct backing_dev_info(sysfs文件系统基于内存,无须数据同步)
  13. err = sysfs_inode_init();
  14. if (err)
  15. goto out_err;
  16. err = register_filesystem(&sysfs_fs_type); //注册文件系统
  17. if (!err) { //成功返回零
  18. sysfs_mnt = kern_mount(&sysfs_fs_type); //挂载文件系统,不过没有将sysfs_mnt所指的结构体实例插入到挂载树中
  19. if (IS_ERR(sysfs_mnt)) {
  20. printk(KERN_ERR "sysfs: could not mount!\n");
  21. err = PTR_ERR(sysfs_mnt);
  22. sysfs_mnt = NULL;
  23. unregister_filesystem(&sysfs_fs_type);
  24. goto out_err;
  25. }
  26. } else
  27. goto out_err;
  28. out:
  29. return err;
  30. out_err:
  31. kmem_cache_destroy(sysfs_dir_cachep);
  32. sysfs_dir_cachep = NULL;
  33. goto out;
  34. }

在用户空间一般都将sysfs文件系统挂载在/sys目录,而这里也有一次通过kern_mount函数的挂载,这样的话
sysfs文件系统就会挂载两次?其实是没有的,后者的挂载并没有将当前的struct 
vfsmount结构体实例插入到挂载树中,而是保存在全局指针sysfs_mnt中,并且会与用户空间挂载sysfs文件系统时所创建的struct

vfsmount结构体实例共享相同的超级块。因此,无论用户空间的挂载操作是否执行,sysfs文件系统都会存在于内核之中(编译内核有配置CONFIG_SYSFS选项)。sysfs文件系统的struct 
file_system_type结构体实例如下所示:

  1. /* fs/sysfs/mount.c */
  2. static struct file_system_type sysfs_fs_type = {
  3. .name       = "sysfs",
  4. .mount      = sysfs_mount,
  5. .kill_sb    = sysfs_kill_sb,
  6. };
  7. static struct dentry *sysfs_mount(struct file_system_type *fs_type,
  8. int flags, const char *dev_name, void *data)
  9. {
  10. struct sysfs_super_info *info;
  11. enum kobj_ns_type type;
  12. struct super_block *sb;
  13. int error;
  14. info = kzalloc(sizeof(*info), GFP_KERNEL); //分配私有数据所用内存
  15. if (!info)
  16. return ERR_PTR(-ENOMEM);
  17. //构建超级块私有数据,用于sysfs_test_super和sysfs_set_super函数
  18. for (type = KOBJ_NS_TYPE_NONE; type < KOBJ_NS_TYPES; type++)
  19. info->ns[type] = kobj_ns_current(type);
  20. sb = sget(fs_type, sysfs_test_super, sysfs_set_super, info); //查找或创建超级块
  21. if (IS_ERR(sb) || sb->s_fs_info != info)
  22. kfree(info);
  23. if (IS_ERR(sb))
  24. return ERR_CAST(sb);
  25. if (!sb->s_root) { //如果根目录项为空指针,则说明超级块sb是新创建的
  26. sb->s_flags = flags;
  27. error = sysfs_fill_super(sb, data, flags & MS_SILENT ? 1 : 0); //填充超级块
  28. if (error) {
  29. deactivate_locked_super(sb);
  30. return ERR_PTR(error);
  31. }
  32. sb->s_flags |= MS_ACTIVE;
  33. }
  34. return dget(sb->s_root);
  35. }

其中,sysfs_test_super函数用于判断struct  sysfs_super_info结构体数据是否相同以便确认是否可以共享超级块,源代码如下所示:

  1. /* include/linux/kobject_ns.h */
  2. enum kobj_ns_type {
  3. KOBJ_NS_TYPE_NONE = 0,
  4. KOBJ_NS_TYPE_NET,
  5. KOBJ_NS_TYPES
  6. };
  7. /* fs/sysfs/sysfs.h */
  8. struct sysfs_super_info {
  9. const void *ns[KOBJ_NS_TYPES];
  10. };
  11. /* fs/sysfs/mount.c */
  12. static int sysfs_test_super(struct super_block *sb, void *data)
  13. {
  14. struct sysfs_super_info *sb_info = sysfs_info(sb); //sb->s_fs_info
  15. struct sysfs_super_info *info = data;
  16. enum kobj_ns_type type;
  17. int found = 1;
  18. for (type = KOBJ_NS_TYPE_NONE; type < KOBJ_NS_TYPES; type++) {
  19. if (sb_info->ns[type] != info->ns[type]) //只要有任何一项的值不相同则函数返回0
  20. found = 0;
  21. }
  22. return found;
  23. }

sysfs_fill_super函数主要用于创建sysfs文件系统根目录所对应的目录项及其i节点,并且将sysfs_root作为根目录项的私有数据。源代码如下所示:

  1. /* fs/sysfs/mount.c */
  2. static int sysfs_fill_super(struct super_block *sb, void *data, int silent)
  3. {
  4. struct inode *inode;
  5. struct dentry *root;
  6. sb->s_blocksize = PAGE_CACHE_SIZE; //与内存页大小相同
  7. sb->s_blocksize_bits = PAGE_CACHE_SHIFT;
  8. sb->s_magic = SYSFS_MAGIC;
  9. sb->s_op = &sysfs_ops;
  10. sb->s_time_gran = 1;
  11. //创建并初始化根目录的i节点
  12. mutex_lock(&sysfs_mutex);
  13. inode = sysfs_get_inode(sb, &sysfs_root);
  14. mutex_unlock(&sysfs_mutex);
  15. if (!inode) {
  16. pr_debug("sysfs: could not get root inode\n");
  17. return -ENOMEM;
  18. }
  19. //创建并初始化sysfs文件系统的根目录项并关联根目录的i节点
  20. root = d_alloc_root(inode);
  21. if (!root) {
  22. pr_debug("%s: could not get root dentry!\n",__func__);
  23. iput(inode);
  24. return -ENOMEM;
  25. }
  26. root->d_fsdata = &sysfs_root;  //根目录项的d_fsdata成员指向sysfs文件系统的根数据项sysfs_root
  27. sb->s_root = root;
  28. return 0;
  29. }
  30. struct sysfs_dirent sysfs_root = {
  31. .s_name     = "",
  32. .s_count    = ATOMIC_INIT(1),
  33. .s_flags    = SYSFS_DIR | (KOBJ_NS_TYPE_NONE << SYSFS_NS_TYPE_SHIFT),
  34. .s_mode     = S_IFDIR | S_IRWXU | S_IRUGO | S_IXUGO, //目录文件,访问权限0755
  35. .s_ino      = 1,   //起始i节点号为1
  36. };
  37. static const struct super_operations sysfs_ops = { //超级块操作函数
  38. .statfs        = simple_statfs,
  39. .drop_inode    = generic_delete_inode,
  40. .evict_inode    = sysfs_evict_inode,
  41. };

3、文件系统最核心的内容是要看其i节点是如何构建的,sysfs文件系统使用sysfs_init_inode函数来创建。源代码如下所示:

  1. /* fs/sysfs/inode.c */
  2. static void sysfs_init_inode(struct sysfs_dirent *sd, struct inode *inode)
  3. {
  4. struct bin_attribute *bin_attr;
  5. inode->i_private = sysfs_get(sd); //指向引用计数递增之后的sd
  6. inode->i_mapping->a_ops = &sysfs_aops; //地址空间操作函数
  7. inode->i_mapping->backing_dev_info = &sysfs_backing_dev_info; //后备存储介质的相关信息
  8. inode->i_op = &sysfs_inode_operations;
  9. set_default_inode_attr(inode, sd->s_mode);
  10. sysfs_refresh_inode(sd, inode);
  11. switch (sysfs_type(sd)) { //struct sysfs_dirent类型
  12. case SYSFS_DIR: //目录
  13. inode->i_op = &sysfs_dir_inode_operations;
  14. inode->i_fop = &sysfs_dir_operations;
  15. break;
  16. case SYSFS_KOBJ_ATTR: //文本文件
  17. inode->i_size = PAGE_SIZE; //文件大小固定为一页内存
  18. inode->i_fop = &sysfs_file_operations;
  19. break;
  20. case SYSFS_KOBJ_BIN_ATTR: //二进制文件
  21. bin_attr = sd->s_bin_attr.bin_attr;
  22. inode->i_size = bin_attr->size;
  23. inode->i_fop = &bin_fops;
  24. break;
  25. case SYSFS_KOBJ_LINK: //符号链接文件
  26. inode->i_op = &sysfs_symlink_inode_operations;
  27. break;
  28. default:
  29. BUG();
  30. }
  31. unlock_new_inode(inode);
  32. }
  33. static inline void set_default_inode_attr(struct inode * inode, mode_t mode)
  34. {
  35. inode->i_mode = mode;
  36. inode->i_atime = inode->i_mtime = inode->i_ctime = CURRENT_TIME;
  37. }
  38. static void sysfs_refresh_inode(struct sysfs_dirent *sd, struct inode *inode)
  39. {
  40. struct sysfs_inode_attrs *iattrs = sd->s_iattr;
  41. inode->i_mode = sd->s_mode;
  42. if (iattrs) { //sd->s_iattr为真
  43. //从iattrs->ia_iattr拷贝ia_uid、ia_gid、ia_atime、ia_mtime和ia_ctime等成员的值给i节点相应的成员
  44. set_inode_attr(inode, &iattrs->ia_iattr);
  45. security_inode_notifysecctx(inode,
  46. iattrs->ia_secdata,
  47. iattrs->ia_secdata_len); //安全检测模块
  48. }
  49. if (sysfs_type(sd) == SYSFS_DIR)
  50. inode->i_nlink = sysfs_count_nlink(sd); //计算目录的硬链接数目(等于子目录数+2)
  51. }

四种structsysfs_dirent类型对应三种文件类型,其中SYSFS_KOBJ_ATTR和SYSFS_KOBJ_BIN_ATTR都为普通文件。文件系统针对不同的文件类型,i节点操作函数(struct 
inode_operations)和文件内容操作函数(struct 
file_operations)都会有不同的实现,并且其中的函数也是根据文件类型来决定是否实现(大部分成员为空指针)。

3.1、目录的i节点操作函数和文件内容操作函数分别为sysfs_dir_inode_operations和sysfs_dir_operations,其中i节点操作函数的create、mkdir和rmdir等成员都为空指针,表示sysfs文件系统的目录或文件无法从用户空间通过系统调用来创建和删除。对于sysfs文件系统中的目录来说,i节点操作函数最重要的是查找函数sysfs_lookup,文件内容操作函数最重要的是遍历目录函数sysfs_readdir。源代码如下所示:

  1. /* fs/sysfs/dir.c */
  2. const struct inode_operations sysfs_dir_inode_operations = {
  3. .lookup     = sysfs_lookup,
  4. .permission = sysfs_permission,
  5. .setattr    = sysfs_setattr,
  6. .getattr    = sysfs_getattr,
  7. .setxattr   = sysfs_setxattr,
  8. };
  9. const struct file_operations sysfs_dir_operations = {
  10. .read       = generic_read_dir,
  11. .readdir    = sysfs_readdir,
  12. .release    = sysfs_dir_release,
  13. .llseek     = generic_file_llseek,
  14. };
  15. static struct dentry * sysfs_lookup(struct inode *dir, struct dentry *dentry,
  16. struct nameidata *nd) //dir为父目录的i节点,dentry为被查找对象的目录项(这时为d_alloc函数操作之后的状态)
  17. {
  18. struct dentry *ret = NULL;
  19. struct dentry *parent = dentry->d_parent; //父目录项
  20. struct sysfs_dirent *parent_sd = parent->d_fsdata;
  21. struct sysfs_dirent *sd;
  22. struct inode *inode;
  23. enum kobj_ns_type type;
  24. const void *ns;
  25. mutex_lock(&sysfs_mutex);
  26. //获取命名空间
  27. type = sysfs_ns_type(parent_sd);
  28. ns = sysfs_info(dir->i_sb)->ns[type];
  29. //从其parent_sd->s_dir.children链表中查找相同命名空间并且名字相同的子项
  30. sd = sysfs_find_dirent(parent_sd, ns, dentry->d_name.name);
  31. if (!sd) { //该子项不存在
  32. ret = ERR_PTR(-ENOENT);
  33. goto out_unlock;
  34. }
  35. //从inode_hashtable哈希表中查找该子项对应的i节点,若不存在则构建一个新的i节点实例
  36. inode = sysfs_get_inode(dir->i_sb, sd);
  37. if (!inode) { //构建失败
  38. ret = ERR_PTR(-ENOMEM);
  39. goto out_unlock;
  40. }
  41. //对于目录来说,只能有一个目录项,并且只有在其为空目录或者为当前文件系统的根目录时才有可能被设置DCACHE_UNHASHED状态
  42. //当i节点为IS_ROOT和DCACHE_DISCONNECTED时,d_find_alias函数返回真
  43. ret = d_find_alias(inode);
  44. if (!ret) {
  45. d_set_d_op(dentry, &sysfs_dentry_ops); //sysfs_dentry_ops为sysfs文件系统目录项的操作函数
  46. dentry->d_fsdata = sysfs_get(sd);
  47. d_add(dentry, inode);   //调用d_instantiate函数将目录项dentry插入到inode->i_dentry链表并且dentry->d_inode指向i节点inode
  48. //调用d_rehash函数将目录项dentry插入到dentry_hashtable哈希表中
  49. } else {
  50. d_move(ret, dentry); //重新使用所返回的目录项ret并与目录项dentry交换d_name等数据
  51. iput(inode); //销毁i节点
  52. }
  53. out_unlock:
  54. mutex_unlock(&sysfs_mutex);
  55. return ret;
  56. }

对于目录来说,只能有一个目录项别名。

  1. /* fs/sysfs/dir.c */
  2. static int sysfs_readdir(struct file * filp, void * dirent, filldir_t filldir)
  3. {
  4. struct dentry *dentry = filp->f_path.dentry; //当前目录所对应的目录项
  5. struct sysfs_dirent * parent_sd = dentry->d_fsdata;
  6. struct sysfs_dirent *pos = filp->private_data;
  7. enum kobj_ns_type type;
  8. const void *ns;
  9. ino_t ino;
  10. //获取命名空间
  11. type = sysfs_ns_type(parent_sd);
  12. ns = sysfs_info(dentry->d_sb)->ns[type];
  13. //当前目录
  14. if (filp->f_pos == 0) {
  15. ino = parent_sd->s_ino;
  16. if (filldir(dirent, ".", 1, filp->f_pos, ino, DT_DIR) == 0)
  17. filp->f_pos++;
  18. }
  19. //父目录
  20. if (filp->f_pos == 1) {
  21. if (parent_sd->s_parent)
  22. ino = parent_sd->s_parent->s_ino;
  23. else
  24. ino = parent_sd->s_ino; //文件系统根目录将自身当作父目录
  25. if (filldir(dirent, "..", 2, filp->f_pos, ino, DT_DIR) == 0)
  26. filp->f_pos++;
  27. }
  28. mutex_lock(&sysfs_mutex);
  29. for (pos = sysfs_dir_pos(ns, parent_sd, filp->f_pos, pos); //这时filp->f_pos等于2,pos为NULL
  30. pos;
  31. pos = sysfs_dir_next_pos(ns, parent_sd, filp->f_pos, pos)) { //遍历当前目录
  32. const char * name;
  33. unsigned int type;
  34. int len, ret;
  35. name = pos->s_name;
  36. len = strlen(name);
  37. ino = pos->s_ino;
  38. type = dt_type(pos); //文件类型
  39. filp->f_pos = ino; //将i节点号作为上一个struct linux_dirent实例中d_off成员的值
  40. filp->private_data = sysfs_get(pos);
  41. mutex_unlock(&sysfs_mutex);
  42. ret = filldir(dirent, name, len, filp->f_pos, ino, type);
  43. mutex_lock(&sysfs_mutex);
  44. if (ret < 0)
  45. break;
  46. }
  47. mutex_unlock(&sysfs_mutex);
  48. if ((filp->f_pos > 1) && !pos) { //遍历完全
  49. filp->f_pos = INT_MAX;
  50. filp->private_data = NULL;
  51. }
  52. return 0;
  53. }
  54. static struct sysfs_dirent *sysfs_dir_pos(const void *ns,
  55. struct sysfs_dirent *parent_sd, ino_t ino, struct sysfs_dirent *pos)
  56. {
  57. if (pos) {
  58. int valid = !(pos->s_flags & SYSFS_FLAG_REMOVED) && //非SYSFS_FLAG_REMOVED项
  59. pos->s_parent == parent_sd &&  //属于目录parent_sd
  60. ino == pos->s_ino;
  61. sysfs_put(pos);
  62. if (!valid)
  63. pos = NULL; //重新遍历该目录
  64. }
  65. if (!pos && (ino > 1) && (ino < INT_MAX)) { //这时pos为空指针,且i节点号的大小在有效范围内
  66. pos = parent_sd->s_dir.children;
  67. while (pos && (ino > pos->s_ino)) //过滤掉i节点号比它小的
  68. pos = pos->s_sibling;
  69. }
  70. while (pos && pos->s_ns && pos->s_ns != ns) //过滤掉不是相同命名空间的
  71. pos = pos->s_sibling;
  72. return pos;
  73. }
  74. static struct sysfs_dirent *sysfs_dir_next_pos(const void *ns,
  75. struct sysfs_dirent *parent_sd, ino_t ino, struct sysfs_dirent *pos)
  76. {
  77. pos = sysfs_dir_pos(ns, parent_sd, ino, pos);
  78. if (pos)
  79. pos = pos->s_sibling; //获取下一个项
  80. while (pos && pos->s_ns && pos->s_ns != ns)
  81. pos = pos->s_sibling;
  82. return pos;
  83. }

当在用户空间通过ls等命令查看sysfs文件系统的目录时,会通过系统调用getdents调用vfs_readdir,然后再调用sysfs_readdir函数。其中参数filp表示该目录打开之后的文件指针,dirent实际为struct 
getdents_callback类型的数据,filldir为回调函数指针,指向同名的filldir函数。源代码如下所示:

  1. /* include/linux/kernel.h */
  2. #define __ALIGN_KERNEL(x, a)        __ALIGN_KERNEL_MASK(x, (typeof(x))(a) - 1)
  3. #define __ALIGN_KERNEL_MASK(x, mask)    (((x) + (mask)) & ~(mask))
  4. #define ALIGN(x, a)     __ALIGN_KERNEL((x), (a))
  5. /* include/linux/compiler-gcc4.h */
  6. #define __compiler_offsetof(a,b) __builtin_offsetof(a,b)  //GCC编译器内置函数,计算成员偏移量
  7. /* include/linux/stddef.h */
  8. #undef offsetof
  9. #ifdef __compiler_offsetof
  10. #define offsetof(TYPE,MEMBER) __compiler_offsetof(TYPE,MEMBER)
  11. #else
  12. #define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
  13. #endif
  14. /* fs/readdir.c */
  15. struct linux_dirent {
  16. unsigned long   d_ino; //i节点号
  17. unsigned long   d_off; //偏移量,无实际意义,在sysfs文件系统的实现中,它的值为上一个所遍历的文件或目录的i节点号(最后一个为INT_MAX)
  18. unsigned short  d_reclen; //整个struct linux_dirent实例的长度(经过对齐修正)
  19. char        d_name[1]; //文件名,大小由实际文件名的长度决定(空字符结尾)
  20. };
  21. struct getdents_callback {
  22. struct linux_dirent __user * current_dir;   //初始化为系统调用传入的内存地址
  23. struct linux_dirent __user * previous;      //指向上一个struct linux_dirent实例
  24. int count;  //初始化为系统调用传入的内存的总大小(字节数)
  25. int error;  //保存错误码
  26. };
  27. static int filldir(void * __buf, const char * name, int namlen, loff_t offset,
  28. u64 ino, unsigned int d_type)
  29. {
  30. struct linux_dirent __user * dirent;
  31. struct getdents_callback * buf = (struct getdents_callback *) __buf;
  32. unsigned long d_ino;
  33. //功能等效于(len / sizeof(long) + (len % sizeof(long) ? 1 : 0)) * sizeof(long),其中len为ALIGN的第一个参数的值
  34. int reclen = ALIGN(offsetof(struct linux_dirent, d_name) + namlen + 2,  //其中加2个字节的意义是:一个字节用来存储字符串的结束符,
  35. sizeof(long));                                                      //另一个字节用来存储文件类型。
  36. buf->error = -EINVAL;
  37. if (reclen > buf->count) //剩余的内存不够
  38. return -EINVAL;
  39. d_ino = ino;
  40. if (sizeof(d_ino) < sizeof(ino) && d_ino != ino) { //d_ino的数据类型较小且导致数据溢出
  41. buf->error = -EOVERFLOW;
  42. return -EOVERFLOW;
  43. }
  44. dirent = buf->previous;
  45. if (dirent) {
  46. if (__put_user(offset, &dirent->d_off)) //以当前的offset值填充上一个struct linux_dirent实例的d_off成员
  47. goto efault;
  48. }
  49. dirent = buf->current_dir;
  50. if (__put_user(d_ino, &dirent->d_ino)) //i节点号
  51. goto efault;
  52. if (__put_user(reclen, &dirent->d_reclen)) //当前struct linux_dirent实例的总大小(字节数)
  53. goto efault;
  54. if (copy_to_user(dirent->d_name, name, namlen)) //拷贝文件名
  55. goto efault;
  56. if (__put_user(0, dirent->d_name + namlen)) //在文件名后加字符串结束符'\0'
  57. goto efault;
  58. if (__put_user(d_type, (char __user *) dirent + reclen - 1)) //最后一个字节用来保存文件类型信息
  59. goto efault;
  60. buf->previous = dirent;
  61. dirent = (void __user *)dirent + reclen;
  62. buf->current_dir = dirent; //指向下一个尚未使用的struct linux_dirent实例
  63. buf->count -= reclen; //计算剩余内存数量
  64. return 0;
  65. efault:
  66. buf->error = -EFAULT;
  67. return -EFAULT;
  68. }

3.2、sysfs文件系统针对文本文件和二进制文件实现了不同的文件内容操作函数,而i节点操作函数则相同。源代码如下所示:

  1. /* fs/sysfs/inode.c */
  2. static const struct inode_operations sysfs_inode_operations ={
  3. .permission = sysfs_permission,
  4. .setattr    = sysfs_setattr,
  5. .getattr    = sysfs_getattr,
  6. .setxattr   = sysfs_setxattr,
  7. };
  8. /* fs/sysfs/file.c */
  9. const struct file_operations sysfs_file_operations = { //文件文件
  10. .read       = sysfs_read_file,
  11. .write      = sysfs_write_file,
  12. .llseek     = generic_file_llseek,
  13. .open       = sysfs_open_file,
  14. .release    = sysfs_release,
  15. .poll       = sysfs_poll,
  16. };
  17. /* fs/sysfs/bin.c */
  18. const struct file_operations bin_fops = { //二进制文件
  19. .read       = read,
  20. .write      = write,
  21. .mmap       = mmap,
  22. .llseek     = generic_file_llseek,
  23. .open       = open,
  24. .release    = release,
  25. };

针对普通文件,最重要的是观察它的文件内容操作函数的实现,如open、read和write等函数。下面则以文本文件为例,看看sysfs文件系统是如何读写文件的。源代码如下所示:

  1. /* include/linux/limits.h */
  2. #define PATH_MAX        4096
  3. /* fs/sysfs/file.c */
  4. static char last_sysfs_file[PATH_MAX];
  5. static int sysfs_open_file(struct inode *inode, struct file *file)
  6. {
  7. struct sysfs_dirent *attr_sd = file->f_path.dentry->d_fsdata; //sysfs数据项,表示当前被打开的文件
  8. struct kobject *kobj = attr_sd->s_parent->s_dir.kobj; //所属目录所对应的内核对象
  9. struct sysfs_buffer *buffer;
  10. const struct sysfs_ops *ops;
  11. int error = -EACCES;
  12. char *p;
  13. //获得该文件的全路径并依次保存在last_sysfs_file数组的尾部
  14. p = d_path(&file->f_path, last_sysfs_file, sizeof(last_sysfs_file));
  15. if (!IS_ERR(p))
  16. memmove(last_sysfs_file, p, strlen(p) + 1); //将路径移动到数组的开头
  17. //获取活动引用计数
  18. if (!sysfs_get_active(attr_sd))
  19. return -ENODEV;
  20. if (kobj->ktype && kobj->ktype->sysfs_ops) //内核对象针对所属属性的读写函数必须存在
  21. ops = kobj->ktype->sysfs_ops;
  22. else {
  23. WARN(1, KERN_ERR "missing sysfs attribute operations for "
  24. "kobject: %s\n", kobject_name(kobj));
  25. goto err_out;
  26. }
  27. if (file->f_mode & FMODE_WRITE) { //写文件
  28. if (!(inode->i_mode & S_IWUGO) || !ops->store) //需S_IWUGO访问权限且store函数必须定义
  29. goto err_out;
  30. }
  31. if (file->f_mode & FMODE_READ) { //读文件
  32. if (!(inode->i_mode & S_IRUGO) || !ops->show) //需S_IRUGO访问权限且show函数必须定义
  33. goto err_out;
  34. }
  35. //每次打开的文本文件都对应一个struct sysfs_buffer实例
  36. error = -ENOMEM;
  37. buffer = kzalloc(sizeof(struct sysfs_buffer), GFP_KERNEL);
  38. if (!buffer)
  39. goto err_out;
  40. mutex_init(&buffer->mutex);
  41. buffer->needs_read_fill = 1;
  42. buffer->ops = ops;
  43. file->private_data = buffer; //保存在文件指针的私有数据中
  44. //分配struct sysfs_open_dirent结构体实例
  45. error = sysfs_get_open_dirent(attr_sd, buffer);
  46. if (error)
  47. goto err_free;
  48. //打开成功,释放活动引用计数
  49. sysfs_put_active(attr_sd);
  50. return 0;
  51. err_free:
  52. kfree(buffer);
  53. err_out:
  54. sysfs_put_active(attr_sd);
  55. return error;
  56. }
  57. static DEFINE_SPINLOCK(sysfs_open_dirent_lock);
  58. static int sysfs_get_open_dirent(struct sysfs_dirent *sd,
  59. struct sysfs_buffer *buffer)
  60. {
  61. struct sysfs_open_dirent *od, *new_od = NULL;
  62. retry:
  63. spin_lock_irq(&sysfs_open_dirent_lock);
  64. if (!sd->s_attr.open && new_od) { //每个文本文件的struct sysfs_dirent都有一个open成员
  65. sd->s_attr.open = new_od; //指向新分配并初始化的struct sysfs_open_dirent实例
  66. new_od = NULL;
  67. }
  68. od = sd->s_attr.open;
  69. if (od) {
  70. atomic_inc(&od->refcnt); //递增引用计数
  71. //将buffer作为链表元素插入到od->buffers链表(该链表中元素的数量就是该文本文件正被打开的次数)
  72. list_add_tail(&buffer->list, &od->buffers);
  73. }
  74. spin_unlock_irq(&sysfs_open_dirent_lock);
  75. if (od) {
  76. kfree(new_od);
  77. return 0;
  78. }
  79. //为struct sysfs_open_dirent结构体实例分配内存
  80. new_od = kmalloc(sizeof(*new_od), GFP_KERNEL);
  81. if (!new_od)
  82. return -ENOMEM;
  83. //初始化成员
  84. atomic_set(&new_od->refcnt, 0);
  85. atomic_set(&new_od->event, 1);
  86. init_waitqueue_head(&new_od->poll);
  87. INIT_LIST_HEAD(&new_od->buffers);
  88. goto retry;
  89. }

在sysfs文件系统中,文本文件使用struct sysfs_elem_attr来表示,而该结构体有一个struct 
sysfs_open_dirent类型的open成员,主要用于管理每次打开并读/写该文件时所使用的内存(struct 
sysfs_buffer)。源代码如下所示:

  1. /* fs/sysfs/file.c */
  2. struct sysfs_open_dirent {
  3. atomic_t        refcnt; //打开次数
  4. atomic_t        event;
  5. wait_queue_head_t   poll; //等待队列
  6. struct list_head    buffers; //sysfs_buffer.list的链表
  7. };
  8. struct sysfs_buffer {
  9. size_t          count;  //数据大小(字节数)
  10. loff_t          pos;    //偏移量
  11. char            * page; //指向一页内存,用于存储数据
  12. const struct sysfs_ops  * ops;  //操作函数
  13. struct mutex        mutex;      //互斥锁
  14. int         needs_read_fill;    //是否已填充数据
  15. int         event;
  16. struct list_head    list;   //插入所属的struct sysfs_open_dirent的buffers链表
  17. };

sysfs_read_file为sysfs文件系统文本文件的读函数,源代码如下所示:

  1. /* fs/sysfs/file.c */
  2. static ssize_t
  3. sysfs_read_file(struct file *file, char __user *buf, size_t count, loff_t *ppos)
  4. {
  5. struct sysfs_buffer * buffer = file->private_data;  //调用sysfs_open_file时所生成的
  6. ssize_t retval = 0;
  7. mutex_lock(&buffer->mutex);
  8. if (buffer->needs_read_fill || *ppos == 0) { //尚未获取数据或者文件偏移量为零
  9. retval = fill_read_buffer(file->f_path.dentry,buffer); //一次读取
  10. if (retval) //获取数据失败(返回零表示成功)
  11. goto out;
  12. }
  13. pr_debug("%s: count = %zd, ppos = %lld, buf = %s\n",
  14. __func__, count, *ppos, buffer->page);
  15. //将count个字节的数据拷贝到用户空间内存buf,buffer->count表示可拷贝数据的最大字节数(可能须要多次拷贝才能读取完整个内存)
  16. retval = simple_read_from_buffer(buf, count, ppos, buffer->page,
  17. buffer->count);
  18. out:
  19. mutex_unlock(&buffer->mutex);
  20. return retval;
  21. }
  22. static int fill_read_buffer(struct dentry * dentry, struct sysfs_buffer * buffer)
  23. {
  24. struct sysfs_dirent *attr_sd = dentry->d_fsdata; //sysfs数据项
  25. struct kobject *kobj = attr_sd->s_parent->s_dir.kobj; //所属内核对象
  26. const struct sysfs_ops * ops = buffer->ops;
  27. int ret = 0;
  28. ssize_t count;
  29. if (!buffer->page) //分配存储数据的内存
  30. buffer->page = (char *) get_zeroed_page(GFP_KERNEL);
  31. if (!buffer->page) //内存分配失败
  32. return -ENOMEM;
  33. if (!sysfs_get_active(attr_sd)) //获取活动引用计数
  34. return -ENODEV;
  35. buffer->event = atomic_read(&attr_sd->s_attr.open->event);
  36. count = ops->show(kobj, attr_sd->s_attr.attr, buffer->page);  //从该内核对象相应的属性中获取数据并保存到刚才所分配的内存中
  37. sysfs_put_active(attr_sd); //释放活动引用计数
  38. if (count >= (ssize_t)PAGE_SIZE) { //至多能读取PAGE_SIZE - 1个字节
  39. print_symbol("fill_read_buffer: %s returned bad count\n",
  40. (unsigned long)ops->show);
  41. /* Try to struggle along */
  42. count = PAGE_SIZE - 1;
  43. }
  44. if (count >= 0) {
  45. buffer->needs_read_fill = 0; //表示填充过数据
  46. buffer->count = count;
  47. } else {
  48. ret = count; //获取失败
  49. }
  50. return ret;
  51. }
  52. /* fs/libfs.c */
  53. ssize_t simple_read_from_buffer(void __user *to, size_t count, loff_t *ppos,
  54. const void *from, size_t available)
  55. {
  56. loff_t pos = *ppos;
  57. size_t ret;
  58. if (pos < 0) //文件偏移量不能为负数
  59. return -EINVAL;
  60. if (pos >= available || !count) //已无数据可读或者是想读取零个字节
  61. return 0;
  62. if (count > available - pos)  //只剩available - pos个字节
  63. count = available - pos;
  64. ret = copy_to_user(to, from + pos, count); //拷贝数据
  65. if (ret == count) //返回值为没有拷贝成功的字节数
  66. return -EFAULT;
  67. count -= ret; //获得count个字节的数据
  68. *ppos = pos + count; //更新文件偏移量
  69. return count;
  70. }

sysfs_write_file为sysfs文件系统的写函数,源代码如下所示:

  1. /* fs/sysfs/file.c */
  2. static ssize_t
  3. sysfs_write_file(struct file *file, const char __user *buf, size_t count, loff_t *ppos) //ppos为文件偏移量
  4. {
  5. struct sysfs_buffer * buffer = file->private_data; //调用sysfs_open_file时所生成的
  6. ssize_t len;
  7. mutex_lock(&buffer->mutex);
  8. len = fill_write_buffer(buffer, buf, count); //一次写入,从用户空间内存buf中拷贝count个字节到内核空间
  9. if (len > 0) //成功拷贝len个字节
  10. len = flush_write_buffer(file->f_path.dentry, buffer, len); //根据获得的数据更新该内核对象相应的属性
  11. if (len > 0)
  12. *ppos += len; //更改文件偏移量,对buffer->page中的数据无意义,后面写入的数据会覆盖前面写入的数据
  13. mutex_unlock(&buffer->mutex);
  14. return len;
  15. }
  16. static int
  17. fill_write_buffer(struct sysfs_buffer * buffer, const char __user * buf, size_t count)
  18. {
  19. int error;
  20. if (!buffer->page) //分配存储数据的内存
  21. buffer->page = (char *)get_zeroed_page(GFP_KERNEL);
  22. if (!buffer->page) //内存分配失败
  23. return -ENOMEM;
  24. if (count >= PAGE_SIZE) //写入的数据最多只能为PAGE_SIZE - 1个字节
  25. count = PAGE_SIZE - 1;
  26. error = copy_from_user(buffer->page,buf,count); //拷贝数据,成功函数返回零
  27. buffer->needs_read_fill = 1;
  28. buffer->page[count] = 0; //字符串结束符'\0'
  29. return error ? -EFAULT : count;
  30. }
  31. static int
  32. flush_write_buffer(struct dentry * dentry, struct sysfs_buffer * buffer, size_t count)
  33. {
  34. struct sysfs_dirent *attr_sd = dentry->d_fsdata; //sysfs数据项,这里表示一个文本文件
  35. struct kobject *kobj = attr_sd->s_parent->s_dir.kobj; //所属内核对象
  36. const struct sysfs_ops * ops = buffer->ops;
  37. int rc;
  38. if (!sysfs_get_active(attr_sd)) //获取活动引用计数
  39. return -ENODEV;
  40. rc = ops->store(kobj, attr_sd->s_attr.attr, buffer->page, count); //调用store函数
  41. sysfs_put_active(attr_sd); //释放活动引用计数
  42. return rc;
  43. }

关闭文件时,打开、读/写文件时所分配的内存都会释放,并不会一直存在于内核中。

3.3、对于符号链接文件来说,它没有文件内容操作函数,只有i节点操作函数,其中最重要的函数为符号链接文件的解析函数sysfs_follow_link,源代码如下所示:

  1. /* fs/sysfs/symlink.c */
  2. const struct inode_operations sysfs_symlink_inode_operations = {
  3. .setxattr   = sysfs_setxattr,
  4. .readlink   = generic_readlink,
  5. .follow_link    = sysfs_follow_link,
  6. .put_link   = sysfs_put_link,
  7. .setattr    = sysfs_setattr,
  8. .getattr    = sysfs_getattr,
  9. .permission = sysfs_permission,
  10. };
  11. static void *sysfs_follow_link(struct dentry *dentry, struct nameidata *nd)
  12. {
  13. int error = -ENOMEM;
  14. unsigned long page = get_zeroed_page(GFP_KERNEL); //分配内存
  15. if (page) {
  16. error = sysfs_getlink(dentry, (char *) page);
  17. if (error < 0)  //成功时sysfs_getlink返回零
  18. free_page((unsigned long)page);
  19. }
  20. nd_set_link(nd, error ? ERR_PTR(error) : (char *)page); //保存获得的相对路径或错误码
  21. return NULL;
  22. }
  23. static int sysfs_getlink(struct dentry *dentry, char * path)
  24. {
  25. struct sysfs_dirent *sd = dentry->d_fsdata; //sysfs数据项
  26. struct sysfs_dirent *parent_sd = sd->s_parent; //父sysfs数据项
  27. struct sysfs_dirent *target_sd = sd->s_symlink.target_sd; //所链接到的sysfs数据项
  28. int error;
  29. mutex_lock(&sysfs_mutex);
  30. error = sysfs_get_target_path(parent_sd, target_sd, path); //获取从链接文件到链接对象的相对路径
  31. mutex_unlock(&sysfs_mutex);
  32. return error;
  33. }
  34. //sysfs链接文件是两个内核对象之间的链接,也就是目录之间的链接
  35. static int sysfs_get_target_path(struct sysfs_dirent *parent_sd,
  36. struct sysfs_dirent *target_sd, char *path)
  37. {
  38. struct sysfs_dirent *base, *sd;
  39. char *s = path;
  40. int len = 0;
  41. base = parent_sd;
  42. while (base->s_parent) { //sysfs的根数据项为sysfs_root,该数据项的s_parent成员为空指针
  43. sd = target_sd->s_parent;
  44. while (sd->s_parent && base != sd)  //如果base一直不等于sd,则循环直到根数据项才会停止
  45. sd = sd->s_parent;
  46. if (base == sd) //两者相等,这时链接文件和被链接对象的直接或间接的父目录相同
  47. break;
  48. strcpy(s, "../");  //拷贝字符串“../”,意味着两者不是同在这一目录下
  49. s += 3;
  50. base = base->s_parent; //接下来将比较链接文件的上一级目录的数据项
  51. }
  52. //这时,base已指向链接文件和被链接对象首个共有的直接或间接的父目录的数据项
  53. //计算整个路径的长度
  54. sd = target_sd;
  55. while (sd->s_parent && sd != base) {
  56. len += strlen(sd->s_name) + 1; //其中的加1表示目录项分隔符“/”所占的字节
  57. sd = sd->s_parent;
  58. }
  59. if (len < 2) //名称为空字符串
  60. return -EINVAL;
  61. len--; //减去一个多余的目录项分隔符所占的字节
  62. if ((s - path) + len > PATH_MAX) //总长度超过path数组的大小
  63. return -ENAMETOOLONG;
  64. //从被链接对象开始以倒序的方式拷贝目录项名称,直到base数据项(但不包括它的)
  65. sd = target_sd;
  66. while (sd->s_parent && sd != base) {
  67. int slen = strlen(sd->s_name);
  68. len -= slen;
  69. strncpy(s + len, sd->s_name, slen); //拷贝名称
  70. if (len) //等于0时表示最后一个名称的前面不需要再加分隔符
  71. s[--len] = '/'; //在该名称前加目录项分隔符“/”
  72. sd = sd->s_parent; //接着处理上一级目录
  73. }
  74. return 0;
  75. }

读取符号链接文件内容的函数generic_readlink主要就是靠解析函数来实现的,源代码如下所示:

  1. /* fs/namei.c */
  2. int generic_readlink(struct dentry *dentry, char __user *buffer, int buflen)
  3. {
  4. struct nameidata nd;
  5. void *cookie;
  6. int res;
  7. nd.depth = 0;
  8. cookie = dentry->d_inode->i_op->follow_link(dentry, &nd);
  9. if (IS_ERR(cookie)) //返回错误码
  10. return PTR_ERR(cookie);
  11. res = vfs_readlink(dentry, buffer, buflen, nd_get_link(&nd)); //通过nd_get_link获取follow_link保存的路径或错误码
  12. if (dentry->d_inode->i_op->put_link)
  13. dentry->d_inode->i_op->put_link(dentry, &nd, cookie); //这里put_link指向sysfs_put_link函数
  14. return res;
  15. }
  16. int vfs_readlink(struct dentry *dentry, char __user *buffer, int buflen, const char *link)
  17. {
  18. int len;
  19. len = PTR_ERR(link);
  20. if (IS_ERR(link)) //这里的link也有可能是错误码
  21. goto out;
  22. len = strlen(link);
  23. if (len > (unsigned) buflen) //路径长度比传入的内存大
  24. len = buflen;
  25. if (copy_to_user(buffer, link, len)) //拷贝到用户空间内存(但不包括字符串结束符)
  26. len = -EFAULT;
  27. out:
  28. return len;
  29. }
  30. /* fs/sysfs/symlink.c */
  31. static void sysfs_put_link(struct dentry *dentry, struct nameidata *nd, void *cookie)
  32. {
  33. char *page = nd_get_link(nd);
  34. if (!IS_ERR(page)) //非错误码
  35. free_page((unsigned long)page); //释放内存
  36. }

其中,sysfs_put_link函数执行与sysfs_follow_link函数相反的操作,这里只是释放由sysfs_follow_link函数分配的内存。

4、对于sysfs文件系统来说,在用户空间只能读写文件的内容,而无法创建或删除文件或目录,只能在内核中通过sysfs_create_dir、sysfs_create_file等等函数来实现。

4.1、内核对象(struct  kobject)对应于sysfs文件系统中的目录,可使用sysfs_create_dir函数来创建,源代码如下所示:

  1. /* fs/sysfs/dir.c */
  2. int sysfs_create_dir(struct kobject * kobj)
  3. {
  4. enum kobj_ns_type type;
  5. struct sysfs_dirent *parent_sd, *sd;
  6. const void *ns = NULL;
  7. int error = 0;
  8. BUG_ON(!kobj);
  9. if (kobj->parent) //父内核对象为空时,则在sysfs文件系统的根目录下创建目录
  10. parent_sd = kobj->parent->sd;
  11. else
  12. parent_sd = &sysfs_root;
  13. //获取命名空间及其类型
  14. if (sysfs_ns_type(parent_sd))
  15. ns = kobj->ktype->namespace(kobj);
  16. type = sysfs_read_ns_type(kobj);
  17. error = create_dir(kobj, parent_sd, type, ns, kobject_name(kobj), &sd);
  18. if (!error) //成功则返回零
  19. kobj->sd = sd;  //保存相应的sysfs数据项
  20. return error;
  21. }
  22. /* include/linux/kobject.h */
  23. static inline const char *kobject_name(const struct kobject *kobj)
  24. {
  25. return kobj->name;
  26. }
  27. /* fs/sysfs/dir.c */
  28. static int create_dir(struct kobject *kobj, struct sysfs_dirent *parent_sd,
  29. enum kobj_ns_type type, const void *ns, const char *name,
  30. struct sysfs_dirent **p_sd)
  31. {
  32. umode_t mode = S_IFDIR| S_IRWXU | S_IRUGO | S_IXUGO; //文件类型为目录,访问权限为0755
  33. struct sysfs_addrm_cxt acxt;
  34. struct sysfs_dirent *sd;
  35. int rc;
  36. //分配sysfs数据项并初始化
  37. sd = sysfs_new_dirent(name, mode, SYSFS_DIR); //数据项类型为目录
  38. if (!sd)
  39. return -ENOMEM;
  40. sd->s_flags |= (type << SYSFS_NS_TYPE_SHIFT); //命名空间类型,占用s_flags倒数第二个8位
  41. sd->s_ns = ns; //命名空间
  42. sd->s_dir.kobj = kobj; //关联内核对象
  43. sysfs_addrm_start(&acxt, parent_sd); //加锁并携带父数据项
  44. rc = sysfs_add_one(&acxt, sd); //关联父数据项
  45. sysfs_addrm_finish(&acxt); //解锁
  46. if (rc == 0)  //成功返回
  47. *p_sd = sd;
  48. else
  49. sysfs_put(sd); //释放数据项
  50. return rc;
  51. }
  52. struct sysfs_dirent *sysfs_new_dirent(const char *name, umode_t mode, int type)
  53. {
  54. char *dup_name = NULL;
  55. struct sysfs_dirent *sd;
  56. if (type & SYSFS_COPY_NAME) { //目录或符号链接文件需要拷贝文件名,它们对应的都是内核对象
  57. name = dup_name = kstrdup(name, GFP_KERNEL); //分配内存并拷贝文件名
  58. if (!name) //当不为空指针时则表示name指向新分配的内存
  59. return NULL;
  60. }
  61. sd = kmem_cache_zalloc(sysfs_dir_cachep, GFP_KERNEL); //从sysfs_dir_cachep缓存中分配sysfs数据项
  62. if (!sd)
  63. goto err_out1;
  64. if (sysfs_alloc_ino(&sd->s_ino)) //分配i节点号
  65. goto err_out2;
  66. //引用计数
  67. atomic_set(&sd->s_count, 1);
  68. atomic_set(&sd->s_active, 0);
  69. sd->s_name = name; //目录项名称
  70. sd->s_mode = mode; //文件类型及访问权限
  71. sd->s_flags = type; //sysfs数据项类型,占用s_flags低8位
  72. return sd;
  73. err_out2:
  74. kmem_cache_free(sysfs_dir_cachep, sd);
  75. err_out1:
  76. kfree(dup_name);
  77. return NULL;
  78. }
  79. int sysfs_add_one(struct sysfs_addrm_cxt *acxt, struct sysfs_dirent *sd)
  80. {
  81. int ret;
  82. ret = __sysfs_add_one(acxt, sd);
  83. if (ret == -EEXIST) { //同名数据项已经存在则输出告警信息
  84. char *path = kzalloc(PATH_MAX, GFP_KERNEL);
  85. WARN(1, KERN_WARNING
  86. "sysfs: cannot create duplicate filename '%s'\n",
  87. (path == NULL) ? sd->s_name :
  88. strcat(strcat(sysfs_pathname(acxt->parent_sd, path), "/"),
  89. sd->s_name)); //数据项的全路径
  90. kfree(path);
  91. }
  92. return ret;
  93. }
  94. int __sysfs_add_one(struct sysfs_addrm_cxt *acxt, struct sysfs_dirent *sd)
  95. {
  96. struct sysfs_inode_attrs *ps_iattr;
  97. if (sysfs_find_dirent(acxt->parent_sd, sd->s_ns, sd->s_name)) //查找该父目录下是否存在同名的数据项
  98. return -EEXIST; //已存在则返回错误码
  99. sd->s_parent = sysfs_get(acxt->parent_sd); //递增父数据项的引用计数,然后指向该父数据项
  100. sysfs_link_sibling(sd); //加入到父数据项的子数据项链表
  101. //更新父数据项的时间戳
  102. ps_iattr = acxt->parent_sd->s_iattr;
  103. if (ps_iattr) {
  104. struct iattr *ps_iattrs = &ps_iattr->ia_iattr;
  105. ps_iattrs->ia_ctime = ps_iattrs->ia_mtime = CURRENT_TIME;
  106. }
  107. return 0;
  108. }
  109. struct sysfs_dirent *sysfs_find_dirent(struct sysfs_dirent *parent_sd,
  110. const void *ns,
  111. const unsigned char *name)
  112. {
  113. struct sysfs_dirent *sd;
  114. for (sd = parent_sd->s_dir.children; sd; sd = sd->s_sibling) { //遍历父目录
  115. if (ns && sd->s_ns && (sd->s_ns != ns)) //查找同一命名空间
  116. continue;
  117. if (!strcmp(sd->s_name, name)) //同名sysfs数据项
  118. return sd;
  119. }
  120. return NULL;
  121. }
  122. static void sysfs_link_sibling(struct sysfs_dirent *sd)
  123. {
  124. struct sysfs_dirent *parent_sd = sd->s_parent;
  125. struct sysfs_dirent **pos;
  126. BUG_ON(sd->s_sibling);
  127. for (pos = &parent_sd->s_dir.children; *pos; pos = &(*pos)->s_sibling) {
  128. if (sd->s_ino < (*pos)->s_ino) //升序排列
  129. break;
  130. }
  131. sd->s_sibling = *pos;
  132. *pos = sd;
  133. }

4.2、内核对象的属性(struct  attribute)对应于sysfs文件系统中的文本文件,可使用sysfs_create_file函数来创建,源代码如下所示:

  1. /* fs/sysfs/file.c */
  2. int sysfs_create_file(struct kobject * kobj, const struct attribute * attr)
  3. {
  4. BUG_ON(!kobj || !kobj->sd || !attr); //三者必须为真
  5. return sysfs_add_file(kobj->sd, attr, SYSFS_KOBJ_ATTR); //数据项类型为SYSFS_KOBJ_ATTR,对应sysfs文件系统中的文本文件
  6. }
  7. int sysfs_add_file(struct sysfs_dirent *dir_sd, const struct attribute *attr,
  8. int type)
  9. {
  10. return sysfs_add_file_mode(dir_sd, attr, type, attr->mode); //访问权限由内核对象的属性自身配置
  11. }
  12. int sysfs_add_file_mode(struct sysfs_dirent *dir_sd,
  13. const struct attribute *attr, int type, mode_t amode)
  14. {
  15. umode_t mode = (amode & S_IALLUGO) | S_IFREG; //文件类型为普通文件
  16. struct sysfs_addrm_cxt acxt;
  17. struct sysfs_dirent *sd;
  18. int rc;
  19. //分配sysfs数据项并初始化
  20. sd = sysfs_new_dirent(attr->name, mode, type);
  21. if (!sd)
  22. return -ENOMEM;
  23. sd->s_attr.attr = (void *)attr; //保存内核对象属性
  24. sysfs_dirent_init_lockdep(sd); //初始化死锁检测模块
  25. sysfs_addrm_start(&acxt, dir_sd); //dir_sd对应于属性文件所在的目录
  26. rc = sysfs_add_one(&acxt, sd);
  27. sysfs_addrm_finish(&acxt);
  28. if (rc) //失败则销毁sysfs数据项
  29. sysfs_put(sd);
  30. return rc;
  31. }

sysfs文件系统中的二进制文件通过sysfs_create_bin_file函数来创建,符号链接文件通过sysfs_create_link函数来创建。

概述sysfs文件系统【转】的更多相关文章

  1. [中英对照]The sysfs Filesystem | sysfs文件系统

    The sysfs Filesystem | sysfs文件系统 Abstract | 摘要 sysfs is a feature of the Linux 2.6 kernel that allow ...

  2. Linux设备模型——设备驱动模型和sysfs文件系统解读

    本文将对Linux系统中的sysfs进行简单的分析,要分析sysfs就必须分析内核的driver-model(驱动模型),两者是紧密联系的.在分析过程中,本文将以platform总线和spi主控制器的 ...

  3. sysfs文件系统的建立【转】

    http://blog.csdn.net/dndxhej/article/details/7434615 对sysfs和设备模型有了解的都会知道sysfs实际是为了将设备模型导出到用户空间的一个内存文 ...

  4. proc文件系统、sysfs文件系统、kobject操作

    Proc文件系统是提供一个接口给用户,让用户可以查看系统运行的一些状态信息,让用户修改内核的一些参数,比方说printk的打印级别就可以通过proc去修改 Sysfs文件系统, Sysfs is a ...

  5. sysfs文件系统学习--sysfs

    一.sysfs简介1.sysfs就是利用VFS的接口去读写kobject的层次结构,建立起来的文件系统.其更新与删除是那些xxx_register()/unregister()做的事 情.从sysfs ...

  6. Sysfs文件系统与Linux设备模型

    转:http://www.360doc.com/content/11/1218/16/1299815_173168170.shtml sysfs把连接在系统上的设备和总线组织成为一个分级的目录及文件, ...

  7. sysfs文件系统

    3 sysfs文件系统 sysfs是一个基于内存的文件系统,它的作用是将内核信息以文件的方式提供给用户程序使用.该文件系统的目录层次结构严格按照内核的数据结构组织.除了二进制文件外(只有特殊场合才使用 ...

  8. Linux 内核sysfs 文件系统符号连接

    sysfs 文件系统有通常的树结构, 反映它代表的 kobjects 的层次组织. 但是内核中对象 间的关系常常比那个更加复杂. 例如, 一个 sysfs 子树 (/sys/devices )代表所有 ...

  9. linux sysfs文件系统

    个人理解:sysfs向用户空间展示了驱动设备的层次结构.我们都知道设备和对应的驱动都是由内核管理的,这些对于用户空间是不可见的.现在通过sysfs,可以在用户空间直观的了解设备驱动的层次结构. 我们来 ...

随机推荐

  1. 如何更好的使用JAVA线程池

    这篇文章结合Doug Lea大神在JDK1.5提供的JCU包,分别从线程池大小参数的设置.工作线程的创建.空闲线程的回收.阻塞队列的使用.任务拒绝策略.线程池Hook等方面来了解线程池的使用,其中涉及 ...

  2. 转换成json字符串,与json字符串转换成java类型都要先转换成json对象

    转换成json字符串,与json字符串转换成java类型都要先转换成json对象

  3. Hadoop基于Protocol Buffer的RPC实现代码分析-Server端--转载

    原文地址:http://yanbohappy.sinaapp.com/?p=110 最新版本的Hadoop代码中已经默认了Protocol buffer(以下简称PB,http://code.goog ...

  4. BZOJ3747 POI2015Kinoman(线段树)

    考虑固定左端点,求出该情况下能获得的最大值.于是每次可以在某数第一次出现的位置加上其价值,第二次出现的位置减掉其价值,查询前缀最大值就可以了.每次移动左端点在线段树上更新即可. #include< ...

  5. C++解析(10):struct和class的区别

    0.目录 1.默认访问级别 2.默认继承方式 2.1 分别独立继承 2.2 struct继承class 2.3 class继承struct 3.小结 1.默认访问级别 在用struct定义类时,所有成 ...

  6. NOIP2017 【游记】

    一年过去,想起去年还是个傻b[今年也是],心里总是无限的感慨. 脑海里是日日夜夜在机房的身影,一题一题AC的激情 我等今年等了许久,虽然我是个蒟蒻,但我有梦想的憧憬 鲲鹏展翅翼向天,扶摇直上九万里. ...

  7. bzoj2396: 神奇的矩阵(矩阵乘法+随机化)

    这题n三方显然会GG... 运用矩阵乘法的性质A*B*R=A*(B*R)=C*R,于是随机化出一个一列的R,就可以把复杂度降低成n方...大概率是不会错的 #include<iostream&g ...

  8. 解题:由乃OI 2018 五彩斑斓的世界

    题面 写在前面的扯淡: 分块的总体学习告一段落,这算是分块集中学习的最后一题么:以后当然也可能会写,就是零零散散的题了=.= 在洛谷上搜ynoi发现好像只有这道题和 由乃OI 2018 未来日记 是分 ...

  9. 【神仙题】【CF28D】 Don't fear, DravDe is kind

    传送门 Description 一个有N辆卡车的车队从城市Z驶向城市3,来到了一条叫做"恐惧隧道"的隧道.在卡车司机中,有传言说怪物DravDe在那条隧道里搜寻司机.有些司机害怕先 ...

  10. jsp 的 7 个动作指令

    动作指令与编译指令不同,编译指令是通知 Servlet 引擎的处理消息,而动作指令只是运行时的动作.编译指令在将 JSP 编译成 Servlet 时起作用:而处理指令通常可替换成 JSP 脚本,它只是 ...