FindFLEX.cmake 8.5 KB

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