2009-04-17 63 views
2

我有下面的代码蟒蛇的ctypes和sysctl的

import sys 
from ctypes import * 
from ctypes.util import find_library 

libc = cdll.LoadLibrary(find_library("c")) 
CTL_KERN = 1 
KERN_SHMMAX = 34 
sysctl_names = { 
    'memory_shared_buffers' : (CTL_KERN, KERN_SHMMAX), 
    } 

def posix_sysctl_long(name): 
    _mem = c_uint64(0) 
    _arr = c_int * 2 
    _name = _arr() 
    _name[0] = c_int(sysctl_names[name][0]) 
    _name[1] = c_int(sysctl_names[name][1]) 
    result = libc.sysctl(_name, byref(_mem), c_size_t(sizeof(_mem)), None, c_size_t(0)) 
    if result != 0: 
     raise Exception('sysctl returned with error %s' % result) 
    return _mem.value 

print posix_sysctl_long('memory_shared_buffers') 

产生以下结果:

Traceback (most recent call last): 
    File "test.py", line 23, in <module> 
    print posix_sysctl_long('memory_shared_buffers') 
    File "test.py", line 20, in posix_sysctl_long 
    raise Exception('sysctl returned with error %s' % result) 
Exception: sysctl returned with error -1 

我客串我做错了什么。什么是正确的调用约定?我怎么知道究竟出了什么问题?

回答

6

您没有向sysctl函数提供正确的值。关于sysctl()的参数的详细信息可以在here找到。

这里是你的错误:

  • 你已经忘记了nlen参数(第二个参数)
  • oldlenp参数是一个指向大小,而不是直接的大小

这里是正确的功能(有小的改进):

def posix_sysctl_long(name): 
    _mem = c_uint64(0) 
    _def = sysctl_names[name] 
    _arr = c_int * len(_def) 
    _name = _arr() 
    for i, v in enumerate(_def): 
     _name[i] = c_int(v) 
    _sz = c_size_t(sizeof(_mem)) 
    result = libc.sysctl(_name, len(_def), byref(_mem), byref(_sz), None, c_size_t(0)) 
    if result != 0: 
     raise Exception('sysctl returned with error %s' % result) 
    return _mem.value 
+0

非常感谢。事实上,我忘了第二个参数:-( 我的原始版本使用sysctlbyname,那里的参数是不需要的。不幸的是,似乎linux不支持该功能。 – Mauli 2009-04-17 13:11:42