ObsPluginHelpers.cmake 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444
  1. if(POLICY CMP0087)
  2. cmake_policy(SET CMP0087 NEW)
  3. endif()
  4. set(OBS_STANDALONE_PLUGIN_DIR ${CMAKE_SOURCE_DIR}/release)
  5. set(INCLUDED_LIBOBS_CMAKE_MODULES ON)
  6. include(GNUInstallDirs)
  7. if(${CMAKE_SYSTEM_NAME} STREQUAL "Darwin")
  8. set(OS_MACOS ON)
  9. set(OS_POSIX ON)
  10. elseif(${CMAKE_SYSTEM_NAME} MATCHES "Linux|FreeBSD|OpenBSD")
  11. set(OS_POSIX ON)
  12. string(TOUPPER "${CMAKE_SYSTEM_NAME}" _SYSTEM_NAME_U)
  13. set(OS_${_SYSTEM_NAME_U} ON)
  14. elseif(${CMAKE_SYSTEM_NAME} STREQUAL "Windows")
  15. set(OS_WINDOWS ON)
  16. set(OS_POSIX OFF)
  17. endif()
  18. # Old-Style plugin detected, find "modern" libobs variant instead and set global
  19. # include directories to fix "bad" plugin behaviour
  20. if(DEFINED LIBOBS_INCLUDE_DIR AND NOT TARGET OBS::libobs)
  21. message(
  22. DEPRECATION
  23. "You are using an outdated method of adding 'libobs' to your project. Refer to the updated wiki on how to build and export 'libobs' and use it in your plugin projects."
  24. )
  25. find_package(libobs REQUIRED)
  26. if(TARGET OBS::libobs)
  27. set_target_properties(OBS::libobs PROPERTIES IMPORTED_GLOBAL TRUE)
  28. message(STATUS "OBS: Using modern libobs target")
  29. add_library(libobs ALIAS OBS::libobs)
  30. if(OS_WINDOWS)
  31. add_library(w32-pthreads ALIAS OBS::w32-pthreads)
  32. endif()
  33. endif()
  34. endif()
  35. if(CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT AND (OS_WINDOWS OR OS_MACOS))
  36. set(CMAKE_INSTALL_PREFIX
  37. ${OBS_STANDALONE_PLUGIN_DIR}
  38. CACHE STRING "Directory to install OBS plugin after building" FORCE)
  39. endif()
  40. if(NOT CMAKE_BUILD_TYPE)
  41. set(CMAKE_BUILD_TYPE
  42. "RelWithDebInfo"
  43. CACHE STRING
  44. "OBS build type [Release, RelWithDebInfo, Debug, MinSizeRel]" FORCE)
  45. set_property(CACHE CMAKE_BUILD_TYPE PROPERTY STRINGS Release RelWithDebInfo
  46. Debug MinSizeRel)
  47. endif()
  48. if(NOT QT_VERSION)
  49. set(QT_VERSION
  50. "5"
  51. CACHE STRING "OBS Qt version [5, 6]" FORCE)
  52. set_property(CACHE QT_VERSION PROPERTY STRINGS 5 6)
  53. endif()
  54. macro(find_qt)
  55. set(oneValueArgs VERSION)
  56. set(multiValueArgs COMPONENTS COMPONENTS_WIN COMPONENTS_MAC COMPONENTS_LINUX)
  57. cmake_parse_arguments(FIND_QT "" "${oneValueArgs}" "${multiValueArgs}"
  58. ${ARGN})
  59. if(OS_WINDOWS)
  60. find_package(
  61. Qt${FIND_QT_VERSION}
  62. COMPONENTS ${FIND_QT_COMPONENTS} ${FIND_QT_COMPONENTS_WIN}
  63. REQUIRED)
  64. elseif(OS_MACOS)
  65. find_package(
  66. Qt${FIND_QT_VERSION}
  67. COMPONENTS ${FIND_QT_COMPONENTS} ${FIND_QT_COMPONENTS_MAC}
  68. REQUIRED)
  69. else()
  70. find_package(
  71. Qt${FIND_QT_VERSION}
  72. COMPONENTS ${FIND_QT_COMPONENTS} ${FIND_QT_COMPONENTS_LINUX}
  73. REQUIRED)
  74. endif()
  75. foreach(_COMPONENT IN LISTS FIND_QT_COMPONENTS FIND_QT_COMPONENTS_WIN
  76. FIND_QT_COMPONENTS_MAC FIND_QT_COMPONENTS_LINUX)
  77. if(NOT TARGET Qt::${_COMPONENT} AND TARGET
  78. Qt${FIND_QT_VERSION}::${_COMPONENT})
  79. add_library(Qt::${_COMPONENT} INTERFACE IMPORTED)
  80. set_target_properties(
  81. Qt::${_COMPONENT} PROPERTIES INTERFACE_LINK_LIBRARIES
  82. "Qt${FIND_QT_VERSION}::${_COMPONENT}")
  83. endif()
  84. endforeach()
  85. endmacro()
  86. find_qt(VERSION ${QT_VERSION} COMPONENTS Widgets Core)
  87. file(RELATIVE_PATH RELATIVE_INSTALL_PATH ${CMAKE_SOURCE_DIR}
  88. ${CMAKE_INSTALL_PREFIX})
  89. file(RELATIVE_PATH RELATIVE_BUILD_PATH ${CMAKE_SOURCE_DIR} ${CMAKE_BINARY_DIR})
  90. # Set up OS-specific environment and helper functions
  91. if(OS_MACOS)
  92. find_program(CCACHE_PROGRAM "ccache")
  93. set(CCACHE_SUPPORT
  94. ON
  95. CACHE BOOL "Enable ccache support")
  96. mark_as_advanced(CCACHE_PROGRAM)
  97. if(CCACHE_PROGRAM AND CCACHE_SUPPORT)
  98. set(CMAKE_CXX_COMPILER_LAUNCHER ${CCACHE_PROGRAM})
  99. set(CMAKE_C_COMPILER_LAUNCHER ${CCACHE_PROGRAM})
  100. set(CMAKE_OBJC_COMPILER_LAUNCHER ${CCACHE_PROGRAM})
  101. set(CMAKE_OBJCXX_COMPILER_LAUNCHER ${CCACHE_PROGRAM})
  102. set(CMAKE_CUDA_COMPILER_LAUNCHER ${CCACHE_PROGRAM}) # CMake 3.9+
  103. endif()
  104. set(CMAKE_OSX_ARCHITECTURES
  105. "x86_64"
  106. CACHE STRING
  107. "OBS build architecture for macOS - x86_64 required at least")
  108. set_property(CACHE CMAKE_OSX_ARCHITECTURES PROPERTY STRINGS x86_64 arm64
  109. "x86_64;arm64")
  110. set(CMAKE_OSX_DEPLOYMENT_TARGET
  111. "10.13"
  112. CACHE STRING "OBS deployment target for macOS - 10.15+ required")
  113. set_property(CACHE CMAKE_OSX_DEPLOYMENT_TARGET PROPERTY STRINGS 10.15 11 12)
  114. set(OBS_BUNDLE_CODESIGN_IDENTITY
  115. "-"
  116. CACHE STRING "OBS code signing identity for macOS")
  117. set(OBS_CODESIGN_ENTITLEMENTS
  118. ${CMAKE_SOURCE_DIR}/cmake/bundle/macos/entitlements.plist
  119. CACHE INTERNAL "Path to codesign entitlements plist")
  120. set(OBS_CODESIGN_LINKER
  121. ON
  122. CACHE BOOL "Enable linker code-signing on macOS (macOS 11+ required)")
  123. # Xcode configuration
  124. if(XCODE)
  125. # Tell Xcode to pretend the linker signed binaries so that editing with
  126. # install_name_tool preserves ad-hoc signatures. This option is supported by
  127. # codesign on macOS 11 or higher. See CMake Issue 21854:
  128. # https://gitlab.kitware.com/cmake/cmake/-/issues/21854
  129. set(CMAKE_XCODE_GENERATE_SCHEME ON)
  130. if(OBS_CODESIGN_LINKER)
  131. set(CMAKE_XCODE_ATTRIBUTE_OTHER_CODE_SIGN_FLAGS "-o linker-signed")
  132. endif()
  133. endif()
  134. # Set default options for bundling on macOS
  135. set(CMAKE_MACOSX_RPATH ON)
  136. set(CMAKE_SKIP_BUILD_RPATH OFF)
  137. set(CMAKE_BUILD_WITH_INSTALL_RPATH OFF)
  138. set(CMAKE_INSTALL_RPATH "@executable_path/../Frameworks/")
  139. set(CMAKE_INSTALL_RPATH_USE_LINK_PATH OFF)
  140. function(setup_plugin_target target)
  141. if(NOT DEFINED MACOSX_PLUGIN_GUI_IDENTIFIER)
  142. message(
  143. FATAL_ERROR
  144. "No 'MACOSX_PLUGIN_GUI_IDENTIFIER' set, but is required to build plugin bundles on macOS - example: 'com.yourname.pluginname'"
  145. )
  146. endif()
  147. if(NOT DEFINED MACOSX_PLUGIN_BUNDLE_VERSION)
  148. message(
  149. FATAL_ERROR
  150. "No 'MACOSX_PLUGIN_BUNDLE_VERSION' set, but is required to build plugin bundles on macOS - example: '25'"
  151. )
  152. endif()
  153. if(NOT DEFINED MACOSX_PLUGIN_SHORT_VERSION_STRING)
  154. message(
  155. FATAL_ERROR
  156. "No 'MACOSX_PLUGIN_SHORT_VERSION_STRING' set, but is required to build plugin bundles on macOS - example: '1.0.2'"
  157. )
  158. endif()
  159. set(MACOSX_PLUGIN_BUNDLE_NAME
  160. "${target}"
  161. PARENT_SCOPE)
  162. set(MACOSX_PLUGIN_BUNDLE_VERSION
  163. "${MACOSX_BUNDLE_BUNDLE_VERSION}"
  164. PARENT_SCOPE)
  165. set(MACOSX_PLUGIN_SHORT_VERSION_STRING
  166. "${MACOSX_BUNDLE_SHORT_VERSION_STRING}"
  167. PARENT_SCOPE)
  168. set(MACOSX_PLUGIN_EXECUTABLE_NAME
  169. "${target}"
  170. PARENT_SCOPE)
  171. set(MACOSX_PLUGIN_BUNDLE_TYPE
  172. "BNDL"
  173. PARENT_SCOPE)
  174. install(
  175. TARGETS ${target}
  176. LIBRARY DESTINATION "."
  177. COMPONENT obs_plugins
  178. NAMELINK_COMPONENT ${target}_Development)
  179. set(_COMMAND
  180. "${CMAKE_INSTALL_NAME_TOOL} \\
  181. -change ${CMAKE_PREFIX_PATH}/lib/QtWidgets.framework/Versions/${QT_VERSION}/QtWidgets @rpath/QtWidgets.framework/Versions/${QT_VERSION}/QtWidgets \\
  182. -change ${CMAKE_PREFIX_PATH}/lib/QtCore.framework/Versions/${QT_VERSION}/QtCore @rpath/QtCore.framework/Versions/${QT_VERSION}/QtCore \\
  183. -change ${CMAKE_PREFIX_PATH}/lib/QtGui.framework/Versions/${QT_VERSION}/QtGui @rpath/QtGui.framework/Versions/${QT_VERSION}/QtGui \\
  184. \\\"\${CMAKE_INSTALL_PREFIX}/${target}.plugin/Contents/MacOS/${target}\\\""
  185. )
  186. install(CODE "execute_process(COMMAND /bin/sh -c \"${_COMMAND}\")"
  187. COMPONENT obs_plugins)
  188. if(NOT XCODE)
  189. set(_COMMAND
  190. "/usr/bin/codesign --force \\
  191. --sign \\\"${OBS_BUNDLE_CODESIGN_IDENTITY}\\\" \\
  192. --options runtime \\
  193. --entitlements \\\"${CMAKE_CURRENT_FUNCTION_LIST_DIR}/bundle/macOS/entitlements.plist\\\" \\
  194. \\\"${CMAKE_INSTALL_PREFIX}/${target}.plugin\\\"")
  195. install(CODE "execute_process(COMMAND /bin/sh -c \"${_COMMAND}\")"
  196. COMPONENT obs_plugins)
  197. endif()
  198. set_target_properties(
  199. ${target}
  200. PROPERTIES
  201. BUNDLE ON
  202. BUNDLE_EXTENSION "plugin"
  203. OUTPUT_NAME ${target}
  204. MACOSX_BUNDLE_INFO_PLIST
  205. "${CMAKE_CURRENT_FUNCTION_LIST_DIR}/bundle/macOS/Plugin-Info.plist.in"
  206. XCODE_ATTRIBUTE_PRODUCT_BUNDLE_IDENTIFIER
  207. "${MACOSX_PLUGIN_GUI_IDENTIFIER}"
  208. XCODE_ATTRIBUTE_CODE_SIGN_IDENTITY "${OBS_BUNDLE_CODESIGN_IDENTITY}"
  209. XCODE_ATTRIBUTE_CODE_SIGN_ENTITLEMENTS
  210. "${CMAKE_CURRENT_FUNCTION_LIST_DIR}/bundle/macOS/entitlements.plist")
  211. add_custom_command(
  212. TARGET ${target}
  213. POST_BUILD
  214. COMMAND
  215. /bin/sh -c
  216. "codesign --force --sign \"-\" $<$<BOOL:${OBS_CODESIGN_LINKER}>:--options linker-signed >\"$<TARGET_BUNDLE_DIR:${target}>\""
  217. COMMENT "Codesigning ${target}"
  218. VERBATIM)
  219. install_bundle_resources(${target})
  220. endfunction()
  221. function(install_bundle_resources target)
  222. if(EXISTS ${CMAKE_CURRENT_SOURCE_DIR}/data)
  223. file(GLOB_RECURSE _DATA_FILES "${CMAKE_CURRENT_SOURCE_DIR}/data/*")
  224. foreach(_DATA_FILE IN LISTS _DATA_FILES)
  225. file(RELATIVE_PATH _RELATIVE_PATH ${CMAKE_CURRENT_SOURCE_DIR}/data/
  226. ${_DATA_FILE})
  227. get_filename_component(_RELATIVE_PATH ${_RELATIVE_PATH} PATH)
  228. target_sources(${target} PRIVATE ${_DATA_FILE})
  229. set_source_files_properties(
  230. ${_DATA_FILE} PROPERTIES MACOSX_PACKAGE_LOCATION
  231. Resources/${_RELATIVE_PATH})
  232. string(REPLACE "\\" "\\\\" _GROUP_NAME ${_RELATIVE_PATH})
  233. source_group("Resources\\${_GROUP_NAME}" FILES ${_DATA_FILE})
  234. endforeach()
  235. endif()
  236. endfunction()
  237. else()
  238. if(CMAKE_SIZEOF_VOID_P EQUAL 8)
  239. set(_ARCH_SUFFIX 64)
  240. else()
  241. set(_ARCH_SUFFIX 32)
  242. endif()
  243. set(OBS_OUTPUT_DIR ${CMAKE_BINARY_DIR}/rundir)
  244. if(OS_POSIX)
  245. find_program(CCACHE_PROGRAM "ccache")
  246. set(CCACHE_SUPPORT
  247. ON
  248. CACHE BOOL "Enable ccache support")
  249. mark_as_advanced(CCACHE_PROGRAM)
  250. if(CCACHE_PROGRAM AND CCACHE_SUPPORT)
  251. set(CMAKE_CXX_COMPILER_LAUNCHER ${CCACHE_PROGRAM})
  252. set(CMAKE_C_COMPILER_LAUNCHER ${CCACHE_PROGRAM})
  253. set(CMAKE_OBJC_COMPILER_LAUNCHER ${CCACHE_PROGRAM})
  254. set(CMAKE_OBJCXX_COMPILER_LAUNCHER ${CCACHE_PROGRAM})
  255. set(CMAKE_CUDA_COMPILER_LAUNCHER ${CCACHE_PROGRAM}) # CMake 3.9+
  256. endif()
  257. option(LINUX_PORTABLE "Build portable version (Linux)" OFF)
  258. if(NOT LINUX_PORTABLE)
  259. set(OBS_LIBRARY_DESTINATION ${CMAKE_INSTALL_LIBDIR})
  260. set(OBS_PLUGIN_DESTINATION ${OBS_LIBRARY_DESTINATION}/obs-plugins)
  261. set(CMAKE_INSTALL_RPATH ${CMAKE_INSTALL_PREFIX}/lib)
  262. set(OBS_DATA_DESTINATION ${CMAKE_INSTALL_DATAROOTDIR}/obs)
  263. else()
  264. set(OBS_LIBRARY_DESTINATION bin/${_ARCH_SUFFIX}bit)
  265. set(OBS_PLUGIN_DESTINATION obs-plugins/${_ARCH_SUFFIX}bit)
  266. set(CMAKE_INSTALL_RPATH
  267. "$ORIGIN/" "${CMAKE_INSTALL_PREFIX}/${OBS_LIBRARY_DESTINATION}")
  268. set(OBS_DATA_DESTINATION "data")
  269. endif()
  270. if(OS_LINUX)
  271. set(CPACK_PACKAGE_NAME "${CMAKE_PROJECT_NAME}")
  272. set(CPACK_DEBIAN_PACKAGE_MAINTAINER "${LINUX_MAINTAINER_EMAIL}")
  273. set(CPACK_PACKAGE_VERSION "${CMAKE_PROJECT_VERSION}")
  274. set(CPACK_GENERATOR "DEB")
  275. set(CPACK_DEBIAN_PACKAGE_DEPENDS
  276. "obs-studio (>= 27.0.0), libqt5core5a (>= 5.9.0~beta), libqt5gui5 (>= 5.3.0), libqt5widgets5 (>= 5.7.0)"
  277. )
  278. set(CPACK_OUTPUT_FILE_PREFIX ${CMAKE_SOURCE_DIR}/release)
  279. if(NOT LINUX_PORTABLE)
  280. set(CPACK_SET_DESTDIR ON)
  281. endif()
  282. include(CPack)
  283. endif()
  284. else()
  285. set(OBS_LIBRARY_DESTINATION "bin/${_ARCH_SUFFIX}bit")
  286. set(OBS_LIBRARY32_DESTINATION "bin/32bit")
  287. set(OBS_LIBRARY64_DESTINATION "bin/64bit")
  288. set(OBS_PLUGIN_DESTINATION "obs-plugins/${_ARCH_SUFFIX}bit")
  289. set(OBS_PLUGIN32_DESTINATION "obs-plugins/32bit")
  290. set(OBS_PLUGIN64_DESTINATION "obs-plugins/64bit")
  291. set(OBS_DATA_DESTINATION "data")
  292. endif()
  293. function(setup_plugin_target target)
  294. set_target_properties(${target} PROPERTIES PREFIX "")
  295. install(
  296. TARGETS ${target}
  297. RUNTIME DESTINATION "${OBS_PLUGIN_DESTINATION}"
  298. COMPONENT ${target}_Runtime
  299. LIBRARY DESTINATION "${OBS_PLUGIN_DESTINATION}"
  300. COMPONENT ${target}_Runtime
  301. NAMELINK_COMPONENT ${target}_Development)
  302. install(
  303. FILES $<TARGET_FILE:${target}>
  304. DESTINATION $<CONFIG>/${OBS_PLUGIN_DESTINATION}
  305. COMPONENT obs_rundir
  306. EXCLUDE_FROM_ALL)
  307. if(OS_WINDOWS)
  308. install(
  309. FILES $<TARGET_PDB_FILE:${target}>
  310. CONFIGURATIONS "RelWithDebInfo" "Debug"
  311. DESTINATION ${OBS_PLUGIN_DESTINATION}
  312. COMPONENT ${target}_Runtime
  313. OPTIONAL)
  314. install(
  315. FILES $<TARGET_PDB_FILE:${target}>
  316. CONFIGURATIONS "RelWithDebInfo" "Debug"
  317. DESTINATION $<CONFIG>/${OBS_PLUGIN_DESTINATION}
  318. COMPONENT obs_rundir
  319. OPTIONAL EXCLUDE_FROM_ALL)
  320. endif()
  321. if(MSVC)
  322. target_link_options(
  323. ${target}
  324. PRIVATE
  325. "LINKER:/OPT:REF"
  326. "$<$<NOT:$<EQUAL:${CMAKE_SIZEOF_VOID_P},8>>:LINKER\:/SAFESEH\:NO>"
  327. "$<$<CONFIG:DEBUG>:LINKER\:/INCREMENTAL:NO>"
  328. "$<$<CONFIG:RELWITHDEBINFO>:LINKER\:/INCREMENTAL:NO>")
  329. endif()
  330. setup_target_resources(${target} obs-plugins/${target})
  331. if(OS_WINDOWS AND DEFINED OBS_BUILD_DIR)
  332. setup_target_for_testing(${target} obs-plugins/${target})
  333. endif()
  334. add_custom_command(
  335. TARGET ${target}
  336. POST_BUILD
  337. COMMAND
  338. "${CMAKE_COMMAND}" -DCMAKE_INSTALL_PREFIX=${OBS_OUTPUT_DIR}
  339. -DCMAKE_INSTALL_COMPONENT=obs_rundir
  340. -DCMAKE_INSTALL_CONFIG_NAME=$<CONFIG> -P
  341. ${CMAKE_CURRENT_BINARY_DIR}/cmake_install.cmake
  342. COMMENT "Installing to plugin rundir"
  343. VERBATIM)
  344. endfunction()
  345. function(setup_target_resources target destination)
  346. if(EXISTS ${CMAKE_CURRENT_SOURCE_DIR}/data)
  347. install(
  348. DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}/data/
  349. DESTINATION ${OBS_DATA_DESTINATION}/${destination}
  350. USE_SOURCE_PERMISSIONS
  351. COMPONENT obs_plugins)
  352. install(
  353. DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}/data
  354. DESTINATION $<CONFIG>/${OBS_DATA_DESTINATION}/${destination}
  355. USE_SOURCE_PERMISSIONS
  356. COMPONENT obs_rundir
  357. EXCLUDE_FROM_ALL)
  358. endif()
  359. endfunction()
  360. if(OS_WINDOWS)
  361. function(setup_target_for_testing target destination)
  362. install(
  363. FILES $<TARGET_FILE:${target}>
  364. DESTINATION $<CONFIG>/${OBS_PLUGIN_DESTINATION}
  365. COMPONENT obs_testing
  366. EXCLUDE_FROM_ALL)
  367. install(
  368. FILES $<TARGET_PDB_FILE:${target}>
  369. CONFIGURATIONS "RelWithDebInfo" "Debug"
  370. DESTINATION $<CONFIG>/${OBS_PLUGIN_DESTINATION}
  371. COMPONENT obs_testing
  372. OPTIONAL EXCLUDE_FROM_ALL)
  373. install(
  374. DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}/data/
  375. DESTINATION $<CONFIG>/${OBS_DATA_DESTINATION}/${destination}
  376. USE_SOURCE_PERMISSIONS
  377. COMPONENT obs_testing
  378. EXCLUDE_FROM_ALL)
  379. add_custom_command(
  380. TARGET ${target}
  381. POST_BUILD
  382. COMMAND
  383. "${CMAKE_COMMAND}" -DCMAKE_INSTALL_PREFIX=${OBS_BUILD_DIR}/rundir
  384. -DCMAKE_INSTALL_COMPONENT=obs_testing
  385. -DCMAKE_INSTALL_CONFIG_NAME=$<CONFIG> -P
  386. ${CMAKE_CURRENT_BINARY_DIR}/cmake_install.cmake
  387. COMMENT "Installing to OBS test directory"
  388. VERBATIM)
  389. endfunction()
  390. endif()
  391. endif()