cmake does not properly interpret the SYSTEM marker in lists of include directories. this causes issues downstream when using dunecontrol / the in-tree cmake config files.
380 lines
15 KiB
CMake
380 lines
15 KiB
CMake
project(opm-common C CXX)
|
|
cmake_minimum_required (VERSION 2.8)
|
|
|
|
option(SIBLING_SEARCH "Search for other modules in sibling directories?" ON)
|
|
list(APPEND CMAKE_MODULE_PATH ${PROJECT_SOURCE_DIR}/cmake/Modules)
|
|
set(OPM_MACROS_ROOT ${PROJECT_SOURCE_DIR})
|
|
|
|
option(ENABLE_ECL_INPUT "Enable eclipse input support?" ON)
|
|
option(ENABLE_ECL_OUTPUT "Enable eclipse output support?" ON)
|
|
option(ENABLE_MOCKSIM "Build the mock simulator for io testing" ON)
|
|
option(OPM_ENABLE_PYTHON "Enable python bindings?" OFF)
|
|
option(OPM_INSTALL_PYTHON "Enable python bindings?" OFF)
|
|
option(OPM_ENABLE_EMBEDDED_PYTHON "Enable python bindings?" OFF)
|
|
|
|
# Output implies input
|
|
if(ENABLE_ECL_OUTPUT)
|
|
set(ENABLE_ECL_INPUT ON)
|
|
endif()
|
|
|
|
# And likewise, no input means no output
|
|
if(NOT ENABLE_ECL_INPUT)
|
|
set(ENABLE_ECL_OUTPUT OFF)
|
|
endif()
|
|
|
|
|
|
# not the same location as most of the other projects; this hook overrides
|
|
macro (dir_hook)
|
|
endmacro (dir_hook)
|
|
|
|
# We need to define this variable in the installed cmake config file.
|
|
set(OPM_PROJECT_EXTRA_CODE_INSTALLED "#ENABLE_ECL_INPUT is needed by opm-common-prereq.cmake
|
|
set(ENABLE_ECL_INPUT ${ENABLE_ECL_INPUT})
|
|
set(OPM_MACROS_ROOT ${CMAKE_INSTALL_PREFIX}/share/opm)
|
|
list(APPEND CMAKE_MODULE_PATH \${OPM_MACROS_ROOT}/cmake/Modules)
|
|
include(OpmPackage) #Make macros availabe after find_package(opm-common)")
|
|
|
|
set(OPM_PROJECT_EXTRA_CODE_INTREE "#ENABLE_ECL_INPUT is needed by opm-common-prereq.cmake
|
|
set(ENABLE_ECL_INPUT ${ENABLE_ECL_INPUT})
|
|
set(OPM_MACROS_ROOT ${OPM_MACROS_ROOT})
|
|
list(APPEND CMAKE_MODULE_PATH \${OPM_MACROS_ROOT}/cmake/Modules)
|
|
include(OpmPackage) #Make macros availabe after find_package(opm-common)")
|
|
if(ENABLE_ECL_OUTPUT)
|
|
set(OPM_PROJECT_EXTRA_CODE_INSTALLED "${OPM_PROJECT_EXTRA_CODE_INSTALLED}
|
|
set(COMPARE_ECL_COMMAND ${CMAKE_INSTALL_PREFIX}/bin${${name}_VER_DIR}/compareECL)
|
|
set(OPM_PACK_COMMAND ${CMAKE_INSTALL_PREFIX}/bin${${name}_VER_DIR}/opmpack)")
|
|
|
|
set(OPM_PROJECT_EXTRA_CODE_INTREE "${OPM_PROJECT_EXTRA_CODE_INTREE}
|
|
set(COMPARE_ECL_COMMAND ${PROJECT_BINARY_DIR}/bin/compareECL)
|
|
set(OPM_PACK_COMMAND ${PROJECT_BINARY_DIR}/bin/opmpack)")
|
|
endif()
|
|
|
|
# project information is in dune.module. Read this file and set variables.
|
|
# we cannot generate dune.module since it is read by dunecontrol before
|
|
# the build starts, so it makes sense to keep the data there then.
|
|
include (OpmInit)
|
|
OpmSetPolicies()
|
|
|
|
# Look for the opm-tests repository; if found the variable
|
|
# HAVE_OPM_TESTS will be set to true.
|
|
include(Findopm-tests)
|
|
|
|
# list of prerequisites for this particular project; this is in a
|
|
# separate file (in cmake/Modules sub-directory) because it is shared
|
|
# with the find module
|
|
include (${project}-prereqs)
|
|
|
|
# read the list of components from this file (in the project directory);
|
|
# it should set various lists with the names of the files to include
|
|
include (CMakeLists_files.cmake)
|
|
|
|
macro (config_hook)
|
|
if(ENABLE_ECL_INPUT)
|
|
if(NOT cjson_FOUND)
|
|
list(APPEND EXTRA_INCLUDES ${PROJECT_SOURCE_DIR}/external/cjson)
|
|
endif()
|
|
|
|
# For this project
|
|
add_definitions(-DFMT_HEADER_ONLY)
|
|
include_directories(SYSTEM ${PROJECT_SOURCE_DIR}/external/fmtlib/include)
|
|
include_directories(${EXTRA_INCLUDES} ${PROJECT_BINARY_DIR}/include)
|
|
list(APPEND EXTRA_INCLUDES ${PROJECT_SOURCE_DIR}/external/fmtlib/include)
|
|
|
|
# For downstreams
|
|
list(APPEND EXTRA_INCLUDES ${PROJECT_BINARY_DIR}/include)
|
|
set(OPM_PROJECT_EXTRA_CODE_INTREE "${OPM_PROJECT_EXTRA_CODE_INTREE}
|
|
list(APPEND opm-common_INCLUDE_DIRS ${EXTRA_INCLUDES})")
|
|
if(ENABLE_ECL_INPUT)
|
|
set(OPM_PROJECT_EXTRA_CODE_INTREE "${OPM_PROJECT_EXTRA_CODE_INTREE}
|
|
set(HAVE_ECL_INPUT 1)")
|
|
set(OPM_PROJECT_EXTRA_CODE_INSTALLED "${OPM_PROJECT_EXTRA_CODE_INSTALLED}
|
|
set(HAVE_ECL_INPUT 1)")
|
|
endif()
|
|
if(ENABLE_ECL_OUTPUT)
|
|
set(OPM_PROJECT_EXTRA_CODE_INTREE "${OPM_PROJECT_EXTRA_CODE_INTREE}
|
|
set(HAVE_ECL_OUTPUT 1)")
|
|
set(OPM_PROJECT_EXTRA_CODE_INSTALLED "${OPM_PROJECT_EXTRA_CODE_INSTALLED}
|
|
set(HAVE_ECL_OUTPUT 1)")
|
|
endif()
|
|
|
|
# Configure boost targets for old cmake
|
|
include(cmake/Modules/BoostTargets.cmake)
|
|
|
|
if (HAVE_DYNAMIC_BOOST_TEST)
|
|
set_target_properties(Boost::unit_test_framework PROPERTIES INTERFACE_COMPILE_DEFINITIONS BOOST_TEST_DYN_LINK=1)
|
|
endif()
|
|
endif()
|
|
endmacro (config_hook)
|
|
|
|
macro (prereqs_hook)
|
|
endmacro (prereqs_hook)
|
|
|
|
macro (sources_hook)
|
|
if(ENABLE_ECL_INPUT)
|
|
# Keyword generation
|
|
include(GenerateKeywords.cmake)
|
|
|
|
# Append generated sources
|
|
list(INSERT opm-common_SOURCES 0 ${PROJECT_BINARY_DIR}/ParserInit.cpp)
|
|
foreach (name A B C D E F G H I J K L M N O P Q R S T U V W X Y Z)
|
|
list(INSERT opm-common_SOURCES 0 ${PROJECT_BINARY_DIR}/ParserKeywords/${name}.cpp)
|
|
endforeach()
|
|
endif()
|
|
set_source_files_properties(src/opm/parser/eclipse/Python/Python.cpp
|
|
PROPERTIES COMPILE_FLAGS -Wno-shadow)
|
|
endmacro (sources_hook)
|
|
|
|
macro (fortran_hook)
|
|
endmacro (fortran_hook)
|
|
|
|
macro (files_hook)
|
|
endmacro (files_hook)
|
|
|
|
macro (tests_hook)
|
|
if(ENABLE_ECL_INPUT)
|
|
include(ExtraTests.cmake)
|
|
endif()
|
|
endmacro (tests_hook)
|
|
|
|
macro (install_hook)
|
|
install(DIRECTORY ${PROJECT_BINARY_DIR}/include/
|
|
DESTINATION include
|
|
PATTERN *.hpp)
|
|
endmacro (install_hook)
|
|
|
|
# If opm-common is configured to embed the python interpreter we must make sure
|
|
# that all downstream modules link libpython transitively. Due to the required
|
|
# integration with Python+cmake machinery provided by pybind11 this is done by
|
|
# manually adding to the opm-common_LIBRARIES variable here, and not in the
|
|
# OpmnLibMain function. Here only the library dependency is implemented, the
|
|
# bulk of the python configuration is further down in the file.
|
|
if (OPM_ENABLE_PYTHON)
|
|
find_package(PythonInterp REQUIRED)
|
|
if (OPM_ENABLE_EMBEDDED_PYTHON)
|
|
find_package(PythonLibs REQUIRED)
|
|
list(APPEND opm-common_LIBRARIES ${PYTHON_LIBRARY})
|
|
endif()
|
|
endif()
|
|
|
|
|
|
# all setup common to the OPM library modules is done here
|
|
include (OpmLibMain)
|
|
|
|
if (ENABLE_MOCKSIM)
|
|
add_library(mocksim
|
|
msim/src/msim.cpp)
|
|
target_link_libraries(mocksim opmcommon)
|
|
target_include_directories(mocksim PUBLIC msim/include)
|
|
add_executable(msim examples/msim.cpp)
|
|
target_link_libraries(msim mocksim)
|
|
|
|
if (Boost_UNIT_TEST_FRAMEWORK_FOUND)
|
|
set(_libs mocksim opmcommon
|
|
${Boost_UNIT_TEST_FRAMEWORK_LIBRARY})
|
|
|
|
foreach( test test_msim test_msim_ACTIONX test_msim_EXIT)
|
|
opm_add_test(${test} SOURCES tests/msim/${test}.cpp
|
|
LIBRARIES ${_libs}
|
|
WORKING_DIRECTORY ${PROJECT_BINARY_DIR}/tests
|
|
CONDITION ${HAVE_ECL_INPUT})
|
|
endforeach()
|
|
endif()
|
|
endif()
|
|
|
|
# Build the compare utilities
|
|
if(ENABLE_ECL_INPUT)
|
|
add_executable(compareECL
|
|
test_util/EclFilesComparator.cpp
|
|
test_util/EclRegressionTest.cpp
|
|
test_util/compareECL.cpp
|
|
)
|
|
|
|
add_executable(convertECL
|
|
test_util/convertECL.cpp
|
|
)
|
|
|
|
add_executable(summary
|
|
test_util/summary.cpp
|
|
)
|
|
|
|
foreach(target compareECL convertECL summary)
|
|
target_link_libraries(${target} opmcommon)
|
|
install(TARGETS ${target} DESTINATION bin)
|
|
endforeach()
|
|
|
|
# Add the tests
|
|
set(_libs opmcommon
|
|
${Boost_UNIT_TEST_FRAMEWORK_LIBRARY})
|
|
|
|
opm_add_test(test_EclFilesComparator
|
|
CONDITION
|
|
ENABLE_ECL_INPUT AND Boost_UNIT_TEST_FRAMEWORK_FOUND
|
|
SOURCES
|
|
tests/test_EclFilesComparator.cpp
|
|
test_util/EclFilesComparator.cpp
|
|
LIBRARIES
|
|
${_libs}
|
|
WORKING_DIRECTORY
|
|
${PROJECT_BINARY_DIR}/tests
|
|
)
|
|
|
|
opm_add_test(test_EclRegressionTest
|
|
CONDITION
|
|
ENABLE_ECL_INPUT AND Boost_UNIT_TEST_FRAMEWORK_FOUND
|
|
SOURCES
|
|
tests/test_EclRegressionTest.cpp
|
|
test_util/EclFilesComparator.cpp
|
|
test_util/EclRegressionTest.cpp
|
|
LIBRARIES
|
|
${_libs}
|
|
WORKING_DIRECTORY
|
|
${PROJECT_BINARY_DIR}/tests
|
|
)
|
|
|
|
foreach(test test_EclIO test_EGrid test_ERft test_ERst test_ESmry)
|
|
opm_add_test(${test} CONDITION ENABLE_ECL_INPUT AND Boost_UNIT_TEST_FRAMEWORK_FOUND
|
|
LIBRARIES ${_libs}
|
|
WORKING_DIRECTORY ${PROJECT_BINARY_DIR}/tests)
|
|
endforeach()
|
|
endif()
|
|
|
|
# Install build system files
|
|
install(DIRECTORY cmake DESTINATION share/opm)
|
|
|
|
# Install tab completion skeleton
|
|
install(FILES etc/opm_bash_completion.sh.in DESTINATION share/opm/etc)
|
|
|
|
if (OPM_ENABLE_PYTHON)
|
|
# -------------------------------------------------------------------------
|
|
# 1: Wrap C++ functionality in Python
|
|
if (EXISTS "/etc/debian_version")
|
|
set(PYTHON_PACKAGE_PATH "dist-packages")
|
|
else()
|
|
set(PYTHON_PACKAGE_PATH "site-packages")
|
|
endif()
|
|
set(PYTHON_INSTALL_PREFIX "lib/python${PYTHON_VERSION_MAJOR}.${PYTHON_VERSION_MINOR}/${PYTHON_PACKAGE_PATH}" CACHE STRING "Subdirectory to install Python modules in")
|
|
|
|
make_directory(${PROJECT_BINARY_DIR}/python)
|
|
get_target_property(_opmcommon_include_dirs opmcommon INCLUDE_DIRECTORIES)
|
|
list(APPEND _opmcommon_include_dirs ${_ecl_include_dirs})
|
|
string(REPLACE ";" ":" _setup_include_dirs "${_opmcommon_include_dirs}")
|
|
|
|
get_target_property(_opmcommon_lib_dirs opmcommon LINK_DIRECTORIES)
|
|
if (CMAKE_PREFIX_PATH)
|
|
list(APPEND _opmcommon_lib_dirs ${PROJECT_BINARY_DIR}/lib ${CMAKE_PREFIX_PATH}/${CMAKE_INSTALL_LIBDIR})
|
|
else()
|
|
list(APPEND _opmcommon_lib_dirs ${PROJECT_BINARY_DIR}/lib)
|
|
endif()
|
|
string(REPLACE ";" ":" _setup_lib_dirs "${_opmcommon_lib_dirs}")
|
|
|
|
if (USE_RUNPATH)
|
|
set (_python_rpath_list)
|
|
if (CMAKE_PREFIX_PATH)
|
|
foreach(path ${CMAKE_PREFIX_PATH})
|
|
list(APPEND _python_rpath_list "${path}/${CMAKE_INSTALL_LIBDIR}")
|
|
endforeach()
|
|
endif()
|
|
|
|
if (BUILD_SHARED_LIBS)
|
|
list(APPEND _python_rpath_list "${CMAKE_INSTALL_PREFIX}/${CMAKE_INSTALL_LIBDIR}")
|
|
endif()
|
|
|
|
if (_python_rpath_list)
|
|
string(REPLACE ";" ":" _rpath "${_python_rpath_list}")
|
|
set( _rpath_arg "--rpath=${_rpath}")
|
|
else()
|
|
set(_rpath_arg "")
|
|
endif()
|
|
else()
|
|
set( _rpath_arg "")
|
|
endif()
|
|
|
|
set(opm-common_PYTHON_PACKAGE_VERSION ${OPM_PYTHON_PACKAGE_VERSION_TAG})
|
|
|
|
# Generate versioned setup.py
|
|
configure_file (${PROJECT_SOURCE_DIR}/python/setup.py.in ${PROJECT_BINARY_DIR}/python/setup.py)
|
|
file(COPY ${PROJECT_SOURCE_DIR}/python/README.md DESTINATION ${PROJECT_BINARY_DIR}/python)
|
|
execute_process(COMMAND ${PYTHON_EXECUTABLE} target_name.py
|
|
WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}/python
|
|
OUTPUT_VARIABLE python_lib_target)
|
|
|
|
add_custom_target(copy_python ALL
|
|
COMMAND ${PYTHON_EXECUTABLE} ${PROJECT_SOURCE_DIR}/python/install.py ${PROJECT_SOURCE_DIR}/python ${PROJECT_BINARY_DIR} 0)
|
|
|
|
add_custom_command(OUTPUT python/opm/${python_lib_target}
|
|
DEPENDS ${PYTHON_CXX_DEPENDS}
|
|
DEPENDS copy_python
|
|
COMMAND ${PYTHON_EXECUTABLE} ${PROJECT_BINARY_DIR}/python/setup.py
|
|
build
|
|
build_ext
|
|
--build-lib=${PROJECT_BINARY_DIR}/python
|
|
--library-dirs=${_setup_lib_dirs}
|
|
${_rpath_arg}
|
|
--include-dirs=${_setup_include_dirs}
|
|
WORKING_DIRECTORY ${PROJECT_BINARY_DIR}/python
|
|
COMMENT "Building python bindings at python/opm/${python_lib_target}")
|
|
add_custom_target(opmcommon_python ALL DEPENDS python/opm/${python_lib_target})
|
|
add_dependencies(opmcommon_python opmcommon)
|
|
|
|
# The install target is based on manually copying the python file tree to the
|
|
# installation area with a small installation script 'install.py'. Would have
|
|
# preferred to use standard setup.py install, but the setup.py based solution
|
|
# refuses to install to a location which the current python executable can not
|
|
# load from, and the use of eggs in the setup.py based installation makes
|
|
# debugging quite difficult.
|
|
#
|
|
# Since the installation of Python code is nonstandard it is protected by an
|
|
# extra cmake switch, OPM_INSTALL_PYTHON. If you prefer you can still invoke
|
|
# setup.py install manually - optionally with the generated script
|
|
# setup-install.sh - and completely bypass cmake in the installation phase.
|
|
if (OPM_INSTALL_PYTHON)
|
|
install( CODE "execute_process(COMMAND ${PYTHON_EXECUTABLE} ${PROJECT_BINARY_DIR}/python/install.py ${PROJECT_BINARY_DIR}/python/opm ${DEST_PREFIX}${CMAKE_INSTALL_PREFIX}/${PYTHON_INSTALL_PREFIX} 1)")
|
|
endif()
|
|
|
|
# Observe that if the opmcommon library has been built as a shared library the
|
|
# python library opmcommon_python will in general not find it runtime while
|
|
# testing.
|
|
add_test(NAME python_tests
|
|
WORKING_DIRECTORY ${PROJECT_BINARY_DIR}/python
|
|
COMMAND ${CMAKE_COMMAND} -E env LD_LIBRARY_PATH=${PROJECT_BINARY_DIR}/lib ${PYTHON_EXECUTABLE} setup.py build_ext --dry-run --build-lib ${PROJECT_BINARY_DIR}/python test
|
|
)
|
|
|
|
set_target_properties(opmcommon PROPERTIES POSITION_INDEPENDENT_CODE ON)
|
|
set_directory_properties(PROPERTIES ADDITIONAL_MAKE_CLEAN_FILES ${PROJECT_BINARY_DIR}/python)
|
|
|
|
# -------------------------------------------------------------------------
|
|
# Let cmake configure some small shell scripts which can be used to simplify
|
|
# building, testing and installation of the Python extensions.
|
|
configure_file(python/setup-build.sh.in tmp/setup-build.sh)
|
|
file( COPY ${PROJECT_BINARY_DIR}/tmp/setup-build.sh
|
|
DESTINATION ${PROJECT_BINARY_DIR}
|
|
FILE_PERMISSIONS OWNER_READ OWNER_WRITE OWNER_EXECUTE )
|
|
|
|
configure_file(python/setup-package.sh.in tmp/setup-package.sh)
|
|
file( COPY ${PROJECT_BINARY_DIR}/tmp/setup-package.sh
|
|
DESTINATION ${PROJECT_BINARY_DIR}
|
|
FILE_PERMISSIONS OWNER_READ OWNER_WRITE OWNER_EXECUTE )
|
|
|
|
configure_file(python/setup-test.sh.in tmp/setup-test.sh)
|
|
file( COPY ${PROJECT_BINARY_DIR}/tmp/setup-test.sh
|
|
DESTINATION ${PROJECT_BINARY_DIR}
|
|
FILE_PERMISSIONS OWNER_READ OWNER_WRITE OWNER_EXECUTE )
|
|
|
|
configure_file(python/setup-install.sh.in tmp/setup-install.sh)
|
|
file( COPY ${PROJECT_BINARY_DIR}/tmp/setup-install.sh
|
|
DESTINATION ${PROJECT_BINARY_DIR}
|
|
FILE_PERMISSIONS OWNER_READ OWNER_WRITE OWNER_EXECUTE )
|
|
|
|
configure_file(python/enable-python.sh.in enable-python.sh)
|
|
|
|
# -------------------------------------------------------------------------
|
|
# 2: Embed the Python interpreter for keywords like PYACTION and PYINPUT
|
|
if (OPM_ENABLE_EMBEDDED_PYTHON)
|
|
add_subdirectory(python/pybind11)
|
|
target_include_directories(opmcommon SYSTEM PRIVATE "python/pybind11/include;${PYTHON_INCLUDE_DIRS}")
|
|
target_link_libraries(opmcommon PUBLIC ${PYTHON_LIBRARY})
|
|
|
|
add_definitions(-DEMBEDDED_PYTHON)
|
|
endif()
|
|
endif()
|