美文网首页Linux驱动
L4. ATTR节点应用

L4. ATTR节点应用

作者: 拂去尘世尘 | 来源:发表于2020-06-21 14:06 被阅读0次

    1.ATTR介绍

    应用层与内核驱动层的交互,一般是通过驱动节点的读写来实现。即驱动开发人员在完成驱动设备的创建后,同时会创建对应的节点,且提供节点的访问函数,以便应用层开发调用。驱动提供接口的方法有注册file_operation结构体,另一种方法就是本文要记录的建立ATTR节点。
    使用DEVICE_ATTR,可以实现驱动在sys目录自动创建文件,我们只需要实现show和store函数即可。然后在应用层就能通过cat和echo命令来对sys创建出来的文件进行读写驱动设备,实现交互。

    2.基本原理概述

    2.1 DEVICE_ATTR()

    DEVICE_ATTR()定义位于Android/kernel-4.14/include/linux/device.h

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

    __ATTR()位于Android/kernel-4.14/include/linux/sysfs.h

    #define __ATTR_PREALLOC(_name, _mode, _show, _store) {                  \
       .attr = {.name = __stringify(_name),                                 \
                 .mode = SYSFS_PREALLOC | VERIFY_OCTAL_PERMISSIONS(_mode) },\
       .show   = _show,                                                     \
       .store   = _store,                                                   \
    }
    

    发现DEVICE_ATTR()只是内核定义的一个宏,使用此宏作用相当于定义并填充了一个device_attribute类型的结构体dev_attr_xx。其中需要注意的是_mode参数表示文件权限,一般是给0664,权限比0664高会报错(也可以用S_IWUSR(用户可写),S_IRUSR(用户可读)等宏代替)。

    将DEVICE_ATTR()宏完全展开

    #define DEVICE_ATTR(_name, _mode, _show, _store)         \
    struct device_attribute dev_attr_##_name = {           \
       .attr = {.name = __stringify(_name),                  \
                 .mode = SYSFS_PREALLOC | VERIFY_OCTAL_PERMISSIONS(_mode) },\
       .show   = _show,                                        \
       .store   = _store,                                        \
    }
    

    2.2 device_attribute

    device_attribute同样定义位于Android/kernel-4.14/include/linux/device.h

    /* 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);
    };
    

    device_attribute结构体内包含一个attribute 和 show、store函数。其中show接口实现adb内 cat读取显示功能,store实现的是echo 写入功能。
    因此在驱动中,需要实现show、store接口功能。

    2.3 device_create_file

    在实例完struct device_attribute dev_attr_xx后,需要将此结构体通过device_create_file注册到内核中/sys/bus/platform/devices下。定义在kernel-4.14/drivers/base/core.c

    int device_create_file(struct device *dev,
                 const struct device_attribute *attr)
    {
       int error = 0;
       if (dev) {
          WARN(((attr->attr.mode & S_IWUGO) && !attr->store),
             "Attribute %s: write permission without 'store'\n",
             attr->attr.name);
          WARN(((attr->attr.mode & S_IRUGO) && !attr->show),
             "Attribute %s: read permission without 'show'\n",
             attr->attr.name);
          error = sysfs_create_file(&dev->kobj, &attr->attr);
       }
       return error;
    }
    EXPORT_SYMBOL_GPL(device_create_file);
    

    3.实际操作

    首先在内核创建一个驱动文件,这里命名为attr_test。
    增加节点读接口

    /创建ATTR可读节点接口/

    static ssize_t show_attr_test(struct device *dev, struct device_attribute *attr, char *buf)
    {
    int ret = 0;
    
        ret = snprintf(buf, strlen(global_char), "%s\n", global_char);
    printk("%s:%d: Entry %s, buf:%s", __FILE__, __LINE__, __func__, buf);
    
        return ret;
    }
    

    /* 创建ATTR可写节点接口 */

    static ssize_t store_attr_test(struct device *dev, struct device_attribute *attr,
                 const char *buf, size_t count)
    {
        int ret = 0;
       
        memset(global_char, 0x00, sizeof(global_char));
        ret = snprintf(global_char, sizeof(global_char), "%s", buf);
        printk("%s:%d: Entry %s", __FILE__, __LINE__, __func__);
    
        return ret;
    }
    

    填充到结构体dev_attr_attr_test

    DEVICE_ATTR(attr_test, 0664, show_attr_test, store_attr_test);
    

    注册到内核中

    #if 0
    //默认创建节点 注册节点: /sys/class/bus/platform/attr_test
    ret = device_create_file(&pdev->dev, &dev_attr_attr_test); 
    #else
    //自定义路径/sys/class/diy_class/diy_device/attr_test
    diy_class = class_create(THIS_MODULE, "diy_class");
    diy_device = device_create(diy_class, NULL,
                                                MKDEV(0, 0),
                                                NULL, "diy_device");
    ret = device_create_file(diy_device, &dev_attr_attr_test);
    #endif
    

    注意:在实现show、store时,返回值必须为具体的字符长度,若返回0则会出导致数据传输出现问题,cat、echo功能也会失败。

    2020-06-21

    相关文章

      网友评论

        本文标题:L4. ATTR节点应用

        本文链接:https://www.haomeiwen.com/subject/icxaxktx.html