python中使用ctypes调用so传参设置

问题

近日在作一组声纹聚类时,使用了另外一团队同窗开发的声纹距离算法。该算法对外提供的是一组so包,须要使用方本身去使用。在python中调用纯so包通常使用ctypes类库,用起来看起来简单但也有很多细节容易犯错。本次使用过程当中,就遇到传参的问题。python

目标so库中对外export的函数是大体以下的三个函数:算法

void* create_handler();
    int extract_feature(void* hander);
    bool destroy(void* handler);

这三个函数使用起来倒也简单,顺序使用就能够了。但发现写成以下形式的python代码后,执行会直接segment fault。函数

import sys
    import ctypes
    
    so = ctypes.CDLL("./lib/libbase.so")
    p = so.create_handler()
    feature = so.extract_feature(p)
    so.destroy(p)

解决

这段代码中p是int类型,由void*自动转来,在ctyeps中这种转型自己是没问题的。segment fault发生在extract_feature函数调用中,问题应当出在参数上,回传的handler已经不是原来的pointer了,致使访问指针出错。指针

查阅ctypes的文档后,发现ctypes能够声明so库中函数的参数,返回类型。试了试,显式声明后问题获得了解决,证实咱们的猜测是对的,确实指针发生了变化。修改后代码以下:rest

import sys
    import ctypes
    
    so = ctypes.CDLL("./lib/libbase.so")
    so.create_handler.restype=ctypes.c_void_p
    so.extract_feature.argtypes=[ctypes.c_void_p]
    so.destroy.argtypes=[ctypes.c_void_p]
    
    p = so.create_handler()
    feature = so.extract_feature(p)
    so.destroy(p)

结论:

ctypes中传递指针类型参数须要显式声明c函数的参数,返回类型。
相关文章
相关标签/搜索