python - SWIG Python C/C++。结果模块是空的,没有错误

标签 python c++ c swig

对于 future 的任何人

解决方案在问题和回复的评论中。


我正在尝试为 musly 创建一个 python 接口(interface)软件 (C/C++),更具体地说是用于 libmusly,带有 swig。我正在使用一个构建目录,该目录是链接存储库中的/libmusly 目录,其中包含同一存储库中的/include 目录。

我正在使用以下名为 lib.i 的 swig 接口(interface),因为我想要访问的方法位于 lib.cpp 文件中。我会提前说最后的部分让我觉得问题出在这个文件中。

/* lib.i */ 
%module libmusly 
%{ 
typedef struct {
    void* method;
    char* method_name;
    void* decoder;
    char* decoder_name; } musly_jukebox;

typedef float musly_track;

typedef int musly_trackid;

const char* musly_version();

void musly_debug(
        int level);

const char* musly_jukebox_listmethods();

const char* musly_jukebox_listdecoders();

const char* musly_jukebox_aboutmethod(
        musly_jukebox* jukebox);

musly_jukebox* musly_jukebox_poweron(
        const char* method,
        const char* decoder);   int musly_jukebox_similarity(
        musly_jukebox* jukebox,
        musly_track* seed_track,
        musly_trackid seed_trackid,
        musly_track** tracks,
        musly_trackid* trackids,
        int num_tracks,
        float* similarities);

musly_track* musly_track_alloc(
        musly_jukebox* jukebox);

void musly_track_free(
        musly_track* track);

int musly_track_size(
        musly_jukebox* jukebox);

int musly_track_analyze_audiofile(
        musly_jukebox* jukebox,
        const char* audiofile,
        float excerpt_length,
        float excerpt_start,
        musly_track* track);    
%}

我正在使用以下文件来构建和链接它:

#!/bin/bash

echo "Generating swig wrapper file." 
swig3.0 -c++ -python lib.i

echo "Compiling." 
c++ -fPIC -c -lstdc++ -DNDEBUG -Dlibmusly_EXPORTS
-DHAVE_AVUTIL_CHANNEL_LAYOUT -DMUSLY_VERSION=\"0.2\" -I./ -I./include/ -I./libresample/ -I./decoders/ -I./kissfft/ -I./methods/ -I/usr/include/python3.5m/ -I/usr/include/eigen3/ -I/usr/include/x86_64-linux-gnu lib_wrap.cxx kissfft/kiss_fft.c kissfft/kiss_fftr.c methods/mandelellis.cpp methods/timbre.cpp decoders/libav.cpp resampler.cpp plugins.cpp method.cpp decoder.cpp windowfunction.cpp powerspectrum.cpp melspectrum.cpp discretecosinetransform.cpp mfcc.cpp gaussianstatistics.cpp mutualproximity.cpp lib.cpp

c++ -fPIC -c -lstdc++ -DNDEBUG -Dlibmusly_EXPORTS
-DHAVE_AVUTIL_CHANNEL_LAYOUT -DMUSLY_VERSION=\"0.2\" -I./ -I./include/ -I./libresample/ -I./decoders/ -I./kissfft/ -I./methods/ -I/usr/include/python3.5m/ -I/usr/include/eigen3/ -I/usr/include/x86_64-linux-gnu libresample/filterkit.c libresample/resamplesubs.c libresample/resample.c 

echo "Linking." 
cc  -fPIC   -shared -Wl,-soname,libmusly_resample.so
-o libmusly_resample.so libresample/filterkit.o libresample/resamplesubs.o libresample/resample.o 

c++ -fPIC  -shared -Wl,-soname,_libmusly.so -o _libmusly.so kiss_fft.o kiss_fftr.o mandelellis.o timbre.o libav.o resampler.o plugins.o method.o decoder.o windowfunction.o powerspectrum.o melspectrum.o discretecosinetransform.o mfcc.o gaussianstatistics.o mutualproximity.o libmusly_resample.so lib.o lib_wrap.o -lavcodec
-lavformat -lavutil -lpython3.5m

标题中提到我可以将生成的模块导入到我链接的 python 解释器中,3.5m,没有错误。但该模块除了标准方法外没有其他方法,这些方法以“_”(下划线)和一些 swig 相关方法开始和结束。

