ioctl系统调用

2024-05-15 23:08
文章标签 系统 调用 ioctl

本文主要是介绍ioctl系统调用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. 什么是ioctlioctl是设备驱动程序中对设备的I/O通道进行管理的函数。所谓对I/O通道进行管理,就是对设备的一些特性进行控制,例如串口的传输波特率、马达的转速等等。它的调用个数如下:int ioctl(int fd, ind cmd, …);其中fd是用户程序打开设备时使用open函数返回的文件标示符,cmd是用户程序对设备的控制命令,至于后面的省略号,那是一些补充参数,一般最多一个,这个参数的有无和cmd的意义相关。ioctl函数是文件结构中的一个属性分量,就是说如果你的驱动程序提供了对ioctl的支持,用户就可以在用户程序中使用ioctl函数来控制设备的I/O通道。2. 3. 预定义命令  有一些ioctl命令是由内核识别的,当这些命令用于自己的设备时,他们会在我们自己的文件操作被调用之前被解码. 因此, 如果你选择一个ioctl命令编号和系统预定义的相同时,你永远不会看到该命令的请求,而且因为ioctl 号之间的冲突,应用程序的行为将无法预测。预定义命令分为 3 类:(1)用于任何文件(常规, 设备, FIFO和socket) 的命令(2)只用于常规文件的命令(3)特定于文件系统类型的命令 下列 ioctl 命令是预定义给任何文件,包括设备特定文件:FIOCLEX :设置 close-on-exec 标志(File IOctl Close on EXec)。FIONCLEX :清除 close-no-exec 标志(File IOctl Not CLose on EXec)。FIOQSIZE :这个命令返回一个文件或者目录的大小; 当用作一个设备文件, 但是, 它返回一个 ENOTTY 错误。FIONBIO:"File IOctl Non-Blocking I/O"(在"阻塞和非阻塞操作"一节中描述)。4. cmd命令构成。  在Linux核心中是这样定义一个命令码的:――――――――――――――――――――――――――――――――――――――| 设备类型 | 序列号 | 方向 | 数据尺寸 ||----------|--------|------|----------|| 8 bit  |  8 bit | 2 bit| 8~14 bit ||----------|--------|------|----------|设备类型 : '0'~'9','a'~'z','A'~'Z',补充:实际上只要8位的数都是可以的。又称为 幻数,魔数 在内核文档Ioctl-number.txt (documentation\ioctl)有详细的描述,哪些数已经被使用了。序列号   : 就是你驱动中命令的序号,从0开始,也可以从其他数开始。方向     :以用户空间为参照:ioctl(fd,cmd);       没有arg参数,不存在数据方向ioctl(fd,cmd,arg);   arg是一个值或地址--->写入内核驱动中---写方向ioctl(fd,cmd,arg);   arg是地址  --->内核驱动中修改arg的值---读方向ioctl(fd,cmd,arg);   arg是地址  --->内核驱动中取arg的内容,然后修改arg的值---读写方向int arg = 10;ioctl(fd,cmd,&arg);驱动中可以取得arg的值来使用,也可以修改arg的值,这种情况属于数据双向。手工构造,例子: cmd0: (‘k’ << 24 | 0 << 16 | 1 << 14 | 4 << 0)cmd1: (‘k’ << 24 | 1 << 16 | 2 << 14 | 4 << 0) 是否可以自动构造??????数据尺寸 :arg 大小。这样一来,一个命令就变成了一个整数形式的命令码。但是命令码非常的不直观,所以Linux Kernel中提供了一些宏,这些宏可根据便于理解的字符串生成命令码,或者是从命令码得到一些用户可以理解的字符串以标明这个命令对应的设备类型、设备序列号、数据传送方向和数据传输尺寸。1、定义命令:内核提供了一些宏来帮助定义命令://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_IOC         _IO(MEM_IOC_MAGIC, 0)#define MEM_IOCSET      _IOW(MEM_IOC_MAGIC,1,int)#define MEM_IOCGQSET    _IOR(MEM_IOC_MAGIC,2, int)同时为发方便验证命令合法性,内核也实现了相应的域提取宏,如下: _IOC_NR(cmd), _IOC_TYPE(cmd), _IOC_SIZE(cmd), _IOC_DIR(cmd)  这几个宏用来取得 cmd 命令中的域,其作用如下:_IOC_NR(cmd)  : 读取基数域值 (bit0~ bit7)_IOC_TYPE(cmd): 读取魔数域值 (bit8 ~ bit15)_IOC_SIZE(cmd): 读取数据大小域值 (bit16 ~ bit29)_IOC_DIR(cmd) : 获取读写属性域值 (bit30 ~ bit31)   2、实现命令:定义好了命令,下一步就是要实现ioctl函数了,ioctl的实现包括三个技术环节:1)返回值;ioctl函数的实现是根据命令执行的一个switch语句,但是,当命令不能匹配任何一个设备所支持的命令时,通常返回-EINVAL(非法参数);2)参数使用;用户使用  int ioctl(int fd,unsinged long cmd,...)  时,...就是要传递的参数;再通过  int (*ioctl)(struct inode *inode, struct file *filp, unsigned int cmd, unsigned long arg)中的arg传递;如果arg是一个整数,可以直接使用;如果是指针,我们必须确保这个用户地址是有效的,因此,使用之前需要进行正确检查。内部有检查的,不需要检测的:copy_from_usercopy_to_userget_userput_user需要检测的函数:__get_user__put_user检测函数access_ok():static inline int access_ok(int type, const void *addr, unsigned long size)/**type :是VERIFY_READ 或者VERIFY_WRITE用来表明是读用户内存还是写用户内存;*addr:是要操作的用户内存地址;*size:是操作的长度。如果ioctl需要从用户空间读一个整数,那么size参数就等于sizeof(int);*返回值:Access_ok返回一个布尔值:1,是成功(存取没问题);0,是失败,ioctl返回-EFAULT;*/3)命令操作;switch(cmd){case:... ...}5 .程序范例作业:实现标准ioctl接口,控制led,读取led状态,移动led文件指针。//复位文件指针;          ---                                                                                 
//移动文件指针;          arg                                                                                         
//查询当前led文件指针    arg                                                                                 
//设置当前led文件指针    arg                                                                                 //点亮当前led            ---                                                                                     
//熄灭当前led            ---                                                                                 
//根据当前arg参数设置当前led状态  arg                                                                        
//全部开led              ---                                                                                           
//全部关led              ---                                                                                 //示例用户空间和内核空间同时传递数据的情况  arg                                                              

