我正在开发一个迷你模块,以便在 python 中为我的大学作业(不是说这是一个实际的作业,只是为了澄清)提供低级网络接口访问权限。实际的分配是在 C 中完成的,但在那之后我决定使用 ctypes 进行一些练习并启动 lib,目标是 Linux 和 Python 2.7。
我已经知道 Python 3.3 公开了很多我试图涵盖的功能,并且自 arround 2.3 版左右以来它已经有一个 fcntl.ioctl 模块(确切的版本并不重要,只是它存在在 2.7 中),但这个任务是为了学习。
我遇到的问题非常简单。我已经根据 C 对应物的布局定义了以下类(为了清楚起见,请忍受长片段):
class struct_ifmap(Structure):
_fields_ = [
('mem_start', c_ulong),
('mem_end', c_ulong),
('base_addr', c_ushort),
('irq', c_ubyte),
('dma', c_ubyte),
('port', c_ubyte),]
class union_ifreq_anon(Union):
_fields_ = [
('ifr_addr', struct_sockaddr),
('ifr_dstaddr', struct_sockaddr),
('ifr_broadaddr', struct_sockaddr),
('ifr_netmask', struct_sockaddr),
('ifr_hwaddr', struct_sockaddr),
('ifr_flags', c_short),
('ifr_ifindex', c_int),
('ifr_metric', c_int),
('ifr_mtu', c_int),
('ifr_map', struct_ifmap),
('ifr_slave', c_char),
('ifr_newname', c_char),
('ifr_data', c_char_p),
('raw', c_short * 15),]
class struct_ifreq(Structure):
__IFNAMSIZ = 16
_anonymous_ = [
('u'),]
_fields_ = [
('ifr_name', c_char * __IFNAMSIZ),
('u', union_ifreq_anon),]
以及以下功能:
def _getfahwdraddr(name):
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM,
socket.IPPROTO_IP)
ifr = struct_ifreq()
ifr.ifr_name = name
result = _libc.ioctl(sock.fileno(), 0x8927, pointer(ifr))
if result != 0:
raise OSError(get_errno())
return ''.join(['%02x:' % pair for pair in \
ifr.ifr_hwaddr.sa_data[0:6]])[:-1]
这给出的输出是 -6c:39:-1b:XX:XX:XX,这与真正的 MAC 不对应,即 94:39:e5:XX:XX:XX。一个线索可能是某些元素是正确的,但正如您所看到的,输出是错误的。我不知道为什么。
令人惊讶的是,我在 SO 中发现了一个类似的问题,其目的几乎相同,但使用了打包结构和 python 自己的 ioctl。它只是工作:获取 MAC 地址。不过,我的问题是如何正确表示和使用 ctype 结构。