1、前言

在Linux驱动程序编写中,使用DEVICE_ATTR宏,可以定义一个struct device_attribute设备属性,并使用sysfs的API函数,便可以在设备目录下创建出属性文件,当我们在驱动程序中实现了show和store函数后,便可以使用cat和echo命令对创建出来的设备属性文件进行读写,从而达到控制设备的功能。

2、宏DEVICE_ATTR定义

在讲解DEVICE_ATTR宏之前,先了解一些基本的结构体,首先是struct attribute结构体,其定义在include/linux/device.h中,结构体定义如下所示:

struct attribute {
const char *name;
umode_t mode;
#ifdef CONFIG_DEBUG_LOCK_ALLOC
bool ignore_lockdep:;
struct lock_class_key *key;
struct lock_class_key skey;
#endif
};

该结构体有两个重要的成员,分别是name和mode,其中name代表属性的名称,一般表示为文件名,mode代表该属性的读写权限,也就是属性文件的读写权限。

关于文件的权限详解,可以参考下面的链接:

https://blog.csdn.net/DLUTBruceZhang/article/details/8658475

接下来要了解的结构体为struct device_attribute,该结构体的定义在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);
};

该结构体其实是struct attribute结构体的进一步封装,并提供了两个函数指针,show函数用于读取设备的属性文件,而store则是用于写设备的属性文件,当我们在Linux的驱动程序中实现了这两个函数后,便可以使用cat和echo命令对设备属性文件进行读写操作。

了解了一下基本的结构体,接下来,进一步分析宏DEVICE_ATTR的实现,在Linux内核源码中,宏DEVICE_ATTR的定义在include/linux/device.h文件中,如下:

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

而__ATTR宏的定义在include/linux/sysfs.h文件中,如下:

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

通过上面的宏展开可以发现,其实宏DEVICE_ATTR实现的功能就是定义了一个struct device_attribute结构体变量dev_attr_name,并对里面的成员进行初始化,包括struct attribute结构体里面的name和mode成员变量,然后还有实现属性文件读写的show和store函数赋值,非常简单。

3、使用示例

接下来对宏DEVICE_ATTR使用示例进行分析,该示例在内核中将100个字节虚拟成一个设备,在驱动中实现设备属性文件的读写函数,示例如下:

首先是设备属性的定义,以及设备属性文件读写函数的实现:

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, , 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(, "mydevice", &myfops);
if (major < ) {
ret = major;
return ret;
} myclass = class_create(THIS_MODULE, "myclass");
if (IS_ERR(myclass)) {
ret = -EBUSY;
goto fail;
} mydevice = device_create(myclass, NULL, MKDEV(major, ), NULL, "mydevice");
if (IS_ERR(mydevice)) {
class_destroy(myclass);
ret = -EBUSY;
goto fail;
} ret = sysfs_create_file(&mydevice->kobj, &dev_attr_mydevice.attr);
if (ret < )
return ret; return ; fail:
unregister_chrdev(major, "mydevice");
return ret;
}

函数首先调用register_chrdev()完成一个主设备号的动态申请,设备的名称为mydevice,然后调用class_create()和device_create()在sysfs中动态创建出设备所属的类myclass和mydevice设备,需要注意的是,这两个函数调用后,要对返回的结果进行错误检测,最后,使用sysfs的API函数sysfs_create_file()在sysfs中创建出设备的属性文件,完成驱动模块的加载。

接下来是该模块的卸载函数,函数的代码如下所示:

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

模块的卸载函数与模块的加载函数是相对的操作,需要调用device_destory()和class_destory()对模块加载创建的myclass和mydevice进行销毁,然后调用unregister_chrdev()将动态分配的主设备号进行释放。

驱动程序完整代码如下:

#include <linux/module.h>
#include <linux/init.h>
#include <linux/sysfs.h>
#include <linux/string.h>
#include <linux/device.h>
#include <linux/fs.h> static char mybuf[] = "mydevice";
static int major;
static struct class *myclass; 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, , mydevice_show, mydevice_store); static struct file_operations myfops = {
.owner = THIS_MODULE,
}; static int __init mydevice_init(void)
{
int ret;
struct device *mydevice; major = register_chrdev(, "mydevice", &myfops);
if (major < ) {
ret = major;
return ret;
} myclass = class_create(THIS_MODULE, "myclass");
if (IS_ERR(myclass)) {
ret = -EBUSY;
goto fail;
} mydevice = device_create(myclass, NULL, MKDEV(major, ), NULL, "mydevice");
if (IS_ERR(mydevice)) {
class_destroy(myclass);
ret = -EBUSY;
goto fail;
} ret = sysfs_create_file(&mydevice->kobj, &dev_attr_mydevice.attr);
if (ret < )
return ret; return ; fail:
unregister_chrdev(major, "mydevice");
return ret;
} static void __exit mydevice_exit(void)
{
device_destroy(myclass, MKDEV(major, ));
class_destroy(myclass);
unregister_chrdev(major, "mydevice");
} module_init(mydevice_init);
module_exit(mydevice_exit); MODULE_DESCRIPTION("A simplest driver");
MODULE_LICENSE("GPL v2");
MODULE_AUTHOR("HLY");

4、测试结果

将驱动程序进行编译后,将驱动模块进行加载,并查看创建出来的属性文件,使用cat和echo命令进行读写测试:

# make
# insmod simple-device.ko
# cd /sys/devices/virtual/myclass/mydevice/
# ls –al ./

结果如下所示,可以看到创建出来的属性文件mydevice:

使用cat和echo命令进行文件读写测试:

# cat mydevice
# echo "I am a simplest driver." > mydevice
# cat mydevice

