我正在尝试与 Python 中的 mdb 库交互,使用 SWIG 根据 mdb 附带的头文件生成绑定。这是我第一次尝试这样做,当最终在 python 中加载共享库时,我遇到了一个未定义的符号。
这是我的接口定义文件:
$ cat pymdb.i
%module pymdb
%{
#include "mdbtools.h"
%}
%include "mdbtools.h"
生成 C 代码会引发以下警告:
$ swig -python -o pymdb.c -I/usr/include pymdb.i
/usr/include/mdbtools.h:187: Warning 451: Setting a const char * variable may leak memory.
/usr/include/mdbtools.h:188: Warning 451: Setting a const char * variable may leak memory.
/usr/include/mdbtools.h:189: Warning 451: Setting a const char * variable may leak memory.
/usr/include/mdbtools.h:190: Warning 451: Setting a const char * variable may leak memory.
/usr/include/mdbtools.h:191: Warning 451: Setting a const char * variable may leak memory.
/usr/include/mdbtools.h:192: Warning 451: Setting a const char * variable may leak memory.
/usr/include/mdbtools.h:193: Warning 451: Setting a const char * variable may leak memory.
编译 C 代码,一切看起来都很好:
$ gcc -c -fPIC -I/usr/include/python2.7 -I/usr/include/glib-2.0 -I/usr/lib/x86_64-linux-gnu/glib-2.0/include pymdb.c -o pymdb.o
将其链接到 /usr/lib/x86_64-linux-gnu/libmdb.so.2.0.0 文件(Debian 多架构):
$ ld -shared -lmdb pymdb.o -o _pymdb.so
这就是我得到未定义符号的地方,在python中加载.so:
$ echo "import pymdb" |python
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "pymdb.py", line 26, in <module>
_pymdb = swig_import_helper()
File "pymdb.py", line 22, in swig_import_helper
_mod = imp.load_module('_pymdb', fp, pathname, description)
ImportError: ./_pymdb.so: undefined symbol: read_pg_if_16
mdbtools.h 中有 read_pg_if_16 的声明:
extern guint16 read_pg_if_16(MdbHandle *mdb, int *cur_pos);
我可以在 _pymdb.so 文件表中看到未定义的符号:
$ objdump -T _pymdb.so |grep read_pg_if_16
0000000000000000 D *UND* 0000000000000000 read_pg_if_16
但 libmdb.so.2 不提供此符号:
$ objdump -T /usr/lib/x86_64-linux-gnu/libmdb.so.2 |grep read_pg_if_16
会不会是我的接口文件中的错误?或者这可能是 libmdb.so.2 的问题,没有暴露这个符号?我对 C 和链接的东西没有那么深入,所以在这一点上我对如何着手解决这个问题有点迷茫。