一、 什么是ioctl。
ioctl是设备驱动程序中对设备的I/O通道进行管理的函数。所谓对I/O通道进行管理,就
是对设备的一些特性进行控制,例如串口的传输波特率、马达的转速等等。它的调用个数
如下:
int ioctl(int fd, ind cmd, …);
其中fd就是用户程序打开设备时使用open函数返回的文件标示符,cmd就是用户程序对设
备的控制命令,至于后面的省略号,那是一些补充参数,一般最多一个,有或没有是和
cmd的意义相关的。
ioctl函数是文件结构中的一个属性分量,就是说如果你的驱动程序提供了对ioctl的支
持,用户就可以在用户程序中使用ioctl函数控制设备的I/O通道。
二、 ioctl的必要性
如果不用ioctl的话,也可以实现对设备I/O通道的控制,但那就是蛮拧了。例如,我们可
以在驱动程序中实现write的时候检查一下是否有特殊约定的数据流通过,如果有的话,
那么后面就跟着控制命令(一般在socket编程中常常这样做)。但是如果这样做的话,会
导致代码分工不明,程序结构混乱,程序员自己也会头昏眼花的。
所以,我们就使用ioctl来实现控制的功能。要记住,用户程序所作的只是通过命令码告
诉驱动程序它想做什么,至于怎么解释这些命令和怎么实现这些命令,这都是驱动程序要
做的事情。
三、 ioctl如何实现
这是一个很麻烦的问题,我是能省则省。要说清楚它,没有四五千字是不行的,所以我这
里是不可能把它说得非常清楚了,不过如果有读者对用户程序怎么和驱动程序联系起来感
兴趣的话,可以看我前一阵子写的《write的奥秘》。读者只要把write换成ioctl,就知
道用户程序的ioctl是怎么和驱动程序中的ioctl实现联系在一起的了。
我这里说一个大概思路,因为我觉得《Linux设备驱动程序》这本书已经说的非常清楚
了,但是得化一些时间来看。
在驱动程序中实现的ioctl函数体内,实际上是有一个switch{case}结构,每一个case对
应一个命令码,做出一些相应的操作。怎么实现这些操作,这是每一个程序员自己的事
情,因为设备都是特定的,这里也没法说。关键在于怎么样组织命令码,因为在ioctl中
命令码是唯一联系用户程序命令和驱动程序支持的途径。
命令码的组织是有一些讲究的,因为我们一定要做到命令和设备是一一对应的,这样才不
会将正确的命令发给错误的设备,或者是把错误的命令发给正确的设备,或者是把错误的
命令发给错误的设备。这些错误都会导致不可预料的事情发生,而当程序员发现了这些奇
怪的事情的时候,再来调试程序查找错误,那将是非常困难的事情。
所以在Linux核心中是这样定义一个命令码的:
____________________________________
| 设备类型 | 序列号 | 方向 |数据尺寸|
|----------|--------|------|--------|
| 8 bit | 8 bit |2 bit |8~14 bit|
|----------|--------|------|--------|
这样一来,一个命令就变成了一个整数形式的命令码。但是命令码非常的不直观,所以
Linux Kernel中提供了一些宏,这些宏可根据便于理解的字符串生成命令码,或者是从
命令码得到一些用户可以理解的字符串以标明这个命令对应的设备类型、设备序列号、数
据传送方向和数据传输尺寸。
这些宏我就不在这里解释了,具体的形式请读者察看Linux核心源代码中的和,文件里给
除了这些宏完整的定义。这里我只多说一个地方,那就是""幻数""。
幻数是一个字母,数据长度也是8,所以就用一个特定的字母来标明设备类型,这和用一
个数字是一样的,只是更加利于记忆和理解。就是这样,再没有更复杂的了。
更多的说了也没有,读者还是看一看源代码吧,推荐各位阅读《Linux 设备驱动程序》所
带源代码中的short一例,因为它比较短小,功能比较简单,可以看明白ioctl的功能和细
节。
四、 cmd参数如何得出
这里确实要说一说,cmd参数在用户程序端由一些宏根据设备类型、序列号、传送方向、
数据尺寸等生成,这个整数通过系统调用传递到内核中的驱动程序,再由驱动程序使用解
码宏从这个整数中得到设备的类型、序列号、传送方向、数据尺寸等信息,然后通过
switch{case}结构进行相应的操作。
要透彻理解,只能是通过阅读源代码,我这篇文章实际上只是一个引子。Cmd参数的组织
还是比较复杂的,我认为要搞熟它还是得花不少时间的,但是这是值得的,驱动程序中最
难的是对中断的理解。
五、 小结
ioctl其实没有什么很难的东西需要理解,关键是理解cmd命令码是怎么在用户程序里生成
并在驱动程序里解析的,程序员最主要的工作量在switch{case}结构中,因为对设备的
I/O控制都是通过这一部分的代码实现的。
分享到:
相关推荐
"Linux 下的 ioctl 函数" ioctl 函数是 Linux 驱动程序中对设备的 I/O 通道进行管理的函数,它可以控制设备的一些特性,例如串口的传输波特率、马达的转速等等。ioctl 函数的调用格式为 int ioctl(int fd, int cmd,...
ioctl 函数用法详解 ioctl 函数是一个系统调用函数...ioctl 函数是 Linux 系统中一个非常重要的系统调用函数,用于控制设备或文件描述符的参数。它可以用来执行各种设备操作,如获取或设置设备参数、执行设备命令等。
"Linux下的ioctl函数详解" Linux操作系统中,ioctl函数是一个非常重要的概念,它是设备驱动程序中对设备的I/O通道进行管理的函数。下面,我们将详细讨论ioctl函数的定义、必要性、实现方式、cmd参数的生成等方面。 ...
在Linux内核中,`ioctl`函数的实现依赖于文件操作结构体`file_operations`中的`unlocked_ioctl`或`compat_ioctl`成员函数。当用户空间调用`ioctl`时,系统会找到对应的设备驱动,并调用其中定义的`ioctl`处理函数。 ...
Linux内核编程是学习操作系统相关知识的必备知识,了解内核的体质结构和编程机制是非常重要的。 itle的一部分,设备文件操作系统是Linux内核编程的重要组件之一。设备文件操作系统是指在Linux系统中负责管理硬件...
ioctl函数的原型如下所示: ```c int ioctl(int __fd, unsigned long int __request, ...); ``` - `__fd`:代表文件描述符,通常是在调用open或类似函数时获得的。 - `__request`:这是一个无符号长整型参数,用于...
Linux内核函数库是Linux操作...学习并掌握这些Linux内核函数不仅有助于提升开发能力,还能帮助解决实际问题,如优化系统性能、调试内核模块、编写高效的服务程序等。深入研究Linux内核,能够让你在IT领域更具竞争力。
ioctl函数在Linux操作系统中扮演着重要的角色,它是设备驱动程序和应用程序之间进行通信的主要机制之一。这个函数允许用户空间的应用程序对设备进行控制操作,比如配置硬件参数、获取设备状态等,而不仅仅是简单的...
Linux 设备驱动程序头文件及内核函数说明总结 标题:头文件及内核函数说明总结-Linux 设备驱动程序 描述:Linux 设备驱动程序头文件及内核函数相关定义,包括常用头文件、内核函数、设备驱动程序编写等内容。 ...
### Linux内核编程知识点概述 #### 1. Hello, World - **背景**: 在学习编程时,“Hello, World”程序通常作为入门的第一个程序。对于Linux内核编程而言,也是如此。 - **基本结构**: 一个典型的内核模块至少包含...
通过对LINUX内核和驱动编程的学习,我们可以更加灵活地控制嵌入式系统的硬件资源,从而实现更复杂的系统功能。无论是对于个人项目还是商业应用,掌握这些技能都将大有裨益。希望本文能为你在探索LINUX内核的世界中...
本篇文章旨在建立对Linux内核文件系统整体架构的理解,并重点关注虚拟文件系统(Virtual File System, VFS)的实现机制。 ##### 基本概念 1. **块设备(Block Device)**:磁盘或其他存储介质,在Linux中被视为块设备...
Linux 内核模块与_proc 文件系统.pdf Linux 内核模块是 Linux 操作系统中的一种关键机制,它允许用户动态地修改内核、加载自己编写的程序,而不需要每次都编译内核。这种机制极大地改善了 Linux 的灵活性。 Linux ...
在Linux内核中,ioctl函数主要在`<linux/ioctl.h>`头文件中定义。 1. **ioctl命令定义** 每个ioctl命令都有一个唯一的命令标识符,由三个参数组合而成:主设备号、次设备号和命令代码。命令标识符通常定义为`#...
《Linux内核API完全参考手册(第2版)》是一本深入探讨Linux内核API的权威指南,专为系统开发者、驱动程序工程师以及对Linux内核感兴趣的程序员设计。该手册全面覆盖了Linux内核的各种接口,是理解并利用Linux内核...
在hello_mod驱动中,ioctl函数被用来控制字符串的处理方式,例如设置语言类型(如Chinese、English或Pinyin),这样当用户通过write函数写入数据后,read函数将返回经过处理的结果。 为了实现这些功能,驱动程序...
- **基本结构**:一个简单的Linux内核模块通常包含两个函数:`init_module()` 和 `cleanup_module()`。 - `init_module()`:当模块加载到内核时被调用,用于初始化内核中的资源或修改内核的行为。 - `cleanup_...
《Linux内核编程入门》这本书覆盖了Linux内核编程的核心知识和技术,通过对每个章节的详细讲解和示例代码的分析,读者可以系统地学习到如何编写内核模块、驱动程序以及理解内核的工作原理。无论是初学者还是有一定...