一、内核原型(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函数的更多相关文章

  1. Linux2.6 内核的 Initrd 机制解析(转)

    from: https://www.ibm.com/developerworks/cn/linux/l-k26initrd/ 简介: Linux 的 initrd 技术是一个非常普遍使用的机制,lin ...

  2. Linux2.6 内核的 Initrd 机制解析

    文章来自:www.ibm.com/developerworks/cn/linux/l-k26initrd/ 1.什么是 Initrd initrd 的英文含义是 boot loader initial ...

  3. Linux2.6内核实现的是NPTL

    NPTL是一个1×1的线程模型,即一个线程对于一个操作系统的调度进程,优点是非常简单.而其他一些操作系统比如Solaris则是MxN的,M对应创建的线程数,N对应操作系统可以运行的实体.(N<M ...

  4. Linux2.6内核进程调度系列--scheduler_tick()函数1.总体思想

    参考的是ULK第三版,Linux2.6.11.12内核版本. 调度程序依靠几个函数来完成调度工作,其中最重要的第一个函数是scheduler_tick函数,主要步骤如下: /** * 维持当前最新的t ...

  5. Linux2.6内核进程调度系列--scheduler_tick()函数3.更新普通进程的时间片

    RT /** * 运行到此,说明进程是普通进程.现在开始更新普通进程的时间片. */ /* 首先递减普通进程的时间片计数器.如果用完,继续执行以下操作 */ if (!--p->time_sli ...

  6. Linux2.6内核进程调度系列--scheduler_tick()函数2.更新实时进程的时间片

    RT /** * 递减当前进程的时间片计数器,并检查是否已经用完时间片. * 由于进程的调度类型不同,函数所执行的操作也有很大差别. */ /* 如果是实时进程,就进一步根据是FIFO还是RR类型的实 ...

  7. Linux2.6内核--进程调度理论

    从1991年Linux的第1版到后来的2.4内核系列,Linux的调度程序都相当简陋,设计近乎原始,见0.11版内核进程调度.当然它很容易理解,但是它在众多可运行进程或者多处理器的环境下都难以胜任. ...

  8. linux2.6内核链表

    一.        链表数据结构简介      链表是一种常用的组织有序数据的数据结构,它通过指针将一系列数据节点连接成一条数据链,是线性表的一种重要实现方式.相对于数组,链表具有更好的动态性,建立链 ...

  9. 【转载】linux2.6内核initrd机制解析

    题记 很久之前就分析过这部分内容,但是那个时候不够深入,姑且知道这么个东西存在,到底怎么用,来龙去脉咋回事就不知道了.前段时间工作上遇到了一个initrd的问题,没办法只能再去研究研究,还好,有点眉目 ...

随机推荐

  1. 编写一个函数func(),将此函数的输入参数(int型)逆序输出显示,如54321 –> 12345,要求使用递归,并且函数体代码不超过8行

    public class Test{ //中间变量 private String res = "0"; //方法 public int func(int i){ if(i>0 ...

  2. python-day6 常见算法 python内置模块

    1.冒泡排序 需求:请按照从小到大对列表 [13, 22, 6, 99, 11] 进行排序 原理:相邻两个值进行比较,将较大的值放在右侧,依次比较! li=[39,11,43,88,765,9]for ...

  3. centos 6.5源码编译安装subversion 1.8.10

    一.简介 CentOS 6.5的yum源可以安装的SVN客户端版本太低了,1.6.11,所以需要升级到1.8.10,而官网有没有找到1.8.10的安装包,只能选择源码编译安装. 二.安装步骤 参考官网 ...

  4. Apache Commons-pool实现对象池(包括带key对象池)

    Commons-pool是一个apache开源组织下的众多项目的一个.其被广泛地整合到众多需要对象池功能的项目中. 官网:http://commons.apache.org/proper/common ...

  5. Codeforces Round #260 (Div. 2) B

    Description Fedya studies in a gymnasium. Fedya's maths hometask is to calculate the following expre ...

  6. jar文件签名

    1.生成密钥: keytool -genkey -keystore key.keystore -alias key -validity 3650 将在当前目录下生成一个key.keystore文件, ...

  7. 一、spring——helloWorld

    1.添加jar包,如下图所示: 2.建立spring项目,如下图所示: 3.验证,如下图所示:

  8. android“设置”里的版本号

    在文件 packages/apps/Settings/src/com/android/settings/DeviceInfoSettings.java 中 setStringSummary(" ...

  9. MTK+Android编译

    1. 修改recovery代码 比如mediatek\custom\itek82_wet_kk\recovery\inc\cust_keys.h ./mk r k ./mk recoveryimage ...

  10. Web自定义协议,BS端启动CS端,

    实例 1.准备CS项目,windows窗体应用程序,拖进来一个label控件来接受BS的参数,并显示,右击生成,复制该文件的bin目录下的exe,例如放在以下路径,例如C:\\simu\\下, 2.编 ...