sysfs是Linux内核提供的一个虚拟文件系统,用于将内核对象、属性和它们之间的关系导出到用户空间。在驱动程序中创建sysfs接口可以让用户空间程序方便地获取设备信息或配置设备参数。
#include <linux/sysfs.h>
#include <linux/kobject.h>
static ssize_t my_attr_show(struct kobject *kobj, struct kobj_attribute *attr, char *buf);
static ssize_t my_attr_store(struct kobject *kobj, struct kobj_attribute *attr, const char *buf, size_t count);
static struct kobj_attribute my_attr = __ATTR(my_attribute, 0644, my_attr_show, my_attr_store);
static struct attribute *my_attrs[] = {
&my_attr.attr,
NULL,
};
static struct attribute_group my_attr_group = {
.attrs = my_attrs,
};
static ssize_t my_attr_show(struct kobject *kobj, struct kobj_attribute *attr, char *buf)
{
// 从驱动中读取数据并写入buf
return sprintf(buf, "%d\n", some_value);
}
static ssize_t my_attr_store(struct kobject *kobj, struct kobj_attribute *attr, const char *buf, size_t count)
{
unsigned long val;
int ret;
ret = kstrtoul(buf, 10, &val);
if (ret < 0)
return ret;
// 使用val更新驱动状态
some_value = val;
return count;
}
static struct class *my_class;
// 在init函数中
my_class = class_create(THIS_MODULE, "my_class");
if (IS_ERR(my_class)) {
return PTR_ERR(my_class);
}
// 为每个设备创建
sysfs_create_group(&dev->kobj, &my_attr_group);
// 在exit函数中
class_destroy(my_class);
static struct kobject *my_kobj;
// 在init函数中
my_kobj = kobject_create_and_add("my_kobject", kernel_kobj);
if (!my_kobj) {
return -ENOMEM;
}
sysfs_create_group(my_kobj, &my_attr_group);
// 在exit函数中
kobject_put(my_kobj);
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/sysfs.h>
#include <linux/kobject.h>
static int my_value = 0;
static ssize_t my_attr_show(struct kobject *kobj, struct kobj_attribute *attr, char *buf)
{
return sprintf(buf, "%d\n", my_value);
}
static ssize_t my_attr_store(struct kobject *kobj, struct kobj_attribute *attr, const char *buf, size_t count)
{
int ret;
unsigned long val;
ret = kstrtoul(buf, 10, &val);
if (ret < 0)
return ret;
my_value = val;
return count;
}
static struct kobj_attribute my_attr = __ATTR(my_value, 0644, my_attr_show, my_attr_store);
static struct attribute *my_attrs[] = {
&my_attr.attr,
NULL,
};
static struct attribute_group my_attr_group = {
.attrs = my_attrs,
};
static struct kobject *my_kobj;
static int __init my_init(void)
{
int ret;
my_kobj = kobject_create_and_add("my_module", kernel_kobj);
if (!my_kobj) {
return -ENOMEM;
}
ret = sysfs_create_group(my_kobj, &my_attr_group);
if (ret) {
kobject_put(my_kobj);
return ret;
}
printk(KERN_INFO "sysfs interface created\n");
return 0;
}
static void __exit my_exit(void)
{
kobject_put(my_kobj);
printk(KERN_INFO "sysfs interface removed\n");
}
module_init(my_init);
module_exit(my_exit);
MODULE_LICENSE("GPL");
static struct bin_attribute my_bin_attr = {
.attr = {
.name = "binary_data",
.mode = 0644,
},
.read = my_bin_read,
.write = my_bin_write,
.size = MAX_SIZE,
};
// 然后使用 sysfs_create_bin_file() 创建
static DEVICE_ATTR(my_device_attr, 0644, dev_attr_show, dev_attr_store);
// 然后使用 device_create_file() 创建
可以使用 sysfs_add_file_to_group()
在运行时动态添加属性。
通过sysfs接口,用户空间可以通过简单的文件操作(cat/echo等)与驱动程序交互,这是一种非常方便的用户空间-内核空间通信机制。