339 lines
11 KiB
CMake
339 lines
11 KiB
CMake
#------------------------------------------------------------------------------
|
|
# CLING - the C++ LLVM-based InterpreterG :)
|
|
#
|
|
# This file is dual-licensed: you can choose to license it under the University
|
|
# of Illinois Open Source License or the GNU Lesser General Public License. See
|
|
# LICENSE.TXT for details.
|
|
#------------------------------------------------------------------------------
|
|
|
|
set(CLING_DEPEND_LIBS
|
|
clangCodeGen
|
|
clangDriver
|
|
clangFrontend
|
|
clangParse
|
|
clangSema
|
|
clangAnalysis
|
|
clangEdit
|
|
clangRewrite
|
|
clangRewriteFrontend
|
|
clangSerialization
|
|
clangAST
|
|
clangBasic
|
|
clangLex
|
|
CACHE STRING "Dependency Clang libraries for Cling"
|
|
)
|
|
|
|
set(LIBS
|
|
clingUtils
|
|
)
|
|
|
|
set(LLVM_LINK_COMPONENTS
|
|
analysis
|
|
core
|
|
coroutines
|
|
coverage
|
|
executionengine
|
|
ipo
|
|
lto
|
|
mc
|
|
object
|
|
option
|
|
orcjit
|
|
runtimedyld
|
|
scalaropts
|
|
support
|
|
target
|
|
transformutils
|
|
binaryformat
|
|
${LLVM_TARGETS_TO_BUILD}
|
|
)
|
|
|
|
# clingInterpreter depends on Options.inc to be tablegen-ed
|
|
# (target ClangDriverOptions) from in-tree builds.
|
|
set(CLING_DEPENDS)
|
|
if(TARGET ClangDriverOptions)
|
|
set(CLING_DEPENDS ClangDriverOptions)
|
|
endif()
|
|
# clangSema will make sure all of the dependencies of clingInterpreter are met.
|
|
if(TARGET clangSema)
|
|
set(CLING_DEPENDS "${CLING_DEPENDS};clangSema")
|
|
endif()
|
|
|
|
|
|
add_cling_library(clingInterpreter OBJECT
|
|
AutoSynthesizer.cpp
|
|
AutoloadCallback.cpp
|
|
ASTTransformer.cpp
|
|
BackendPasses.cpp
|
|
CheckEmptyTransactionTransformer.cpp
|
|
CIFactory.cpp
|
|
ClangInternalState.cpp
|
|
ClingCodeCompleteConsumer.cpp
|
|
ClingPragmas.cpp
|
|
DeclCollector.cpp
|
|
DeclExtractor.cpp
|
|
DefinitionShadower.cpp
|
|
DeclUnloader.cpp
|
|
DeviceKernelInliner.cpp
|
|
DynamicLibraryManager.cpp
|
|
DynamicLibraryManagerSymbol.cpp
|
|
DynamicLookup.cpp
|
|
DynamicExprInfo.cpp
|
|
Exception.cpp
|
|
ExternalInterpreterSource.cpp
|
|
ForwardDeclPrinter.cpp
|
|
IncrementalCUDADeviceCompiler.cpp
|
|
IncrementalExecutor.cpp
|
|
IncrementalJIT.cpp
|
|
IncrementalParser.cpp
|
|
Interpreter.cpp
|
|
InterpreterCallbacks.cpp
|
|
InvocationOptions.cpp
|
|
LookupHelper.cpp
|
|
NullDerefProtectionTransformer.cpp
|
|
RequiredSymbols.cpp
|
|
Transaction.cpp
|
|
TransactionUnloader.cpp
|
|
ValueExtractionSynthesizer.cpp
|
|
Value.cpp
|
|
ValuePrinter.cpp
|
|
ValuePrinterSynthesizer.cpp
|
|
|
|
DEPENDS
|
|
${CLING_DEPENDS}
|
|
|
|
LINK_LIBS
|
|
${CLING_DEPEND_LIBS}
|
|
${LIBS}
|
|
${CMAKE_DL_LIBS}
|
|
)
|
|
|
|
|
|
if (UNIX)
|
|
set_source_files_properties(Exception.cpp COMPILE_FLAGS "-fexceptions -frtti")
|
|
set_source_files_properties(Interpreter.cpp COMPILE_FLAGS "-fexceptions")
|
|
|
|
# Remove all -I from CMAKE_CXX_FLAGS
|
|
string(REPLACE ";" " " __flags "${CMAKE_CXX_FLAGS}")
|
|
string(REGEX REPLACE "-I[^ ]+" "" CLING_COMPILER_FLAGS_NO_I "${__flags}")
|
|
|
|
option(CLING_CXX_PATH "Compiler cling will invoke for c++ headers." "")
|
|
option(CLING_CXX_HEADERS "Path cling will use for c++ headers." "")
|
|
|
|
function(stripNewLine strVal varName)
|
|
string(STRIP "${strVal}" strVal)
|
|
string(REGEX REPLACE "\\n$" "" strVal "${strVal}")
|
|
SET(${varName} ${strVal} PARENT_SCOPE)
|
|
endfunction()
|
|
|
|
if(NOT CLING_CXX_PATH)
|
|
# Remove absolute path from CMAKE_CXX_COMPILER
|
|
get_filename_component(_name ${CMAKE_CXX_COMPILER} NAME)
|
|
get_filename_component(_path ${CMAKE_CXX_COMPILER} PATH)
|
|
# This should probably be more general...but how?
|
|
if(_name STREQUAL "ccache" OR _name STREQUAL "distcc")
|
|
separate_arguments(_arg_list UNIX_COMMAND "${CMAKE_CXX_COMPILER_ARG1}")
|
|
if (_arg_list)
|
|
list(GET _arg_list 0 _name)
|
|
string(STRIP "${_name}" _name)
|
|
if (APPLE)
|
|
execute_process(COMMAND xcrun -f ${_name}
|
|
OUTPUT_VARIABLE CLING_CXX_FOUND
|
|
OUTPUT_STRIP_TRAILING_WHITESPACE)
|
|
stripNewLine("${CLING_CXX_FOUND}" CLING_CXX_FOUND)
|
|
else()
|
|
find_program(_cling_cxx_path "${_name}")
|
|
execute_process(COMMAND ${_cling_cxx_path} -xc++ -E -v /dev/null
|
|
OUTPUT_QUIET ERROR_VARIABLE _cling_cxx_path)
|
|
|
|
if (CMAKE_CXX_COMPILER_ID STREQUAL "GNU")
|
|
execute_process(
|
|
COMMAND echo ${_cling_cxx_path}
|
|
COMMAND grep "COLLECT_GCC="
|
|
OUTPUT_VARIABLE _cling_cxx_path)
|
|
string(REPLACE "COLLECT_GCC=" "" _cling_cxx_path "${_cling_cxx_path}")
|
|
|
|
elseif(CMAKE_CXX_COMPILER_ID MATCHES "Clang")
|
|
execute_process(
|
|
COMMAND echo ${_cling_cxx_path}
|
|
COMMAND grep "/${_name}.*\" -cc1"
|
|
OUTPUT_VARIABLE _cling_clng_path)
|
|
|
|
if(NOT _cling_clng_path)
|
|
execute_process(
|
|
COMMAND echo ${_cling_cxx_path}
|
|
COMMAND grep "/clang.*\" -cc1"
|
|
OUTPUT_VARIABLE _cling_clng_path)
|
|
endif()
|
|
|
|
separate_arguments(_arg_list UNIX_COMMAND "${_cling_clng_path}")
|
|
if (_arg_list)
|
|
list(GET _arg_list 0 _cling_cxx_path)
|
|
endif()
|
|
endif()
|
|
|
|
stripNewLine("${_cling_cxx_path}" _cling_cxx_path)
|
|
set(CLING_CXX_FOUND "${_cling_cxx_path}")
|
|
endif()
|
|
|
|
if (NOT EXISTS "${CLING_CXX_FOUND}")
|
|
find_program(CLING_CXX_FOUND "${_name}")
|
|
endif()
|
|
else()
|
|
set(CLING_CXX_FOUND "")
|
|
set(_name "")
|
|
endif()
|
|
|
|
if (EXISTS ${CLING_CXX_FOUND})
|
|
set(CLING_CXX_PATH ${CLING_CXX_FOUND})
|
|
get_filename_component(_name ${CLING_CXX_PATH} NAME)
|
|
get_filename_component(_path ${CLING_CXX_PATH} PATH)
|
|
else()
|
|
set(CLING_CXX_PATH "${CMAKE_CXX_COMPILER} ${CMAKE_CXX_COMPILER_ARG1}")
|
|
if(_name)
|
|
set(CLING_CXX_RLTV "${_name}")
|
|
endif()
|
|
set(_path "__THISREALLYBETTERNOTBEINPATH_THANKS__")
|
|
endif()
|
|
else()
|
|
# FIXME: In some ccache setups we can have a soft link pointing to ccache
|
|
# binary. Eg. /usr/local/gcc -> /usr/bin/ccache. Resolving the realpath
|
|
# we will get to the ccache and not the intended compiler binary. This
|
|
# could be fixed if we run 'gcc -###' which will give us the correct info.
|
|
get_filename_component(_realpath ${CMAKE_CXX_COMPILER} REALPATH)
|
|
get_filename_component(_name ${_realpath} NAME)
|
|
get_filename_component(_path ${_realpath} PATH)
|
|
endif()
|
|
|
|
# Test if path compiler is on PATH.
|
|
string(REPLACE ":" ";" _pathlist $ENV{PATH})
|
|
foreach (_pathcomp ${_pathlist})
|
|
get_filename_component(_pathcomp ${_pathcomp} REALPATH)
|
|
if (_path STREQUAL _pathcomp)
|
|
# This adds a lot of unneccessary flags, but may be useful if there's
|
|
# a flag that should be passed to cling.
|
|
set(CLING_CXX_RLTV ${_name})
|
|
break()
|
|
endif()
|
|
endforeach()
|
|
|
|
# FIXME: Perhaps CLING_CXX_RLTV should have a better name?
|
|
if(NOT CLING_CXX_RLTV AND NOT CLING_CXX_PATH)
|
|
# We got nothing, just use whatever CMake is using.
|
|
set(CLING_CXX_PATH ${CMAKE_CXX_COMPILER})
|
|
endif()
|
|
|
|
# If CMAKE_CXX_FLAGS contains --gcc-toolchain= then that should be passed on
|
|
string(FIND "${CMAKE_CXX_FLAGS}" "--gcc-toolchain=" cling_gcc_toolchain)
|
|
if ("${cling_gcc_toolchain}" GREATER -1)
|
|
# TODO Refactor these two into common function
|
|
if (CLING_CXX_PATH)
|
|
string(FIND "${CLING_CXX_PATH}" "--gcc-toolchain=" cling_gcc_toolchain)
|
|
if ("${cling_gcc_toolchain}" EQUAL -1)
|
|
set(CLING_CXX_PATH_ARGS "--gcc-toolchain=${gcctoolchain}")
|
|
endif()
|
|
endif()
|
|
if (CLING_CXX_RLTV)
|
|
string(FIND "${CLING_CXX_RLTV}" "--gcc-toolchain=" cling_gcc_toolchain)
|
|
if ("${cling_gcc_toolchain}" EQUAL -1)
|
|
set(CLING_CXX_RLTV "${CLING_CXX_RLTV} --gcc-toolchain=${gcctoolchain}")
|
|
endif()
|
|
endif()
|
|
endif()
|
|
endif()
|
|
|
|
if(NOT CLING_CXX_HEADERS)
|
|
if (CLING_CXX_PATH)
|
|
execute_process(COMMAND ${CLING_CXX_PATH} ${CLING_CXX_PATH_ARGS} -xc++ -E -v /dev/null
|
|
OUTPUT_QUIET ERROR_VARIABLE CLING_CXX_HEADERS)
|
|
set(CLING_CXX_PATH "${CLING_CXX_PATH} ${CLING_CXX_PATH_ARGS}")
|
|
else()
|
|
# convert CMAKE_CXX_FLAGS to a list for execute_process
|
|
string(REPLACE "-fdiagnostics-color=always" "" cling_tmp_arg_list ${CMAKE_CXX_FLAGS})
|
|
string(REPLACE "-fcolor-diagnosics" "" cling_tmp_arg_list ${cling_tmp_arg_list})
|
|
string(REPLACE " " ";" cling_tmp_arg_list ${cling_tmp_arg_list})
|
|
execute_process(COMMAND ${CMAKE_CXX_COMPILER} ${CMAKE_CXX_COMPILER_ARG1} ${cling_tmp_arg_list} -xc++ -E -v /dev/null
|
|
OUTPUT_QUIET ERROR_VARIABLE CLING_CXX_HEADERS)
|
|
endif()
|
|
|
|
execute_process(
|
|
COMMAND echo ${CLING_CXX_HEADERS}
|
|
COMMAND sed -n -e /^.include/,\$\{ -e /^\\\ \\\/.*++/p -e \}
|
|
OUTPUT_VARIABLE CLING_CXX_HEADERS)
|
|
|
|
stripNewLine("${CLING_CXX_HEADERS}" CLING_CXX_HEADERS)
|
|
endif()
|
|
|
|
if (NOT EXISTS ${CLING_CXX_HEADERS})
|
|
string(REPLACE "\n" ";" _cxx_inc_paths ${CLING_CXX_HEADERS})
|
|
foreach(_cxx_inc_path ${_cxx_inc_paths})
|
|
string(STRIP "${_cxx_inc_path}" _cxx_inc_path)
|
|
if (NOT EXISTS ${_cxx_inc_path})
|
|
set(_cxx_inc_join "")
|
|
break()
|
|
endif()
|
|
if(_cxx_inc_join)
|
|
set(_cxx_inc_join "${_cxx_inc_join}:${_cxx_inc_path}")
|
|
else()
|
|
set(_cxx_inc_join "${_cxx_inc_path}")
|
|
endif()
|
|
endforeach()
|
|
set(CLING_CXX_HEADERS "${_cxx_inc_join}")
|
|
if (NOT CLING_CXX_HEADERS)
|
|
MESSAGE(WARNING "Cannot determine location of C++ headers for runtime.")
|
|
endif()
|
|
endif()
|
|
|
|
MESSAGE(STATUS "Cling will look for C++ headers in '${CLING_CXX_HEADERS}' at runtime.")
|
|
|
|
# In modules builds we 'mount' our own stl modulemap for libstdc++. In order to do this,
|
|
# we need to know where is ROOT/cling STL.
|
|
set_property(GLOBAL PROPERTY ROOT_CLING_CXX_HEADERS_LOCATION "${CLING_CXX_HEADERS}")
|
|
|
|
# FIXME: We should use file(GENERATE) cmake command.
|
|
file(WRITE ${CMAKE_CURRENT_BINARY_DIR}/cling-compiledata.h.in
|
|
"
|
|
#define CLING_CXX_INCL \"${CLING_CXX_HEADERS}\"
|
|
#define CLING_INCLUDE_PATHS \"${CLING_INCLUDE_PATHS}\"
|
|
")
|
|
if (CMAKE_OSX_SYSROOT)
|
|
file(APPEND ${CMAKE_CURRENT_BINARY_DIR}/cling-compiledata.h.in
|
|
"
|
|
#define CLING_OSX_SYSROOT \"${CMAKE_OSX_SYSROOT}\"
|
|
")
|
|
endif()
|
|
if (CLING_CXX_PATH)
|
|
MESSAGE(STATUS "And if not found, will invoke: '${CLING_CXX_PATH}' for them.")
|
|
file(APPEND ${CMAKE_CURRENT_BINARY_DIR}/cling-compiledata.h.in
|
|
"
|
|
#define CLING_CXX_PATH \"${CLING_CXX_PATH} ${CMAKE_CXX_FLAGS_NO_I} ${CMAKE_CXX_FLAGS_${uppercase_CMAKE_BUILD_TYPE}}\"
|
|
")
|
|
endif()
|
|
if (CLING_CXX_RLTV)
|
|
MESSAGE(STATUS "And then fallback to: '${CLING_CXX_RLTV}'")
|
|
file(APPEND ${CMAKE_CURRENT_BINARY_DIR}/cling-compiledata.h.in
|
|
"
|
|
#define CLING_CXX_RLTV \"${CLING_CXX_RLTV} ${CMAKE_CXX_FLAGS_NO_I} ${CMAKE_CXX_FLAGS_${uppercase_CMAKE_BUILD_TYPE}}\"
|
|
")
|
|
endif()
|
|
else()
|
|
file(APPEND ${CMAKE_CURRENT_BINARY_DIR}/cling-compiledata.h.in
|
|
"
|
|
#define CLING_INCLUDE_PATHS \"${CLING_INCLUDE_PATHS}\"
|
|
#define CLING_UCRT_VERSION \"$ENV{UCRTVersion}\"
|
|
")
|
|
endif()
|
|
|
|
# Make sure this goes last so so we can pick up any changes that occured
|
|
# Also means cling-compiledata.h.in should be edited never cling-compiledata.h
|
|
|
|
add_custom_command(OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/cling-compiledata.h
|
|
COMMAND ${CMAKE_COMMAND} -E copy_if_different
|
|
${CMAKE_CURRENT_BINARY_DIR}/cling-compiledata.h.in
|
|
${CMAKE_CURRENT_BINARY_DIR}/cling-compiledata.h
|
|
MAIN_DEPENDENCY ${CMAKE_CURRENT_BINARY_DIR}/cling-compiledata.h.in
|
|
COMMENT "Updating cling-compiledata.h")
|
|
|
|
add_file_dependencies(${CMAKE_CURRENT_SOURCE_DIR}/CIFactory.cpp
|
|
${CMAKE_CURRENT_BINARY_DIR}/cling-compiledata.h)
|