FindFLEX.cmake 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. # - Find flex executable and provides a macro to generate custom build rules
  2. #
  3. # The module defines the following variables:
  4. # FLEX_FOUND - true is flex executable is found
  5. # FLEX_EXECUTABLE - the path to the flex executable
  6. # FLEX_VERSION - the version of flex
  7. # FLEX_LIBRARIES - The flex libraries
  8. # FLEX_INCLUDE_DIRS - The path to the flex headers
  9. #
  10. # The minimum required version of flex can be specified using the
  11. # standard syntax, e.g. find_package(FLEX 2.5.13)
  12. #
  13. #
  14. # If flex is found on the system, the module provides the macro:
  15. # FLEX_TARGET(Name FlexInput FlexOutput [COMPILE_FLAGS <string>])
  16. # which creates a custom command to generate the <FlexOutput> file from
  17. # the <FlexInput> file. If COMPILE_FLAGS option is specified, the next
  18. # parameter is added to the flex command line. Name is an alias used to
  19. # get details of this custom command. Indeed the macro defines the
  20. # following variables:
  21. # FLEX_${Name}_DEFINED - true is the macro ran successfully
  22. # FLEX_${Name}_OUTPUTS - the source file generated by the custom rule, an
  23. # alias for FlexOutput
  24. # FLEX_${Name}_INPUT - the flex source file, an alias for ${FlexInput}
  25. #
  26. # Flex scanners oftenly use tokens defined by Bison: the code generated
  27. # by Flex depends of the header generated by Bison. This module also
  28. # defines a macro:
  29. # ADD_FLEX_BISON_DEPENDENCY(FlexTarget BisonTarget)
  30. # which adds the required dependency between a scanner and a parser
  31. # where <FlexTarget> and <BisonTarget> are the first parameters of
  32. # respectively FLEX_TARGET and BISON_TARGET macros.
  33. #
  34. # ====================================================================
  35. # Example:
  36. #
  37. # find_package(BISON)
  38. # find_package(FLEX)
  39. #
  40. # BISON_TARGET(MyParser parser.y ${CMAKE_CURRENT_BINARY_DIR}/parser.cpp)
  41. # FLEX_TARGET(MyScanner lexer.l ${CMAKE_CURRENT_BINARY_DIR}/lexer.cpp)
  42. # ADD_FLEX_BISON_DEPENDENCY(MyScanner MyParser)
  43. #
  44. # include_directories(${CMAKE_CURRENT_BINARY_DIR})
  45. # add_executable(Foo
  46. # Foo.cc
  47. # ${BISON_MyParser_OUTPUTS}
  48. # ${FLEX_MyScanner_OUTPUTS}
  49. # )
  50. # ====================================================================
  51. #=============================================================================
  52. # Copyright 2009 Kitware, Inc.
  53. # Copyright 2006 Tristan Carel
  54. #
  55. # Distributed under the OSI-approved BSD License (the "License");
  56. # see accompanying file Copyright.txt for details.
  57. #
  58. # This software is distributed WITHOUT ANY WARRANTY; without even the
  59. # implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  60. # See the License for more information.
  61. #=============================================================================
  62. # (To distribute this file outside of CMake, substitute the full
  63. # License text for the above reference.)
  64. find_program(FLEX_EXECUTABLE flex DOC "path to the flex executable")
  65. mark_as_advanced(FLEX_EXECUTABLE)
  66. find_library(FL_LIBRARY NAMES fl
  67. DOC "Path to the fl library")
  68. find_path(FLEX_INCLUDE_DIR FlexLexer.h
  69. DOC "Path to the flex headers")
  70. mark_as_advanced(FL_LIBRARY FLEX_INCLUDE_DIR)
  71. set(FLEX_INCLUDE_DIRS ${FLEX_INCLUDE_DIR})
  72. set(FLEX_LIBRARIES ${FL_LIBRARY})
  73. if(FLEX_EXECUTABLE)
  74. execute_process(COMMAND ${FLEX_EXECUTABLE} --version
  75. OUTPUT_VARIABLE FLEX_version_output
  76. ERROR_VARIABLE FLEX_version_error
  77. RESULT_VARIABLE FLEX_version_result
  78. OUTPUT_STRIP_TRAILING_WHITESPACE)
  79. if(NOT ${FLEX_version_result} EQUAL 0)
  80. if(FLEX_FIND_REQUIRED)
  81. message(SEND_ERROR "Command \"${FLEX_EXECUTABLE} --version\" failed with output:\n${FLEX_version_output}\n${FLEX_version_error}")
  82. else()
  83. message("Command \"${FLEX_EXECUTABLE} --version\" failed with output:\n${FLEX_version_output}\n${FLEX_version_error}\nFLEX_VERSION will not be available")
  84. endif()
  85. else()
  86. # older versions of flex printed "/full/path/to/executable version X.Y"
  87. # newer versions use "basename(executable) X.Y"
  88. get_filename_component(FLEX_EXE_NAME "${FLEX_EXECUTABLE}" NAME)
  89. string(REGEX REPLACE "^.*${FLEX_EXE_NAME}\"? (version )?([0-9]+[^ ]*)$" "\\2"
  90. FLEX_VERSION "${FLEX_version_output}")
  91. unset(FLEX_EXE_NAME)
  92. endif()
  93. #============================================================
  94. # FLEX_TARGET (public macro)
  95. #============================================================
  96. #
  97. macro(FLEX_TARGET Name Input Output)
  98. set(FLEX_TARGET_usage "FLEX_TARGET(<Name> <Input> <Output> [COMPILE_FLAGS <string>]")
  99. if(${ARGC} GREATER 3)
  100. if(${ARGC} EQUAL 5)
  101. if("${ARGV3}" STREQUAL "COMPILE_FLAGS")
  102. set(FLEX_EXECUTABLE_opts "${ARGV4}")
  103. separate_arguments(FLEX_EXECUTABLE_opts)
  104. else()
  105. message(SEND_ERROR ${FLEX_TARGET_usage})
  106. endif()
  107. else()
  108. message(SEND_ERROR ${FLEX_TARGET_usage})
  109. endif()
  110. endif()
  111. add_custom_command(OUTPUT ${Output}
  112. COMMAND ${FLEX_EXECUTABLE}
  113. ARGS ${FLEX_EXECUTABLE_opts} -o${Output} ${Input}
  114. DEPENDS ${Input}
  115. COMMENT "[FLEX][${Name}] Building scanner with flex ${FLEX_VERSION}"
  116. WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR})
  117. set(FLEX_${Name}_DEFINED TRUE)
  118. set(FLEX_${Name}_OUTPUTS ${Output})
  119. set(FLEX_${Name}_INPUT ${Input})
  120. set(FLEX_${Name}_COMPILE_FLAGS ${FLEX_EXECUTABLE_opts})
  121. endmacro()
  122. #============================================================
  123. #============================================================
  124. # ADD_FLEX_BISON_DEPENDENCY (public macro)
  125. #============================================================
  126. #
  127. macro(ADD_FLEX_BISON_DEPENDENCY FlexTarget BisonTarget)
  128. if(NOT FLEX_${FlexTarget}_OUTPUTS)
  129. message(SEND_ERROR "Flex target `${FlexTarget}' does not exists.")
  130. endif()
  131. if(NOT BISON_${BisonTarget}_OUTPUT_HEADER)
  132. message(SEND_ERROR "Bison target `${BisonTarget}' does not exists.")
  133. endif()
  134. set_source_files_properties(${FLEX_${FlexTarget}_OUTPUTS}
  135. PROPERTIES OBJECT_DEPENDS ${BISON_${BisonTarget}_OUTPUT_HEADER})
  136. endmacro()
  137. #============================================================
  138. endif()
  139. include(${CMAKE_CURRENT_LIST_DIR}/FindPackageHandleStandardArgs.cmake)
  140. FIND_PACKAGE_HANDLE_STANDARD_ARGS(FLEX REQUIRED_VARS FLEX_EXECUTABLE
  141. VERSION_VAR FLEX_VERSION)
  142. # FindFLEX.cmake ends here