DEVICE_ATTR设置设备属性
程序员文章站
2022-04-29 18:42:33
添加sysfs接口 使用device_create_file/ device_create_group 添加static DEVICE_ATTR(fm1288, 0666, fm1288_show, fm1288_store); ......
为了在sysfs下生成可控节点,方便上层调用。
sysfs是一个基于ram的文件系统,它和kobject一起,可以将kernel的数据结构导出到用户空间,以文件目录结构的形式,提供对这些数据结构(以及数据结构的属性)的访问支持。linux设备模型(4)_sysfs
原型:
#define device_attr(_name, _mode, _show, _store) struct device_attribute dev_attr_##_name = __attr(_name, _mode, _show, _store)
类似的还有driver_attr
,bus_attr
,class_attr
。区别就是,device_attr
对应的文件在/sys/devices/目录中对应的device下面。而其他几个分别在driver
,bus
,class
中对应的目录下。(待确认)
我实际测试,用device_attr
增加的i2c设备,节点是在/sys/bus/i2c/devices/0-0060/fm1288/
下的。按照上面的说法,应该fm1288
节点应该是在/sys/device
下
用法:
static ssize_t fm1288_store(struct device *dev, struct device_attribute *attr, const char *buf, size_t count) { struct i2c_client *i2c = container_of(dev, struct i2c_client, dev); unsigned int reg = 0, write = 0; __u16 val = 0; int ret; sscanf(buf,"%x %x %x", &write, ®, (unsigned int *)&val);//用sscanf读取sysfs写入的值 echo "1 2 3" > fm1288_mode fm2018_write(i2c, reg, val); fm2018_read(i2c, reg, &val); } fm1288_showstatic ssize_t fm1288_show(struct device *dev, struct device_attribute *attr, const char *buf, size_t count) { //read mode ssprintf("%d", mode); } static device_attr(fm1288, s_iwusr,//模式可以只读0444,只写0222,或者读写都行0666 fm1288_show, //null,不需要该函数可以写null fm1288_store); //注意_show和_store的命名一般习惯都是来自name+ _show / _store static int fm2018_i2c_probe(struct i2c_client *i2c, const struct i2c_device_id *id) { struct device *dev = &i2c->dev; ret = device_create_file(&i2c->dev, &dev_attr_register); if (ret) { dev_err(&i2c->dev, "failed to create sysfs files\n"); return ret; } }
也可以写多个属性,这时,使用sysfs_create_group
注册就很方便,代替device_create_file
static device_attr(xtilt, s_irugo | s_iwugo, show_tabletxtilt, store_tabletxtilt); static device_attr(ytilt, s_irugo | s_iwugo, show_tabletytilt, store_tabletytilt); static struct attribute *aiptek_attributes[] = { &dev_attr_xtilt.attr,//名字 dev_attr_name.attr &dev_attr_ytilt.attr, null }; static struct attribute_group aiptek_attribute_group = { .attrs = aiptek_attributes, }; aiptek_probe(struct usb_interface *intf, const struct usb_device_id *id) { int err = -enomem; err = sysfs_create_group(&intf->dev.kobj, &aiptek_attribute_group); if (err) { dev_warn(&intf->dev, "cannot create sysfs group err: %d\n", err); goto fail3; } fail3: return -1; }