linux中sysfs创建设备节点的方法和DEVICE_ATTR

使用DEVICE_ATTR宏,可以定义一个struct device_attribute设备属性,使用函数sysfs_create_group或sysfs_create_file便可以在设备目录下创建具有show和store方法的节点。能方便的进行调试。

一、使用DEVICE_ATTR构建device attribute

下面将顺着我们直接使用的DEVICE_ATTR来分析一下,这个宏究竟都做了哪些事情。

 DEVICE_ATTR的定义:

1

2

#define DEVICE_ATTR(_name, _mode, _show, _store) \

    struct device_attribute dev_attr_##_name = __ATTR(_name, _mode, _show, _store)

 而__ATTR宏的定义在include/linux/sysfs.h文件中,如下:

1

2

3

4

5

6

#define __ATTR(_name, _mode, _show, _store) {                \

    .attr = {.name = __stringify(_name),                \

         .mode = VERIFY_OCTAL_PERMISSIONS(_mode) },        \

    .show    = _show,                        \

    .store    = _store,                        \

}

 那么struct device_attribute的定义又是怎么样的呢?该结构体的定义在include /linux/device.h,其定义如下:

1

2

3

4

5

6

7

8

/* interface for exporting device attributes */

struct device_attribute {

    struct attribute    attr;

    ssize_t (*show)(struct device *dev, struct device_attribute *attr,

            char *buf);

    ssize_t (*store)(struct device *dev, struct device_attribute *attr,

             const char *buf, size_t count);

};

 而其中的struct attribute的定义在include/linux/device.h中,如下:

1

2

3

4

5

6

7

8

9

struct attribute {

    const char        *name;

    umode_t            mode;

#ifdef CONFIG_DEBUG_LOCK_ALLOC

    bool            ignore_lockdep:1;

    struct lock_class_key    *key;

    struct lock_class_key    skey;

#endif

};

 总结一下:

DEVICE_ATTR(_name, _mode, _show, _store)等价于:

1

2

3

4

5

6

struct device_attribute dev_attr_##_name = {

    .attr = {.name = __stringify(_name),

             .mode = VERIFY_OCTAL_PERMISSIONS(_mode)},

    .show = _show,

    .store = _store,

}

 其中:.show和.store的类型定义如下:

 show函数的详细描述:

1

2

ssize_t (*show)(struct device *dev, struct device_attribute *attr,

        char *buf);

  •  入参buf是需要我们填充的string即我们cat属性节点时要显示的内容;
  • 函数的返回值是我们填充buf的长度,且长度应当小于一个页面的大小(4096字节);
  • 其他参数一般不用关心。

例如:

1

2

3

4

5

static ssize_t show_my_device(struct device *dev,

                              struct device_attribute *attr, char *buf) //cat命令时,将会调用该函数

{

    return sprintf(buf, "%s\n", mybuf);

}

store函数的详细描述:

1

2

ssize_t (*store)(struct device *dev, struct device_attribute *attr,

          const char *buf, size_t count);

  •  入参buf是用户传入的字符串,即echo到属性节点的内容;
  • 入参count是buf中字符串的长度。
  • 函数的返回值通常返回count即可。
  • 其他参数一般不用关心。

例如:

1

2

3

4

5

6

7

static ssize_t store_my_device(struct device *dev,

                             struct device_attribute *attr,

                             const char *buf, size_t count) //echo命令时,将会调用该函数

{

    sprintf(mybuf, "%s", buf);

    return count;

}

mode的权限定义在kernel/include/uapi/linux/stat.h中。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

#define S_IRWXU 00700 //用户可读写和执行

#define S_IRUSR 00400//用户可读

#define S_IWUSR 00200//用户可写

#define S_IXUSR 00100//用户可执行

#define S_IRWXG 00070//用户组可读写和执行

#define S_IRGRP 00040//用户组可读

#define S_IWGRP 00020//用户组可写

#define S_IXGRP 00010//用户组可执行

#define S_IRWXO 00007//其他可读写和执行

#define S_IROTH 00004//其他可读

#define S_IWOTH 00002//其他可写

#define S_IXOTH 00001//其他可执行

  至此,我们已经定义好了.show和.store函数,那么就可以使用DEVICE_ATTR了。

1

static DEVICE_ATTR(my_device_test, S_IWUSR | S_IRUSR, show_my_device, store_my_device);

二、将device attribute添加到sysfs中

上面我们已经创建好了所需要的device attribute,下面就要将这些attribute添加到sysfs中了,此处可用的函数由sysfs_create_file和sysfs_create_group。

1、只有一个节点的创建sysfs_create_file

此处在驱动的probe或module_init函数调用sysfs_create_file即可,在module_exit或remove函数中调用sysfs_remove_file来移除节点。

1

2