我确实尝试导入 libmusly 和 _libmusly,无论如何前者总是包含后者。它们里面的方法不一样,但是都存在同样的问题,我的方法都没有。

编译后目录下的文件为:

./:
build.sh
decoder.cpp
decoder.h
decoder.o
decoders
discretecosinetransform.cpp
discretecosinetransform.h
discretecosinetransform.o
files.txt
filterkit.o
gaussian.h
gaussianstatistics.cpp
gaussianstatistics.h
gaussianstatistics.o
idpool.h
include
kissfft
kiss_fft.o
kiss_fftr.o
libav.o
lib.cpp
lib.i
libmusly.py
libmusly_resample.so
_libmusly.so
lib.o
libresample
lib_wrap.cxx
lib_wrap.o
mandelellis.o
melspectrum.cpp
melspectrum.h
melspectrum.o
method.cpp
method.h
method.o
methods
mfcc.cpp
mfcc.h
mfcc.o
mutualproximity.cpp
mutualproximity.h
mutualproximity.o
plugins.cpp
plugins.h
plugins.o
powerspectrum.cpp
powerspectrum.h
powerspectrum.o
resample.o
resampler.cpp
resampler.h
resampler.o
resamplesubs.o
timbre.o
windowfunction.cpp
windowfunction.h
windowfunction.o

./decoders:
libav.cpp
libav.h

./include:
CMakeLists.txt
minilog.h
minilog.h.gch
musly

./include/musly:
musly.h
musly.h.gch
musly_types.h

./kissfft:
CHANGELOG
COPYING
kiss_fft.c
_kiss_fft_guts.h
kiss_fft.h
kiss_fftr.c
kiss_fftr.h
README
README.simd
TIPS

./libresample:
CMakeLists.txt
filterkit.c
filterkit.h
filterkit.o
libresample.h
LICENSE.txt
README.txt
resample.c
resample_defs.h
resample.o
resamplesubs.c
resamplesubs.o

./methods:
mandelellis.cpp
mandelellis.h
timbre.cpp
timbre.h

libmusly.py 的内容是:

# This file was automatically generated by SWIG (http://www.swig.org).
# Version 3.0.10
#
# Do not make changes to this file unless you know what you are doing--modify
# the SWIG interface file instead.





from sys import version_info as _swig_python_version_info
if _swig_python_version_info >= (2, 7, 0):
    def swig_import_helper():
        import importlib
        pkg = __name__.rpartition('.')[0]
        mname = '.'.join((pkg, '_libmusly')).lstrip('.')
        try:
            return importlib.import_module(mname)
        except ImportError:
            return importlib.import_module('_libmusly')
    _libmusly = swig_import_helper()
    del swig_import_helper
elif _swig_python_version_info >= (2, 6, 0):
    def swig_import_helper():
        from os.path import dirname
        import imp
        fp = None
        try:
            fp, pathname, description = imp.find_module('_libmusly', [dirname(__file__)])
        except ImportError:
            import _libmusly
            return _libmusly
        if fp is not None:
            try:
                _mod = imp.load_module('_libmusly', fp, pathname, description)
            finally:
                fp.close()
            return _mod
    _libmusly = swig_import_helper()
    del swig_import_helper
else:
    import _libmusly
del _swig_python_version_info
try:
    _swig_property = property
except NameError:
    pass  # Python < 2.2 doesn't have 'property'.

try:
    import builtins as __builtin__
except ImportError:
    import __builtin__

def _swig_setattr_nondynamic(self, class_type, name, value, static=1):
    if (name == "thisown"):
        return self.this.own(value)
    if (name == "this"):
        if type(value).__name__ == 'SwigPyObject':
            self.__dict__[name] = value
            return
    method = class_type.__swig_setmethods__.get(name, None)
    if method:
        return method(self, value)
    if (not static):
        if _newclass:
            object.__setattr__(self, name, value)
        else:
            self.__dict__[name] = value
    else:
        raise AttributeError("You cannot add attributes to %s" % self)


def _swig_setattr(self, class_type, name, value):
    return _swig_setattr_nondynamic(self, class_type, name, value, 0)


def _swig_getattr(self, class_type, name):
    if (name == "thisown"):
        return self.this.own()
    method = class_type.__swig_getmethods__.get(name, None)
    if method:
        return method(self)
    raise AttributeError("'%s' object has no attribute '%s'" % (class_type.__name__, name))