这篇关于ioctl系统调用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/993177

相关文章

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

Python中Tensorflow无法调用GPU问题的解决方法

《Python中Tensorflow无法调用GPU问题的解决方法》文章详解如何解决TensorFlow在Windows无法识别GPU的问题,需降级至2.10版本,安装匹配CUDA11.2和cuDNN... 当用以下代码查看GPU数量时,gpuspython返回的是一个空列表,说明tensorflow没有找到

python如何调用java的jar包

《python如何调用java的jar包》这篇文章主要为大家详细介绍了python如何调用java的jar包,文中的示例代码简洁易懂,具有一定的借鉴价值,有需要的小伙伴可以参考一下... 目录一、安装包二、使用步骤三、代码演示四、自己写一个jar包五、打包步骤六、方法补充一、安装包pip3 install

linux重启命令有哪些? 7个实用的Linux系统重启命令汇总

《linux重启命令有哪些?7个实用的Linux系统重启命令汇总》Linux系统提供了多种重启命令,常用的包括shutdown-r、reboot、init6等,不同命令适用于不同场景,本文将详细... 在管理和维护 linux 服务器时,完成系统更新、故障排查或日常维护后,重启系统往往是必不可少的步骤。本文

Mac系统下卸载JAVA和JDK的步骤

《Mac系统下卸载JAVA和JDK的步骤》JDK是Java语言的软件开发工具包,它提供了开发和运行Java应用程序所需的工具、库和资源,:本文主要介绍Mac系统下卸载JAVA和JDK的相关资料,需... 目录1. 卸载系统自带的 Java 版本检查当前 Java 版本通过命令卸载系统 Java2. 卸载自定

基于Python实现一个简单的题库与在线考试系统

《基于Python实现一个简单的题库与在线考试系统》在当今信息化教育时代,在线学习与考试系统已成为教育技术领域的重要组成部分,本文就来介绍一下如何使用Python和PyQt5框架开发一个名为白泽题库系... 目录概述功能特点界面展示系统架构设计类结构图Excel题库填写格式模板题库题目填写格式表核心数据结构

Linux系统中的firewall-offline-cmd详解(收藏版)

《Linux系统中的firewall-offline-cmd详解(收藏版)》firewall-offline-cmd是firewalld的一个命令行工具,专门设计用于在没有运行firewalld服务的... 目录主要用途基本语法选项1. 状态管理2. 区域管理3. 服务管理4. 端口管理5. ICMP 阻断

Java调用C#动态库的三种方法详解

《Java调用C#动态库的三种方法详解》在这个多语言编程的时代,Java和C#就像两位才华横溢的舞者,各自在不同的舞台上展现着独特的魅力,然而,当它们携手合作时,又会碰撞出怎样绚丽的火花呢?今天,我们... 目录方法1:C++/CLI搭建桥梁——Java ↔ C# 的“翻译官”步骤1:创建C#类库(.NET

Windows 系统下 Nginx 的配置步骤详解

《Windows系统下Nginx的配置步骤详解》Nginx是一款功能强大的软件,在互联网领域有广泛应用,简单来说,它就像一个聪明的交通指挥员,能让网站运行得更高效、更稳定,:本文主要介绍W... 目录一、为什么要用 Nginx二、Windows 系统下 Nginx 的配置步骤1. 下载 Nginx2. 解压

如何确定哪些软件是Mac系统自带的? Mac系统内置应用查看技巧

《如何确定哪些软件是Mac系统自带的?Mac系统内置应用查看技巧》如何确定哪些软件是Mac系统自带的?mac系统中有很多自带的应用,想要看看哪些是系统自带,该怎么查看呢?下面我们就来看看Mac系统内... 在MAC电脑上,可以使用以下方法来确定哪些软件是系统自带的:1.应用程序文件夹打开应用程序文件夹