int __must_check sysfs_create_file(struct kobject *kobj, const struct attribute *attr);

void sysfs_remove_file(struct kobject *kobj, const struct attribute *attr);

以下是全部的代码:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

#include <linux/module.h>

#include <linux/init.h>

#include <linux/platform_device.h>

#include <linux/gpio.h>

#include <linux/delay.h>

static char sysfs_buff[100] = "my_sysfs_test_string";

static ssize_t show_sys_device(struct device *dev,

                              struct device_attribute *attr, char *buf) //cat命令时,将会调用该函数

{

    return sprintf(buf, "%s\n", sysfs_buff);

}

static ssize_t store_sys_device(struct device *dev,

                             struct device_attribute *attr,

                             const char *buf, size_t count) //echo命令时,将会调用该函数

{

    sprintf(sysfs_buff, "%s", buf);

    return count;

}

static DEVICE_ATTR(sys_device_file, S_IWUSR | S_IRUSR, show_sys_device, store_sys_device);

//定义一个名字为sys_device_file的设备属性文件

struct file_operations mytest_ops = {

    .owner = THIS_MODULE,

};

static int major;

static struct class *cls;

struct device *mydev;

static int mytest_init(void)

{

     

    major = register_chrdev(0, "mytest", &mytest_ops);

    cls = class_create(THIS_MODULE, "mytest_class");

    mydev = device_create(cls, 0, MKDEV(major, 0), NULL, "mytest_device"); //创建mytest_device设备

    if (sysfs_create_file(&(mydev->kobj), &dev_attr_sys_device_file.attr))

    //在mytest_device设备目录下创建一个sys_device_file属性文件

        return -1;

    }

    return 0;

}

static void mytest_exit(void)

{

    device_destroy(cls, MKDEV(major, 0));

    class_destroy(cls);

    unregister_chrdev(major, "mytest");

    sysfs_remove_file(&(mydev->kobj), &dev_attr_sys_device_file.attr);

}

module_init(mytest_init);

module_exit(mytest_exit);

MODULE_LICENSE("GPL");

2、多个节点的创建sysfs_create_group

 定义attribute属性结构体数组到属性组中:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

static DEVICE_ATTR(sys_device_file, S_IWUSR | S_IRUSR, show_sys_device, store_sys_device);

static DEVICE_ATTR(sys_device_file0, S_IWUSR | S_IRUSR, show_sys_device, store_sys_device);

static DEVICE_ATTR(sys_device_file1, S_IWUSR | S_IRUSR, show_sys_device, store_sys_device);

static DEVICE_ATTR(sys_device_file2, S_IWUSR | S_IRUSR, show_sys_device, store_sys_device);

static struct attribute *sys_device_attributes[] = {

    &dev_attr_sys_device_file.attr,

    &dev_attr_sys_device_file0.attr,

    &dev_attr_sys_device_file1.attr,

    &dev_attr_sys_device_file2.attr,

    NULL属性结构体数组最后一项必须以NULL结尾。

};

static const struct attribute_group sys_device_attr_group = {

    .attrs = sys_device_attributes,

};

 下面就可以利用sysfs_create_group和sysfs_create_group来添加、移除属性节点组了。

定义如下:

1

2

int sysfs_create_group(struct kobject *kobj, const struct attribute_group *grp)

void sysfs_remove_group(struct kobject *kobj, const struct attribute_group *grp)

全部代码如下:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

#include <linux/module.h>

#include <linux/init.h>

#include <linux/platform_device.h>

#include <linux/gpio.h>

#include <linux/delay.h>

static char sysfs_buff[100] = "my_sysfs_test_string";

static ssize_t show_sys_device(struct device *dev,

                              struct device_attribute *attr, char *buf) //cat命令时,将会调用该函数

{

    return sprintf(buf, "%s\n", sysfs_buff);

}

static ssize_t store_sys_device(struct device *dev,

                             struct device_attribute *attr,

                             const char *buf, size_t count) //echo命令时,将会调用该函数

{

    sprintf(sysfs_buff, "%s", buf);

    return count;

}

//定义多个sys_device_file的设备属性文件

static DEVICE_ATTR(sys_device_file, S_IWUSR | S_IRUSR, show_sys_device, store_sys_device);

static DEVICE_ATTR(sys_device_file0, S_IWUSR | S_IRUSR, show_sys_device, store_sys_device);

static DEVICE_ATTR(sys_device_file1, S_IWUSR | S_IRUSR, show_sys_device, store_sys_device);

static DEVICE_ATTR(sys_device_file2, S_IWUSR | S_IRUSR, show_sys_device, store_sys_device);

static struct attribute *sys_device_attributes[] = {

    &dev_attr_sys_device_file.attr,

    &dev_attr_sys_device_file0.attr,

