- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我需要一个与 ctypes 结构兼容的位域定义,这样我就可以将它用作指向一组内存映射的硬件寄存器的指针,即
class RegFile(Structure):
_fields_ = [
('ticks', c_uint32),
('id', id_bitfield),
]
理想情况下,它们是非常 Pythonic 的东西,并且表现得像字典。我已经设法处理了大部分血淋淋的细节,现在我有了一个工厂函数,它可以像制作命名元组一样制作位域类。下面是一个拆分标准 float 字段的示例。
def make_register(name, fields, basetype=c_uint32):
# Define the underlying bitfield type
bitfield = type(name + '_bitfield', (LittleEndianStructure, ), {})
bitfield._pack_ = 1
bitfield._fields_ = fields
# Set up the union
d = {
'_fields_' : [('base', basetype), ('_b', bitfield)],
'_anonymous_' : ('_b',),
'__iter__' : _register_iter,
'keys' : _register_keys,
'items' : _register_items,
'update' : _register_update
}
return type(name, (Union, ), d)
ieee754_fields = [
('mantissa', c_uint, 23),
('exponent', c_uint, 8),
('sign', c_uint, 1)
]
IEEE754 = make_register('IEEE754', ieee754_fields, c_float)
x = IEEE754()
这工作得体,但不是特别 Pythonic 感觉的语法。理想情况下,我会有一些答案允许我将位域类定义为:
class IEEE754(Register):
"""Individual bitfields of a standard IEEE-754 floating point number."""
_fields_ = ieee754_fields
_basetype_ = c_float
但我一直没能使 Register 类成为现实。看起来 Register 应该继承自 Union 并应用一些元类魔法,但元类与 Union 的冲突就在于此。有什么想法吗?还是我应该坚持使用现有的东西?
最佳答案
您可以通过创建一个扩展 type(ctypes.Union)
的 Register 元类来实现您所描述的内容,如下所示:
class Register(type(ctypes.Union)):
def __new__(mcs, name, bases, dict):
#make the bitfield class
class regBitfield(ctypes.LittleEndianStructure):
_fields_ = dict['_fields_']
#set up the Union
class regUnion(ctypes.Union):
_fields_ = [('base', dict['_basetype_']), ('_b', regBitfield)]
_anonymous_ = ('_b', )
__iter__ = _register_iter
keys = _register_keys
items = _register_items
update = _register_update
#return a subclass of the temporary regUnion class
return type(ctypes.Union).__new__(mcs, name, (regUnion,), dict)
class IEEE754: #for Python3 use class IEEE754(metaclass=Register):
__metaclass__ = Register #omit this line Python3
_fields_ = [('mantissa', ctypes.c_uint, 23),
('exponent', ctypes.c_uint, 8),
('sign', ctypes.c_uint, 1)
]
_basetype_ = ctypes.c_float
这与您的 make_register
函数的工作方式几乎相同,但利用类系统的现有机制和扩展 ctypes.Union
和 ctypes.结构
。具体来说,ctypes.Union
或 ctypes.Structure
子类的任何子类都将扩展 _fields_
列表,而不是而不是更换它。知道这一点,我们可以在最终类扩展的 Register
元类的 __new__
中创建临时类。有了以上:
>>> myFloat = IEEE754(42.0)
>>> print(myFloat.base)
42.0
>>> print(myFloat.mantissa)
2621440L
>>> fltReg._fields_
[('mantissa', <class 'ctypes.c_ulong'>, 23), ('exponent', <class 'ctypes.c_ulong'>, 8), ('sign', <class 'ctypes.c_ulong'>, 1)]
如果正在创建的类型具有多个基数,则 Register.__new__
方法可能会中断。
关于Python ctypes 位域,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20380451/
我似乎找不到任何将 ctypes.c_void_p() 转换为字符串或字节数组的简单示例。有没有简单的衬里可以做到这一点? 最佳答案 给你: import ctypes as ct # set up
在ctypes中,pointer和byref有什么区别?它们似乎都是将指针传递给函数的一种方式,例如作为输出参数。 最佳答案 在功能上,它们是等价的。 然而,python docs请指出 pointe
我知道我应该指定 argtypes对于我的 C/C++ 函数,因为我的某些调用会导致堆栈损坏。 myCfunc.argtypes = [ct.c_void_p, ct.POINTER(ct.c
有没有办法获取指向 ctypes 数组中间元素的指针?示例: lib = ctypes.cdll.LoadLibrary('./lib.so') arr = (ctypes.c_int32 * 100
在我自定义的 TYPO3 Extbase 扩展中,我创建了一个后端模块来管理个人记录。现在我需要一个内容元素来在前端显示记录。 我看到了两种实现此目的的方法: 使用 CType“list”和自定义 l
实际上,我正在尝试将 ctypes 数组转换为 python 列表并返回。 如果找到this thread 。但它假设我们在编译时知道类型。 但是是否可以检索元素的 ctypes 类型? 我有一个 p
我正在将 float 列表转换为具有以下字段的 ctypes Structure 类,然后再将它们传递给 FFI 函数: FFIArray(Structure): _fields_ = [("
我需要将异质数据的二维数组从我的 c dll 返回到 python。 为此目的,我从我的 c dll 返回一个元组的元组。它作为 PyObject 返回 * 这个元组的元组需要作为第一行第一列的 tu
这是不一致的: from ctypes import * class S(Structure): _fields_ = [("x", POINTER(c_int)), ("y", c_int)
我真的希望一些 Python/Ctypes/C 专家可以帮助我解决这个问题,这可能是我在使用 Python 与 C 库交互时正确使用 Ctypes 的类型结构方面缺乏知识。 目标:我需要访问几个使用
我正在尝试调试 python 使用 ctypes 调用 C 函数的代码。我感兴趣的 python 代码中的一行看起来像: returnValue = cfunction() 其中 cfunction
我正在开发 DLL/SO 的 Python 包装器。我已经验证了代码可以调用实际的 DLL 和 SO。我想对我的包装器进行单元测试,而不需要安装底层 DLL/SO。我正在考虑使用 mock 。 我遇到
大家。我在使用 ctypes 和 C 代码时遇到内存分配错误。我想知道内存问题是在 C 内部,还是由 ctypes 使用不当引起的。内存错误是 python(79698) malloc: * erro
我想制作一个笑话程序,首先它打开一个消息框,关闭后另一个消息框出现在随机位置。它会一直这样重复,直到有什么东西终止了它的任务。使用 tkinter 消息框,那么这些消息框就无法被 Hook ,我必须制
我对 python 中的变量大小有疑问,我使用 Ctypes 因为我想要一个 1 字节的数字,但是当我试图在 python 中检查它的大小时(通过 sys.getsize ) 它说它是 80 字节但是
我正在尝试在 python lambda 函数中使用 matplotlib 生成图形。我使用库 mathplotlib 导入了一个图层,但它不起作用。 这个想法是生成一个图形,将其保存为临时文件并上传
我正在尝试使用 C 中的 python ctypes 制作简单的库 blake 哈希函数包装器。但只是为了首先测试我的简单 C 辅助函数是否能正常工作,我编写了小的 python 脚本 blake 哈
图书馆代码(简化版): // package1.go package package1 import "C" func Play(s *C.char) { } 客户代码: // main.go pac
到目前为止,我已经得到了一个不适用于 python 的 DLL,并输入 return: I just can't pass it arguments because I doing it wrong
我有一个具有以下签名的 C 函数: void init(int* argc, char** argv[]); 我想使用 Ctypes 从我的 OCaml 代码中调用此函数,但我想不出一个正确的方法来传
我是一名优秀的程序员,十分优秀!