def _swig_repr(self):
    try:
        strthis = "proxy of " + self.this.__repr__()
    except __builtin__.Exception:
        strthis = ""
    return "<%s.%s; %s >" % (self.__class__.__module__, self.__class__.__name__, strthis,)

try:
    _object = object
    _newclass = 1
except __builtin__.Exception:
    class _object:
        pass
    _newclass = 0

# This file is compatible with both classic and new-style classes.

我在大约 2013 年的 Intel i3 上使用 Debian 9,一切都是从 repos 安装的。 Swig 是版本 3 和 c++ 6.3.0。 swig 文档推荐使用 c++ 来编译 c++,但我确实尝试了其他编译​​器并得到了相同的结果。


当我尝试将以下代码添加到/libmusly/CMakeLists.txt 的末尾并编译时,我得到两个文件“libmusly.py”和“_libmusly.so”,前者我必须重命名为“_libmusly.py” “只会遇到同样的问题。但是生成的 _libmusly.so 比我通过另一种方法获得的要大得多,分别是 14.6MB 和 3.5MB。

FIND_PACKAGE(SWIG REQUIRED) INCLUDE(${SWIG_USE_FILE})

FIND_PACKAGE(PythonLibs) INCLUDE_DIRECTORIES(${LIBMUSLY_INCLUDE}    ${EIGEN3_INCLUDE_DIR}   ${LIBAV_INCLUDE_DIRS}   ${CMAKE_CURRENT_SOURCE_DIR}     ${PYTHON_INCLUDE_PATH})

INCLUDE_DIRECTORIES(${CMAKE_CURRENT_SOURCE_DIR})

SET(CMAKE_SWIG_FLAGS "")

SET_SOURCE_FILES_PROPERTIES(lib.i PROPERTIES CPLUSPLUS ON) SET_SOURCE_FILES_PROPERTIES(lib.i PROPERTIES SWIG_FLAGS "-includeall") SWIG_ADD_MODULE(libmusly python lib.i
    kissfft/kiss_fft.c
    kissfft/kiss_fftr.c
    methods/mandelellis.cpp
    methods/timbre.cpp
    decoders/libav.cpp
    resampler.cpp
    plugins.cpp
    method.cpp
    decoder.cpp
    windowfunction.cpp
    powerspectrum.cpp
    melspectrum.cpp
    discretecosinetransform.cpp
    mfcc.cpp
    gaussianstatistics.cpp
    mutualproximity.cpp
    lib.cpp) SWIG_LINK_LIBRARIES(libmusly
    ${LIBMUSLY_LIBS}
    ${LIBAV_LIBRARIES}
    ${LIBMUSLY_EXTERNAL}
    ${PYTHON_LIBRARIES})

马克回复后更新

%{ %} block 之前使用 %inline 后,我无法再导入生成的模块。解释器提示 undefined symbol ,即使它在 .i 和 .cpp 文件中定义。我认为这可能是一个链接错误,但我不知道是什么。错误:

Traceback (most recent call last):
  File "/home/tuco/Desenvolvimento/libmusly_swig/libmusly.py", line 18, in swig_import_helper
    return importlib.import_module(mname)
  File "/usr/lib/python3.5/importlib/__init__.py", line 126, in import_module
    return _bootstrap._gcd_import(name[level:], package, level)
  File "<frozen importlib._bootstrap>", line 986, in _gcd_import
  File "<frozen importlib._bootstrap>", line 969, in _find_and_load
  File "<frozen importlib._bootstrap>", line 958, in _find_and_load_unlocked
  File "<frozen importlib._bootstrap>", line 666, in _load_unlocked
  File "<frozen importlib._bootstrap>", line 577, in module_from_spec
  File "<frozen importlib._bootstrap_external>", line 914, in create_module
  File "<frozen importlib._bootstrap>", line 222, in _call_with_frames_removed
ImportError: /home/tuco/Desenvolvimento/libmusly_swig/_libmusly.so: undefined symbol: _Z26musly_jukebox_listdecodersv

During handling of the above exception, another exception occurred:

