forked from NOAA-OWP/ngen
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathCMakeLists.txt
357 lines (310 loc) · 13.3 KB
/
CMakeLists.txt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
cmake_minimum_required(VERSION 3.10)
if(MPI_ACTIVE)
set (CMAKE_CXX_COMPILER "mpicxx")
endif()
########### Define project version and use via generated config header
project(ngen VERSION 0.1.0)
configure_file(include/NGenConfig.h.in include/NGenConfig.h)
if("${PROJECT_NAME}" STREQUAL "${CMAKE_PROJECT_NAME}")
option(PACKAGE_TESTS "Build automated tests" ON)
else()
option(PACKAGE_TESTS "Build automated tests")
endif()
if (NOT DEFINED CMAKE_C_COMPILER)
message(STATUS "Checking environment variable 'C' for C compiler")
if (DEFINED ENV{CC})
set(CMAKE_C_COMPILER $ENV{CC})
else()
message(ERROR "'CC' not set - unable to find C++ compiler")
endif()
else()
message(STATUS "Compiler: ${CMAKE_C_COMPILER}")
endif()
if (NOT DEFINED CMAKE_CXX_COMPILER)
message(STATUS "Checking environment variable 'CXX' for C++ compiler")
if (DEFINED ENV{CXX})
set(CMAKE_CXX_COMPILER $ENV{CXX})
else()
message(ERROR "'CXX' not set - unable to find C++ compiler")
endif()
else()
message(STATUS "Compiler: ${CMAKE_CXX_COMPILER}")
endif()
set(CMAKE_CXX_STANDARD 14)
set(CMAKE_CXX_STANDARD_REQUIRED True)
set(CMAKE_CXX_VISIBILITY_PRESET hidden)
include(ExternalProject)
set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} "${CMAKE_SOURCE_DIR}/cmake/modules/")
# Account for different OS and how that impacts shared lib file names
if ("${CMAKE_SYSTEM_NAME}" STREQUAL "Darwin")
set(SHARED_LIB_EXTENSION "dylib")
elseif(WIN32)
error( "Windows platforms are not currently supported" )
else()
set(SHARED_LIB_EXTENSION "so")
endif()
find_package(Git QUIET)
if(GIT_FOUND AND EXISTS "${PROJECT_SOURCE_DIR}/.git")
# Update submodules as needed
option(GIT_SUBMODULE "Check submodules during build" ON)
endif()
function(git_update_submodule PATH)
if(GIT_SUBMODULE)
message(STATUS "Updating submodule ${PATH}")
execute_process(COMMAND ${GIT_EXECUTABLE} submodule update --init -- ${PATH}
WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}
RESULT_VARIABLE GIT_SUBMOD_RESULT
)
if(NOT GIT_SUBMOD_RESULT EQUAL "0")
message(FATAL_ERROR "git submodule update --init for \"${PATH}\" failed with ${GIT_SUBMOD_RESULT}")
endif()
else()
message(FATAL_ERROR "Git not detected: cannot obtain submodule at \"${PATH}\" (but how is this even working?)")
endif()
endfunction()
set(PROJ_ROOT_INCLUDE_DIR ./include)
set(PROJ_ROOT_MODELS_DIR ./models)
set(HYMOD_DIR ${PROJ_ROOT_MODELS_DIR}/hymod)
set(HYMOD_INCLUDE_DIR ${HYMOD_DIR}/include)
# Find the Boost library and configure usage
set(Boost_USE_STATIC_LIBS OFF)
set(Boost_USE_MULTITHREADED ON)
set(Boost_USE_STATIC_RUNTIME OFF)
if(DEFINED ENV{BOOST_ROOT})
#set(Boost_INCLUDE_DIR $ENV{BOOST_ROOT})
set(BOOST_ROOT $ENV{BOOST_ROOT})
else()
# Look for version-specific Boost directory if available (known from Github Actions VM docs)
if(DEFINED ENV{BOOST_ROOT_1_72_0})
set(BOOST_ROOT $ENV{BOOST_ROOT_1_72_0})
set(ENV{BOOST_ROOT} $ENV{BOOST_ROOT_1_72_0})
else()
message(STATUS "No Boost root: $ENV{BOOST_ROOT}")
endif()
endif()
if(MPI_ACTIVE)
find_package(MPI)
add_compile_definitions(NGEN_MPI_ACTIVE)
endif()
if(LSTM_TORCH_LIB_ACTIVE)
find_package(Torch REQUIRED)
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${TORCH_CXX_FLAGS}")
endif()
find_package(Boost 1.72.0 REQUIRED)
# UDUNITS
# Since UDUNITS is currently not really optional (yet) let's make it default to on...
if (NOT (DEFINED UDUNITS_ACTIVE))
message(INFO " UDUNITS_ACTIVE not defined")
if (DEFINED ENV{UDUNITS_ACTIVE})
message(INFO " set as $ENV{UDUNITS_ACTIVE}")
set(UDUNITS_ACTIVE $ENV{UDUNITS_ACTIVE})
else()
set(UDUNITS_ACTIVE true)
endif()
endif()
if(UDUNITS_ACTIVE)
find_package(UDUNITS2)
if(UDUNITS2_LIBRARY)
#add_library(libudunits2 SHARED IMPORTED)
target_include_directories(libudunits2 INTERFACE ${UDUNITS2_INCLUDE})
#include_directories(${UDUNITS2_INCLUDE})
message("INFO Using UDUNITS2 at ${UDUNITS2_LIBRARY} and ${UDUNITS2_INCLUDE}")
else()
set(UDUNITS_DIR ${PROJECT_SOURCE_DIR}/extern/UDUNITS-2)
set(UDUNITS_BUILD_BASE ${CMAKE_CURRENT_BINARY_DIR}/UDUNITS-2)
set(UDUNITS_BUILD_DIR ${UDUNITS_BUILD_BASE}/build)
set(UDUNITS_INSTALL_DIR ${UDUNITS_BUILD_BASE}/build)
git_update_submodule(${UDUNITS_DIR})
ExternalProject_Add(libudunits2_build
PREFIX ${UDUNITS_BUILD_BASE}
SOURCE_DIR ${UDUNITS_DIR}
BINARY_DIR ${UDUNITS_BUILD_DIR}
INSTALL_DIR ${UDUNITS_INSTALL_DIR}
PATCH_COMMAND find ${PROJECT_SOURCE_DIR}/extern/UDUNITS-2 \( -type d -name .git -prune \) -o -type f -name "CMakeLists.txt" -exec sed -i -e "s/CMAKE_MINIMUM_REQUIRED(VERSION 3\.19)/CMAKE_MINIMUM_REQUIRED(VERSION 3.17)/" {} +
#CMAKE_ARGS -DBOOST_ROOT=${BOOST_ROOT} CMAKE_INSTALL_RPATH_USE_LINK_PATH="ON" -DCMAKE_INSTALL_PREFIX:PATH=<INSTALL_DIR>
CMAKE_ARGS -DCMAKE_INSTALL_PREFIX=${UDUNITS_INSTALL_DIR}
)
add_library(libudunits2 SHARED IMPORTED)
add_dependencies(libudunits2 libudunits2_build)
#set_property(TARGET libudunits2 PROPERTY INSTALL_RPATH_USE_LINK_PATH TRUE)
#set(CMAKE_MACOSX_RPATH 1)
message("INFO - Linking libudunits2 to path \"${UDUNITS_INSTALL_DIR}/lib/libudunits2.${SHARED_LIB_EXTENSION}\"")
set_property(TARGET libudunits2 PROPERTY IMPORTED_LOCATION "${UDUNITS_INSTALL_DIR}/lib/libudunits2.${SHARED_LIB_EXTENSION}")
list( APPEND CMAKE_INSTALL_RPATH ${UDUNITS_INSTALL_DIR}/lib ${UDUNITS_INSTALL_DIR}/include )
list( APPEND CMAKE_BUILD_RPATH ${UDUNITS_BUILD_DIR}/lib )
#set_property(TARGET libudunits2 PROPERTY GENERATED TRUE)
target_include_directories(libudunits2 INTERFACE "${UDUNITS_DIR}/lib") # This seems like it should have worked...??
include_directories("${UDUNITS_DIR}/lib")
endif()
#add_compile_definitions("NGEN_UDUNITS2_XML_PATH=\"${UDUNITS_INSTALL_DIR}/share/udunits/udunits2.xml\"")
endif()
add_executable(ngen
src/NGen.cpp
)
# Handle several steps for BMI C library logic and dependencies, at top level, if functionality is turned on
if(BMI_C_LIB_ACTIVE)
# Define associated preprocessor directive
add_compile_definitions(NGEN_BMI_C_LIB_ACTIVE)
endif()
# Configure whether Fortran BMI functionality is active
if (${BMI_FORTRAN_ACTIVE})
add_compile_definitions(NGEN_BMI_FORTRAN_ACTIVE)
endif()
#### Python stuff
# First, define NGEN_ACTIVATE_PYTHON, if not already defined
# Use NGEN_ACTIVATE_PYTHON environment variable if available, or default to true
if (NOT (DEFINED NGEN_ACTIVATE_PYTHON))
message(INFO " NGEN_ACTIVATE_PYTHON not defined")
if (DEFINED ENV{NGEN_ACTIVATE_PYTHON})
message(INFO " set as $ENV{NGEN_ACTIVATE_PYTHON}")
set(NGEN_ACTIVATE_PYTHON $ENV{NGEN_ACTIVATE_PYTHON})
else()
set(NGEN_ACTIVATE_PYTHON true)
endif()
endif()
# If Python set to be active, run steps to make sure it is included properly, and set compiler definition to true
if (${NGEN_ACTIVATE_PYTHON})
add_compile_definitions(ACTIVATE_PYTHON=true)
# If there is a virtual environment directory then look for several things there
if(EXISTS ${PROJECT_SOURCE_DIR}/.venv OR EXISTS ${PROJECT_SOURCE_DIR}/venv)
if (EXISTS ${PROJECT_SOURCE_DIR}/.venv)
set(PY_VENV_DIR ${PROJECT_SOURCE_DIR}/.venv)
else()
set(PY_VENV_DIR ${PROJECT_SOURCE_DIR}/venv)
endif()
# Start with looking for NumPy
file(GLOB_RECURSE NP_COMMON_HEADER ${PY_VENV_DIR}/*npy_common.h)
get_filename_component(NP_HEADER_DIR ${NP_COMMON_HEADER} DIRECTORY)
get_filename_component(NP_INCLUDE_DIR ${NP_HEADER_DIR} DIRECTORY)
set(Python_NumPy_INCLUDE_DIR ${NP_INCLUDE_DIR})
message(INFO " Found numpy include path ${Python_NumPy_INCLUDE_DIR}")
# Also try to get the real python root
message(INFO " Before Python exec ${PYTHON_EXECUTABLE}")
get_filename_component(PYTHON_EXECUTABLE ${PY_VENV_DIR}/bin/python REALPATH)
message(INFO " After Python exec ${PYTHON_EXECUTABLE}")
get_filename_component(PYTHON_BIN ${PYTHON_EXECUTABLE} DIRECTORY)
get_filename_component(PY_ROOT_DIR ${PYTHON_BIN} DIRECTORY)
set(Python_ROOT_DIR "${PY_ROOT_DIR}")
message(INFO " Python root dir to search is ${Python_ROOT_DIR}")
set(Python_FIND_VIRTUALENV ONLY)
# use, i.e. don't skip the full RPATH for the build tree
#set(CMAKE_SKIP_BUILD_RPATH FALSE)
#set(CMAKE_SKIP_BUILD_RPATH TRUE)
#set(CMAKE_INSTALL_RPATH "/Library/Developer/CommandLineTools/Library/Frameworks")
#message(INFO " ORIGIN is ${ORIGIN}")
#SET(CMAKE_INSTALL_RPATH_USE_LINK_PATH TRUE)
# add the automatically determined parts of the RPATH
# which point to directories outside the build tree to the install RPATH
#set(CMAKE_INSTALL_RPATH_USE_LINK_PATH TRUE)
#set(CMAKE_INSTALL_RPATH_USE_LINK_PATH FALSE)
# the RPATH to be used when installing, but only if it's not a system directory
#list(FIND CMAKE_PLATFORM_IMPLICIT_LINK_DIRECTORIES "${CMAKE_INSTALL_PREFIX}/lib" isSystemDir)
#if("${isSystemDir}" STREQUAL "-1")
# set(CMAKE_INSTALL_RPATH "${CMAKE_INSTALL_PREFIX}/lib")
#endif("${isSystemDir}" STREQUAL "-1")
# Alternatively, if there is something in the ENV for the user, try that
elseif(DEFINED ENV{Python_NumPy_INCLUDE_DIR})
set(Python_NumPy_INCLUDE_DIR $ENV{Python_NumPy_INCLUDE_DIR})
endif()
# if(DEFINED ENV{PYTHON_INCLUDE_DIR})
# set(PYTHON_INCLUDE_DIR $ENV{PYTHON_INCLUDE_DIR})
# message(INFO " Found 'PYTHON_INCLUDE_DIR' from ENV ($ENV{PYTHON_INCLUDE_DIR})")
# else()
# message(INFO " Did not find 'PYTHON_INCLUDE_DIR' in ENV")
# endif()
# if(DEFINED ENV{PYTHON_LIBRARIES})
# set(PYTHON_LIBRARIES $ENV{PYTHON_LIBRARIES})
# message(INFO " Found 'PYTHON_LIBRARIES' from ENV ($ENV{PYTHON_LIBRARIES})")
# else()
# message(INFO " Did not find 'PYTHON_LIBRARIES' in ENV")
# endif()
# include_directories(${PYTHON_INCLUDE_DIR})
find_package(Python 3.6.8 REQUIRED COMPONENTS Interpreter Development NumPy)
#message(INFO " the final Py lib is ${Python_LIBRARY}")
#message(INFO " the final Py libraries are ${Python_LIBRARIES}")
#message(INFO " the final Py library dirs are ${Python_LIBRARY_DIRS}")
#message(INFO " the final Py runtime library dirs are ${Python_RUNTIME_LIBRARY_DIRS}")
#message(INFO " the final Py std lib is ${Python_STDLIB}")
#message(INFO " the rpath is ${CMAKE_INSTALL_RPATH}")
add_subdirectory(extern/pybind11)
endif()
target_include_directories(ngen PUBLIC
"${PROJECT_BINARY_DIR}/include" # For generated config header file in binary tree
)
# In some IDEs (e.g., CLion), it is easier to work with the code in the included example of an "external" shared library
# when that code appears to be part of the main project, which can be done by add a statement like the ones below.
#
# However, these should be removed or commented out when running CMake builds and in the committed version of this file.
#
#add_subdirectory("extern/cfe")
#add_subdirectory("extern/test_bmi_c")
add_subdirectory("src/core")
add_dependencies(core libudunits2)
add_subdirectory("src/geojson")
add_subdirectory("src/realizations/catchment")
add_subdirectory("src/models/tshirt")
add_subdirectory("src/models/kernels/reservoir")
add_subdirectory("src/models/kernels/evapotranspiration")
#add_subdirectory("src/forcing")
target_link_libraries(ngen PUBLIC
NGen::core
NGen::core_catchment
#NGen::core_catchment_giuh
NGen::core_nexus
NGen::geojson
NGen::models_tshirt
NGen::realizations_catchment
NGen::kernels_reservoir
NGen::kernels_evapotranspiration
#NGen::kernels_reservoir_timeless
#NGen::forcing
NGen::core_mediator
libudunits2
)
add_executable(partitionGenerator
src/partitionGenerator.cpp
)
target_include_directories(partitionGenerator PUBLIC
"${PROJECT_BINARY_DIR}/include" # For generated config header file in binary tree
)
target_link_libraries(partitionGenerator PUBLIC
core
geojson
)
if(NGEN_ACTIVATE_ROUTING)
add_compile_definitions(NGEN_ROUTING_ACTIVE)
add_subdirectory("src/routing")
target_link_libraries(ngen PUBLIC
NGen::routing
)
endif()
if(LSTM_TORCH_LIB_ACTIVE)
add_compile_definitions(NGEN_LSTM_TORCH_LIB_ACTIVE)
add_subdirectory("src/models/lstm")
target_link_libraries(ngen PUBLIC
NGen::models_lstm
${TORCH_LIBRARIES}
)
endif()
# For automated testing with Google Test
if(PACKAGE_TESTS)
enable_testing()
include(GoogleTest)
add_subdirectory(test)
endif()
if(QUIET)
set(UDUNITS_QUIET true)
set(ET_QUIET true)
set(NGEN_QUIET true)
endif()
if(UDUNITS_QUIET)
add_compile_definitions(UDUNITS_QUIET)
endif()
#add_library(Hymod ${HYMOD_INCLUDE_DIR}/Hymod.h)
#set_target_properties(Hymod PROPERTIES LINKER_LANGUAGE CXX)
#target_compile_features(Hymod PUBLIC cxx_std_14)
#set_target_properties(ngen PROPERTIES LINKER_LANGUAGE CXX)
#
#target_compile_options(ngen PUBLIC -std=c++14 -Wall)
#target_compile_features(ngen PUBLIC cxx_std_14)
#add_subdirectory("src/geojson")