在cython中包装一个c库,其中包含一个python内部c符号

时间:2017-05-03 11:12:15

标签: c cython

所以我正在研究Ox和Python之间的接口。问题是:Ox定义了一个函数c_pow,Cython'mangles'到_Py_c_pow。不幸的是,这个函数是allready defined。在这种情况下,有没有办法防止命名冲突?我可以指定自己的前缀吗?

#"jdmath.h"
...
void   JDCALL c_pow(double xr, double xi, double a, double *yr, double *yi);
...

#"oxcy.pxd"
cdef extern from r'dev\jdmath.h':
    pass

编译器输出:

PS D:\Git\OxPy\oxcy> python setup.py build_ext --inplace
running build_ext
cythoning oxcy.pyx to oxcy.c
building 'oxcy' extension
    ~\AppData\Local\Programs\Common\Microsoft\Visual C++ for Python\9.0\VC\Bin\amd64\cl.exe /c /nologo /Ox /MD /W3 /GS- /DNDEBUG "-IC:\Program Files (x86)\OxMetrics7\ox\bin" "-IC:\Program Files (x86)\OxMetrics7\ox\dev\lib64" "-IC:\Program Files (x86)\OxMetrics7\ox\dev" -IC:\Anaconda2\include -IC:\Anaconda2\PC /Tcoxcy.c /Fobuild\temp.win-amd64-2.7\Release\oxcy.obj
oxcy.c
d:\git\oxpy\oxcy\dev\jdmath.h(111) : warning C4031: second formal parameter list longer than the first list
d:\git\oxpy\oxcy\dev\jdmath.h(111) : warning C4028: formal parameter 1 different from declaration
d:\git\oxpy\oxcy\dev\jdmath.h(111) : warning C4273: '_Py_c_abs' : inconsistent dll linkage
    c:\anaconda2\include\complexobject.h(30) : see previous definition of '_Py_c_abs'
d:\git\oxpy\oxcy\dev\jdmath.h(117) : warning C4031: second formal parameter list longer than the first list
d:\git\oxpy\oxcy\dev\jdmath.h(117) : warning C4028: formal parameter 1 different from declaration
d:\git\oxpy\oxcy\dev\jdmath.h(117) : warning C4028: formal parameter 2 different from declaration
d:\git\oxpy\oxcy\dev\jdmath.h(117) : error C2371: '_Py_c_pow' : redefinition; different basic types
    c:\anaconda2\include\complexobject.h(29) : see declaration of '_Py_c_pow'
error: command '~\\AppData\\Local\\Programs\\Common\\Microsoft\\Visual C++ for Python\\9.0\\VC\\Bin\\amd64\\cl.exe' failed with exit status 2

修改

事实证明,Anaconda定义了一个宏:#define c_pow _Py_c_pow。因此,尽管Cython不应该归咎于修改,但编译仍然会因Python C头中的宏和来自Ox的头部引起的名称冲突而失败。我该如何规避这个问题呢?

1 个答案:

答案 0 :(得分:1)

正如您所发现的,这是由导致重命名的macros defined by Python 2引起的。它们未在Python 3中定义,因此一个选项是升级。

假设您不想升级(这可能是合理的......),那就更难了。最简单的尝试就是添加

#undef c_pow
#undef c_abs

到jdmath.h的顶部。 (或者,如果你不能修改jdmath.h然后创建一个新的头,它取消定义宏并包含jdmath.h然后使用来自Cython的新头)。

希望这样可以正常工作。然而,如果它试图使用Python定义的c_powc_abs本身,那么这可能会破坏Cython中的某些东西(你会得到明显的编译错误)。在这种情况下,您需要创建一个使用新名称创建函数的标题:

// undefine so we can access c_pow and c_abs
#undef c_pow
#undef c_abs

#include "dev/jdmath.h"

inline void not_c_pow(/* fill in args yourself*/) { c_pow(/*args*/); }
inline void not_c_abs(/* fill in args yourself*/) { c_abs(/*args*/); }

// redefine to original state
#define c_pow _Py_c_pow
#define c_abs _Py_c_abs

在Cython中你可以使用not_c_pownot_c_abs来代替它,它应该有用。