Visibility of template function specialisations in

2019-06-02 04:40发布

I am building some C++ code that utilises pybind11 to do some wrapping and create a Python extension module, and I am encountering some undefined symbol problems. Upon further examination, it appears to be because certain template function specialisations are not publically visible in the shared library I am building.

I have a MWE to demonstrate explicitly the problem:

symbol_test.hpp

#include <pybind11/pybind11.h>

template <typename T>
void test_func(T var) {}

template <>
void test_func<int>(int);

template <>
void test_func<pybind11::object>(pybind11::object);

symbol_test.cpp

#include "symbol_test.hpp"

template <>
void test_func<int>(int var) {}

template <>
void test_func<pybind11::object>(pybind11::object var) {}

Build command:

g++ -std=c++14 -fPIC -I/home/farmer/anaconda3/envs/default/include/python2.7 -I/home/farmer/repos/pybind11/include -shared -o testlib.so symbol_test.cpp

Inspect library with nm:

nm testlib.so
...
00000000000024d0 T _Z9test_funcIiEvT_
00000000000024da t _Z9test_funcIN8pybind116objectEEvT_
...

In short, the instance of test_func specialised on "int" is visible, but the instance specialised on "pybind11::object" is not. Why? How do I control this? I need the pybind11::object version of the function to be exported in the library, but I don't know why it isn't.

I ask the question here rather than submit it as a pybind11 bug because I just assume there is something I don't understand about how template functions symbols are exported in shared libraries, rather than it being a deficiency in pybind11.

Edit: I had a thought; pybind11 is a header-only library, so I suppose the code for the pybind11::object class is not getting built or something? And this is affecting the visibility of template functions which use it?

0条回答
登录 后发表回答