UseSWIG.cmake 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303
  1. #.rst:
  2. # UseSWIG
  3. # -------
  4. #
  5. # Defines the following macros for use with SWIG:
  6. #
  7. # ::
  8. #
  9. # SWIG_ADD_MODULE(name language [ files ])
  10. # - Define swig module with given name and specified language
  11. # SWIG_LINK_LIBRARIES(name [ libraries ])
  12. # - Link libraries to swig module
  13. #
  14. # Source files properties on module files can be set before the invocation
  15. # of the SWIG_ADD_MODULE macro to specify special behavior of SWIG.
  16. #
  17. # The source file property CPLUSPLUS calls SWIG in c++ mode, e.g.::
  18. #
  19. # set_property(SOURCE mymod.i PROPERTY CPLUSPLUS ON)
  20. # swig_add_module(mymod python mymod.i)
  21. #
  22. # The source file property SWIG_FLAGS adds custom flags to the SWIG executable.
  23. #
  24. # The source-file property SWIG_MODULE_NAME have to be provided to specify the actual
  25. # import name of the module in the target language if it cannot be scanned automatically
  26. # from source or different from the module file basename.::
  27. #
  28. # set_property(SOURCE mymod.i PROPERTY SWIG_MODULE_NAME mymod_realname)
  29. #
  30. # To get the name of the swig module target library, use: ${SWIG_MODULE_${name}_REAL_NAME}.
  31. #
  32. # Also some variables can be set to specify special behavior of SWIG.
  33. #
  34. # CMAKE_SWIG_FLAGS can be used to add special flags to all swig calls.
  35. #
  36. # Another special variable is CMAKE_SWIG_OUTDIR, it allows one to specify
  37. # where to write all the swig generated module (swig -outdir option)
  38. #
  39. # The name-specific variable SWIG_MODULE_<name>_EXTRA_DEPS may be used to specify extra
  40. # dependencies for the generated modules.
  41. #
  42. # If the source file generated by swig need some special flag you can use::
  43. #
  44. # set_source_files_properties( ${swig_generated_file_fullname}
  45. # PROPERTIES COMPILE_FLAGS "-bla")
  46. #=============================================================================
  47. # Copyright 2004-2009 Kitware, Inc.
  48. # Copyright 2009 Mathieu Malaterre <mathieu.malaterre@gmail.com>
  49. #
  50. # Distributed under the OSI-approved BSD License (the "License");
  51. # see accompanying file Copyright.txt for details.
  52. #
  53. # This software is distributed WITHOUT ANY WARRANTY; without even the
  54. # implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  55. # See the License for more information.
  56. #=============================================================================
  57. # (To distribute this file outside of CMake, substitute the full
  58. # License text for the above reference.)
  59. set(SWIG_CXX_EXTENSION "cxx")
  60. set(SWIG_EXTRA_LIBRARIES "")
  61. set(SWIG_PYTHON_EXTRA_FILE_EXTENSION "py")
  62. #
  63. # For given swig module initialize variables associated with it
  64. #
  65. macro(SWIG_MODULE_INITIALIZE name language)
  66. string(TOUPPER "${language}" swig_uppercase_language)
  67. string(TOLOWER "${language}" swig_lowercase_language)
  68. set(SWIG_MODULE_${name}_LANGUAGE "${swig_uppercase_language}")
  69. set(SWIG_MODULE_${name}_SWIG_LANGUAGE_FLAG "${swig_lowercase_language}")
  70. set(SWIG_MODULE_${name}_REAL_NAME "${name}")
  71. if (";${CMAKE_SWIG_FLAGS};" MATCHES ";-noproxy;")
  72. set (SWIG_MODULE_${name}_NOPROXY TRUE)
  73. endif ()
  74. if("x${SWIG_MODULE_${name}_LANGUAGE}" STREQUAL "xUNKNOWN")
  75. message(FATAL_ERROR "SWIG Error: Language \"${language}\" not found")
  76. elseif("x${SWIG_MODULE_${name}_LANGUAGE}" STREQUAL "xPYTHON" AND NOT SWIG_MODULE_${name}_NOPROXY)
  77. # swig will produce a module.py containing an 'import _modulename' statement,
  78. # which implies having a corresponding _modulename.so (*NIX), _modulename.pyd (Win32),
  79. # unless the -noproxy flag is used
  80. set(SWIG_MODULE_${name}_REAL_NAME "_${name}")
  81. elseif("x${SWIG_MODULE_${name}_LANGUAGE}" STREQUAL "xPERL")
  82. set(SWIG_MODULE_${name}_EXTRA_FLAGS "-shadow")
  83. elseif("x${SWIG_MODULE_${name}_LANGUAGE}" STREQUAL "xCSHARP")
  84. # This makes sure that the name used in the generated DllImport
  85. # matches the library name created by CMake
  86. set(SWIG_MODULE_${name}_EXTRA_FLAGS "-dllimport;${name}")
  87. endif()
  88. endmacro()
  89. #
  90. # For a given language, input file, and output file, determine extra files that
  91. # will be generated. This is internal swig macro.
  92. #
  93. macro(SWIG_GET_EXTRA_OUTPUT_FILES language outfiles generatedpath infile)
  94. set(${outfiles} "")
  95. get_source_file_property(SWIG_GET_EXTRA_OUTPUT_FILES_module_basename
  96. ${infile} SWIG_MODULE_NAME)
  97. if(SWIG_GET_EXTRA_OUTPUT_FILES_module_basename STREQUAL "NOTFOUND")
  98. # try to get module name from "%module foo" syntax
  99. if ( EXISTS ${infile} )
  100. file ( STRINGS ${infile} _MODULE_NAME REGEX "[ ]*%module[ ]*[a-zA-Z0-9_]+.*" )
  101. endif ()
  102. if ( _MODULE_NAME )
  103. string ( REGEX REPLACE "[ ]*%module[ ]*([a-zA-Z0-9_]+).*" "\\1" _MODULE_NAME "${_MODULE_NAME}" )
  104. set(SWIG_GET_EXTRA_OUTPUT_FILES_module_basename "${_MODULE_NAME}")
  105. else ()
  106. # try to get module name from "%module (options=...) foo" syntax
  107. if ( EXISTS ${infile} )
  108. file ( STRINGS ${infile} _MODULE_NAME REGEX "[ ]*%module[ ]*\\(.*\\)[ ]*[a-zA-Z0-9_]+.*" )
  109. endif ()
  110. if ( _MODULE_NAME )
  111. string ( REGEX REPLACE "[ ]*%module[ ]*\\(.*\\)[ ]*([a-zA-Z0-9_]+).*" "\\1" _MODULE_NAME "${_MODULE_NAME}" )
  112. set(SWIG_GET_EXTRA_OUTPUT_FILES_module_basename "${_MODULE_NAME}")
  113. else ()
  114. # fallback to file basename
  115. get_filename_component(SWIG_GET_EXTRA_OUTPUT_FILES_module_basename ${infile} NAME_WE)
  116. endif ()
  117. endif ()
  118. endif()
  119. foreach(it ${SWIG_${language}_EXTRA_FILE_EXTENSION})
  120. set(${outfiles} ${${outfiles}}
  121. "${generatedpath}/${SWIG_GET_EXTRA_OUTPUT_FILES_module_basename}.${it}")
  122. endforeach()
  123. endmacro()
  124. #
  125. # Take swig (*.i) file and add proper custom commands for it
  126. #
  127. macro(SWIG_ADD_SOURCE_TO_MODULE name outfiles infile)
  128. set(swig_full_infile ${infile})
  129. get_filename_component(swig_source_file_name_we "${infile}" NAME_WE)
  130. get_source_file_property(swig_source_file_generated ${infile} GENERATED)
  131. get_source_file_property(swig_source_file_cplusplus ${infile} CPLUSPLUS)
  132. get_source_file_property(swig_source_file_flags ${infile} SWIG_FLAGS)
  133. if("${swig_source_file_flags}" STREQUAL "NOTFOUND")
  134. set(swig_source_file_flags "")
  135. endif()
  136. get_filename_component(swig_source_file_fullname "${infile}" ABSOLUTE)
  137. # If CMAKE_SWIG_OUTDIR was specified then pass it to -outdir
  138. if(CMAKE_SWIG_OUTDIR)
  139. set(swig_outdir ${CMAKE_SWIG_OUTDIR})
  140. else()
  141. set(swig_outdir ${CMAKE_CURRENT_BINARY_DIR})
  142. endif()
  143. if (NOT SWIG_MODULE_${name}_NOPROXY)
  144. SWIG_GET_EXTRA_OUTPUT_FILES(${SWIG_MODULE_${name}_LANGUAGE}
  145. swig_extra_generated_files
  146. "${swig_outdir}"
  147. "${swig_source_file_fullname}")
  148. endif()
  149. set(swig_generated_file_fullname
  150. "${swig_outdir}/${swig_source_file_name_we}")
  151. # add the language into the name of the file (i.e. TCL_wrap)
  152. # this allows for the same .i file to be wrapped into different languages
  153. set(swig_generated_file_fullname
  154. "${swig_generated_file_fullname}${SWIG_MODULE_${name}_LANGUAGE}_wrap")
  155. if(swig_source_file_cplusplus)
  156. set(swig_generated_file_fullname
  157. "${swig_generated_file_fullname}.${SWIG_CXX_EXTENSION}")
  158. else()
  159. set(swig_generated_file_fullname
  160. "${swig_generated_file_fullname}.c")
  161. endif()
  162. #message("Full path to source file: ${swig_source_file_fullname}")
  163. #message("Full path to the output file: ${swig_generated_file_fullname}")
  164. get_directory_property(cmake_include_directories INCLUDE_DIRECTORIES)
  165. list(REMOVE_DUPLICATES cmake_include_directories)
  166. set(swig_include_dirs)
  167. foreach(it ${cmake_include_directories})
  168. set(swig_include_dirs ${swig_include_dirs} "-I${it}")
  169. endforeach()
  170. set(swig_special_flags)
  171. # default is c, so add c++ flag if it is c++
  172. if(swig_source_file_cplusplus)
  173. set(swig_special_flags ${swig_special_flags} "-c++")
  174. endif()
  175. set(swig_extra_flags)
  176. if(SWIG_MODULE_${name}_EXTRA_FLAGS)
  177. set(swig_extra_flags ${swig_extra_flags} ${SWIG_MODULE_${name}_EXTRA_FLAGS})
  178. endif()
  179. add_custom_command(
  180. OUTPUT "${swig_generated_file_fullname}" ${swig_extra_generated_files}
  181. # Let's create the ${swig_outdir} at execution time, in case dir contains $(OutDir)
  182. COMMAND ${CMAKE_COMMAND} -E make_directory ${swig_outdir}
  183. COMMAND "${SWIG_EXECUTABLE}"
  184. ARGS "-${SWIG_MODULE_${name}_SWIG_LANGUAGE_FLAG}"
  185. ${swig_source_file_flags}
  186. ${CMAKE_SWIG_FLAGS}
  187. -outdir ${swig_outdir}
  188. ${swig_special_flags}
  189. ${swig_extra_flags}
  190. ${swig_include_dirs}
  191. -o "${swig_generated_file_fullname}"
  192. "${swig_source_file_fullname}"
  193. MAIN_DEPENDENCY "${swig_source_file_fullname}"
  194. DEPENDS ${SWIG_MODULE_${name}_EXTRA_DEPS}
  195. COMMENT "Swig source")
  196. set_source_files_properties("${swig_generated_file_fullname}" ${swig_extra_generated_files}
  197. PROPERTIES GENERATED 1)
  198. set(${outfiles} "${swig_generated_file_fullname}" ${swig_extra_generated_files})
  199. endmacro()
  200. #
  201. # Create Swig module
  202. #
  203. macro(SWIG_ADD_MODULE name language)
  204. SWIG_MODULE_INITIALIZE(${name} ${language})
  205. set(swig_dot_i_sources)
  206. set(swig_other_sources)
  207. foreach(it ${ARGN})
  208. if(${it} MATCHES "\\.i$")
  209. set(swig_dot_i_sources ${swig_dot_i_sources} "${it}")
  210. else()
  211. set(swig_other_sources ${swig_other_sources} "${it}")
  212. endif()
  213. endforeach()
  214. set(swig_generated_sources)
  215. foreach(it ${swig_dot_i_sources})
  216. SWIG_ADD_SOURCE_TO_MODULE(${name} swig_generated_source ${it})
  217. set(swig_generated_sources ${swig_generated_sources} "${swig_generated_source}")
  218. endforeach()
  219. get_directory_property(swig_extra_clean_files ADDITIONAL_MAKE_CLEAN_FILES)
  220. set_directory_properties(PROPERTIES
  221. ADDITIONAL_MAKE_CLEAN_FILES "${swig_extra_clean_files};${swig_generated_sources}")
  222. add_library(${SWIG_MODULE_${name}_REAL_NAME}
  223. MODULE
  224. ${swig_generated_sources}
  225. ${swig_other_sources})
  226. set_target_properties(${SWIG_MODULE_${name}_REAL_NAME} PROPERTIES NO_SONAME ON)
  227. string(TOLOWER "${language}" swig_lowercase_language)
  228. if ("${swig_lowercase_language}" STREQUAL "octave")
  229. set_target_properties(${SWIG_MODULE_${name}_REAL_NAME} PROPERTIES PREFIX "")
  230. set_target_properties(${SWIG_MODULE_${name}_REAL_NAME} PROPERTIES SUFFIX ".oct")
  231. elseif ("${swig_lowercase_language}" STREQUAL "go")
  232. set_target_properties(${SWIG_MODULE_${name}_REAL_NAME} PROPERTIES PREFIX "")
  233. elseif ("${swig_lowercase_language}" STREQUAL "java")
  234. if (APPLE)
  235. # In java you want:
  236. # System.loadLibrary("LIBRARY");
  237. # then JNI will look for a library whose name is platform dependent, namely
  238. # MacOS : libLIBRARY.jnilib
  239. # Windows: LIBRARY.dll
  240. # Linux : libLIBRARY.so
  241. set_target_properties (${SWIG_MODULE_${name}_REAL_NAME} PROPERTIES SUFFIX ".jnilib")
  242. endif ()
  243. elseif ("${swig_lowercase_language}" STREQUAL "lua")
  244. set_target_properties(${SWIG_MODULE_${name}_REAL_NAME} PROPERTIES PREFIX "")
  245. elseif ("${swig_lowercase_language}" STREQUAL "python")
  246. # this is only needed for the python case where a _modulename.so is generated
  247. set_target_properties(${SWIG_MODULE_${name}_REAL_NAME} PROPERTIES PREFIX "")
  248. # Python extension modules on Windows must have the extension ".pyd"
  249. # instead of ".dll" as of Python 2.5. Older python versions do support
  250. # this suffix.
  251. # http://docs.python.org/whatsnew/ports.html#SECTION0001510000000000000000
  252. # <quote>
  253. # Windows: .dll is no longer supported as a filename extension for extension modules.
  254. # .pyd is now the only filename extension that will be searched for.
  255. # </quote>
  256. if(WIN32 AND NOT CYGWIN)
  257. set_target_properties(${SWIG_MODULE_${name}_REAL_NAME} PROPERTIES SUFFIX ".pyd")
  258. endif()
  259. elseif ("${swig_lowercase_language}" STREQUAL "r")
  260. set_target_properties(${SWIG_MODULE_${name}_REAL_NAME} PROPERTIES PREFIX "")
  261. elseif ("${swig_lowercase_language}" STREQUAL "ruby")
  262. # In ruby you want:
  263. # require 'LIBRARY'
  264. # then ruby will look for a library whose name is platform dependent, namely
  265. # MacOS : LIBRARY.bundle
  266. # Windows: LIBRARY.dll
  267. # Linux : LIBRARY.so
  268. set_target_properties (${SWIG_MODULE_${name}_REAL_NAME} PROPERTIES PREFIX "")
  269. if (APPLE)
  270. set_target_properties (${SWIG_MODULE_${name}_REAL_NAME} PROPERTIES SUFFIX ".bundle")
  271. endif ()
  272. endif ()
  273. endmacro()
  274. #
  275. # Like TARGET_LINK_LIBRARIES but for swig modules
  276. #
  277. macro(SWIG_LINK_LIBRARIES name)
  278. if(SWIG_MODULE_${name}_REAL_NAME)
  279. target_link_libraries(${SWIG_MODULE_${name}_REAL_NAME} ${ARGN})
  280. else()
  281. message(SEND_ERROR "Cannot find Swig library \"${name}\".")
  282. endif()
  283. endmacro()