sysfs接口函数到建立_DEVICE_ATTR

本文深入探讨了sysfs接口在Sensor驱动中的应用,包括DEVICE_ATTR宏的使用、接口创建过程及其实现原理。通过简单步骤展示了如何在adbshell终端查看接口,并解释了接口与硬件操作之间的联系。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

sysfs接口函数到建立_DEVICE_ATTR

 

最近在弄Sensor驱动,看过一个某厂家的成品驱动,里面实现的全都是sysfs接口,hal层利用sysfs生成的接口,对Sensor进行操作。

说道sysfs接口,就不得不提到函数宏 DEVICE_ATTR

原型是#define DEVICE_ATTR(_name, _mode, _show, _store)\

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

函数宏DEVICE_ATTR内封装的是__ATTR(_name,_mode,_show,_stroe)方法,_show表示的是读方法,_stroe表示的是写方法。

当然_ATTR不是独生子女,他还有一系列的姊妹__ATTR_RO宏只有读方法,__ATTR_NULL等等

如对设备的使用  DEVICE_ATTR  ,对总线使用  BUS_ATTR  ,对驱动使用 DRIVER_ATTR  ,对类别 (class) 使用  CLASS_ATTR,  这四个高级的宏来自于<include/linux/device.h>

DEVICE_ATTR  宏声明有四个参数,分别是名称、权限位、读函数、写函数。其中读函数和写函数是读写功能函数的函数名。

如果你完成了DEVICE_ATTR函数宏的填充,下面就需要创建接口了

例如:

   static DEVICE_ATTR(polling, S_IRUGO | S_IWUSR, show_polling, set_polling);
    static struct attribute *dev_attrs[] = {
            &dev_attr_polling.attr,
            NULL,
    };

当你想要实现的接口名字是polling的时候,需要实现结构体struct attribute *dev_attrs[]

其中成员变量的名字必须是&dev_attr_polling.attr

然后再封装

    static struct attribute_group dev_attr_grp = {
            .attrs = dev_attrs,
    };

在利用sysfs_create_group(&pdev->dev.kobj, &dev_attr_grp);创建接口

通过以上简单的三个步骤,就可以在adb shell 终端查看到接口了。当我们将数据 echo 到接口中时,在上层实际上完成了一次 write 操作,对应到 kernel ,调用了驱动中的 “store”。同理,当我们cat 一个 接口时则会调用 “show” 。到这里,只是简单的建立了 android 层到 kernel 的桥梁,真正实现对硬件操作的,还是在 "show" 和 "store" 中完成的。

### DEVICE_ATTR_RW 和 DEVICE_ATTR 的区别 Linux内核提供了多种宏来简化设备属性的创建过程。`DEVICE_ATTR` 是用于定义只读或可写属性的基础宏[^2]。 对于 `DEVICE_ATTR` 宏而言,其语法结构如下: ```c #define DEVICE_ATTR(_name, _mode, _show, _store) ``` 此宏接受四个参数:属性名称 `_name`、权限模式 `_mode` 以及两个函数指针 `_show` 和 `_store` 分别对应于读取操作和写入操作。当只需要实现单向的数据访问(即要么能读不能写,要么能写不能读),可以仅提供相应的回调函数;如果不需要某个方向的操作,则可以把该位置设为空(NULL)。 而 `DEVICE_ATTR_RW` 则是一个更高层次封装版本,在功能上更加具体化。它专门用来声明既支持读又支持写的双向属性,并自动为用户提供默认的读写接口模板。因此,使用起来更为简便快捷。其实现方式本质上还是依赖于基础版的 `DEVICE_ATTR` 来完成实际工作[^1]。 通过比较可以看出,虽然两者都基于相同的底层机制构建而成,但是面向的应用场景有所不同——前者给予开发者更大的灵活性去定制特定行为,后者则更侧重于快速搭建常规性的读写特性。 #### 示例代码展示 以下是利用上述两种不同类型的宏实例化的简单示例: ```c // 使用 DEVICE_ATTR 创建一个名为 "example_attr" 的属性, // 支持读 (read_example_show),也支持写 (write_example_store). static ssize_t read_example_show(struct device *dev, struct device_attribute *attr, char *buf); static ssize_t write_example_store(struct device *dev, struct device_attribute *attr, const char *buf, size_t count); static DEVICE_ATTR(example_attr, 0644, read_example_show, write_example_store); // 使用 DEVICE_ATTR_RW 可以达到相同的效果,但编码量减少。 static ssize_t example_rw_show(struct device *dev, struct device_attribute *attr, char *buf); static ssize_t example_rw_store(struct device *dev, struct device_attribute *attr, const char *buf, size_t count); static DEVICE_ATTR_RW(wakealarm); // 实际应用中的例子来自 rtc-sysfs.c 文件 ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值