forked from SeisSol/SeisSol
-
Notifications
You must be signed in to change notification settings - Fork 0
/
CMakeLists.txt
459 lines (385 loc) · 16 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
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
cmake_minimum_required(VERSION 3.10)
# use <PackageName>_ROOT variables
if(POLICY CMP0074)
cmake_policy(SET CMP0074 NEW)
endif()
# honor CMAKE_REQUIRED_LIBRARIES in the include file check macros
if(POLICY CMP0075)
cmake_policy(SET CMP0075 NEW)
endif()
project(SeisSol LANGUAGES C CXX Fortran)
# set hardware specific definition needed for seissol compilation
# 'process_users_input' returns the following:
#
# switches: HDF5, NETCDF, METIS, MPI, OPENMP, ASAGI, SIONLIB, MEMKIND, PROXY_PYBINDING, ENABLE_PIC_COMPILATION
#
# user's input: HOST_ARCH, DEVICE_ARCH, DEVICE_SUB_ARCH,
# ORDER, NUMBER_OF_MECHANISMS, EQUATIONS,
# PRECISION, DYNAMIC_RUPTURE_METHOD,
# NUMBER_OF_FUSED_SIMULATIONS,
# MEMORY_LAYOUT, COMMTHREAD,
# LOG_LEVEL, LOG_LEVEL_MASTER,
# GEMM_TOOLS_LIST, EXTRA_CXX_FLAGS
#
# derived input: REAL_SIZE_IN_BYTES, ALIGNMENT, ARCH_STRING, NUMBER_OF_QUANTITIES, WITH_GPU
#
include(cmake/process_users_input.cmake)
set(HARDWARE_DEFINITIONS "ALIGNMENT=${ALIGNMENT}"
"REAL_SIZE=${REAL_SIZE_IN_BYTES}"
"ALIGNED_REAL_SIZE=${REAL_SIZE_IN_BYTES}")
# TODO: Move one dir up.
set(CMAKE_MODULE_PATH "${PROJECT_SOURCE_DIR}/cmake" "${CMAKE_ROOT}/Modules")
set(CMAKE_CXX_STANDARD 17)
set(CMAKE_CXX_STANDARD_REQUIRED ON)
set(CMAKE_CXX_EXTENSIONS OFF)
set_property(CACHE CMAKE_BUILD_TYPE PROPERTY STRINGS
"Debug" "Release" "RelWithDebInfo") # MinSizeRel is useless for us
if(NOT CMAKE_BUILD_TYPE)
set(CMAKE_BUILD_TYPE Release)
message(STATUS "Set build type to Release as none was supplied.")
endif()
# Generate version.h
include(GetGitRevisionDescription)
git_describe(PACKAGE_GIT_VERSION --dirty=\ \(modified\) --always)
configure_file("src/version_template.h"
"${CMAKE_CURRENT_BINARY_DIR}/src/version.h")
message(STATUS "Current Git description: " ${PACKAGE_GIT_VERSION})
add_custom_target(build-time-make-directory ALL
COMMAND ${CMAKE_COMMAND} -E make_directory "${CMAKE_CURRENT_BINARY_DIR}/src/generated_code")
set(GENERATED_FILES_FOR_SEISSOL src/generated_code/subroutine.h
src/generated_code/tensor.cpp
src/generated_code/subroutine.cpp
src/generated_code/gpulike_subroutine.cpp
src/generated_code/tensor.h
src/generated_code/init.cpp
src/generated_code/init.h
src/generated_code/kernel.h
src/generated_code/kernel.cpp)
add_custom_command(
COMMAND
"/usr/bin/env" python3 ${CMAKE_CURRENT_SOURCE_DIR}/generated_code/generate.py
"--equations" ${EQUATIONS}
"--matricesDir" ${CMAKE_CURRENT_SOURCE_DIR}/generated_code/matrices
"--outputDir" ${CMAKE_CURRENT_BINARY_DIR}/src/generated_code
"--host_arch" ${HOST_ARCH_STR}
"--device_arch" ${DEVICE_ARCH_STR}
"--device_backend" ${DEVICE_BACKEND}
"--order" ${ORDER}
"--numberOfMechanisms" ${NUMBER_OF_MECHANISMS}
"--memLayout" ${MEMORY_LAYOUT}
"--multipleSimulations" ${NUMBER_OF_FUSED_SIMULATIONS}
"--dynamicRuptureMethod" ${DYNAMIC_RUPTURE_METHOD}
"--PlasticityMethod" ${PLASTICITY_METHOD}
"--gemm_tools" ${GEMM_TOOLS_LIST}
WORKING_DIRECTORY ${CMAKE_SOURCE_DIR}/generated_code
DEPENDS
build-time-make-directory
generated_code/arch.py
generated_code/memlayout.py
generated_code/viscoelastic.py
generated_code/DynamicRupture.py
generated_code/Plasticity.py
generated_code/multSim.py
generated_code/aderdg.py
generated_code/generate.py
generated_code/Point.py
generated_code/elastic.py
generated_code/viscoelastic2.py
generated_code/poroelastic.py
generated_code/anisotropic.py
generated_code/SurfaceDisplacement.py
generated_code/NodalBoundaryConditions.py
generated_code/matrices/matrices_poroelastic.xml
OUTPUT
${GENERATED_FILES_FOR_SEISSOL}
COMMENT "Codegen for tensor stuff."
)
add_custom_target(SeisSol-codegen ALL DEPENDS ${GENERATED_FILES_FOR_SEISSOL})
find_package(GemmTools REQUIRED)
include(src/sources.cmake)
# enforce code generation to run before any other target
add_dependencies(SeisSol-lib SeisSol-codegen)
if (WITH_GPU)
add_dependencies(SeisSol-device-lib SeisSol-codegen)
endif()
if(GemmTools_INCLUDE_DIRS)
target_include_directories(SeisSol-lib PUBLIC ${GemmTools_INCLUDE_DIRS})
endif()
if(GemmTools_LIBRARIES)
target_link_libraries(SeisSol-lib PUBLIC ${GemmTools_LIBRARIES})
endif()
if(GemmTools_COMPILER_DEFINITIONS)
target_compile_definitions(SeisSol-lib PUBLIC ${GemmTools_COMPILER_DEFINITIONS})
endif()
# Find appropriate compiler flags based on the target computer architecture
# and the target compiler
# 'get_arch_specific_flags' returns: CPU_ARCH_FLAGS
include(cmake/cpu_arch_flags.cmake)
get_arch_flags(${HOST_ARCH} ${CMAKE_CXX_COMPILER_ID})
# set hardware/compiler specific definitions and flags
target_compile_definitions(SeisSol-lib PUBLIC ${HARDWARE_DEFINITIONS})
target_compile_options(SeisSol-lib PUBLIC ${CPU_ARCH_FLAGS})
target_compile_definitions(SeisSol-lib PUBLIC LOGLEVEL=${LOG_LEVEL})
target_compile_definitions(SeisSol-lib PUBLIC LOG_LEVEL=${LOG_LEVEL_MASTER}
LOGLEVEL0=${LOG_LEVEL_MASTER})
if (PLASTICITY_METHOD STREQUAL "ip")
target_compile_definitions(SeisSol-lib PUBLIC USE_PLASTICITY_IP)
elseif (PLASTICITY_METHOD STREQUAL "nb")
target_compile_definitions(SeisSol-lib PUBLIC USE_PLASTICITY_NB)
endif()
# enable interproc. opts for small cores
#if cpu in ['knc', 'knl', 'skx']:
# flags.extend(['-ip'])
# Libs
include(ExternalProject)
find_package(easi 1.0.0 REQUIRED)
target_link_libraries(SeisSol-lib PUBLIC easi::easi)
if (OPENMP)
find_package(OpenMP REQUIRED)
target_link_libraries(SeisSol-lib PUBLIC OpenMP::OpenMP_CXX)
target_link_libraries(SeisSol-lib PUBLIC OpenMP::OpenMP_Fortran)
target_compile_definitions(SeisSol-lib PUBLIC OMP OMPI_SKIP_MPICXX)
endif()
if (MPI)
find_package(MPI REQUIRED)
target_include_directories(SeisSol-lib SYSTEM PUBLIC ${MPI_CXX_INCLUDE_PATH})
target_link_libraries(SeisSol-lib PUBLIC MPI::MPI_C)
target_include_directories(SeisSol-lib SYSTEM PUBLIC ${MPI_Fortran_INCLUDE_PATH})
target_link_libraries(SeisSol-lib PUBLIC MPI::MPI_Fortran)
target_compile_definitions(SeisSol-lib PUBLIC USE_MPI PARALLEL)
if (MINI_SEISSOL)
target_compile_definitions(SeisSol-lib PUBLIC USE_MINI_SEISSOL)
endif()
endif()
if (COMMTHREAD)
target_compile_definitions(SeisSol-lib PUBLIC USE_COMM_THREAD)
endif()
if (NUMA_AWARE_PINNING)
target_compile_definitions(SeisSol-lib PUBLIC USE_NUMA_AWARE_PINNING)
find_package(NUMA REQUIRED)
target_include_directories(SeisSol-lib SYSTEM PUBLIC ${NUMA_INCLUDE_DIR})
target_link_libraries(SeisSol-lib PUBLIC ${NUMA_LIBRARY})
endif()
#set(HDF5_PREFER_PARALLEL True)
if (NETCDF)
find_package(NetCDF REQUIRED)
target_include_directories(SeisSol-lib PUBLIC ${NetCDF_INCLUDE_DIRS})
target_link_libraries(SeisSol-lib PUBLIC ${NetCDF_LIBRARY})
target_compile_definitions(SeisSol-lib PUBLIC USE_NETCDF)
endif()
if (HDF5)
if (MPI)
set(HDF5_PREFER_PARALLEL True)
endif()
find_package(HDF5 REQUIRED
COMPONENTS C HL)
target_include_directories(SeisSol-lib PUBLIC ${HDF5_INCLUDE_DIRS})
target_link_libraries(SeisSol-lib PUBLIC ${HDF5_C_HL_LIBRARIES} ${HDF5_C_LIBRARIES})
target_compile_definitions(SeisSol-lib PUBLIC USE_HDF)
endif()
# Parmetis
if (METIS)
find_package(METIS REQUIRED)
if (NOT METIS_64_BIT_INTEGER)
message(WARNING "Found METIS which does not support IDXTYPEWIDTH = 64. But this is required for SeisSol.")
endif()
find_package(ParMETIS REQUIRED)
target_include_directories(SeisSol-lib PUBLIC ${PARMETIS_INCLUDE_DIRS})
target_link_libraries(SeisSol-lib PUBLIC ${PARMETIS_LIBRARIES})
target_compile_definitions(SeisSol-lib PUBLIC USE_METIS)
endif()
find_package(PkgConfig REQUIRED)
if (ASAGI)
# todo warn if netcdf is off
pkg_check_modules(ASAGI REQUIRED IMPORTED_TARGET asagi) # asagi_nompi?
target_compile_definitions(SeisSol-lib PUBLIC USE_ASAGI)
target_link_libraries(SeisSol-lib PUBLIC ${ASAGI_STATIC_LDFLAGS})
target_include_directories(SeisSol-lib PUBLIC ${ASAGI_INCLUDE_DIRS})
target_compile_options(SeisSol-lib PUBLIC ${ASAGI_CFLAGS} ${ASAGI_CFLAGS_OTHER})
endif()
if (MEMKIND)
find_package(Memkind REQUIRED)
target_include_directories(SeisSol-lib PUBLIC ${MEMKIND_INCLUDE_DIR})
target_link_libraries(SeisSol-lib PUBLIC ${MEMKIND_LIBRARIES})
target_compile_definitions(SeisSol-lib PUBLIC USE_MEMKIND)
endif()
if(${EQUATIONS} STREQUAL "poroelastic")
include(CheckLanguage)
check_language(Fortran)
if(CMAKE_Fortran_COMPILER)
enable_language(Fortran)
include(FortranCInterface)
FortranCInterface_HEADER(FC.h MACRO_NAMESPACE "FC_")
find_package(LAPACK REQUIRED)
target_include_directories(SeisSol-lib PUBLIC ${CMAKE_CURRENT_BINARY_DIR})
message(${LAPACK_LIBRARIES})
target_link_libraries(SeisSol-lib PUBLIC ${LAPACK_LIBRARIES})
else()
message(FATAL_ERROR "SeisSol needs a Fortran compiler.")
endif()
endif()
if (WITH_GPU)
add_subdirectory(submodules/Device)
target_include_directories(SeisSol-lib PUBLIC submodules/Device)
target_link_libraries(SeisSol-lib PUBLIC device)
target_compile_definitions(SeisSol-lib PUBLIC ACL_DEVICE)
target_link_libraries(SeisSol-lib PUBLIC SeisSol-device-lib)
endif()
if (INTEGRATE_QUANTITIES)
target_compile_definitions(SeisSol-lib PUBLIC INTEGRATE_QUANTITIES)
endif()
if (ADDRESS_SANITIZER_DEBUG)
target_link_libraries(SeisSol-lib PUBLIC debug
-fno-omit-frame-pointer -fsanitize=address -fsanitize-recover=address
)
endif()
# Note: it is better to include `eigen3` `async` as
# system headers because they emit lot's of warnings
# from clang. Most of them are issues with respect
# to overriden virtual methods
target_include_directories(SeisSol-lib SYSTEM PUBLIC
${CMAKE_CURRENT_SOURCE_DIR}/submodules/eigen3
${CMAKE_CURRENT_SOURCE_DIR}/submodules/async
)
find_package(yaml-cpp REQUIRED
PATHS ${CMAKE_CURRENT_BINARY_DIR})
target_include_directories(SeisSol-lib PUBLIC
${CMAKE_CURRENT_SOURCE_DIR}/src
${CMAKE_CURRENT_SOURCE_DIR}/submodules
${CMAKE_CURRENT_SOURCE_DIR}/submodules/yateto/include
${CMAKE_CURRENT_BINARY_DIR}/src/
)
target_compile_definitions(SeisSol-lib PUBLIC
CONVERGENCE_ORDER=${ORDER}
NUMBER_OF_QUANTITIES=${NUMBER_OF_QUANTITIES}
NUMBER_OF_RELAXATION_MECHANISMS=${NUMBER_OF_MECHANISMS}
ENABLE_MATRIX_PREFETCH
)
# Fortran compliler settings
set(CMAKE_Fortran_FLAGS "${CMAKE_Fortran_FLAGS} -cpp")
if ("${CMAKE_Fortran_COMPILER_ID}" STREQUAL "GNU")
target_compile_options(SeisSol-lib PUBLIC $<$<COMPILE_LANGUAGE:Fortran>:-ffree-line-length-none -fdefault-real-8 -Wno-unused-parameter>)
elseif ("${CMAKE_Fortran_COMPILER_ID}" STREQUAL "Intel")
# todo intel, is needed: -align -align array64byte
# todo -r8 -WB is needed for intel (8 byte precision for reals)
set(CMAKE_Fortran_FLAGS "${CMAKE_Fortran_FLAGS} -cpp -r8 -WB")
endif()
# C++ compiler settings
if ("${CMAKE_CXX_COMPILER_ID}" STREQUAL "GNU")
target_compile_options(SeisSol-lib PUBLIC -pedantic $<$<OR:$<COMPILE_LANGUAGE:CXX>,$<COMPILE_LANGUAGE:C>>:-Wall -Wextra -Wno-unused-parameter -Wno-unknown-pragmas>)
# using GCC
elseif ("${CMAKE_CXX_COMPILER_ID}" STREQUAL "Intel")
# using Intel C++
# todo remove std?, is ffreestanding needed?
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++17")
# Activate interprocedual optimization.
#set_property(TARGET SeisSol-lib PROPERTY INTERPROCEDURAL_OPTIMIZATION TRUE)
elseif ("${CMAKE_CXX_COMPILER_ID}" STREQUAL "Clang")
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++17 -Wall -Wextra -pedantic -Wno-unused-parameter")
endif()
find_package(FILESYSTEM REQUIRED)
target_link_libraries(SeisSol-lib PUBLIC std::filesystem)
# Generated code does only work without red-zone.
if (HAS_REDZONE)
set_source_files_properties(
${CMAKE_CURRENT_BINARY_DIR}/src/generated_code/subroutine.cpp PROPERTIES COMPILE_FLAGS -mno-red-zone
)
endif()
# adjust prefix name of executables
if ("${DEVICE_ARCH_STR}" STREQUAL "none")
set(EXE_NAME_PREFIX "${CMAKE_BUILD_TYPE}_${HOST_ARCH_STR}_${ORDER}_${EQUATIONS}")
else()
set(EXE_NAME_PREFIX "${CMAKE_BUILD_TYPE}_${DEVICE_ARCH_STR}_${DEVICE_BACKEND}_${ORDER}_${EQUATIONS}")
endif()
add_executable(SeisSol-bin src/main.cpp)
target_link_libraries(SeisSol-bin PUBLIC SeisSol-lib)
set_target_properties(SeisSol-bin PROPERTIES OUTPUT_NAME "SeisSol_${EXE_NAME_PREFIX}")
# SeisSol proxy-core
add_library(SeisSol-proxy-core auto_tuning/proxy/src/proxy_seissol.cpp)
target_link_libraries(SeisSol-proxy-core PUBLIC SeisSol-lib)
if (PROXY_PYBINDING)
if ("${CMAKE_CXX_COMPILER_ID}" STREQUAL "Intel")
find_package(pybind11 2.6.2 REQUIRED)
else()
find_package(pybind11 2.3.0 REQUIRED)
endif()
pybind11_add_module(seissol_proxy_bindings auto_tuning/proxy/src/proxy_bindings.cpp)
target_link_libraries(seissol_proxy_bindings PUBLIC SeisSol-proxy-core)
file(COPY auto_tuning/proxy/src/proxy-runners DESTINATION ${CMAKE_CURRENT_BINARY_DIR})
endif()
# C/C++ proxy interface
add_executable(SeisSol-proxy auto_tuning/proxy/src/proxy_main.cpp)
target_link_libraries(SeisSol-proxy PUBLIC SeisSol-proxy-core SeisSol-lib)
set_target_properties(SeisSol-proxy PROPERTIES OUTPUT_NAME "SeisSol_proxy_${EXE_NAME_PREFIX}")
if (LIKWID)
find_package(likwid REQUIRED)
target_compile_definitions(SeisSol-proxy-core PUBLIC LIKWID_PERFMON)
target_link_libraries(SeisSol-proxy-core PRIVATE likwid::likwid)
endif()
if (TESTING)
enable_testing()
include(cmake/doctest.cmake)
# Coverage
if(COVERAGE AND CMAKE_COMPILER_IS_GNUCXX)
include(cmake/CodeCoverage.cmake)
append_coverage_compiler_flags()
setup_target_for_coverage_lcov(
NAME SeisSol-coverage
EXECUTABLE SeisSol-serial-test
EXCLUDE "/usr/*"
"submodules/*"
"*/tests/*"
"external/*"
"*/yaml-cpp-install/*"
)
endif()
set(seissol_test_sources
src/tests/Model/TestModel.cpp
src/tests/Initializer/TestInitializer.cpp
src/tests/Numerical_aux/TestNumerical_aux.cpp
src/tests/Geometry/TestGeometry.cpp
src/tests/Kernel/TestKernel.cpp
src/tests/SourceTerm/TestSourceTerm.cpp
src/tests/Pipeline/TestPipeline.cpp
src/tests/ResultWriter/TestResultWriter.cpp
)
if (TESTING_GENERATED)
set(seissol_test_sources
${seissol_test_sources}
${CMAKE_CURRENT_BINARY_DIR}/src/generated_code/test-kernel.cpp
)
endif()
configure_file(
${CMAKE_CURRENT_SOURCE_DIR}/src/tests/Initializer/time_stepping/mesh.h5
${CMAKE_CURRENT_BINARY_DIR}/Testing/mesh.h5
COPYONLY
)
configure_file(
${CMAKE_CURRENT_SOURCE_DIR}/src/tests/Initializer/time_stepping/material.yaml
${CMAKE_CURRENT_BINARY_DIR}/Testing/material.yaml
COPYONLY
)
configure_file(
${CMAKE_CURRENT_SOURCE_DIR}/src/tests/ResultWriter/receiver_correct.dat
${CMAKE_CURRENT_BINARY_DIR}/Testing/receiver_correct.dat
COPYONLY
)
if (NETCDF)
configure_file(
${CMAKE_CURRENT_SOURCE_DIR}/src/tests/Reader/source_loh.nrf
${CMAKE_CURRENT_BINARY_DIR}/Testing/source_loh.nrf
COPYONLY
)
set(seissol_test_sources ${seissol_test_sources} ${CMAKE_CURRENT_SOURCE_DIR}/src/tests/Reader/ReaderTest.cpp)
endif()
add_executable(SeisSol-serial-test
${seissol_test_sources}
src/tests/test_main.cpp)
target_link_libraries(SeisSol-serial-test PRIVATE SeisSol-lib)
target_include_directories(SeisSol-serial-test PUBLIC external/)
doctest_discover_tests(SeisSol-serial-test)
# Avoid duplicate definition of FLOP counters
target_compile_definitions(SeisSol-serial-test PRIVATE YATETO_TESTING_NO_FLOP_COUNTER)
endif()
# https://blog.kitware.com/static-checks-with-cmake-cdash-iwyu-clang-tidy-lwyu-cpplint-and-cppcheck/
# https://ortogonal.github.io/cmake-clang-tidy/