Linux下对input设备调用ioctl时指定EVIOCGBIT选项时的缓冲区该多大

31 篇文章 1 订阅
17 篇文章 0 订阅

我们有时候需要获取/dev/input目录下的eventX设备支持哪些事件(EV_KEY、EV_REL和EV_ABS等),可以通过ioctl调用指定EVIOCGBIT(ev, len)选项来获取,例如:

ioctl(fd, EVIOCGBIT(0, EV_MAX), buf);

来获取fd设备支持的事件。这涉及到一个问题:buf需要指定多大的长度?
EVIOCGBIT宏的第二个参数是事件标志位最高位可能有多高,例如当前4.16内核版本中该值为0x1f,说明缓冲区最高第0x1f位可能会被置位。因此之前缓冲区是这样指定的:

uint8_t buf[EV_MAX / 8 + 1];

这样的话理论上可以容纳下所有标志位。但是实际执行时(64位机器上)会有栈溢出问题:

$ sudo ./eviocgbit /dev/input/event4
Supported event types:
  Event type 0x00  (Synch Events)
  Event type 0x01  (Keys or Buttons)
*** stack smashing detected ***: <unknown> terminated
Aborted

为什么会这样呢,跟了一下内核代码,发现在计算需要往用户空间拷贝多少字节的数据是在这个函数中计算的:

static int bits_to_user(unsigned long *bits, unsigned int maxbit,
            unsigned int maxlen, void __user *p, int compat)
{
    int len = BITS_TO_LONGS(maxbit) * sizeof(long);

    if (len > maxlen)
        len = maxlen;

    return copy_to_user(p, bits, len) ? -EFAULT : len;
}

其中的maxbit在本例中即为我们指定的EV_MAX,这个函数首先使用BITS_TO_LONGS宏计算出需要几个long型数据能够放下这么多位的数据,然后乘以long的大小,得到缓冲区的大小。虽然下面有使用maxlen限制缓冲区大小,但是maxlen也被指定为了EV_MAX,所以并没有效果。
如上所述,我们在申请缓冲区时也要像内核代码一样以long型数据大小为最小单位,申请n个long型数据大小的缓冲区就没有问题了:

uint8_t evtype_b[(EV_MAX  / (sizeof(long) * 8) + 1) * sizeof(long)];
  • 2
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
IOCTL_READ_PCI_CONFIG 和 IOCTL_WRITE_PCI_CONFIG 是用于读写PCI设备配置空间的Windows I/O控制码。 以下是它们的实现示例: ```c // 读取PCI设备配置空间 BOOL ReadPCIConfig(HANDLE hDevice, DWORD dwBus, DWORD dwSlot, DWORD dwFunction, LPVOID lpBuffer, DWORD dwBytesToRead) { // 构造PCI配置空间的地址 DWORD dwAddress = 0x80000000 | (dwBus << 16) | (dwSlot << 11) | (dwFunction << 8); // 构造输入缓冲区 PCI_CONFIG_INPUT input = {0}; input.dwAddress = dwAddress; input.dwBytesToRead = dwBytesToRead; // 构造输出缓冲区 PCI_CONFIG_OUTPUT output = {0}; // 发送IOCTL请求 DWORD dwBytesReturned = 0; BOOL bSuccess = DeviceIoControl(hDevice, IOCTL_READ_PCI_CONFIG, &input, sizeof(input), &output, sizeof(output), &dwBytesReturned, NULL); // 将结果拷贝到输出缓冲区 if (bSuccess) { memcpy(lpBuffer, output.bResult, dwBytesToRead); } return bSuccess; } // 写入PCI设备配置空间 BOOL WritePCIConfig(HANDLE hDevice, DWORD dwBus, DWORD dwSlot, DWORD dwFunction, LPCVOID lpBuffer, DWORD dwBytesToWrite) { // 构造PCI配置空间的地址 DWORD dwAddress = 0x80000000 | (dwBus << 16) | (dwSlot << 11) | (dwFunction << 8); // 构造输入缓冲区 PCI_CONFIG_INPUT input = {0}; input.dwAddress = dwAddress; input.dwBytesToWrite = dwBytesToWrite; memcpy(input.bData, lpBuffer, dwBytesToWrite); // 发送IOCTL请求 DWORD dwBytesReturned = 0; BOOL bSuccess = DeviceIoControl(hDevice, IOCTL_WRITE_PCI_CONFIG, &input, sizeof(input), NULL, 0, &dwBytesReturned, NULL); return bSuccess; } ``` 这里的 `PCI_CONFIG_INPUT` 和 `PCI_CONFIG_OUTPUT` 是自定义的结构体,定义如下: ```c typedef struct _PCI_CONFIG_INPUT { DWORD dwAddress; // PCI配置空间的地址 DWORD dwBytesToRead; // 要读取的字节数 DWORD dwBytesToWrite; // 要写入的字节数 BYTE bData[256]; // 数据缓冲区 } PCI_CONFIG_INPUT, *PPCI_CONFIG_INPUT; typedef struct _PCI_CONFIG_OUTPUT { BYTE bResult[256]; // 数据缓冲区 } PCI_CONFIG_OUTPUT, *PPCI_CONFIG_OUTPUT; ``` 在实际使用中,需要先打开一个PCI设备的句柄,然后通过该句柄调用 `ReadPCIConfig` 和 `WritePCIConfig` 函数进行读写操作。其中,`dwBus`、`dwSlot` 和 `dwFunction` 分别表示PCI设备的总线号、插槽号和功能号,`lpBuffer` 表示数据缓冲区的指针,`dwBytesToRead` 和 `dwBytesToWrite` 分别表示要读取和写入的字节数。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值