# - Find flex executable and provides a macro to generate custom build rules
#
# The module defines the following variables:
# FLEX_FOUND - true is flex executable is found
# FLEX_EXECUTABLE - the path to the flex executable
# FLEX_VERSION - the version of flex
# FLEX_LIBRARIES - The flex libraries
# FLEX_INCLUDE_DIRS - The path to the flex headers
#
# The minimum required version of flex can be specified using the
# standard syntax, e.g. find_package(FLEX 2.5.13)
#
#
# If flex is found on the system, the module provides the macro:
# FLEX_TARGET(Name FlexInput FlexOutput [COMPILE_FLAGS <string>])
# which creates a custom command to generate the <FlexOutput> file from
# the <FlexInput> file. If COMPILE_FLAGS option is specified, the next
# parameter is added to the flex command line. Name is an alias used to
# get details of this custom command. Indeed the macro defines the
# following variables:
# FLEX_${Name}_DEFINED - true is the macro ran successfully
# FLEX_${Name}_OUTPUTS - the source file generated by the custom rule, an
# alias for FlexOutput
# FLEX_${Name}_INPUT - the flex source file, an alias for ${FlexInput}
#
# Flex scanners oftenly use tokens defined by Bison: the code generated
# by Flex depends of the header generated by Bison. This module also
# defines a macro:
# ADD_FLEX_BISON_DEPENDENCY(FlexTarget BisonTarget)
# which adds the required dependency between a scanner and a parser
# where <FlexTarget> and <BisonTarget> are the first parameters of
# respectively FLEX_TARGET and BISON_TARGET macros.
#
# ====================================================================
# Example:
#
# find_package(BISON)
# find_package(FLEX)
#
# BISON_TARGET(MyParser parser.y ${CMAKE_CURRENT_BINARY_DIR}/parser.cpp)
# FLEX_TARGET(MyScanner lexer.l ${CMAKE_CURRENT_BINARY_DIR}/lexer.cpp)
# ADD_FLEX_BISON_DEPENDENCY(MyScanner MyParser)
#
# include_directories(${CMAKE_CURRENT_BINARY_DIR})
# add_executable(Foo
# Foo.cc
# ${BISON_MyParser_OUTPUTS}
# ${FLEX_MyScanner_OUTPUTS}
# )
# ====================================================================
#=============================================================================
# Copyright 2009 Kitware, Inc.
# Copyright 2006 Tristan Carel
#
# Distributed under the OSI-approved BSD License (the "License");
# see accompanying file Copyright.txt for details.
#
# This software is distributed WITHOUT ANY WARRANTY; without even the
# implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
# See the License for more information.
#=============================================================================
# (To distribute this file outside of CMake, substitute the full
# License text for the above reference.)
find_program ( FLEX_EXECUTABLE NAMES flex win_flex DOC "path to the flex executable" )
mark_as_advanced ( FLEX_EXECUTABLE )
find_library ( FL_LIBRARY NAMES fl
D O C " P a t h t o t h e f l l i b r a r y " )
find_path ( FLEX_INCLUDE_DIR FlexLexer.h
D O C " P a t h t o t h e f l e x h e a d e r s " )
mark_as_advanced ( FL_LIBRARY FLEX_INCLUDE_DIR )
set ( FLEX_INCLUDE_DIRS ${ FLEX_INCLUDE_DIR } )
set ( FLEX_LIBRARIES ${ FL_LIBRARY } )
if ( FLEX_EXECUTABLE )
execute_process ( COMMAND ${ FLEX_EXECUTABLE } --version
O U T P U T _ V A R I A B L E F L E X _ v e r s i o n _ o u t p u t
E R R O R _ V A R I A B L E F L E X _ v e r s i o n _ e r r o r
R E S U L T _ V A R I A B L E F L E X _ v e r s i o n _ r e s u l t
O U T P U T _ S T R I P _ T R A I L I N G _ W H I T E S P A C E )
if ( NOT ${ FLEX_version_result } EQUAL 0 )
if ( FLEX_FIND_REQUIRED )
message ( SEND_ERROR "Command \" ${ FLEX_EXECUTABLE } --version\ " failed with output:\n${FLEX_version_output}\n${FLEX_version_error}" )
else ( )
message ( "Command \" ${ FLEX_EXECUTABLE } --version\ " failed with output:\n${FLEX_version_output}\n${FLEX_version_error}\nFLEX_VERSION will not be available" )
endif ( )
else ( )
# older versions of flex printed "/full/path/to/executable version X.Y"
# newer versions use "basename(executable) X.Y"
get_filename_component ( FLEX_EXE_NAME_WE "${FLEX_EXECUTABLE}" NAME_WE )
get_filename_component ( FLEX_EXE_EXT "${FLEX_EXECUTABLE}" EXT )
string ( REGEX REPLACE "^.*${FLEX_EXE_NAME_WE}(${FLEX_EXE_EXT})?\" ? ( version ) ?([0-9]+[^ ]* ) ( . * ) ? $ " " \ \ 3 "
F L E X _ V E R S I O N " $ { F L E X _ v e r s i o n _ o u t p u t } " )
unset ( FLEX_EXE_EXT )
unset ( FLEX_EXE_NAME_WE )
endif ( )
#============================================================
# FLEX_TARGET (public macro)
#============================================================
#
macro ( FLEX_TARGET Name Input Output )
set ( FLEX_TARGET_usage "FLEX_TARGET(<Name> <Input> <Output> [COMPILE_FLAGS <string>]" )
if ( ${ ARGC } GREATER 3 )
if ( ${ ARGC } EQUAL 5 )
if ( "${ARGV3}" STREQUAL "COMPILE_FLAGS" )
set ( FLEX_EXECUTABLE_opts "${ARGV4}" )
separate_arguments ( FLEX_EXECUTABLE_opts )
else ( )
message ( SEND_ERROR ${ FLEX_TARGET_usage } )
endif ( )
else ( )
message ( SEND_ERROR ${ FLEX_TARGET_usage } )
endif ( )
endif ( )
add_custom_command ( OUTPUT ${ Output }
C O M M A N D $ { F L E X _ E X E C U T A B L E }
A R G S $ { F L E X _ E X E C U T A B L E _ o p t s } - o $ { O u t p u t } $ { I n p u t }
D E P E N D S $ { I n p u t }
C O M M E N T " [ F L E X ] [ $ { N a m e } ] B u i l d i n g s c a n n e r w i t h f l e x $ { F L E X _ V E R S I O N } "
W O R K I N G _ D I R E C T O R Y $ { C M A K E _ C U R R E N T _ S O U R C E _ D I R } )
set ( FLEX_ ${ Name } _DEFINED TRUE )
set ( FLEX_ ${ Name } _OUTPUTS ${ Output } )
set ( FLEX_ ${ Name } _INPUT ${ Input } )
set ( FLEX_ ${ Name } _COMPILE_FLAGS ${ FLEX_EXECUTABLE_opts } )
endmacro ( )
#============================================================
#============================================================
# ADD_FLEX_BISON_DEPENDENCY (public macro)
#============================================================
#
macro ( ADD_FLEX_BISON_DEPENDENCY FlexTarget BisonTarget )
if ( NOT FLEX_ ${ FlexTarget } _OUTPUTS )
message ( SEND_ERROR "Flex target `${FlexTarget}' does not exists." )
endif ( )
if ( NOT BISON_ ${ BisonTarget } _OUTPUT_HEADER )
message ( SEND_ERROR "Bison target `${BisonTarget}' does not exists." )
endif ( )
set_source_files_properties ( ${ FLEX_${FlexTarget } _OUTPUTS}
P R O P E R T I E S O B J E C T _ D E P E N D S $ { B I S O N _ $ { B i s o n T a r g e t } _ O U T P U T _ H E A D E R } )
endmacro ( )
#============================================================
endif ( )
include ( ${ CMAKE_CURRENT_LIST_DIR } /FindPackageHandleStandardArgs.cmake )
FIND_PACKAGE_HANDLE_STANDARD_ARGS ( FLEX REQUIRED_VARS FLEX_EXECUTABLE
V E R S I O N _ V A R F L E X _ V E R S I O N )
# FindFLEX.cmake ends here