Remove/replace many mentions of Jenkins
[alexxy/gromacs.git] / cmake / gmxManageFFTLibraries.cmake
1 #
2 # This file is part of the GROMACS molecular simulation package.
3 #
4 # Copyright (c) 2012,2013,2014,2015,2016 by the GROMACS development team.
5 # Copyright (c) 2017,2018,2019,2020,2021, by the GROMACS development team, led by
6 # Mark Abraham, David van der Spoel, Berk Hess, and Erik Lindahl,
7 # and including many others, as listed in the AUTHORS file in the
8 # top-level source directory and at http://www.gromacs.org.
9 #
10 # GROMACS is free software; you can redistribute it and/or
11 # modify it under the terms of the GNU Lesser General Public License
12 # as published by the Free Software Foundation; either version 2.1
13 # of the License, or (at your option) any later version.
14 #
15 # GROMACS is distributed in the hope that it will be useful,
16 # but WITHOUT ANY WARRANTY; without even the implied warranty of
17 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
18 # Lesser General Public License for more details.
19 #
20 # You should have received a copy of the GNU Lesser General Public
21 # License along with GROMACS; if not, see
22 # http://www.gnu.org/licenses, or write to the Free Software Foundation,
23 # Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA.
24 #
25 # If you want to redistribute modifications to GROMACS, please
26 # consider that scientific software is very special. Version
27 # control is crucial - bugs must be traceable. We will be happy to
28 # consider code for inclusion in the official distribution, but
29 # derived work must not be called official GROMACS. Details are found
30 # in the README & COPYING files - if they are missing, get the
31 # official version at http://www.gromacs.org.
32 #
33 # To help us fund GROMACS development, we humbly ask that you cite
34 # the research papers on the package. Check out http://www.gromacs.org.
35
36 # Manage setup of the different FFT libraries we can use in Gromacs.
37 set(PKG_FFT "")
38 set(PKG_FFT_LIBS "")
39 # Intel 11 and up makes life somewhat easy if you just want to use
40 # all their stuff. It's not easy if you only want some of their
41 # stuff...
42 set(MKL_MANUALLY FALSE)
43 if (GMX_FFT_LIBRARY STREQUAL "MKL" AND NOT GMX_INTEL_LLVM)
44     # The user will have to provide the set of magic libraries in
45     # MKL_LIBRARIES (see below), which we cache (non-advanced), so that they
46     # don't have to keep specifying it, and can easily see that
47     # CMake is still using that information.
48     set(MKL_MANUALLY TRUE)
49 endif()
50 set(MKL_LIBRARIES_FORMAT_DESCRIPTION "Use full paths to library files, in the right order, and separated by semicolons.")
51 gmx_dependent_cache_variable(
52     MKL_LIBRARIES
53     "List of libraries for linking to MKL. ${MKL_LIBRARIES_FORMAT_DESCRIPTION}"
54     STRING ""
55     MKL_MANUALLY)
56 gmx_dependent_cache_variable(
57     MKL_INCLUDE_DIR
58     "Path to mkl.h (non-inclusive)."
59     PATH ""
60     MKL_MANUALLY)
61 if(${GMX_FFT_LIBRARY} STREQUAL "FFTW3")
62     # ${FFTW} must be in upper case
63     if(GMX_DOUBLE)
64         set(FFTW "FFTW")
65     else()
66         set(FFTW "FFTWF")
67     endif()
68
69     if(GMX_BUILD_OWN_FFTW)
70
71         if(MSVC)
72             message(FATAL_ERROR "Cannot build FFTW3 automatically (GMX_BUILD_OWN_FFTW=ON) in Visual Studio")
73         endif()
74         if(CMAKE_GENERATOR STREQUAL "Ninja")
75             message(FATAL_ERROR "Cannot build FFTW3 automatically (GMX_BUILD_OWN_FFTW=ON) with ninja")
76         endif()
77
78         add_subdirectory(src/external/build-fftw)
79         include_directories(BEFORE ${${FFTW}_INCLUDE_DIRS})
80         # libgmxfftw is always built static, so libgromacs does not
81         # have a dependency on anything, so PKG_FFT should be empty
82         set(PKG_FFT "")
83         set(FFT_STATUS_MESSAGE "Using external FFT library - FFTW3 build managed by GROMACS")
84     else()
85         string(TOLOWER "${FFTW}" LOWERFFTW)
86         find_package(FFTW COMPONENTS ${LOWERFFTW})
87
88         if(NOT ${FFTW}_FOUND)
89             MESSAGE(FATAL_ERROR "Cannot find FFTW 3 (with correct precision - libfftw3f for mixed-precision GROMACS or libfftw3 for double-precision GROMACS). Either choose the right precision, choose another FFT(W) library (-DGMX_FFT_LIBRARY), enable the advanced option to let GROMACS build FFTW 3 for you (-DGMX_BUILD_OWN_FFTW=ON), or use the really slow GROMACS built-in fftpack library (-DGMX_FFT_LIBRARY=fftpack).")
90         endif()
91
92         set(PKG_FFT "${${FFTW}_PKG}")
93         include_directories(SYSTEM ${${FFTW}_INCLUDE_DIRS})
94
95         if(NOT WIN32) # Detection doesn't work on Windows
96           if ((${GMX_SIMD_ACTIVE} MATCHES "SSE" OR ${GMX_SIMD_ACTIVE} MATCHES "AVX") AND NOT ${FFTW}_HAVE_SIMD)
97               set(FFT_WARNING_MESSAGE "The fftw library found is compiled without SIMD support, which makes it slow. Consider recompiling it or contact your admin")
98           else()
99               if(${GMX_SIMD_ACTIVE} MATCHES "AVX" AND NOT (${FFTW}_HAVE_SSE OR ${FFTW}_HAVE_SSE2))
100                   # If we end up here we have an AVX Gromacs build, and
101                   # FFTW with SIMD.
102                   set(FFT_WARNING_MESSAGE "The FFTW library was compiled with neither --enable-sse nor --enable-sse2; those would have enabled SSE(2) SIMD instructions. This will give suboptimal performance. You should (re)compile the FFTW library with --enable-sse2 and --enable-avx (and --enable-avx2 or --enable-avx512 if supported).")
103               endif()
104           endif()
105         endif()
106
107         find_path(ARMPL_INCLUDE_DIR "armpl.h" HINTS ${${FFTW}_INCLUDE_DIRS}
108             NO_DEFAULT_PATH
109             NO_CMAKE_ENVIRONMENT_PATH
110             NO_CMAKE_PATH
111             NO_SYSTEM_ENVIRONMENT_PATH
112             NO_CMAKE_SYSTEM_PATH)
113         mark_as_advanced(ARMPL_INCLUDE_DIR)
114         if (ARMPL_INCLUDE_DIR)
115             set(GMX_FFT_ARMPL_FFTW3 1)
116             set(FFT_STATUS_MESSAGE "Using external FFT library - ARM Performance Library (FFTW3 compatibility mode)")
117         else()
118             set(FFT_STATUS_MESSAGE "Using external FFT library - FFTW3")
119         endif()
120     endif()
121     if (NOT GMX_FFT_ARMPL_FFTW3)
122         set(GMX_FFT_FFTW3 1)
123     endif()
124
125     set(FFT_LIBRARIES ${${FFTW}_LIBRARIES})
126 elseif(${GMX_FFT_LIBRARY} STREQUAL "MKL")
127     # Intel compilers make life somewhat easy if you just want to use
128     # all their stuff. It's not easy if you only want some of their
129     # stuff...
130     if (NOT MKL_MANUALLY)
131         # The next line takes care of everything for MKL
132         if (WIN32)
133             set(FFT_LINKER_FLAGS "/Qmkl:sequential")
134         elseif(GMX_INTEL_LLVM AND GMX_INTEL_LLVM_VERSION GREATER_EQUAL 2021020)
135             set(FFT_LINKER_FLAGS "-qmkl=sequential")
136         else()
137             set(FFT_LINKER_FLAGS "-mkl=sequential")
138         endif()
139         # Some versions of icc require this in order that mkl.h can be
140         # found at compile time.
141         list(APPEND EXTRA_C_FLAGS ${FFT_LINKER_FLAGS})
142         list(APPEND EXTRA_CXX_FLAGS ${FFT_LINKER_FLAGS})
143
144         set(MKL_ERROR_MESSAGE "Make sure you have configured your compiler so that ${FFT_LINKER_FLAGS} will work.")
145     else()
146         include_directories(SYSTEM ${MKL_INCLUDE_DIR})
147         set(FFT_LIBRARIES "${MKL_LIBRARIES}")
148         set(MKL_ERROR_MESSAGE "The include path to mkl.h in MKL_INCLUDE_DIR, and the link libraries in MKL_LIBRARIES=${MKL_LIBRARIES} need to match what the MKL documentation says you need for your system: ${MKL_LIBRARIES_FORMAT_DESCRIPTION}")
149         # Convert the semi-colon separated list to a list of
150         # command-line linker arguments so that code using our
151         # pkgconfig setup can use it.
152         string(REGEX REPLACE ";" " " PKG_FFT_LIBS "${MKL_LIBRARIES}")
153     endif()
154
155     # Check MKL works. If we were in a non-global scope, we wouldn't
156     # have to play nicely.
157     set(old_CMAKE_REQUIRED_FLAGS "${CMAKE_REQUIRED_FLAGS}")
158     set(CMAKE_REQUIRED_FLAGS "${FFT_LINKER_FLAGS}")
159     set(old_CMAKE_REQUIRED_LIBRARIES "${CMAKE_REQUIRED_LIBRARIES}")
160     set(CMAKE_REQUIRED_LIBRARIES "${FFT_LIBRARIES}")
161
162     check_function_exists(DftiCreateDescriptor TEST_MKL)
163
164     set(CMAKE_REQUIRED_FLAGS "${old_CMAKE_REQUIRED_FLAGS}")
165     set(CMAKE_REQUIRED_LIBRARIES "${old_CMAKE_REQUIRED_LIBRARIES}")
166
167     if(NOT TEST_MKL)
168         # Hack to help the user vary MKL settings until they work.
169         # TODO Make this logic more useful.
170         unset(TEST_MKL CACHE)
171         message(FATAL_ERROR "Linking with MKL was requested, but was not successful: ${MKL_ERROR_MESSAGE}")
172     endif()
173
174     # Set variables to signal that we have MKL available and should use it for FFTs.
175     set(GMX_FFT_MKL 1)
176     set(HAVE_LIBMKL 1)
177
178     set(FFT_STATUS_MESSAGE "Using external FFT library - Intel MKL")
179 elseif(${GMX_FFT_LIBRARY} STREQUAL "FFTPACK")
180     set(GMX_FFT_FFTPACK 1)
181     set(FFT_STATUS_MESSAGE "Using internal FFT library - fftpack")
182 else()
183     gmx_invalid_option_value(GMX_FFT_LIBRARY)
184 endif()
185 gmx_check_if_changed(FFT_CHANGED GMX_FFT_LIBRARY)
186 if (FFT_CHANGED)
187     if(FFT_WARNING_MESSAGE)
188         message(WARNING "${FFT_WARNING_MESSAGE}")
189     endif()
190     message(STATUS "${FFT_STATUS_MESSAGE}")
191 endif()
192
193 # enable threaded fftw3 if we've found it
194 if(FFTW3_THREADS OR FFTW3F_THREADS)
195     add_definitions(-DFFT5D_FFTW_THREADS)
196 endif()
197