W moim kodzie Pythona (2.7.3) próbuję użyć wywołania ioctl, akceptując długie int (64-bitowe) jako argument. Jestem w systemie 64-bitowym, więc 64-bitowy int ma taki sam rozmiar jak wskaźnik.64-bitowy argument dla fcntl.ioctl()
Mój problem polega na tym, że Python prawdopodobnie nie akceptuje 64-bitowej wartości int jako argumentu dla wywołania fcntl.ioctl(). Z przyjemnością akceptuje 32-bitowy lub 64-bitowy wskaźnik - , ale potrzebuję przekazać 64-bitową int.
Oto mój obsługi ioctl:
static long trivial_driver_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
{
long err = 0;
switch (cmd)
{
case 1234:
printk("=== (%u) Driver got arg %lx; arg<<32 is %lx\n", cmd, arg, arg<<32);
break;
case 5678:
printk("=== (%u) Driver got arg %lx\n", cmd, arg);
break;
default:
printk("=== OH NOES!!! %u %lu\n", cmd, arg);
err = -EINVAL;
}
return err;
}
W istniejącego kodu C, używam wywołanie takiego:
static int trivial_ioctl_test(){
int ret;
int fd = open(DEV_NAME, O_RDWR);
unsigned long arg = 0xffff;
ret = ioctl(fd, 1234, arg); // ===(1234) Driver got arg ffff; arg<<32 is ffff00000000
arg = arg<<32;
ret = ioctl(fd, 5678, arg); // === (5678) Driver got arg ffff00000000
close(fd);
}
W Pythonie I otworzyć plik urządzenia, a następnie pojawia się następujące wyniki:
>>> from fcntl import ioctl
>>> import os
>>> fd = os.open (DEV_NAME, os.O_RDWR, 0666)
>>> ioctl(fd, 1234, 0xffff)
0
>>> arg = 0xffff<<32
>>> # Kernel log: === (1234) Driver got arg ffff; arg<<32 is ffff00000000
>>> # This demonstrates that ioctl() happily accepts a 32-bit int as an argument.
>>> import struct
>>> ioctl(fd, 5678, struct.pack("L",arg))
'\x00\x00\x00\x00\xff\xff\x00\x00'
>>> # Kernel log: === (5678) Driver got arg 7fff9eb1fcb0
>>> # This demonstrates that ioctl() happily accepts a 64-bit pointer as an argument.
>>> ioctl(fd, 5678, arg)
Traceback (most recent call last):
File "<pyshell#10>", line 1, in <module>
ioctl(fd, 5678, arg)
OverflowError: signed integer is greater than maximum
>>> # Kernel log: (no change - OverflowError is within python)
>>> # Oh no! Can't pass a 64-bit int!
>>>
Czy jest jakiś sposób Python może przechodzić m y 64-bitowy argument do ioctl()?
Pomogłoby to w zapewnieniu odtwarzalnego przykładu, jeśli to możliwe.Biorąc pod uwagę, że wywołania 'ioctl()' są specyficzne dla urządzenia, podstawienie 'IOC_GET_VAL' dla rzeczywistego kodu żądania, którego używasz sprawia, że jest to trudne do przetestowania. – Aya
@Aya: Dzięki za komentarz. Jestem nowy dla sterowników urządzeń i mam problem z skonstruowaniem trywialnego, choć funkcjonalnego przykładu. Ale zobaczę, co mogę zrobić. :) – Ziv
W międzyczasie opublikowałem rozwiązanie oparte na 'ctypes'. – Aya