FindFLEX.cmake 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  1. # Distributed under the OSI-approved BSD 3-Clause License. See accompanying
  2. # file Copyright.txt or https://cmake.org/licensing for details.
  3. #[=======================================================================[.rst:
  4. FindFLEX
  5. --------
  6. Find Fast Lexical Analyzer (Flex) executable and provide a macro
  7. to generate custom build rules.
  8. The module defines the following variables:
  9. ``FLEX_FOUND``
  10. True if ``flex`` executable is found.
  11. ``FLEX_EXECUTABLE``
  12. The path to the ``flex`` executable.
  13. ``FLEX_VERSION``
  14. The version of ``flex``.
  15. ``FLEX_LIBRARIES``
  16. The ``flex`` libraries.
  17. ``FLEX_INCLUDE_DIRS``
  18. The path to the ``flex`` headers.
  19. The minimum required version of ``flex`` can be specified using the
  20. standard CMake syntax, e.g. :command:`find_package(FLEX 2.5.13)`.
  21. If ``flex`` is found on the system, the module defines the macro:
  22. .. command:: flex_target
  23. .. code-block:: cmake
  24. flex_target(<Name> <FlexInput> <FlexOutput>
  25. [COMPILE_FLAGS <string>]
  26. [DEFINES_FILE <string>]
  27. )
  28. which creates a custom command to generate the ``<FlexOutput>`` file from
  29. the ``<FlexInput>`` file. ``<Name>`` is an alias used to get details of this
  30. custom command.
  31. The options are:
  32. ``COMPILE_FLAGS <string>``
  33. Space-separated flex options added to the ``flex`` command line.
  34. A :ref:`;-list <CMake Language Lists>` will not work.
  35. ``DEFINES_FILE <string>``
  36. .. versionadded:: 3.5
  37. If flex is configured to output a header file, this option may be used to
  38. specify its name.
  39. .. versionchanged:: 3.17
  40. When :policy:`CMP0098` is set to ``NEW``, ``flex`` runs in the
  41. :variable:`CMAKE_CURRENT_BINARY_DIR` directory.
  42. The macro defines the following variables:
  43. ``FLEX_<Name>_DEFINED``
  44. True if the macro ran successfully.
  45. ``FLEX_<Name>_OUTPUTS``
  46. The source file generated by the custom rule, an alias for ``<FlexOutput>``.
  47. ``FLEX_<Name>_INPUT``
  48. The flex source file, an alias for ``<FlexInput>``.
  49. ``FLEX_<Name>_OUTPUT_HEADER``
  50. The header flex output, if any.
  51. Flex scanners often use tokens defined by Bison: the code generated
  52. by Flex depends of the header generated by Bison. This module also
  53. defines a macro:
  54. .. command:: add_flex_bison_dependency
  55. .. code-block:: cmake
  56. add_flex_bison_dependency(<FlexTarget> <BisonTarget>)
  57. which adds the required dependency between a scanner and a parser
  58. where ``<FlexTarget>`` and ``<BisonTarget>`` are the first parameters of
  59. respectively ``flex_target`` and ``bison_target`` macros.
  60. Examples
  61. ^^^^^^^^
  62. .. code-block:: cmake
  63. find_package(BISON)
  64. find_package(FLEX)
  65. bison_target(MyParser parser.y ${CMAKE_CURRENT_BINARY_DIR}/parser.cpp)
  66. flex_target(MyScanner lexer.l ${CMAKE_CURRENT_BINARY_DIR}/lexer.cpp)
  67. add_flex_bison_dependency(MyScanner MyParser)
  68. include_directories(${CMAKE_CURRENT_BINARY_DIR})
  69. add_executable(Foo
  70. Foo.cc
  71. ${BISON_MyParser_OUTPUTS}
  72. ${FLEX_MyScanner_OUTPUTS}
  73. )
  74. target_link_libraries(Foo ${FLEX_LIBRARIES})
  75. #]=======================================================================]
  76. find_program(FLEX_EXECUTABLE NAMES flex win-flex win_flex DOC "path to the flex executable")
  77. mark_as_advanced(FLEX_EXECUTABLE)
  78. find_library(FL_LIBRARY NAMES fl
  79. DOC "Path to the fl library")
  80. find_path(FLEX_INCLUDE_DIR FlexLexer.h
  81. DOC "Path to the flex headers")
  82. mark_as_advanced(FL_LIBRARY FLEX_INCLUDE_DIR)
  83. set(FLEX_INCLUDE_DIRS ${FLEX_INCLUDE_DIR})
  84. set(FLEX_LIBRARIES ${FL_LIBRARY})
  85. if(FLEX_EXECUTABLE)
  86. execute_process(COMMAND ${FLEX_EXECUTABLE} --version
  87. OUTPUT_VARIABLE FLEX_version_output
  88. ERROR_VARIABLE FLEX_version_error
  89. RESULT_VARIABLE FLEX_version_result
  90. OUTPUT_STRIP_TRAILING_WHITESPACE)
  91. if(NOT ${FLEX_version_result} EQUAL 0)
  92. if(FLEX_FIND_REQUIRED)
  93. message(SEND_ERROR "Command \"${FLEX_EXECUTABLE} --version\" failed with output:\n${FLEX_version_output}\n${FLEX_version_error}")
  94. else()
  95. message("Command \"${FLEX_EXECUTABLE} --version\" failed with output:\n${FLEX_version_output}\n${FLEX_version_error}\nFLEX_VERSION will not be available")
  96. endif()
  97. else()
  98. # older versions of flex printed "/full/path/to/executable version X.Y"
  99. # newer versions use "basename(executable) X.Y"
  100. get_filename_component(FLEX_EXE_NAME_WE "${FLEX_EXECUTABLE}" NAME_WE)
  101. get_filename_component(FLEX_EXE_EXT "${FLEX_EXECUTABLE}" EXT)
  102. string(REGEX REPLACE "^.*${FLEX_EXE_NAME_WE}(${FLEX_EXE_EXT})?\"? (version )?([0-9]+[^ ]*)( .*)?$" "\\3"
  103. FLEX_VERSION "${FLEX_version_output}")
  104. unset(FLEX_EXE_EXT)
  105. unset(FLEX_EXE_NAME_WE)
  106. endif()
  107. #============================================================
  108. # FLEX_TARGET (public macro)
  109. #============================================================
  110. #
  111. macro(FLEX_TARGET Name Input Output)
  112. set(FLEX_TARGET_PARAM_OPTIONS)
  113. set(FLEX_TARGET_PARAM_ONE_VALUE_KEYWORDS
  114. COMPILE_FLAGS
  115. DEFINES_FILE
  116. )
  117. set(FLEX_TARGET_PARAM_MULTI_VALUE_KEYWORDS)
  118. cmake_parse_arguments(
  119. FLEX_TARGET_ARG
  120. "${FLEX_TARGET_PARAM_OPTIONS}"
  121. "${FLEX_TARGET_PARAM_ONE_VALUE_KEYWORDS}"
  122. "${FLEX_TARGET_MULTI_VALUE_KEYWORDS}"
  123. ${ARGN}
  124. )
  125. set(FLEX_TARGET_usage "FLEX_TARGET(<Name> <Input> <Output> [COMPILE_FLAGS <string>] [DEFINES_FILE <string>]")
  126. if(NOT "${FLEX_TARGET_ARG_UNPARSED_ARGUMENTS}" STREQUAL "")
  127. message(SEND_ERROR ${FLEX_TARGET_usage})
  128. else()
  129. cmake_policy(GET CMP0098 _flex_CMP0098
  130. PARENT_SCOPE # undocumented, do not use outside of CMake
  131. )
  132. set(_flex_INPUT "${Input}")
  133. if("x${_flex_CMP0098}x" STREQUAL "xNEWx")
  134. set(_flex_WORKING_DIR "${CMAKE_CURRENT_BINARY_DIR}")
  135. if(NOT IS_ABSOLUTE "${_flex_INPUT}")
  136. set(_flex_INPUT "${CMAKE_CURRENT_SOURCE_DIR}/${_flex_INPUT}")
  137. endif()
  138. else()
  139. set(_flex_WORKING_DIR "${CMAKE_CURRENT_SOURCE_DIR}")
  140. endif()
  141. unset(_flex_CMP0098)
  142. set(_flex_OUTPUT "${Output}")
  143. if(NOT IS_ABSOLUTE ${_flex_OUTPUT})
  144. set(_flex_OUTPUT "${_flex_WORKING_DIR}/${_flex_OUTPUT}")
  145. endif()
  146. set(_flex_TARGET_OUTPUTS "${_flex_OUTPUT}")
  147. set(_flex_EXE_OPTS "")
  148. if(NOT "${FLEX_TARGET_ARG_COMPILE_FLAGS}" STREQUAL "")
  149. set(_flex_EXE_OPTS "${FLEX_TARGET_ARG_COMPILE_FLAGS}")
  150. separate_arguments(_flex_EXE_OPTS)
  151. endif()
  152. set(_flex_OUTPUT_HEADER "")
  153. if(NOT "${FLEX_TARGET_ARG_DEFINES_FILE}" STREQUAL "")
  154. set(_flex_OUTPUT_HEADER "${FLEX_TARGET_ARG_DEFINES_FILE}")
  155. if(IS_ABSOLUTE "${_flex_OUTPUT_HEADER}")
  156. set(_flex_OUTPUT_HEADER_ABS "${_flex_OUTPUT_HEADER}")
  157. else()
  158. set(_flex_OUTPUT_HEADER_ABS "${_flex_WORKING_DIR}/${_flex_OUTPUT_HEADER}")
  159. endif()
  160. list(APPEND _flex_TARGET_OUTPUTS "${_flex_OUTPUT_HEADER_ABS}")
  161. list(APPEND _flex_EXE_OPTS --header-file=${_flex_OUTPUT_HEADER_ABS})
  162. endif()
  163. get_filename_component(_flex_EXE_NAME_WE "${FLEX_EXECUTABLE}" NAME_WE)
  164. add_custom_command(OUTPUT ${_flex_TARGET_OUTPUTS}
  165. COMMAND ${FLEX_EXECUTABLE} ${_flex_EXE_OPTS} -o${_flex_OUTPUT} ${_flex_INPUT}
  166. VERBATIM
  167. DEPENDS ${_flex_INPUT}
  168. COMMENT "[FLEX][${Name}] Building scanner with ${_flex_EXE_NAME_WE} ${FLEX_VERSION}"
  169. WORKING_DIRECTORY ${_flex_WORKING_DIR}
  170. COMMAND_EXPAND_LISTS)
  171. set(FLEX_${Name}_DEFINED TRUE)
  172. set(FLEX_${Name}_OUTPUTS ${_flex_TARGET_OUTPUTS})
  173. set(FLEX_${Name}_INPUT ${_flex_INPUT})
  174. set(FLEX_${Name}_COMPILE_FLAGS ${_flex_EXE_OPTS})
  175. set(FLEX_${Name}_OUTPUT_HEADER ${_flex_OUTPUT_HEADER})
  176. unset(_flex_EXE_NAME_WE)
  177. unset(_flex_EXE_OPTS)
  178. unset(_flex_INPUT)
  179. unset(_flex_OUTPUT)
  180. unset(_flex_OUTPUT_HEADER)
  181. unset(_flex_OUTPUT_HEADER_ABS)
  182. unset(_flex_TARGET_OUTPUTS)
  183. unset(_flex_WORKING_DIR)
  184. endif()
  185. endmacro()
  186. #============================================================
  187. #============================================================
  188. # ADD_FLEX_BISON_DEPENDENCY (public macro)
  189. #============================================================
  190. #
  191. macro(ADD_FLEX_BISON_DEPENDENCY FlexTarget BisonTarget)
  192. if(NOT FLEX_${FlexTarget}_OUTPUTS)
  193. message(SEND_ERROR "Flex target `${FlexTarget}' does not exist.")
  194. endif()
  195. if(NOT BISON_${BisonTarget}_OUTPUT_HEADER)
  196. message(SEND_ERROR "Bison target `${BisonTarget}' does not exist.")
  197. endif()
  198. set_source_files_properties(${FLEX_${FlexTarget}_OUTPUTS}
  199. PROPERTIES OBJECT_DEPENDS ${BISON_${BisonTarget}_OUTPUT_HEADER})
  200. endmacro()
  201. #============================================================
  202. endif()
  203. include(${CMAKE_CURRENT_LIST_DIR}/FindPackageHandleStandardArgs.cmake)
  204. FIND_PACKAGE_HANDLE_STANDARD_ARGS(FLEX REQUIRED_VARS FLEX_EXECUTABLE
  205. VERSION_VAR FLEX_VERSION)