Traceback (most recent call last):
  File "<string>", line 1, in <module>
  File "/home/tuco/Desenvolvimento/libmusly_swig/libmusly.py", line 21, in <module>
    _libmusly = swig_import_helper()
  File "/home/tuco/Desenvolvimento/libmusly_swig/libmusly.py", line 20, in swig_import_helper
    return importlib.import_module('_libmusly')
  File "/usr/lib/python3.5/importlib/__init__.py", line 126, in import_module
    return _bootstrap._gcd_import(name[level:], package, level)
ImportError: /home/tuco/Desenvolvimento/libmusly_swig/_libmusly.so: undefined symbol: _Z26musly_jukebox_listdecodersv

新的 libmusly.py

# This file was automatically generated by SWIG (http://www.swig.org).
# Version 3.0.10
#
# Do not make changes to this file unless you know what you are doing--modify
# the SWIG interface file instead.





from sys import version_info as _swig_python_version_info
if _swig_python_version_info >= (2, 7, 0):
    def swig_import_helper():
        import importlib
        pkg = __name__.rpartition('.')[0]
        mname = '.'.join((pkg, '_libmusly')).lstrip('.')
        try:
            return importlib.import_module(mname)
        except ImportError:
            return importlib.import_module('_libmusly')
    _libmusly = swig_import_helper()
    del swig_import_helper
elif _swig_python_version_info >= (2, 6, 0):
    def swig_import_helper():
        from os.path import dirname
        import imp
        fp = None
        try:
            fp, pathname, description = imp.find_module('_libmusly', [dirname(__file__)])
        except ImportError:
            import _libmusly
            return _libmusly
        if fp is not None:
            try:
                _mod = imp.load_module('_libmusly', fp, pathname, description)
            finally:
                fp.close()
            return _mod
    _libmusly = swig_import_helper()
    del swig_import_helper
else:
    import _libmusly
del _swig_python_version_info
try:
    _swig_property = property
except NameError:
    pass  # Python < 2.2 doesn't have 'property'.

try:
    import builtins as __builtin__
except ImportError:
    import __builtin__

def _swig_setattr_nondynamic(self, class_type, name, value, static=1):
    if (name == "thisown"):
        return self.this.own(value)
    if (name == "this"):
        if type(value).__name__ == 'SwigPyObject':
            self.__dict__[name] = value
            return
    method = class_type.__swig_setmethods__.get(name, None)
    if method:
        return method(self, value)
    if (not static):
        if _newclass:
            object.__setattr__(self, name, value)
        else:
            self.__dict__[name] = value
    else:
        raise AttributeError("You cannot add attributes to %s" % self)


def _swig_setattr(self, class_type, name, value):
    return _swig_setattr_nondynamic(self, class_type, name, value, 0)


def _swig_getattr(self, class_type, name):
    if (name == "thisown"):
        return self.this.own()
    method = class_type.__swig_getmethods__.get(name, None)
    if method:
        return method(self)
    raise AttributeError("'%s' object has no attribute '%s'" % (class_type.__name__, name))


def _swig_repr(self):
    try:
        strthis = "proxy of " + self.this.__repr__()
    except __builtin__.Exception:
        strthis = ""
    return "<%s.%s; %s >" % (self.__class__.__module__, self.__class__.__name__, strthis,)

try:
    _object = object
    _newclass = 1
except __builtin__.Exception:
    class _object:
        pass
    _newclass = 0

class musly_jukebox(_object):
    __swig_setmethods__ = {}
    __setattr__ = lambda self, name, value: _swig_setattr(self, musly_jukebox, name, value)
    __swig_getmethods__ = {}
    __getattr__ = lambda self, name: _swig_getattr(self, musly_jukebox, name)
    __repr__ = _swig_repr
    __swig_setmethods__["method"] = _libmusly.musly_jukebox_method_set
    __swig_getmethods__["method"] = _libmusly.musly_jukebox_method_get
    if _newclass:
        method = _swig_property(_libmusly.musly_jukebox_method_get, _libmusly.musly_jukebox_method_set)
    __swig_setmethods__["method_name"] = _libmusly.musly_jukebox_method_name_set
    __swig_getmethods__["method_name"] = _libmusly.musly_jukebox_method_name_get
    if _newclass:
        method_name = _swig_property(_libmusly.musly_jukebox_method_name_get, _libmusly.musly_jukebox_method_name_set)
    __swig_setmethods__["decoder"] = _libmusly.musly_jukebox_decoder_set
    __swig_getmethods__["decoder"] = _libmusly.musly_jukebox_decoder_get
    if _newclass:
        decoder = _swig_property(_libmusly.musly_jukebox_decoder_get, _libmusly.musly_jukebox_decoder_set)
    __swig_setmethods__["decoder_name"] = _libmusly.musly_jukebox_decoder_name_set
    __swig_getmethods__["decoder_name"] = _libmusly.musly_jukebox_decoder_name_get
    if _newclass:
        decoder_name = _swig_property(_libmusly.musly_jukebox_decoder_name_get, _libmusly.musly_jukebox_decoder_name_set)

    def __init__(self):
        this = _libmusly.new_musly_jukebox()
        try:
            self.this.append(this)
        except __builtin__.Exception:
            self.this = this
    __swig_destroy__ = _libmusly.delete_musly_jukebox
    __del__ = lambda self: None
