summaryrefslogblamecommitdiff
path: root/var/spack/repos/builtin/packages/heffte/package.py
blob: 4516ba7ddd0054df0a7ae133b61f7503f5a2275c (plain) (tree)
1
2
3
4
5
6
7
8
9
                                                                         






                                                                         
                                                     


                                                 
                                                                 

                                                     
                                
                  
 

                                 
                                       
                                                                                               
                                                                                               
                                                                                             
                                                                                             

                                                                                             
                                         
                                            



                                                                                      
                                                                                                







                                                                                
 
                                                                                
                                                                                    
                                                                                   
                                                                   
                                                                                   

                                                                              
                                                                          



                                                                  
                                                                           
                                                                

                                             
                                                                         
 

                                 
                         
                

                                                                    
                                                                   


                                                                     

                                                                         

                                                 
                                                             
 















                                                                                      



                                                                           
                   
























































                                                                             
# Copyright 2013-2021 Lawrence Livermore National Security, LLC and other
# Spack Project Developers. See the top-level COPYRIGHT file for details.
#
# SPDX-License-Identifier: (Apache-2.0 OR MIT)

from spack import *


class Heffte(CMakePackage, CudaPackage, ROCmPackage):
    """Highly Efficient FFT for Exascale"""

    homepage = "https://bitbucket.org/icl/heffte"
    url      = "https://bitbucket.org/icl/heffte/get/v1.0.tar.gz"
    git      = "https://bitbucket.org/icl/heffte.git"

    maintainers = ['mkstoyanov']
    tags = ['e4s']

    test_requires_compiler = True

    version('develop', branch='master')
    version('2.1.0', sha256='527a3e21115231715a0342afdfaf6a8878d2dd0f02f03c92b53692340fd940b9')
    version('2.0.0', sha256='12f2b49a1a36c416eac174cf0cc50e729d56d68a9f68886d8c34bd45a0be26b6')
    version('1.0', sha256='0902479fb5b1bad01438ca0a72efd577a3529c3d8bad0028f3c18d3a4935ca74')
    version('0.2', sha256='4e76ae60982b316c2e873b2e5735669b22620fefa1fc82f325cdb6989bec78d1')
    version('0.1', sha256='d279a03298d2dc76574b1ae1031acb4ea964348cf359273d1afa4668b5bfe748')

    patch('threads10.patch', when='@1.0')
    patch('fortran200.patch', when='@2.0.0')

    variant('shared', default=True, description='Builds with shared libraries')
    variant('fftw', default=False, description='Builds with support for FFTW backend')
    variant('mkl',  default=False, description='Builds with support for MKL backend')
    variant('magma', default=False, description='Use helper methods from the UTK MAGMA library')
    variant('python', default=False, description='Install the Python bindings')
    variant('fortran', default=False, description='Install the Fortran modules')

    depends_on('python@3.0:', when='+python', type=('build', 'run'))
    depends_on('py-mpi4py', when='+python', type=('build', 'run'))
    depends_on('py-numpy', when='+python', type=('build', 'run'))
    depends_on('py-numba', when='+python+cuda', type=('build', 'run'))
    extends('python', when='+python', type=('build', 'run'))

    conflicts('~fftw', when='@:2.1.0~mkl~cuda')  # requires at least one backend
    conflicts('+fftw', when='+mkl@:1.0')  # old API supports at most one CPU backend
    conflicts('^openmpi~cuda', when='+cuda')  # +cuda requires CUDA enabled OpenMPI
    conflicts('~cuda', when='+magma')  # magma requires CUDA or HIP
    conflicts('+rocm', when='@:2.1.0')  # heffte+rocm is in in development in spack
    conflicts('+python', when="@:1.0")  # python support was added post v1.0
    conflicts('+fortran', when="@:1.0")  # fortran support was added post v1.0
    conflicts('+magma', when="@:1.0")  # magma support was added post v1.0

    depends_on('mpi', type=('build', 'run'))

    depends_on('fftw@3.3.8:', when="+fftw", type=('build', 'run'))
    depends_on('intel-mkl@2018.0.128:', when="+mkl", type=('build', 'run'))
    depends_on('cuda@8.0:', when="+cuda", type=('build', 'run'))
    depends_on('hip@3.8.0:', when='+rocm')
    depends_on('rocfft@3.8.0:', when='+rocm')
    depends_on('magma@2.5.3:', when="+cuda+magma", type=('build', 'run'))

    examples_src_dir = 'examples'

    def cmake_args(self):
        args = [
            self.define_from_variant('BUILD_SHARED_LIBS', 'shared'),
            self.define_from_variant('Heffte_ENABLE_CUDA', 'cuda'),
            self.define_from_variant('Heffte_ENABLE_ROCM', 'rocm'),
            self.define_from_variant('Heffte_ENABLE_FFTW', 'fftw'),
            self.define_from_variant('Heffte_ENABLE_MKL', 'mkl'),
            self.define_from_variant('Heffte_ENABLE_MAGMA', 'magma'),
            self.define_from_variant('Heffte_ENABLE_FORTRAN', 'fortran'),
            self.define_from_variant('Heffte_ENABLE_PYTHON', 'python'),
            '-DBUILD_GPU={0:1s}'.format(
                'ON' if ('+cuda' in self.spec and
                         '+fftw' in self.spec) else 'OFF'), ]

        if '+cuda' in self.spec:
            cuda_arch = self.spec.variants['cuda_arch'].value
            if len(cuda_arch) > 0 or cuda_arch[0] != 'none':
                nvcc_flags = ""
                for nvflag in self.cuda_flags(cuda_arch):
                    nvcc_flags += "{0};".format(nvflag)

                args.append('-DCUDA_NVCC_FLAGS={0}'.format(nvcc_flags))

        if '+rocm' in self.spec:
            args.append('-DCMAKE_CXX_COMPILER={0}'.format(self.spec['hip'].hipcc))

            rocm_arch = self.spec.variants['amdgpu_target'].value
            if 'none' not in rocm_arch:
                args.append('-DCMAKE_CXX_FLAGS={0}'.format(self.hip_flags(rocm_arch)))

            # See https://github.com/ROCmSoftwarePlatform/rocFFT/issues/322
            if self.spec.satisfies('^cmake@3.21:'):
                args.append(self.define('__skip_rocmclang', 'ON'))

        return args

    @run_after('install')
    def cache_test_sources(self):
        """Copy the example source files after the package is installed to an
        install test subdirectory for use during `spack test run`."""
        self.cache_extra_test_sources([self.examples_src_dir])

    def test(self):
        cmake_file = join_path(self.install_test_root, 'CMakeLists.txt')
        test_bld_dir = join_path(self.install_test_root, '_build')
        exe_files = []
        mpi_procs = []

        with open(cmake_file, 'w') as cmkf:
            cmkf.write('cmake_minimum_required(VERSION 3.13)\n\n')
            cmkf.write('project(heffte_example LANGUAGES CXX)\n\n')
            cmkf.write('find_package(Heffte REQUIRED)\n\n')
            if '+fftw' in self.spec:
                src_file = '{0}/heffte_example_fftw.cpp'.format(
                    self.examples_src_dir)
                cmkf.write(
                    'add_executable(hf_fftw {0})\n'.format(src_file))
                cmkf.write(
                    'target_link_libraries(hf_fftw Heffte::Heffte)\n')
                exe_files.append('./hf_fftw')
                mpi_procs.append(4)
            if '+cuda' in self.spec:
                src_file = '{0}/heffte_example_cuda.cpp'.format(
                    self.examples_src_dir)
                cmkf.write(
                    'add_executable(hf_cuda {0})\n'.format(src_file))
                cmkf.write(
                    'target_link_libraries(hf_cuda Heffte::Heffte)\n')
                exe_files.append('./hf_cuda')
                mpi_procs.append(4)
            if '+fortran' in self.spec:
                src_file = '{0}/heffte_example_fftw.f90'.format(
                    self.examples_src_dir)
                cmkf.write(
                    'add_executable(hf_fort {0})\n'.format(src_file))
                cmkf.write(
                    'target_link_libraries(hf_fort Heffte::Fortran)\n')
                exe_files.append('./hf_fort')
                mpi_procs.append(2)

        with working_dir(test_bld_dir, create=True):
            cmake('..')
            make()
            test_args = ['-n', '4', ' ']
            mpiexe_f = which('srun', 'mpirun', 'mpiexec')
            if mpiexe_f:
                for exf, procs in zip(exe_files, mpi_procs):
                    test_args[-1] = exf
                    test_args[-2] = str(procs)
                    reason_str = 'Heffte smoke test, exe: {0}'.format(exf)
                    self.run_test(mpiexe_f.command, test_args,
                                  purpose=reason_str)