140

谁能给我解释一下

  1. 是什么IOCTL
  2. 这有什么用途?
  3. 我该如何使用它?
  4. 为什么我不能定义与 功能相同的新功能IOCTL
4

2 回答 2

189

ioctl函数对于实现设备驱动程序以在设备上设置配置很有用。例如,具有检查和设置字体系列、字体大小等配置选项的打印机ioctl可用于获取当前字体以及将字体设置为新字体。用户应用程序用于ioctl向打印机发送代码,告诉它返回当前字体或将字体设置为新字体。

int ioctl(int fd, int request, ...)
  1. fd是文件描述符,由open;返回
  2. request是请求代码。例如GETFONT将从打印机获取当前字体,SETFONT将在打印机上设置字体;
  3. 第三个论点是void *。根据第二个参数,第三个参数可能存在也可能不存在,例如,如果第二个参数是SETFONT,则第三个参数可以是字体名称,例如"Arial";

int request不仅仅是一个宏。用户应用程序需要生成请求代码和设备驱动程序模块来确定必须使用设备上的哪个配置。应用程序使用ioctl设备驱动程序模块中的请求代码发送请求代码,然后使用该请求代码来确定要执行的操作。

请求代码有 4 个主要部分

    1. A Magic number - 8 bits
    2. A sequence number - 8 bits
    3. Argument type (typically 14 bits), if any.
    4. Direction of data transfer (2 bits).  

如果请求代码是SETFONT在打印机上设置字体,则数据传输的方向是从用户应用程序到设备驱动程序模块(用户应用程序将字体名称发送"Arial"到打印机)。如果请求代码是GETFONT,则方向是从打印机到用户应用程序。

为了生成请求代码,Linux 提供了一些预定义的类似函数的宏。

1._IO(MAGIC, SEQ_NO)都是 8 位,0 到 255,例如,假设我们要暂停打印机。这不需要数据传输。所以我们会生成如下的请求代码

#define PRIN_MAGIC 'P'
#define NUM 0
#define PAUSE_PRIN __IO(PRIN_MAGIC, NUM) 

现在ioctl用作

ret_val = ioctl(fd, PAUSE_PRIN);

驱动模块中相应的系统调用将接收代码并暂停打印机。

  1. __IOW(MAGIC, SEQ_NO, TYPE) MAGICand 和SEQ_NO上面一样,并且TYPE给出了下一个参数的类型,回忆一下ioctlis的第三个参数void *。W in__IOW表示数据流是从用户应用程序到驱动模块。例如,假设我们要将打印机字体设置为"Arial".
#define PRIN_MAGIC 'S'
#define SEQ_NO 1
#define SETFONT __IOW(PRIN_MAGIC, SEQ_NO, unsigned long)

更远,

char *font = "Arial";
ret_val = ioctl(fd, SETFONT, font); 

Nowfont是一个指针,这意味着它是一个最好表示为的地址unsigned long,因此第三部分_IOW提到了类型。此外,这个字体地址被传递给在设备驱动模块中实现的相应系统调用 as unsigned long ,我们需要在使用它之前将其转换为正确的类型。内核空间可以访问用户空间,因此这是可行的。另外两个类似函数的宏是__IOR(MAGIC, SEQ_NO, TYPE)__IORW(MAGIC, SEQ_NO, TYPE)数据流将分别从内核空间到用户空间以及双向。

请让我知道这可不可以帮你!

于 2013-04-06T22:11:45.477 回答
108

An ioctl,意思是“输入输出控制”是一种特定于设备的系统调用。Linux 中只有少数几个系统调用(300-400),不足以表达设备可能具有的所有独特功能。因此驱动程序可以定义一个允许用户空间应用程序向其发送命令的 ioctl。但是,ioctl 不是很灵活,而且往往会变得有点混乱(数十个“幻数”,它们是否可以正常工作),并且也可能是不安全的,因为您将缓冲区传递到内核 - 错误的处理可能会破坏事情很容易。

另一种选择是sysfs界面,您可以在其中设置一个文件/sys/并读取/写入该文件以从驱动程序获取信息和向驱动程序获取信息。如何设置的示例:

static ssize_t mydrvr_version_show(struct device *dev,
        struct device_attribute *attr, char *buf)
{
    return sprintf(buf, "%s\n", DRIVER_RELEASE);
}

static DEVICE_ATTR(version, S_IRUGO, mydrvr_version_show, NULL);

在驱动程序设置期间:

device_create_file(dev, &dev_attr_version);

然后,您将在 中为您的设备提供一个文件/sys/,例如,/sys/block/myblk/version用于块驱动程序。

另一种更常用的方法是 netlink,它是一种 IPC(进程间通信)方法,通过 BSD 套接字接口与您的驱动程序通信。例如,WiFi 驱动程序使用它。libnl然后,您使用orlibnl3库从用户空间与它通信。

于 2013-04-04T10:57:03.017 回答