运行结果如下所示,可以看到,能使用cat和echo命令正常完成属性文件的读写了:

5、小节

本文简单介绍了Linux内核中DEVICE_ATTR宏的实现,并使用一个简单的驱动程序示例来介绍了如何在Linux驱动程序中使用DEVICE_ATTR宏以及实现属性文件的读写函数。

参考:

https://blog.csdn.net/hpu11/article/details/83113729

https://blog.csdn.net/DLUTBruceZhang/article/details/8658475

《LINUX设备驱动程序(第三版)》

Linux内核宏DEVICE_ATTR使用的更多相关文章

  1. linux内核宏container_of

    首先来个简单版本 /* given a pointer @ptr to the field @member embedded into type (usually * struct) @type, r ...

  2. linux内核宏container_of前期准备之gcc扩展关键字typeof

    typeof基本介绍 typeof(x) 这是它的使用方法,x可以是数据类型或者表达式.它的作用时期和sizeof类似,就是它是在编译器从高级语言(如C语言)翻译成汇编语言时起作用,这个很重要,稍后会 ...

  3. Linux 内核常见宏定义

    我们在阅读Linux内核是,常见到这些宏 __init, __initdata, __initfunc(), asmlinkage, ENTRY(), FASTCALL()等等.它们定义在 /incl ...

  4. 转载 linux内核 asmlinkage宏

    转载http://blog.chinaunix.net/uid-7390305-id-2057287.html 看一下/usr/include/asm/linkage.h里面的定义:#define a ...

  5. Linux内核中的fastcall和asmlinkage宏

    代码中看见:#define _fastcall 所以了解下fastcall -------------------------------------------------------------- ...

  6. (十)Linux内核中的常用宏container_of

    Container_of在Linux内核中是一个常用的宏,用于从包含在某个结构中的指针获得结构本身的指针,通俗地讲就是通过结构体变量中某个成员的首地址进而获得整个结构体变量的首地址. Containe ...

  7. Linux内核中的常用宏container_of

    Container_of在Linux内核中是一个常用的宏,用于从包含在某个结构中的指针获得结构本身的指针,通俗地讲就是通过结构体变量中某个成员的首地址进而获得整个结构体变量的首地址. Containe ...

  8. 嵌入式C语言自我修养 04:Linux 内核第一宏:container_of

    4.1 typeof 关键字 ANSI C 定义了 sizeof 关键字,用来获取一个变量或数据类型在内存中所占的存储字节数.GNU C 扩展了一个关键字 typeof,用来获取一个变量或表达式的类型 ...

  9. Linux内核中的常用宏container_of其实很简单【转】

    转自:http://blog.csdn.net/npy_lp/article/details/7010752 开发平台:Ubuntu11.04 编 译器:gcc version 4.5.2 (Ubun ...

随机推荐

  1. ThinkPHP各个目录是什么含义ThinkPHP怎么安装和使用

    最近kdchxue看完了smarty之后,想学习下框架,于是乎就选择了ThinkPHP,听说这个框架简单易用,另外还是国产的!所以kdchxue毫不犹豫的就选择了ThinkPHP 了!下面看看Thin ...

  2. Matlab混入模式(Mixin)

    Mixin是一种类,这种类包含了其他类要使用的特性方法,但不必充当其他类的父类.Matlab无疑是支持多继承的.我们可以利用 Matlab 的这种特性,实现一种叫做 Mixin 的类.MixIn的目的 ...

  3. tcp / udp 协议及其实现的socket

    一.tcp协议 1.1 基本知识 特点: 可靠,慢,全双工通信 建立连接时:三次握手 断开连接时:四次挥手 在建立起连接之后 发送的每一条信息都有回执 为了保证数据的完整性,还有重传机制 长连接:会一 ...

  4. PHP实现sha1加密AES算法加密解密数据

    一.加密代码如下: /** * * @param string $string 需要加密的字符串 * @param string $key 密钥 * @return string */ public ...

  5. 记录下vue keep-alive IOS下无法保存滚动scroll位置的问题

    最近 做的项目,遇到了一点小麻烦,就是我一个页面A页面是加载 列表数据 ,B页面是展示详细信息的.A进去B时,缓存A页面. 效果 做出来 后,缓存是缓存数据 了,但是当我A页面的列表数据 好多,要滚动 ...

  6. Android 9.0网络权限适配

    在做Android开发时,使用华为的p20和平板(均为Android 9.0)测试时,发现不能使用WIFI网络,一番郁闷纠结查找后 直接上方法: 在res文件夹下创建xml文件夹,在xml里面创建文件 ...

  7. Android源码分析(十五)----GPS冷启动实现原理分析

    一:原理分析 主要sendExtraCommand方法中传递两个参数, 根据如下源码可以知道第一个参数传递delete_aiding_data,第二个参数传递null即可. @Override pub ...

  8. 设计模式:单例(Sigleton)模式

    题目:设计一个类,我们只能生成该类的一个实例. 只能生成一个实例的类是实现了Singleton(单例)模式的类型. 相关知识: 这种模式涉及到一个单一的类,该类负责创建自己的对象,同时确保只有单个对象 ...

  9. C# Net 通用json转Object(对象)

    C# Net  通用 json 转 Object  对象 C# Net 提取 json 字符串 对象 数组 C# Net json 对象 中有字符串 转为 对象 例如输入:{"1" ...

  10. Linux C/C++编译过程中的各种not declared in this scope

    Linux C/C++编译时经常会"XXX was not declared in this scope" 原因可能是以下几种: 变量名或函数名写错了; 忘记定义了 没有成功链接到 ...