使用CMake Import错误编译嵌入式Python的C ++

时间:2013-07-24 19:15:45

标签: c++ python linux cmake

我正在尝试在使用CMake编译的C ++项目中包含python文件。

首先,我使用这两个文件独立完成此操作:

#include <Python.h>
#include <stdio.h>

int main(int argc, char *argv[])
{
  setenv("PYTHONPATH", ".", 0);
  char hostname[] = "localhost";
  PyObject *pName, *pModule, *pFunc;
  PyObject *pArgs, *pValue;
  Py_Initialize();
  pName = PyString_FromString("GetHostname");
  pModule = PyImport_Import(pName);
  Py_DECREF(pName);

  if(pModule != NULL) {
    pFunc = PyObject_GetAttrString(pModule, "GetHostname");

    if (pFunc && PyCallable_Check(pFunc)) {
      pArgs = PyTuple_New(1);
      pValue = PyString_FromString(hostname);
      PyTuple_SetItem(pArgs, 0, pValue);
      pValue = PyObject_CallObject(pFunc, pArgs);
      Py_DECREF(pArgs);
      if (pValue != NULL) {
    printf("The IP address is %s\n", PyString_AsString(pValue));
    Py_DECREF(pValue);
      }
      else {
    Py_DECREF(pFunc);
    Py_DECREF(pModule);
    PyErr_Print();
    fprintf(stderr, "Call Failed\n");
    return 1;
      }
    }
    else {
      if (PyErr_Occurred())
    PyErr_Print();
      fprintf(stderr, "Cannot find function\n");
    }
    Py_XDECREF(pFunc);
    Py_DECREF(pModule);
  }
  else {
    PyErr_Print();
    fprintf(stderr, "Failed to load file\n");
    return 1;
  }
  Py_Finalize();
  return 0;
}

import socket

def GetHostname(hostname):
    addr = socket.gethostbyname(hostname)
    return addr

当我使用

编译时
g++ $(python-config --cflags) -o test $(python-config --ldflags) ./test.cpp

来自how to link python static library with my c++ program一切正常。

但这包含在使用CMake编译的项目中,我必须做错了,因为在编译后我得到了

Traceback (most recent call last):
  File "/src/GetHostname.py", line 1, in <module>
    import socket
  File "/usr/lib64/python2.6/socket.py", line 46, in <module>
    import _socket
ImportError: /usr/lib64/python2.6/lib-dynload/_socketmodule.so: undefined symbol: PyExc_ValueError

在CMakeLists.txt中我添加了行

find_package( PythonInterp REQUIRED )
find_package( PythonLibs REQUIRED )
include_directories ( ${PYTHON_INCLUDE_DIRS} )
add_library (GetHostname MODULE GetHostname.cc)
target_link_libraries(GetHostname ${PYTHON_LIBRARIES})
CONFIGURE_FILE(${PATH_TO_SOURCE}GetHostname.py ${PATH_TO_BUILD}GetHostname.py COPYONLY)

基于此帖子Python.h: No such file or directory

所有内容都编译但由于错误导致python模块无法加载。我没有在CMake中正确链接python库吗?

任何可以解释失败原因的想法都是受欢迎的。

使用Python 2.6

我知道这可以在C ++中完成,但是这不是我需要包含的唯一python模块,所以用C ++重写它不是我正在寻找的答案。我现在也知道localhost的IP地址是已知的,这仅用于测试目的。

1 个答案:

答案 0 :(得分:2)

所以我终于找到了答案。

问题是当在CMake中链接时,与python库的链接只在本地加载(RTLD_LOCAL),这意味着实际的python脚本没有与python库链接,只有C ++。

要解决此问题,只需在C ++代码中加载具有全局符号解析功能的python库。

dlopen("libpython2.6.so.1.0", RTLD_NOW | RTLD_NOLOAD | RTLD_GLOBAL);

这是一个快速而又脏的修复,因为它特定于python2.6,你应该通过在CMake中根据机器上安装的版本定义变量来使其可移植。

您必须导入dlfcn才能使用dlopen:

#include <dlfcn.h>