Linux内核DEVICE_ATTR_RW宏分析

参考:https://www.cnblogs.com/Cqlismy/p/11454573.html
在Linux驱动中经常会用到DEVICE_ATTR宏,可以定义一个struct device_attribute设备属性,并使用sysfs的API函数,便可以在设备目录下创建出属性文件,当我们在驱动程序中实现了show和store函数后,便可以使用cat和echo命令对创建出来的设备属性文件进行读写,从而达到控制设备的功能。
接下来是相关的基本结构体:
struct attribute{}
里面有两个重要的成员:
name:属性的名称
mode:属性的读写权限
struct device_attribute{}对struct attribute结构体进一步封装,提供了两个函数指针
show函数:用于读取设备的属性文件
store函数:用于写设备的属性文件
可以使用cat和echo命令对设备属性文件进行读写操作

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

宏DEVICE_ATTR实现的功能就是定义了一个struct device_attribute结构体变量dev_attr_name,并对里面的成员进行初始化,包括struct attribute结构体里面的name和mode成员变量,然后还有实现属性文件读写的show和store函数赋值。
案例:

//读操作
static ssize_t mydevice_show(struct device *dev, struct device_attribute *attr,
        char *buf)
{
    return sprintf(buf, "%s\n", mybuf);
}
//写操作
static ssize_t mydevice_store(struct device *dev, struct device_attribute *attr,
         const char *buf, size_t count)
{
    sprintf(mybuf, "%s", buf);

    return count;
}
static DEVICE_ATTR(mydevice, 0644, mydevice_show, mydevice_store);//定义一个自己的属性文件,并赋予权限

以上,使用宏DEVICE_ATTR定义了一个mydevice的属性文件,而且这个属性文件的读写权限为:文件所拥有者可读写,组和其他人只能读。代码还实现了该属性文件的读写函数mydevice_show()和mydevice_store(),当在用户空间中使用cat和echo命令时,将会调用到驱动程序中实现的两个函数。
模块加载函数,当模块加载时将会被调用

static int __init mydevice_init(void)//初始化
{
    int ret;
    struct device *mydevice;
    
    major = register_chrdev(0, "mydevice", &myfops); //1、主设备号动态申请
    if (major < 0) {
        ret = major;
        return ret;
    }
    
    myclass = class_create(THIS_MODULE, "myclass"); //动态创建设备所属 的类myclass
    if (IS_ERR(myclass)) {
        ret = -EBUSY;
        goto fail;
    }
    
    mydevice = device_create(myclass, NULL, MKDEV(major, 0), NULL, "mydevice");//动态创建mydevice设备
    if (IS_ERR(mydevice)) {                   //错误检测
        class_destroy(myclass);
        ret = -EBUSY;
        goto fail;                                      //出现问题goto fail进行对注册的设备号的注销
    }

    ret = sysfs_create_file(&mydevice->kobj, &dev_attr_mydevice.attr);//在sysfs中创建出设备的属性文件
    if (ret < 0)
        return ret;

    return 0;

fail:
    unregister_chrdev(major, "mydevice");               
    return ret;
}

进行主设备的动态申请并调用class_dreate()和device_create()函数动态创建类和设备,并且对其进行错误检测,组后使用sysfs'的API函数sysfs_create_file()在sysfs中创建出设备的属性文件,完成驱动模块的加载。
模块卸载函数
需要调用device_destory()和class_destory()对模块加载创建的myclass和mydevice进行销毁,然后调用unregister_chrdev()将动态分配的主设备号进行释放。

static void __exit mydevice_exit(void)
{
    device_destroy(myclass, MKDEV(major, 0));
    class_destroy(myclass);
    unregister_chrdev(major, "mydevice");
}

测试:

编译之后通过cd /sys/devices/virtual/myclass/mydevice/可以在其目录下看到新建的mydevice设备属性文件
通过cat和echo来进行读写
cat mydevice
echo "I am a simplest driver." > mydevice
cat mydevice


你可能感兴趣的:(Linux内核DEVICE_ATTR_RW宏分析)