1

不确定是否有人在这里有任何想法,我以前从未见过。我正在编写一个存根来测试我的内核模块,当我检查用户空间中命令的值时,我得到的值与查看内核空间时不同。

部分存根:

#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <fcntl.h>
#include <sys/stat.h>
#include "ain.h"
#include "ain_ioctl.h"
#define AI_DEVICE   "/dev/ain"

void main()
{
    int fd, error, ioctl_par = 0;
    char* dev;
    long ret;

    dev = AI_DEVICE;

    printf("Starting driver test\n");

    fd = open(dev, O_RDWR);
    if (fd < 0) {
    /* Failed to open -> Print error-message and exit */
    printf("%s failed to open, error: %s\n", dev, strerror(errno));
    }

    printf("Doing the IOCTL now... cmd: %d\n", AIN_IOC_GET_AN0_CONF);
    fflush(stdout);

    ret = ioctl(fd, AIN_IOC_GET_AN0_CONF, &ioctl_par);

ain_ioctl.h 文件:

#define AIN_IOC_MAGIC  'e'
#define AIN_IOC_GET_AN0_CONF    _IOR(AIN_IOC_MAGIC, 46, int)

内核中的 ioctl 例程:

int ain_ioctl (struct inode * inodep, struct file * filp, unsigned int cmd, unsigned long arg)
{
    printk("In the ain_ioctl function, cmd: %d. type: %d, dir: %d, nr: %d, size: %d\n", 
        cmd, _IOC_TYPE(cmd), _IOC_DIR(cmd), _IOC_NR(cmd), _IOC_SIZE(cmd));

    printk("Testing against command: %d. type: %d, dir: %d, nr: %d, size: %d\n",
        AIN_IOC_GET_AN0_CONF, _IOC_TYPE(AIN_IOC_GET_AN0_CONF), _IOC_DIR(AIN_IOC_GET_AN0_CONF), 
        _IOC_NR(AIN_IOC_GET_AN0_CONF), _IOC_SIZE(AIN_IOC_GET_AN0_CONF));

现在我希望用户空间打印中的输出与内核中相同。并在内核中的第一组打印到第二组。然而,这不是我所看到的......

输出:

mike@linux-4puc:~> ./a.out 
Starting driver test
Doing the IOCTL now... cmd: -2147195602

mike@linux-4puc:~> dmesg | tail
[75253.205136] In the ain_ioctl function, cmd: -1078168112. type: 117, dir: 2, nr: 208, size: 16316
[75253.205140] Testing against            cmd: -2147195602. type: 101, dir: 2, nr: 46, size: 4

任何人都知道为什么当我通过 ioctl 命令将命令传递给内核时,与我只是通过硬编码来检查值时(就像我在打印中所做的那样),我的命令为什么表现不同?

我在构建时看到的唯一警告似乎与 ioctl 调用无关:

makedepend: warning: ignoring option -Wall 
makedepend: warning: ignoring option -Wall 
makedepend: warning: ain.c (reading /usr/src/linux/include/linux/compiler-gcc.h), line 94: incomplete include == "#include gcc_header(__GNUC__)" 
makedepend: warning: ain.c (reading /usr/src/linux/include/linux/string.h, line 13): cannot find include file "stdarg.h" 

谢谢。

4

2 回答 2

4

-1078168112(为什么不以十六进制打印这些?)看起来像一个堆栈指针。可能&ioctl_par。这表明您的 ioctl 方法接收的参数与您预期的不同。

在当前的内核源代码中,我看到 ioctl 方法采用 3 个参数,而不是 4 个。4 参数 ioctl 似乎是一个较旧的接口。

您在模块编译期间是否收到任何警告?注意他们!

于 2012-08-15T16:09:54.727 回答
1

Alan Curry 的答案不是完整的“正确”答案,但它引导我找到了解决方案。该命令的十六进制值很差,所以我查看了内核中的其他 ioctl 调用。

我拥有的系统基于较旧的 2.4X 内核,我正在将其更新为 3.1。这里的问题是 ioctl 调用的参数列表。在参数列表中包含 inode 指针会导致问题,因为它将文件指针作为命令。

适当的解决方案:

long ain_ioctl (struct file * filp, unsigned int cmd, unsigned long arg) { 
    ...
于 2012-08-16T13:28:15.273 回答