一、内核原型(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. return break continue

    return 程序返回,不再执行下面的代码(结束当前的方法 直接返回) break 跳出总上一层循环,不再执行循环(结束当前的循环体)continue 跳出本次循环,继续执行下次循环(结束正在执行的循 ...

  2. Codeforces 741A:Arpa's loud Owf and Mehrdad's evil plan(LCM+思维)

    http://codeforces.com/problemset/problem/741/A 题意:有N个人,第 i 个人有一个 a[i],意味着第 i 个人可以打电话给第 a[i] 个人,所以如果第 ...

  3. 20145227《Java程序设计》第3次实验报告

    20145227<Java程序设计>第3次实验报告 实验步骤与内容 一.实验内容 XP基础 XP核心实践 相关工具 二.实验过程 (一)敏捷开发与XP 1.XP是以开发符合客户需要的软件为 ...

  4. ACM题目————滑雪

    Description Michael喜欢滑雪百这并不奇怪, 因为滑雪的确很刺激.可是为了获得速度,滑的区域必须向下倾斜,而且当你滑到坡底,你不得不再次走上坡或者等待升降机来载你.Michael想知道 ...

  5. SDUT 2608:Alice and Bob

    Alice and Bob Time Limit: 1000ms   Memory limit: 65536K  有疑问?点这里^_^ 题目描述 Alice and Bob like playing ...

  6. 错误Mybatis 元素类型为 "resultMap" 的内容必须匹配 "(constructor?,id*,result*,association*,collection*,discriminat

    今天算是见识了什么事顺序的重要性. 在使用mybatis时由于联合了其他的表,用到了resultMap,之后外加association这一项.可是在替换对应字段的位置上加上association总是报 ...

  7. Elevator 分类: HDU 2015-06-19 21:52 13人阅读 评论(0) 收藏

    Elevator Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Subm ...

  8. css3 转换transfrom 过渡transition 和两个@

    做了一个demo.用到一些css3的动画,还是不太熟练,总结了一下.  1. -webkit-font-smoothing: antialiased;       -webkit-font-smoot ...

  9. Linux crond定时任务

    第1章 Crond是什么? Crond是linux系统用来定期执行命令或指定程序任务的一种服务或软件.一般情况下,我们安装完Centos5/6linux操作系统之后,默认便会启动Crond任务调度服务 ...

  10. SqlSever基础 delete 删除符合特定条件的元素

    y 镇场诗:---大梦谁觉,水月中建博客.百千磨难,才知世事无常.---今持佛语,技术无量愿学.愿尽所学,铸一良心博客.---------------------------------------- ...