linux设备驱动之ioctl控制
大部分驱动除了具有读写的能力之外,还需要具有对硬件控制的能力。
用户空间:ioctl
用户程序使用ioctl系统调用来控制设备。用户程序只是通过命令码告诉驱动程序想做什么,至于怎么解释这些命令和怎么实现这些命令,这都是驱动程序要做的事情。
- 函数原型:int ioctl(int fd, unsigned long cmd, …);
- 参数解析:
1. fd:打开的文件描述符
2. cmd:控制命令码
3. …:可选参数,具体内容依赖于cmd
驱动程序:ioctl
函数原型:
int (*ioctl)(struct inode *inode, struct file *filp, unsigned int cmd, unsigned long arg);
参数解析:
- inode和filp两个指针对应于应用程序传递的文件描述符fd,和传递open方法的参数一样。
- cmd:由用户空间直接不经修改的传递给驱动程序
- arg:可选参数
核心操作:命令码
在ioctl中命令码是唯一联系用户程序和驱动程序的途径
在驱动程序中的ioctl函数体内,有一个switch{case}结构,每一个case对应一个命令码,做出一些相应的操作。程序员要做的就是实现这些操作。
在Linux内核中,是这样定义一个命令码:
设备类型 | 序列号 | 方向 | 数据尺寸 |
---|---|---|---|
8 bit | 8 bit | 2 bit | 8~14 bit |
定义命令
命令码很不直观,因此内核提供了一些宏来帮助定义命令码,这些宏可以根据便于理解的字符串生成命令码,
或者从命令码得到一些可以理解的字符串来标明这个命令对应的设备类型,设备序列号,数据传输方向和数据传输尺寸。
如下宏:
//type是设备类型,nr是序列号,datatype是数据类型,比如int
_IO(type, nr)//没有参数的命令,常用来打印信息
_IOR(type, nr, datatype)//从驱动中读数据
_IOW(type, nr, datatype)//向驱动写数据
_IOWR(type, nr, datatype)//读写数据,双向传输
定义命令例子:
#define MEM_IOC_MAGIC 'm' //设备类型
#define MEM_IOCSET _IOW(MEM_IOC_MAGIC, 0, int) //向驱动写数据
#define MEM_IOCGET _IOR(MEM_IOC_MAGIC, 1, int) //从驱动读数据
实现命令
实现命令包括三个技术环节:
- 返回值:当switch语句的命令不能匹配任何一个设备支持的命令时,通常返回-EINVAL(非法参数)
- 参数使用:
<1>用户程序使用ioctl(int fd, unsigned long cmd, …)的时候,三个点..就是要传递的参数
<2>驱动程序通过int (*ioctl)(struct inode, struct file *filp, unsigned int cmd, unsigned long arg)中的arg传递;
<3>如果arg是一个整数,可以直接使用。如果是指针,必须确保这个用户地址有效,因此使用之前必须进行检查。
内部有检测