musly_jukebox_swigregister = _libmusly.musly_jukebox_swigregister
musly_jukebox_swigregister(musly_jukebox)


def musly_version():
    return _libmusly.musly_version()
musly_version = _libmusly.musly_version

def musly_debug(level):
    return _libmusly.musly_debug(level)
musly_debug = _libmusly.musly_debug

def musly_jukebox_listmethods():
    return _libmusly.musly_jukebox_listmethods()
musly_jukebox_listmethods = _libmusly.musly_jukebox_listmethods

#def musly_jukebox_listdecoders():
#    return _libmusly.musly_jukebox_listdecoders()
#musly_jukebox_listdecoders = _libmusly.musly_jukebox_listdecoders

def musly_jukebox_aboutmethod(jukebox):
    return _libmusly.musly_jukebox_aboutmethod(jukebox)
musly_jukebox_aboutmethod = _libmusly.musly_jukebox_aboutmethod

def musly_jukebox_poweron(method, decoder):
    return _libmusly.musly_jukebox_poweron(method, decoder)
musly_jukebox_poweron = _libmusly.musly_jukebox_poweron

def musly_jukebox_similarity(jukebox, seed_track, seed_trackid, tracks, trackids, num_tracks, similarities):
    return _libmusly.musly_jukebox_similarity(jukebox, seed_track, seed_trackid, tracks, trackids, num_tracks, similarities)
musly_jukebox_similarity = _libmusly.musly_jukebox_similarity

def musly_track_alloc(jukebox):
    return _libmusly.musly_track_alloc(jukebox)
musly_track_alloc = _libmusly.musly_track_alloc

def musly_track_free(track):
    return _libmusly.musly_track_free(track)
musly_track_free = _libmusly.musly_track_free

def musly_track_size(jukebox):
    return _libmusly.musly_track_size(jukebox)
musly_track_size = _libmusly.musly_track_size

def musly_track_analyze_audiofile(jukebox, audiofile, excerpt_length, excerpt_start, track):
    return _libmusly.musly_track_analyze_audiofile(jukebox, audiofile, excerpt_length, excerpt_start, track)
musly_track_analyze_audiofile = _libmusly.musly_track_analyze_audiofile
# This file is compatible with both classic and new-style classes.

最佳答案

.i文件中,%{%}之间的代码直接包含在wrapper中,没有任何修改。模块级别的代码由 SWIG 解析并创建包装函数。例如:

%module test

%{
int func(int a, int b) { return a + b; }  // declares a function that will appear in test_wrap.c
%}

int func(int a, int b);  // SWIG creates a wrapper function in test_wrap.c 

您还可以执行 %inline %{ %} 声明和创建包装器:

%module test

%inline %{
int func(int a, int b) { return a + b; }
%}

关于python - SWIG Python C/C++。结果模块是空的,没有错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54446816/

相关文章:

android - 与其他字符相比,c++/Arduino 是否以不同方式处理 '?'

c++ - MFC 扩展 CFileDialog

c - 如何在 C 中将数组转换为二维矩阵而不分配额外的内存?

c - STM32 USB VCP(虚拟串口)

python - 将数据从 `tf.data.Dataset` 分发给多个工作人员(例如 Horovod)

python - 从处理函数中检索数据/异常

Python Tkinter 条目小部件不接受输入

c++ - std::iterator 将其用作类变量时出现段错误

python - PyTorch 函数签名中的\* 是什么意思?

c - 使用 mongo c 驱动程序获取整个集合的游标