alias.hpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. // Copyright 2014 Renato Tegon Forti, Antony Polukhin.
  2. // Copyright 2015 Antony Polukhin.
  3. //
  4. // Distributed under the Boost Software License, Version 1.0.
  5. // (See accompanying file LICENSE_1_0.txt
  6. // or copy at http://www.boost.org/LICENSE_1_0.txt)
  7. #ifndef BOOST_DLL_ALIAS_HPP
  8. #define BOOST_DLL_ALIAS_HPP
  9. #include <boost/config.hpp>
  10. #include <boost/static_assert.hpp>
  11. #include <boost/predef/compiler.h>
  12. #include <boost/predef/os.h>
  13. #include <boost/dll/detail/aggressive_ptr_cast.hpp>
  14. #ifdef BOOST_HAS_PRAGMA_ONCE
  15. # pragma once
  16. #endif
  17. /// \file boost/dll/alias.hpp
  18. /// \brief Includes alias methods and macro. You can include this header or
  19. /// boost/dll/shared_library.hpp to reduce dependencies
  20. /// in case you do not use the refcountable functions.
  21. namespace boost { namespace dll {
  22. #ifdef BOOST_DLL_DOXYGEN
  23. /// Define this macro to explicitly specify translation unit in which alias must be instantiated.
  24. /// See section 'Limitations' for more info. You may find usage examples in source codes of almost each tutorial.
  25. /// Must be used in code, when \forcedmacrolink{BOOST_DLL_FORCE_NO_WEAK_EXPORTS} is defined
  26. #define BOOST_DLL_FORCE_ALIAS_INSTANTIATION
  27. /// Define this macro to disable exporting weak symbols and start using the \forcedmacrolink{BOOST_DLL_FORCE_ALIAS_INSTANTIATION}.
  28. /// This may be usefull for working around linker problems or to test your program for compatability with linkers that do not support export of weak symbols.
  29. #define BOOST_DLL_FORCE_NO_WEAK_EXPORTS
  30. #endif
  31. #if BOOST_COMP_MSVC
  32. #define BOOST_DLL_SELECTANY __declspec(selectany)
  33. #define BOOST_DLL_SECTION(SectionName, Permissions) \
  34. BOOST_STATIC_ASSERT_MSG( \
  35. sizeof(#SectionName) < 10, \
  36. "Some platforms require section names to be at most 8 bytest" \
  37. ); \
  38. __pragma(section(#SectionName, Permissions)) __declspec(allocate(#SectionName)) \
  39. /**/
  40. #else // #if BOOST_COMP_MSVC
  41. #if BOOST_OS_WINDOWS || BOOST_OS_ANDROID || BOOST_COMP_IBM
  42. // There are some problems with mixing `__dllexport__` and `weak` using MinGW
  43. // See https://sourceware.org/bugzilla/show_bug.cgi?id=17480
  44. //
  45. // Android had an issue with exporting weak symbols
  46. // https://code.google.com/p/android/issues/detail?id=70206
  47. #define BOOST_DLL_SELECTANY
  48. #else // #if BOOST_OS_WINDOWS
  49. /*!
  50. * \brief Macro that allows linker to select any occurrence of this symbol instead of
  51. * failing with 'multiple definitions' error at linktime.
  52. *
  53. * This macro does not work on Android, IBM XL C/C++ and MinGW+Windows
  54. * because of linker problems with exporting weak symbols
  55. * (See https://code.google.com/p/android/issues/detail?id=70206, https://sourceware.org/bugzilla/show_bug.cgi?id=17480)
  56. */
  57. #define BOOST_DLL_SELECTANY __attribute__((weak))
  58. #endif // #if BOOST_OS_WINDOWS
  59. // TODO: improve section permissions using following info:
  60. // http://stackoverflow.com/questions/6252812/what-does-the-aw-flag-in-the-section-attribute-mean
  61. #if !BOOST_OS_MACOS && !BOOST_OS_IOS
  62. /*!
  63. * \brief Macro that puts symbol to a specific section. On MacOS all the sections are put into "__DATA" segment.
  64. * \param SectionName Name of the section. Must be a valid C identifier without quotes not longer than 8 bytes.
  65. * \param Permissions Can be "read" or "write" (without quotes!).
  66. */
  67. #define BOOST_DLL_SECTION(SectionName, Permissions) \
  68. BOOST_STATIC_ASSERT_MSG( \
  69. sizeof(#SectionName) < 10, \
  70. "Some platforms require section names to be at most 8 bytest" \
  71. ); \
  72. __attribute__ ((section (#SectionName))) \
  73. /**/
  74. #else // #if !BOOST_OS_MACOS && !BOOST_OS_IOS
  75. #define BOOST_DLL_SECTION(SectionName, Permissions) \
  76. BOOST_STATIC_ASSERT_MSG( \
  77. sizeof(#SectionName) < 10, \
  78. "Some platforms require section names to be at most 8 bytest" \
  79. ); \
  80. __attribute__ ((section ( "__DATA," #SectionName))) \
  81. /**/
  82. #endif // #if #if !BOOST_OS_MACOS && !BOOST_OS_IOS
  83. #endif // #if BOOST_COMP_MSVC
  84. // Alias - is just a variable that pointers to original data
  85. //
  86. // A few attempts were made to avoid additional indirection:
  87. // 1)
  88. // // Does not work on Windows, work on Linux
  89. // extern "C" BOOST_SYMBOL_EXPORT void AliasName() {
  90. // reinterpret_cast<void (*)()>(Function)();
  91. // }
  92. //
  93. // 2)
  94. // // Does not work on Linux (changes permissions of .text section and produces incorrect DSO)
  95. // extern "C" BOOST_SYMBOL_EXPORT void* __attribute__ ((section(".text#")))
  96. // func_ptr = *reinterpret_cast<std::ptrdiff_t**>(&foo::bar);
  97. //
  98. // 3) // requires mangled name of `Function`
  99. // // AliasName() __attribute__ ((weak, alias ("Function")))
  100. //
  101. // // hard to use
  102. // `#pragma comment(linker, "/alternatename:_pWeakValue=_pDefaultWeakValue")`
  103. /*!
  104. * \brief Makes an alias name for exported function or variable.
  105. *
  106. * This macro is useful in cases of long mangled C++ names. For example some `void boost::foo(std::sting)`
  107. * function name will change to something like `N5boostN3foosE` after mangling.
  108. * Importing function by `N5boostN3foosE` name does not looks user friendly, especially assuming the fact
  109. * that different compilers have different mangling schemes. AliasName is the name that won't be mangled
  110. * and can be used as a portable import name.
  111. *
  112. *
  113. * Can be used in any namespace, including global. FunctionOrVar must be fully qualified,
  114. * so that address of it could be taken. Multiple different aliases for a single variable/function
  115. * are allowed.
  116. *
  117. * Make sure that AliasNames are unique per library/executable. Functions or variables
  118. * in global namespace must not have names same as AliasNames.
  119. *
  120. * Same AliasName in different translation units must point to the same FunctionOrVar.
  121. *
  122. * Puts all the aliases into the \b "boostdll" read only section of the binary. Equal to
  123. * \forcedmacrolink{BOOST_DLL_ALIAS_SECTIONED}(FunctionOrVar, AliasName, boostdll).
  124. *
  125. * \param FunctionOrVar Function or variable for which an alias must be made.
  126. * \param AliasName Name of the alias. Must be a valid C identifier.
  127. *
  128. * \b Example:
  129. * \code
  130. * namespace foo {
  131. * void bar(std::string&);
  132. *
  133. * BOOST_DLL_ALIAS(foo::bar, foo_bar)
  134. * }
  135. *
  136. * BOOST_DLL_ALIAS(foo::bar, foo_bar_another_alias_name)
  137. * \endcode
  138. *
  139. * \b See: \forcedmacrolink{BOOST_DLL_ALIAS_SECTIONED} for making alias in a specific section.
  140. */
  141. #define BOOST_DLL_ALIAS(FunctionOrVar, AliasName) \
  142. BOOST_DLL_ALIAS_SECTIONED(FunctionOrVar, AliasName, boostdll) \
  143. /**/
  144. #if ((BOOST_COMP_GNUC && BOOST_OS_WINDOWS) || BOOST_OS_ANDROID || BOOST_COMP_IBM || defined(BOOST_DLL_FORCE_NO_WEAK_EXPORTS)) \
  145. && !defined(BOOST_DLL_FORCE_ALIAS_INSTANTIATION) && !defined(BOOST_DLL_DOXYGEN)
  146. #define BOOST_DLL_ALIAS_SECTIONED(FunctionOrVar, AliasName, SectionName) \
  147. namespace _autoaliases { \
  148. extern "C" BOOST_SYMBOL_EXPORT const void *AliasName; \
  149. } /* namespace _autoaliases */ \
  150. /**/
  151. #define BOOST_DLL_AUTO_ALIAS(FunctionOrVar) \
  152. namespace _autoaliases { \
  153. extern "C" BOOST_SYMBOL_EXPORT const void *FunctionOrVar; \
  154. } /* namespace _autoaliases */ \
  155. /**/
  156. #else
  157. // Note: we can not use `aggressive_ptr_cast` here, because in that case GCC applies
  158. // different permissions to the section and it causes Segmentation fault.
  159. // Note: we can not use `boost::addressof()` here, because in that case GCC
  160. // may optimize away the FunctionOrVar instance and we'll get a pointer to unexisting symbol.
  161. /*!
  162. * \brief Same as \forcedmacrolink{BOOST_DLL_ALIAS} but puts alias name into the user specified section.
  163. *
  164. * \param FunctionOrVar Function or variable for which an alias must be made.
  165. * \param AliasName Name of the alias. Must be a valid C identifier.
  166. * \param SectionName Name of the section. Must be a valid C identifier without quotes not longer than 8 bytes.
  167. *
  168. * \b Example:
  169. * \code
  170. * namespace foo {
  171. * void bar(std::string&);
  172. *
  173. * BOOST_DLL_ALIAS_SECTIONED(foo::bar, foo_bar, sect_1) // section "sect_1" now exports "foo_bar"
  174. * }
  175. * \endcode
  176. *
  177. */
  178. #define BOOST_DLL_ALIAS_SECTIONED(FunctionOrVar, AliasName, SectionName) \
  179. namespace _autoaliases { \
  180. extern "C" BOOST_SYMBOL_EXPORT const void *AliasName; \
  181. BOOST_DLL_SECTION(SectionName, read) BOOST_DLL_SELECTANY \
  182. const void * AliasName = reinterpret_cast<const void*>(reinterpret_cast<intptr_t>( \
  183. &FunctionOrVar \
  184. )); \
  185. } /* namespace _autoaliases */ \
  186. /**/
  187. /*!
  188. * \brief Exports variable or function with unmangled alias name.
  189. *
  190. * This macro is useful in cases of long mangled C++ names. For example some `void boost::foo(std::sting)`
  191. * function name will change to something like `N5boostN3foosE` after mangling.
  192. * Importing function by `N5boostN3foosE` name does not looks user friendly, especially assuming the fact
  193. * that different compilers have different mangling schemes.*
  194. *
  195. * Must be used in scope where FunctionOrVar declared. FunctionOrVar must be a valid C name, which means that
  196. * it must not contain `::`.
  197. *
  198. * Functions or variables
  199. * in global namespace must not have names same as FunctionOrVar.
  200. *
  201. * Puts all the aliases into the \b "boostdll" read only section of the binary. Almost same as
  202. * \forcedmacrolink{BOOST_DLL_ALIAS}(FunctionOrVar, FunctionOrVar).
  203. *
  204. * \param FunctionOrVar Function or variable for which an unmangled alias must be made.
  205. *
  206. * \b Example:
  207. * \code
  208. * namespace foo {
  209. * void bar(std::string&);
  210. * BOOST_DLL_AUTO_ALIAS(bar)
  211. * }
  212. *
  213. * \endcode
  214. *
  215. * \b See: \forcedmacrolink{BOOST_DLL_ALIAS} for making an alias with different names.
  216. */
  217. #define BOOST_DLL_AUTO_ALIAS(FunctionOrVar) \
  218. namespace _autoaliases { \
  219. BOOST_DLL_SELECTANY const void * dummy_ ## FunctionOrVar \
  220. = reinterpret_cast<const void*>(reinterpret_cast<intptr_t>( \
  221. &FunctionOrVar \
  222. )); \
  223. extern "C" BOOST_SYMBOL_EXPORT const void *FunctionOrVar; \
  224. BOOST_DLL_SECTION(boostdll, read) BOOST_DLL_SELECTANY \
  225. const void * FunctionOrVar = dummy_ ## FunctionOrVar; \
  226. } /* namespace _autoaliases */ \
  227. /**/
  228. #endif
  229. }} // namespace boost::dll
  230. #endif // BOOST_DLL_ALIAS_HPP