Linux内核宏DEVICE_ATTR使用
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使用的更多相关文章
- linux内核宏container_of
首先来个简单版本 /* given a pointer @ptr to the field @member embedded into type (usually * struct) @type, r ...
- linux内核宏container_of前期准备之gcc扩展关键字typeof
typeof基本介绍 typeof(x) 这是它的使用方法,x可以是数据类型或者表达式.它的作用时期和sizeof类似,就是它是在编译器从高级语言(如C语言)翻译成汇编语言时起作用,这个很重要,稍后会 ...
- Linux 内核常见宏定义
我们在阅读Linux内核是,常见到这些宏 __init, __initdata, __initfunc(), asmlinkage, ENTRY(), FASTCALL()等等.它们定义在 /incl ...
- 转载 linux内核 asmlinkage宏
转载http://blog.chinaunix.net/uid-7390305-id-2057287.html 看一下/usr/include/asm/linkage.h里面的定义:#define a ...
- Linux内核中的fastcall和asmlinkage宏
代码中看见:#define _fastcall 所以了解下fastcall -------------------------------------------------------------- ...
- (十)Linux内核中的常用宏container_of
Container_of在Linux内核中是一个常用的宏,用于从包含在某个结构中的指针获得结构本身的指针,通俗地讲就是通过结构体变量中某个成员的首地址进而获得整个结构体变量的首地址. Containe ...
- Linux内核中的常用宏container_of
Container_of在Linux内核中是一个常用的宏,用于从包含在某个结构中的指针获得结构本身的指针,通俗地讲就是通过结构体变量中某个成员的首地址进而获得整个结构体变量的首地址. Containe ...
- 嵌入式C语言自我修养 04:Linux 内核第一宏:container_of
4.1 typeof 关键字 ANSI C 定义了 sizeof 关键字,用来获取一个变量或数据类型在内存中所占的存储字节数.GNU C 扩展了一个关键字 typeof,用来获取一个变量或表达式的类型 ...
- Linux内核中的常用宏container_of其实很简单【转】
转自:http://blog.csdn.net/npy_lp/article/details/7010752 开发平台:Ubuntu11.04 编 译器:gcc version 4.5.2 (Ubun ...
随机推荐
- java中的%取模
在java中的 % 实际上是取余. 下面为数学概念上的取余和取模: 对于整型数a,b来说,取模运算或者求余运算的方法都是: 1.求 整数商: c = a/b; 2.计算模或者余数: r = a - ...
- OO_BLOG4_UML系列学习
目录 Unit4 作业分析 作业 4-1 UML类图解析器UmlInteraction 作业 4-2 扩展解析器(UML顺序图.UML状态图解析,基本规则验证) 架构设计及OO方法理解的演进 测试理解 ...
- python高级编程——网络编程(二)
UDP 概念和介绍 UDP --- 用户数据报协议,是一个无连接的简单的面向数据报的运输层协议. UDP不提供可靠性,它只是把应用程序传给IP层的数据报发送出去,但是并 不能保证它们能到达目的地. U ...
- Vue – 基础学习(4):事件修饰符
Vue – 基础学习(3):事件修饰符
- Flutter 徐徐图之(一)—— 从搭建开发环境到 Hello World
一.环境变量 由于众所周知的原因,在国内访问 Flutter 有时可能会受到限制,所以在开发之前,需要先配置环境变量 MacOS: 编辑 ~/.bash_profile 文件 vim ~/.bash_ ...
- 两个概念:CCA和LDA
典型相关性分析(CCA) https://blog.csdn.net/Mbx8X9u/article/details/78824216 典型关联分析(Canonical Correlation Ana ...
- Django框架(十五)-- cookie和session组件
一.cookie 1.cookie的由来 HTTP协议是无状态的.无状态的意思是每次请求都是独立的,它的执行情况和结果与前面的请求和之后的请求都无直接关系,它不会受前面的请求响应情况直接影响,也不会直 ...
- Linux进程管理之ps
Linux 是一种动态系统,能够适应不断变化的计算需求.下面介绍一些 Linux 所提供的工具来进行进程的查看与控制,掌握这些让我们能在某些进程出现异常的时候及时查看相关的指标,从而解决问题. 进程管 ...
- Make Rounddog Happy(2019年杭电多校第十场1011+HDU6701+启发式分治)
目录 题目链接 题意 思路 代码 题目链接 传送门 题意 求有多少个子区间满足\(a_l,a_{l+1},\dots,a_r\)均不相同且\(max(a_l,a_{l+1},\dots,a_r)-(r ...
- Django Model的外键自关联‘self'和设置'unique_together'
在django的model定义中,有时需要某个Field引用当前定义的Model,比如一个部门(Department)的Model,它有一个字段是上级部门(super_department),上级部门 ...