    &dev_attr_sys_device_file1.attr,

    &dev_attr_sys_device_file2.attr,

    NULL属性结构体数组最后一项必须以NULL结尾。

};

static const struct attribute_group sys_device_attr_group = {

    .attrs = sys_device_attributes,

};

struct file_operations mytest_ops = {

    .owner = THIS_MODULE,

};

static int major;

static struct class *cls;

struct device *mydev;

static int mytest_init(void)

{

     

    major = register_chrdev(0, "mytest", &mytest_ops);

    cls = class_create(THIS_MODULE, "mytest_class");

    mydev = device_create(cls, 0, MKDEV(major, 0), NULL, "mytest_device"); //创建mytest_device设备

    if (sysfs_create_group(&(mydev->kobj), &sys_device_attr_group))

    {

        //在mytest_device设备目录下创建多个sys_device_file属性文件

        return -1;

    }

    return 0;

}

static void mytest_exit(void)

{

    device_destroy(cls, MKDEV(major, 0));

    class_destroy(cls);

    unregister_chrdev(major, "mytest");

    sysfs_remove_group(&(mydev->kobj), &sys_device_attr_group);

}

module_init(mytest_init);

module_exit(mytest_exit);

MODULE_LICENSE("GPL");

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/16489.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Sping源码(八)—Spring事件驱动

观察者模式 在介绍Spring的事件驱动之前&#xff0c;先简单的介绍一下设计模式中的观察者模式。 在一个简单的观察者模式只需要观察者和被观察者两个元素。简单举个栗子&#xff1a; 以警察盯梢犯罪嫌疑人的栗子来说&#xff1a; 其中犯罪嫌疑人为被观察者元素而 警察和军人为…

【启程Golang之旅】基本变量与类型讲解

欢迎来到Golang的世界&#xff01;在当今快节奏的软件开发领域&#xff0c;选择一种高效、简洁的编程语言至关重要。而在这方面&#xff0c;Golang&#xff08;又称Go&#xff09;无疑是一个备受瞩目的选择。在本文中&#xff0c;带领您探索Golang的世界&#xff0c;一步步地了…

Fortran:forpy 嵌入Python

Fortran嵌入Python 利用forpy库&#xff0c;可以在Fortran程序内嵌入Python. program test_forpyuse forpy_modimplicit noneinteger::ierrierrforpy_initialize()!!---------------!!list!!---------------blocktype(list)::my_listierrlist_create(my_list)ierrmy_list%app…

【JVM】内存区域划分 | 类加载的过程 | 双亲委派机制 | 垃圾回收机制

文章目录 JVM一、内存区域划分1.方法区&#xff08;1.7之前&#xff09;/ 元数据区&#xff08;1.8开始&#xff09;2.堆3.栈4.程序计数器常见面试题&#xff1a; 二、类加载的过程1.类加载的基本流程1.加载2.验证3.准备4.解析5.初始化 2.双亲委派模型类加载器找.class文件的过…

Qml:第一个qml程序

//第一个qml程序import QtQuickText {height: 300text: "Hello World"width: 500 }推荐一个零声学院项目课&#xff0c;个人觉得老师讲得不错&#xff0c;分享给大家&#xff1a; 零声白金学习卡&#xff08;含基础架构/高性能存储/golang云原生/音视频/Linux内核&am…

[JDK工具-5] jinfo jvm配置信息工具

文章目录 1. 介绍2. 打印所有的jvm标志信息 jinfo -flags pid3. 打印指定的jvm参数信息 jinfo -flag InitialHeapSize pid4. 启用或者禁用指定的jvm参数 jinfo -flags [|-]HeapDumpOnOutOfMemoryError pid5. 打印系统参数信息 jinfo -sysprops pid6. 打印以上所有配置信息 jinf…

WordPress安装memcached提升网站速度

本教程使用环境为宝塔 第一步、服务器端安装memcached扩展 在网站使用的php上安装memcached扩展 第二步&#xff1a;在 WordPress 网站后台中&#xff0c;安装插件「Memcached Is Your Friend」 安装完成后启用该插件&#xff0c;在左侧工具-中点击Memcached 查看是否提示“U…

Leetcode - 398周赛

目录 一&#xff0c;3151. 特殊数组 I 二&#xff0c;3152. 特殊数组 II 三&#xff0c;3153. 所有数对中数位不同之和 四&#xff0c;3154. 到达第 K 级台阶的方案数 一&#xff0c;3151. 特殊数组 I 本题就是判断一个数组是否是奇偶相间的&#xff0c;如果是&#xff0c;…

「贪心算法」最大数

力扣原题链接&#xff0c;点击跳转。 有一个整数数组&#xff0c;我们可以按照任意顺序把这些数拼接成一个新的整数&#xff0c;如2、3和10可以拼接为2310、3102、2103等等。能拼出来的最大整数是多少呢&#xff1f;由于这个数可能非常大&#xff0c;所以结果是一个字符串。 …

