4

tl;博士

这适用于 GNU 版本的 libc(尚未尝试使用 uclibc)

from ctypes import *

libc = CDLL('libc.so.6')

class uts_struct(Structure):
    _fields_ = [ ('sysname', c_char * 65),
                 ('nodename', c_char * 65),
                 ('release', c_char * 65),
                 ('version', c_char * 65),
                 ('machine', c_char * 65),
                 ('domain', c_char * 65) ]

gnar = uts_struct()

libc.uname(byref(gnar))

print gnar.nodename

原帖

以下代码段错误;我不确定我做错了什么。

from ctypes import *

libc = CDLL('libc.so.6')

class uts_struct(Structure):
    _fields_ = [ ('sysname', c_char_p),
                 ('nodename', c_char_p),
                 ('release', c_char_p),
                 ('version', c_char_p),
                 ('machine', c_char_p) ]

utsname = uts_struct()
libc.uname(byref(utsname))

print utsname.sysname

这做同样的事情:

from ctypes import *

libc = CDLL('libc.so.6')

class uts_struct(Structure):
    _fields_ = [ ('sysname', c_char_p),
                 ('nodename', c_char_p),
                 ('release', c_char_p),
                 ('version', c_char_p),
                 ('machine', c_char_p) ]

utsname = uts_struct()
utsname_pointer = pointer(utsname)
libc.uname(utsname_pointer)

print utsname.sysname

我一定是搞砸了一些基本的东西......

(我知道os.uname(),这只是一个理解练习,我失败了)

我在这里引用了 uname 手册:http://www.cl.cam.ac.uk/cgi-bin/manpage ? 2+uname

我究竟做错了什么?


编辑:

感谢Nemo,我能够获取数据;

>>> from ctypes import *
>>> libc = CDLL('libc.so.6')
>>> gnar = create_string_buffer(512)
>>> libc.uname(byref(gnar))
0
>>> print gnar.value
Linux
>>> 

但是,我假设我只得到“Linux”,因为这些项目是 NULL 分隔的,调节器字符串也是如此。有什么方法可以阅读NULL吗?


编辑2:

根据 Nemos 的评论,我已经尝试过这个 - 这不起作用,但我认为这可能是朝着正确方向迈出的一步......错误:

Traceback (most recent call last):
  File "gnar.py", line 18, in <module>
    utsname = uts_struct(gnar)
TypeError: incompatible types, c_char_Array_512 instance instead of c_char_p instance

这只是不可行吗?

from ctypes import *

libc = CDLL('libc.so.6')

class uts_struct(Structure):
    _fields_ = [ ('sysname', c_char_p),
                 ('nodename', c_char_p),
                 ('release', c_char_p),
                 ('version', c_char_p),
                 ('machine', c_char_p) ]

gnar = create_string_buffer(512)
libc.uname(byref(gnar))
utsname = uts_struct(gnar)

Edit3:(我要写有史以来最长的帖子……=P)

from ctypes import *
libc = CDLL('libc.so.6')
class uts_struct(Structure):
    _fields_ = [ ('sysname', c_char * 65),
                 ('nodename', c_char * 65),
                 ('release', c_char * 65),
                 ('version', c_char * 65),
                 ('machine', c_char * 65) ]
gnar = uts_struct()
libc.uname(byref(gnar))
print gnar.machine

这可行,但是,它在打印值后会出现段错误......


最终编辑:

以下作品 - 我当然使用的是 GNU 版本的 libc。(我在 Ubuntu 机器上)所以添加域的字段就是停止段错误所需​​要的。事后看来是有道理的。:)

from ctypes import *

libc = CDLL('libc.so.6')

class uts_struct(Structure):
    _fields_ = [ ('sysname', c_char * 65),
                 ('nodename', c_char * 65),
                 ('release', c_char * 65),
                 ('version', c_char * 65),
                 ('machine', c_char * 65),
                 ('domain', c_char * 65) ]

gnar = uts_struct()
libc.uname(byref(gnar))
print gnar.nodename
4

2 回答 2

2

根据这个 uname 手册页,该结构包含实现定义大小的数组,而不是 char* (c_char_p)。您是否查看过中的结构定义sys/utsname.h?您必须匹配确切的结构定义。数据类型可能应该是c_char * n其中n是 中找到的该字段的数组大小sys/utsname.h

或者,您应该能够使用 访问第一次编辑中的所有字符串print gnar.raw,只要缓冲区足够大以容纳整个结构。

于 2011-06-03T00:41:39.913 回答
2

utsname 结构中的字段不是指针;它们是“未指定大小的数组”。

因此,字符串在结构中背靠背打包并以空值结尾。

我不知道如何在 Python 中表示这一点。但我建议您从 uname() 以外的其他东西开始进行实验。:-)

[更新]

517366245708 十进制是 0x78756E694C,这是“xuniL”的 ascii。这在 64 位 little-endian 机器上实际上是有意义的......

但问题是您正在使用 c_char_p 创建一个指针,然后在调用 byref() 时传递一个指向该指针的指针。所以 uname() 正在填写您的指针(而不是它指向的内容)。更糟糕的是,它在那里放置了超过 8 个字节,所以你当前的代码正在破坏内存。

您需要弄清楚如何分配一个大小为 a 的内存块struct utsname,然后将指向该内存块的指针传递给uname () 函数,然后找出该块内的哪些偏移量对应于结构中的哪些字段。我不确定使用 Python 有多少可能......

[第二次更新]

这更好......但是现在您必须查看数组中的偏移量。如果这是一个典型的 Linux 系统,每个字段是 65 个字节(是的,真的)。因此,您需要开始将 65 个字节读入字符串。我不知道你是否可以调用string.index这个东西......

于 2011-06-02T22:19:35.307 回答