dlopen在boost python模块中失败

时间:2013-10-29 16:00:14

标签: python boost boost-python dlopen

我的目标是使用boost python创建一个python扩展模块。问题是,当共享库中的代码使用dlopen访问相同库中的符号时,它会失败(详情如下)。看起来模块中的符号没有加载到python进程的符号表中

我在共享库myTest.so

中有以下代码
// File 1
extern "C" void target_func() {
    std::cout << "Works!" << std::endl;
}

// File 2
typedef void (*Func) ();
void run() {
    void *handle = dlopen(0, RTLD_NOW | RTLD_GLOBAL)
    Func *method = (Func *)dlsym(handle, "target_func");
    // check if method is NULL and fail
}

它包含在一个boost python模块中,如图所示 -

BOOST_PYTHON_MODULE(myTest) {
    def("run", run);
}

导入python模块并执行

时,上面的代码失败了
import myTest
myTest.run()

undefined symbol: target_func

然而,当我将myTest.so与调用run的main函数链接时,会出现 NO 问题

int main(int argc, char **argv) {
    run();
}

Output: Works!

1 个答案:

答案 0 :(得分:5)

默认情况下,解释程序不会以跨扩展名共享符号的方式加载扩展程序。可以通过sys.setdlopenflags()设置解释器使用的dlopen()标志来修改此问题。文档说明:

  

要在扩展模块之间共享符号,请调用sys.setdlopenflags(dl.RTLD_NOW | dl.RTLD_GLOBAL)


以下是使用默认设置加载扩展程序的解释器:

>>> import myTest
>>> myTest.run()
/usr/bin/python: undefined symbol: target_func

修改解释器的dlopen()标志后会成功:

>>> import sys
>>> import dl
>>> sys.setdlopenflags(dl.RTLD_NOW | dl.RTLD_GLOBAL)
>>> import myTest
>>> myTest.run()
Works!