Удаление набора символов после сборки (сохраняя только заданный набор) из динамической библиотеки C++ ⇐ Linux
-
Anonymous
Удаление набора символов после сборки (сохраняя только заданный набор) из динамической библиотеки C++
Я создаю динамическую библиотеку (на C++), которая экспортирует ряд символов.
эти экспортированные символы (очевидно) соответствуют API моей библиотеки. однако моя библиотека также экспортирует множество дополнительных символов (которые вообще не упоминаются в моих общедоступных заголовках).
чтобы сохранить небольшой двоичный интерфейс моей библиотеки (предотвратить конфликты имен; не засорять пространство символов; не допустить, чтобы потребители библиотеки случайно использовали символ, который может исчезнуть в следующем выпуске ), я могу сделать экспортированные символы явными с помощью сценария версии, например (на самом деле сценарий версии намного сложнее; в любом случае он включает в себя символы и подстановочные знаки, очищенные от C++):
{ Глобальный: внешний "С++" { /* общедоступный API использует пространство имен «MyLib::» */ MyLib::*; }; /* префиксы изменения версии g++ для 'typeinfo' */ _ЗТИ*; _ЗТФ*; _ЗТС*; /* Префиксы изменения версии g++ для 'vtable' */ _ЗТТ*; _ЗТВ*; местный: *; }; а затем вызвать компоновщик с этими дополнительными флагами -fvisibility=hidden -Wl,--version-script=mylib.ver
круто.
Теперь в библиотеку также входит набор тестов. Чтобы иметь возможность писать краткие тесты, в модульных тестах используются символы, которые не являются частью общедоступного API.
Удаление закрытых символов из библиотеки не позволит модульным тестам (которые связаны с библиотекой) найти эти символы. Поэтому набор тестов становится непригодным для использования.
Один очевидный способ обойти эту проблему — использовать в модульных тестах только общедоступные символы. Но это значительно усложнит модульные тесты, а результаты тестов станет сложнее интерпретировать.
Другим возможным решением было бы использовать не динамическую библиотеку для модульных тестов, а какую-либо другую библиотеку (например, статическую библиотеку или вариант библиотеки, который был связан без скрипта версии). Мне это не нравится, так как я хотел бы протестировать настоящую библиотеку, а не какую-то ее замену.
Третье решение (которое мне нравится больше всего) — отложить удаление закрытых символов до развертывания библиотеки: набор тестов будет использовать сборку локальной библиотеки (со всеми символами), но во время make install (или какой-либо другой явный шаг после сборки) ненужные символы удаляются.
К сожалению, я понятия не имею, как это реализовать.
[*]версия-скрипт — это функция компоновщика. [*]инструменты, такие как strip, делают что-то другое (есть флаг -N для (GNU?) Strip, который должен удалять символ, но, похоже, это не так. работать даже для простой C-библиотеки (без искажения символов C++)
Итак: есть ли способ удалить символы (сохранив только известный набор) из библиотеки на этапе после сборки? И если да, то как это сделать с помощью искажения символов C++ и использования подстановочных знаков?
Я создаю динамическую библиотеку (на C++), которая экспортирует ряд символов.
эти экспортированные символы (очевидно) соответствуют API моей библиотеки. однако моя библиотека также экспортирует множество дополнительных символов (которые вообще не упоминаются в моих общедоступных заголовках).
чтобы сохранить небольшой двоичный интерфейс моей библиотеки (предотвратить конфликты имен; не засорять пространство символов; не допустить, чтобы потребители библиотеки случайно использовали символ, который может исчезнуть в следующем выпуске ), я могу сделать экспортированные символы явными с помощью сценария версии, например (на самом деле сценарий версии намного сложнее; в любом случае он включает в себя символы и подстановочные знаки, очищенные от C++):
{ Глобальный: внешний "С++" { /* общедоступный API использует пространство имен «MyLib::» */ MyLib::*; }; /* префиксы изменения версии g++ для 'typeinfo' */ _ЗТИ*; _ЗТФ*; _ЗТС*; /* Префиксы изменения версии g++ для 'vtable' */ _ЗТТ*; _ЗТВ*; местный: *; }; а затем вызвать компоновщик с этими дополнительными флагами -fvisibility=hidden -Wl,--version-script=mylib.ver
круто.
Теперь в библиотеку также входит набор тестов. Чтобы иметь возможность писать краткие тесты, в модульных тестах используются символы, которые не являются частью общедоступного API.
Удаление закрытых символов из библиотеки не позволит модульным тестам (которые связаны с библиотекой) найти эти символы. Поэтому набор тестов становится непригодным для использования.
Один очевидный способ обойти эту проблему — использовать в модульных тестах только общедоступные символы. Но это значительно усложнит модульные тесты, а результаты тестов станет сложнее интерпретировать.
Другим возможным решением было бы использовать не динамическую библиотеку для модульных тестов, а какую-либо другую библиотеку (например, статическую библиотеку или вариант библиотеки, который был связан без скрипта версии). Мне это не нравится, так как я хотел бы протестировать настоящую библиотеку, а не какую-то ее замену.
Третье решение (которое мне нравится больше всего) — отложить удаление закрытых символов до развертывания библиотеки: набор тестов будет использовать сборку локальной библиотеки (со всеми символами), но во время make install (или какой-либо другой явный шаг после сборки) ненужные символы удаляются.
К сожалению, я понятия не имею, как это реализовать.
[*]версия-скрипт — это функция компоновщика. [*]инструменты, такие как strip, делают что-то другое (есть флаг -N для (GNU?) Strip, который должен удалять символ, но, похоже, это не так. работать даже для простой C-библиотеки (без искажения символов C++)
Итак: есть ли способ удалить символы (сохранив только известный набор) из библиотеки на этапе после сборки? И если да, то как это сделать с помощью искажения символов C++ и использования подстановочных знаков?
Мобильная версия