Неопределенные символы с add_swig_library - PullRequest
1 голос
/ 31 октября 2019

Я попытался написать простой пример использования SWIG с CMake, но получаю следующее сообщение об ошибке:

Undefined symbols for architecture x86_64:
"_My_variable", referenced from:
  _Swig_var_My_variable_get in examplePYTHON_wrap.c.o
  _Swig_var_My_variable_set in examplePYTHON_wrap.c.o
"_fact", referenced from:
  __wrap_fact in examplePYTHON_wrap.c.o
"_get_time", referenced from:
  __wrap_get_time in examplePYTHON_wrap.c.o
"_my_mod", referenced from:
  __wrap_my_mod in examplePYTHON_wrap.c.o
ld: symbol(s) not found for architecture x86_64

Сценарий CMake:

cmake_minimum_required(VERSION 3.8)

cmake_policy(SET CMP0078 OLD)
cmake_policy(SET CMP0086 OLD)

project(example C)

find_package(PythonLibs 3 REQUIRED)
find_package(PythonInterp ${PYTHONLIBS_VERSION_STRING} REQUIRED)

message(STATUS "Python executable is ${PYTHON_EXECUTABLE}")
message("PYTHONLIBS_VERSION_STRING: ${PYTHONLIBS_VERSION_STRING}")

include_directories(${PYTHON_INCLUDE_PATH} ${CMAKE_SOURCE_DIR})

find_package(SWIG REQUIRED)
include(${SWIG_USE_FILE})

message("CMAKE_SWIG_FLAGS: ${CMAKE_SWIG_FLAGS}")

swig_add_library(example TYPE SHARED LANGUAGE python SOURCES example.i)
swig_link_libraries(example ${PYTHON_LIBRARIES})

Мой пример SWIGзависит от двух файлов.

Во-первых, есть файл C (example.c):

#include <time.h>

double My_variable = 3.0;

int fact(int n) 
{
  if (n <= 1)
    return 1;
  else
    return n * fact(n-1);
}

int my_mod(int x, int y)
{
  return (x%y);
}

char *get_time()
{
  time_t ltime;
  time(&ltime);
  return ctime(&ltime);
}

Существует также файл SWIG (example.i):

%module example

%{
/* Put header files here or function declarations like below */
extern double My_variable;
extern int fact(int n);
extern int my_mod(int x, int y);
extern char *get_time();
%}

extern double My_variable;
extern int fact(int n);
extern int my_mod(int x, int y);
extern char *get_time();

Дляпричина, по которой я не понимаю, функции и переменные из example.c не распознаются при использовании CMake. Тем не менее, все отлично работает без CMake:

swig -python example.i
gcc -c example.c example_wrap.c -I<path_to_directory_containing_Python.h>
gcc -shared example.o example_wrap.o -o _example.so 

Я попытался сравнить эти строки с подробным выводом

make 

, но последние гораздо сложнее, чем первый.

РЕДАКТИРОВАТЬ: Это показывает, что CMake эффективно вызывает Swig:

 /usr/local/Cellar/cmake/3.15.3/bin/cmake -E env SWIG_LIB=/usr/local/Cellar/swig/4.0.1/share/swig/4.0.1 /usr/local/bin/swig -python -outdir /Users/myself/forexample/swig/build -interface _example -I/usr/local/Frameworks/Python.framework/Versions/3.7/include/python3.7m -I/Users/myself/forexample/swig -o /Users/myself/forexample/swig/build/CMakeFiles/_example.dir/examplePYTHON_wrap.c /Users/myself/forexample/swig/example.i

1 Ответ

2 голосов
/ 31 октября 2019

Ваш CMake, похоже, не ссылается на example.c вообще;Вот почему все ваши определения функций не могут быть найдены. Ошибка:

Undefined symbols for architecture x86_64:

- это красная сельдь , так как в действительности не существует несоответствия архитектуры;эти символы не были определены вообще . Попробуйте изменить ваш CMake, добавив в него библиотеку для кода C, а затем свяжите его с вашей библиотекой SWIG. Примерно так должно помочь устранить ошибки:

cmake_minimum_required(VERSION 3.8)

cmake_policy(SET CMP0078 OLD)
cmake_policy(SET CMP0086 OLD)

project(example C)

find_package(PythonLibs 3 REQUIRED)
find_package(PythonInterp ${PYTHONLIBS_VERSION_STRING} REQUIRED)

message(STATUS "Python executable is ${PYTHON_EXECUTABLE}")
message("PYTHONLIBS_VERSION_STRING: ${PYTHONLIBS_VERSION_STRING}")

include_directories(${PYTHON_INCLUDE_PATH} ${CMAKE_SOURCE_DIR})

find_package(SWIG REQUIRED)
include(${SWIG_USE_FILE})

message("CMAKE_SWIG_FLAGS: ${CMAKE_SWIG_FLAGS}")

add_library(MyCLib example.c)

swig_add_library(example TYPE SHARED LANGUAGE python SOURCES example.i)
swig_link_libraries(example MyCLib ${PYTHON_LIBRARIES})
...