linux2.6内核compat_ioctl函数
一、内核原型(linux2.6.28-7)
long (*compat_ioctl)(struct tty_struct *tty, struct file * file,
unsigned int cmd, unsigned long arg);
implement ioctl processing for 32 bit process on 64 bit system
Optional
二、What is compat_ioctl
There is one more method called as "compat_ioctl()" that a 64 bit driver
has to implement. It gets called when 64 bit kernel gets ioctl() call
from 32 bit user.
Tasks to be done by compat_ioctl() :
1. Acquire BKL, since kernel calls compat_ioctl without BKL.
2. 32 to 64 bit conversion for long and pointer objects passed by user
3. Process input data, get results.
4. 64 to 32 bit conversion in order to pass the output data back to user
5. Release BKL
三、中文档案
Linux 64Bit 下的 ioctl和compat_ioctl ioctl32 Unknown cmd fd
前段时间将我们的程序移植到Mips64的Linux 2.6环境下,做可行性试验。
由于用户态程程序规模太大,而且之前没有对64bit的情况做考虑,
所以,用户态程序任然使用32位模式编译,内核运行在64bit。
我们有一个内核模块之前是在2.4.21下的,拿到2.6后,把部分api做了些更改,直接编译并加载。
用户态程序在调用ioctl的时候,总是失败。
dmesg看一下内核信息,有如下类似信息:
ioctl32(add_vopp:9767): Unknown cmd fd(3) cmd(80048f00){00} arg(ff240ae0)
后来在内核中的ioctl中添加debug代码,发现根本没调用到内核中的ioctl函数。
经过查找,发现了以下资源
The new way of ioctl()
32 bit user/64 bit kernel
What is compat_ioctl ()
more on compat_ioctl
产生问题的分析:
我们的程序通过Linux的
ssize_t read(int fd, void *buf, size_t count);
系统调用从虚拟设备中读取内核中。
使用
int ioctl(int fd, int request, ...);
系统调用来对设备进行控制,并传递一些数据。
ioctl是我们扩展非标准系统调用的手段。
read系统调用,对应内核中struct file_operations 结构中的
ssize_t (*read) (struct file *filp, char *buf, size_t count, loff_t *f_pos)
当用户态位32bit, 内核位64bit的时候,
用户态的程序,和内核态的程序,在数据类型不一致。
比如指针,用户态的指针实际上是unsigned long 4Byte
内核态的指针是unsigned ong: 8Byte.
对于这种不一致,从用户态陷入内核态的时候,大部分标准调用的参数已经做了相应的转化。
不存在问题。比如ssize_t,内核态和用户态不一致,
对于这种已知类型,内核代码已经做了转换,因为他知道该怎么转,
所以我们的程序调用read,write,等系统调用,都能正确的返回结果。
再来看看ioctl系统调用,
用户态系统调用,int ioctl(int fd, int request, ...);
内核中对应的函数
int (*ioctl) (struct inode *inode, struct file *filp, unsigned int cmd, unsigned long arg)
request是请求号,用来区分不同得到请求。后面的可变参数随便选。
如果想传入或传出自定义结构,可以传个指针类型。
如果没数据传递,可以空着。
当然也可以传这个int或char之类的类型来向内核传递数据。
问题来了,内核不知道你传递的是那种类型,他无法给你转化。
总结一下:
1). Linux2.6对64bit kernel 在struct file_operation中增加了一个成员
long (*compat_ioctl) (struct file *filp, unsigned int cmd, unsigned long arg);
用来提供32位用户态程序的方法,
可以通过filp->f_dentry->d_inode方法获得。
3). 关于ioctl的请求号,经常是通过宏定义生成的
举例如下:
#define IoctlGetNetDeviceInfo _IOWR(Ioctl_Magic, 1, struct_NetDeviceInfo)
#define IoctlNetQueueInit _IOR(Ioctl_Magic, 4, struct_NetDeviceListen)
#define IoctlNetQueueDestroy _IO(Ioctl_Magic, 5)
注意_IOWR和IOR宏, 他们的最后一个参数是一个数据类型,展开时会包含sizeof()操作,
于是32bit用户态程序和64bit内核之间,生成的ioctl的request号很可能就会不同,
在compat函数中switch()的时候,就会switch不到。
要想办法避免:
提供64bit和32bit大小一致的结构。
在用户态下提供一个伪结构,伪结构和内核内的结构宽度一致。
用_IO宏,而不用_IOWR或_IOR宏,反正只是为了得到一个号码,实际传输数据大小,自己心理有数,内核代码处理好就行了。
4). 如果compat收到的最后参数arg是一个用户态指针, 它在用户态是32位的,在内核中为了保证安全,
可以使用compat_ptr(art)宏将其安全的转化为一个64位的指针(仍然是用户指针)
linux2.6内核compat_ioctl函数的更多相关文章
- Linux2.6 内核的 Initrd 机制解析(转)
from: https://www.ibm.com/developerworks/cn/linux/l-k26initrd/ 简介: Linux 的 initrd 技术是一个非常普遍使用的机制,lin ...
- Linux2.6 内核的 Initrd 机制解析
文章来自:www.ibm.com/developerworks/cn/linux/l-k26initrd/ 1.什么是 Initrd initrd 的英文含义是 boot loader initial ...
- Linux2.6内核实现的是NPTL
NPTL是一个1×1的线程模型,即一个线程对于一个操作系统的调度进程,优点是非常简单.而其他一些操作系统比如Solaris则是MxN的,M对应创建的线程数,N对应操作系统可以运行的实体.(N<M ...
- Linux2.6内核进程调度系列--scheduler_tick()函数1.总体思想
参考的是ULK第三版,Linux2.6.11.12内核版本. 调度程序依靠几个函数来完成调度工作,其中最重要的第一个函数是scheduler_tick函数,主要步骤如下: /** * 维持当前最新的t ...
- Linux2.6内核进程调度系列--scheduler_tick()函数3.更新普通进程的时间片
RT /** * 运行到此,说明进程是普通进程.现在开始更新普通进程的时间片. */ /* 首先递减普通进程的时间片计数器.如果用完,继续执行以下操作 */ if (!--p->time_sli ...
- Linux2.6内核进程调度系列--scheduler_tick()函数2.更新实时进程的时间片
RT /** * 递减当前进程的时间片计数器,并检查是否已经用完时间片. * 由于进程的调度类型不同,函数所执行的操作也有很大差别. */ /* 如果是实时进程,就进一步根据是FIFO还是RR类型的实 ...
- Linux2.6内核--进程调度理论
从1991年Linux的第1版到后来的2.4内核系列,Linux的调度程序都相当简陋,设计近乎原始,见0.11版内核进程调度.当然它很容易理解,但是它在众多可运行进程或者多处理器的环境下都难以胜任. ...
- linux2.6内核链表
一. 链表数据结构简介 链表是一种常用的组织有序数据的数据结构,它通过指针将一系列数据节点连接成一条数据链,是线性表的一种重要实现方式.相对于数组,链表具有更好的动态性,建立链 ...
- 【转载】linux2.6内核initrd机制解析
题记 很久之前就分析过这部分内容,但是那个时候不够深入,姑且知道这么个东西存在,到底怎么用,来龙去脉咋回事就不知道了.前段时间工作上遇到了一个initrd的问题,没办法只能再去研究研究,还好,有点眉目 ...
随机推荐
- ActiveMQ 安装异常
解决方式: 1.确认计算机主机名名称没有下划线: 2.如果是win7,停止ICS(运行-->services.msc找到Internet Connection Sharing (ICS)服务,改 ...
- 通过HtmlEmail 发送邮件
今天第一次来上海市虹口图书馆上自习,感觉还是很爽的.自己撸代码学会了发送邮件.啥都不说了,直接撸代码吧! 首先 必须引进来三个jar包: compile 'javax.mail:mail:1.4.7' ...
- class打包成exe方式
首先运行dos命令 jar -cvf **.jar *.class 把所有文件打包成jar包 用解压器打开jar包 打开meta-inf文件夹 打开用记事本打开main.mf文件 增加语句Main-C ...
- 如何在html添加一个搜索框和一个按钮?
<INPUT TYPE="text" id="k"><INPUT TYPE="button" VALUE="ok ...
- 20150625_Andriod_02_ListView2_多条目显示_选中
android listview 参考地址: http://www.cnblogs.com/zhengbeibei/archive/2013/05/14/3078805.html http://xy ...
- Out of Hay
Out of Hay Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 13094 Accepted: 5078 Descripti ...
- 排序练习——找出前m大的数字 分类: 排序 2015-06-08 09:33 21人阅读 评论(0) 收藏
排序练习--找出前m大的数字 Time Limit: 1000ms Memory limit: 65536K 有疑问?点这里^_^ 题目描述 给定n个数字,找出前m大的数字. 输入 多组输 ...
- 基本的MFC多线程
多线程程序我本来是能够来写的.但是由于在图像处理的过程中,对于这方面知识使用的比较少,造成重复忘记的情况.这里再次进行整理学习,特别注重和“图像处理”理清关系,为下一步使用奠定基础. 这里实现的是工作 ...
- Uva 1626,括号序列
题目链接:https://uva.onlinejudge.org/external/16/1626.pdf 题意: 给定一个字符串,看是否括号匹配,不匹配加括号,加最少的括号使得匹配.输出该结果. 分 ...
- 利用JAVA反射机制将JSON数据转换成JAVA对象
net.sf.json.JSONObject为我们提供了toBean方法用来转换为JAVA对象, 功能更为强大, 这里借鉴采用JDK的反射机制, 作为简单的辅助工具使用, 有些数据类型需要进行转 ...