Linux下的调试器 : gdb指令详解

&#x1fa90;&#x1fa90;&#x1fa90;欢迎来到程序员餐厅&#x1f4ab;&#x1f4ab;&#x1f4ab; 主厨&#xff1a;邪王真眼 主厨的主页&#xff1a;Chef‘s blog 所属专栏&#xff1a;青果大战linux 总有光环在陨落&#xff0c;总有新星在闪烁 gdb是什么 gdn是linu…

开源大模型与闭源大模型,你更看好哪一方?

开源大模型与闭源大模型&#xff0c;你更看好哪一方&#xff1f; 简介&#xff1a;评价一个AI模型“好不好”“有没有发展”&#xff0c;首先就躲不掉“开源”和“闭源”两条发展路径。对于这两条路径&#xff0c;你更看好哪一种呢&#xff1f; 1.方向一&#xff1a;数据隐私 …

springBoot项目中的static和templates文件夹

SpringBoot里面没有我们之前常规web开发的WebContent&#xff08;WebApp&#xff09;&#xff0c;它只有src目录在src/main/resources下面有两个文件夹 static和templates springboot默认 static中放静态页面&#xff0c;而templates中放动态页面。但是webapp文件夹可以有&…

基于webpack+Vue3+JavaScript+antd+less+axios技术栈实现所有组件全局自动化注册

一、前言 最近在做一个项目&#xff0c;因为项目比较小&#xff0c;用户群体年龄跨度较大&#xff0c;同时对兼容性的要求较高&#xff0c;所以选择webpackVue3JavaScriptantdlessaxios的技术栈&#xff0c;在开发的当中发现一个问题&#xff0c;原来在vue2当中&#xff0c;可…

react native 下载功能实现

RN 下载 demo iOS 安装必要的包 react-native-fs 下载使用react-native-permissions 获取权限react-native-share 保存下载的内容到手机 修改 podfile 文件 # Resolve react_native_pods.rb with node to allow for hoisting # require Pod::Executable.execute_command(n…

英伟达的GPU(3)

上节内容&#xff1a;英伟达的GPU(2) (qq.com) 书接上文&#xff0c;上文我们讲到CUDA编程体系和硬件的关系&#xff0c;也留了一个小问题CUDA core以外的矩阵计算能力是咋提供的 本节介绍一下Tensor Core 上节我们介绍了CUDA core&#xff0c;或者一般NPU&#xff0c;CPU执行…

【大数据面试题】31 Flink 有哪些重启方法

一步一个脚印&#xff0c;一天一道面试题 Flink 提供了几种不同的重启方法&#xff1a; 异常自动从 Checkpoint 重启: Checkpoint是Flink的另一种状态快照机制&#xff0c;它比Savepoint更为频繁&#xff0c;提供了细粒度的状态恢复点。通过配置Checkpoint&#xff0c;Flink会…

pyqt QMainWindow菜单栏

pyqt QMainWindow菜单栏 pyqt QMainWindow菜单栏效果代码 pyqt QMainWindow菜单栏 QMainWindow 是 PyQt中的一个核心类&#xff0c;它提供了一个主应用程序窗口&#xff0c;通常包含菜单栏、工具栏、状态栏、中心窗口&#xff08;通常是一个 QWidget 或其子类&#xff09;等。…

头歌OpenGauss数据库-I.复杂查询第4关:计算平均分

本关任务:根据提供的表和数据,查询平均成绩小于60分的同学的学生编号(s_id)、学生姓名(s_name)和平均成绩(avg_score),要求平均成绩保留2位小数点。(注意:包括有成绩的和无成绩的) student表数据: s_ids_names_sex01Mia女02Riley男03Aria女04Lucas女05Oliver男06C…

【数据结构/C语言】深入理解 双向链表

&#x1f493; 博客主页&#xff1a;倔强的石头的CSDN主页 &#x1f4dd;Gitee主页&#xff1a;倔强的石头的gitee主页 ⏩ 文章专栏&#xff1a;数据结构与算法 在阅读本篇文章之前&#xff0c;您可能需要用到这篇关于单链表详细介绍的文章 【数据结构/C语言】深入理解 单链表…

[vue error] vue3中使用同名简写报错 ‘v-bind‘ directives require an attribute value

错误详情 错误信息 ‘v-bind’ directives require an attribute value.eslintvue/valid-v-bind 错误原因 默认情况下&#xff0c;ESLint 将同名缩写视为错误。此外&#xff0c;Volar 扩展可能需要更新以支持 Vue 3.4 中的新语法。 解决方案 更新 Volar 扩展 安装或更新 …