Swig - привязка метода, возвращающего ссылку на указатель класса директора - PullRequest
2 голосов
/ 13 марта 2019

Хорошо, название немного длинное, но я не могу найти более короткое :) Итак, позвольте мне объяснить.

У меня есть кодовая база C ++, где у нас есть несколько контейнерных классов. Эти классы имеют методы доступа, возвращающие элементы по ссылке или константной ссылке. Затем в другой части кодовой базы у меня выделены контейнеры с кучей object, которые используют функцию директора.

И я не могу понять, как специализировать наши контейнерные классы для object: для всех методов контейнера, возвращающих ссылки на элементы (в данном случае ссылка на указатель object), Swig генерирует оболочку код, который не компилируется из-за недопустимого dynamic_cast (в основном он пытается преобразовать object ** в Swig::Director *)

Мне удалось воспроизвести проблему с помощью следующего кода.

test.h

#ifndef TEST_H
#define TEST_H

template< typename T >
class Vector
{

public:

    inline Vector(void)
        : m_Data(nullptr)
        , m_Size(0)
    {
    }

    inline ~Vector(void)
    {
        delete [] m_Data;
    }

    inline void add(const T & item)
    {
        T * data = new T [m_Size + 1];
        for (int i = 0; i < m_Size; ++i)
        {
            data[i] = std::move(m_Data[i]);
        }
        delete [] m_Data;
        m_Data = data;
        m_Data[m_Size++] = item;
    }

    inline const T& item(int index) const
    {
        return m_Data[index];
    }

    inline int count(void) const
    {
        return m_Size;
    }

private:

    T * m_Data;
    int m_Size;

};

class Foo
{

public:

    Foo(void) = default;
    virtual ~Foo(void) = default;

    virtual const char * method(void) const
    {
        return "Foo::method";
    }

};

class Cache
{

public:

    static void add(Foo * item = nullptr)
    {
        m_Cache.add(item == nullptr ? new Foo() : item);
    }

    static const Vector< Foo * > & get(void)
    {
        return m_Cache;
    }

    static Foo * get(int index)
    {
        return m_Cache.item(index);
    }

private:

    static Vector< Foo * > m_Cache;

};

Vector< Foo * > Cache::m_Cache;

#endif // TEST_H

core.i

%module(directors="1") core

// we want to be able to inherit Foo in Python
%feature("director") Foo;

// generate wrappers
%include "Test.h"

// specialize Vector for Foo
%template(FooVector) Vector<Foo*>;

// when compiling the wrapper code, include those
%{
#include "Test.h"
%}

Если вы сгенерируете модуль Python (swig.exe -python -c++ core.i), он работает нормально, но сгенерированный файл core_wrap.cxx не может быть собран, поскольку сгенерированный код оболочки для Vector::item содержит недопустимый dynamic_cast от Foo ** до Swig::Director *

Неправильная строка (где результат типа Foo **)

  director = SWIG_DIRECTOR_CAST(result);

И если я вручную исправлю это так:

  director = SWIG_DIRECTOR_CAST(*result);

Затем модуль правильно компилируется, и все работает нормально.

Так что в основном мой вопрос: это ошибка в Swig? Я делаю что-то неправильно? Есть ли обходной путь, чтобы сказать Swig правильно разыменовать мой Foo ** результат перед приведением к Swig::Director *?

Любая помощь приветствуется:)

1 Ответ

1 голос
/ 20 марта 2019

Я нашел решение вашей проблемы.

В некоторых ситуациях использование функции директора и собственного RTTI приводит к коду, который не компилируется из коробки, например, при использовании distutils из Python.

Решение, которое работает для вашего примера, заключается в определении SWIG_DIRECTOR_NORTTI при компиляции кода, например, для компилятора GNU добавьте -DSWIG_DIRECTOR_NORTTI к CXXFLAGS

...