# Copyright 2012-2017 The Meson development team # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # http://www.apache.org/licenses/LICENSE-2.0 # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. from typing import List import subprocess, os from pathlib import Path from .c import CCompiler from .compilers import ( CompilerType, apple_buildtype_linker_args, gnulike_buildtype_args, gnulike_buildtype_linker_args, gnu_optimization_args, clike_debug_args, Compiler, GnuCompiler, ClangCompiler, ElbrusCompiler, IntelCompiler, PGICompiler ) from mesonbuild.mesonlib import ( EnvironmentException, MachineChoice, is_osx, LibType ) class FortranCompiler(Compiler): library_dirs_cache = CCompiler.library_dirs_cache program_dirs_cache = CCompiler.library_dirs_cache find_library_cache = CCompiler.library_dirs_cache def __init__(self, exelist, version, is_cross, exe_wrapper=None, **kwargs): self.language = 'fortran' Compiler.__init__(self, exelist, version, **kwargs) cc = CCompiler(exelist, version, is_cross, exe_wrapper, **kwargs) self.id = 'unknown' self.is_cross = cc.is_cross self.exe_wrapper = cc.exe_wrapper def get_display_language(self): return 'Fortran' def needs_static_linker(self): return CCompiler.needs_static_linker(self) def get_always_args(self): return CCompiler.get_always_args(self) def get_linker_debug_crt_args(self): return CCompiler.get_linker_debug_crt_args(self) def get_no_stdinc_args(self): return CCompiler.get_no_stdinc_args(self) def get_no_stdlib_link_args(self): return CCompiler.get_no_stdlib_link_args(self) def get_warn_args(self, level): return CCompiler.get_warn_args(self, level) def get_no_warn_args(self): return CCompiler.get_no_warn_args(self) def get_soname_args(self, *args): return CCompiler.get_soname_args(self, *args) def sanity_check(self, work_dir: Path, environment): """ Check to be sure a minimal program can compile and execute with this compiler & platform. """ work_dir = Path(work_dir) source_name = work_dir / 'sanitycheckf.f90' binary_name = work_dir / 'sanitycheckf' if binary_name.is_file(): binary_name.unlink() source_name.write_text('print *, "Fortran compilation is working."; end') if environment.is_cross_build() and not self.is_cross: for_machine = MachineChoice.BUILD else: for_machine = MachineChoice.HOST extra_flags = environment.coredata.get_external_args(for_machine, self.language) extra_flags += environment.coredata.get_external_link_args(for_machine, self.language) # %% build the test executable pc = subprocess.Popen(self.exelist + extra_flags + [str(source_name), '-o', str(binary_name)]) pc.wait() if pc.returncode != 0: raise EnvironmentException('Compiler %s can not compile programs.' % self.name_string()) if self.is_cross: if self.exe_wrapper is None: # Can't check if the binaries run so we have to assume they do return cmdlist = self.exe_wrapper + [str(binary_name)] else: cmdlist = [str(binary_name)] # %% Run the test executable try: pe = subprocess.Popen(cmdlist, stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL) pe.wait() if pe.returncode != 0: raise EnvironmentException('Executables created by Fortran compiler %s are not runnable.' % self.name_string()) except OSError: raise EnvironmentException('Executables created by Fortran compiler %s are not runnable.' % self.name_string()) def get_std_warn_args(self, level): return FortranCompiler.std_warn_args def get_buildtype_args(self, buildtype): return gnulike_buildtype_args[buildtype] def get_optimization_args(self, optimization_level): return gnu_optimization_args[optimization_level] def get_debug_args(self, is_debug): return clike_debug_args[is_debug] def get_buildtype_linker_args(self, buildtype): if is_osx(): return apple_buildtype_linker_args[buildtype] return gnulike_buildtype_linker_args[buildtype] def split_shlib_to_parts(self, fname): return CCompiler.split_shlib_to_parts(self, fname) def build_rpath_args(self, *args): return CCompiler.build_rpath_args(self, *args) def get_dependency_gen_args(self, outtarget, outfile): return [] def depfile_for_object(self, objfile): return CCompiler.depfile_for_object(self, objfile) def get_depfile_suffix(self): return CCompiler.get_depfile_suffix(self) def get_exelist(self): return CCompiler.get_exelist(self) def get_linker_exelist(self): return CCompiler.get_linker_exelist(self) def get_preprocess_only_args(self): return ['-cpp'] + CCompiler.get_preprocess_only_args(self) def get_compile_only_args(self): return CCompiler.get_compile_only_args(self) def get_no_optimization_args(self): return CCompiler.get_no_optimization_args(self) def get_compiler_check_args(self): return CCompiler.get_compiler_check_args(self) def get_output_args(self, target): return CCompiler.get_output_args(self, target) def get_linker_output_args(self, outputname): return CCompiler.get_linker_output_args(self, outputname) def get_coverage_args(self): return CCompiler.get_coverage_args(self) def get_coverage_link_args(self): return CCompiler.get_coverage_link_args(self) def get_werror_args(self): return CCompiler.get_werror_args(self) def get_std_exe_link_args(self): return CCompiler.get_std_exe_link_args(self) def get_include_args(self, path, is_system): return CCompiler.get_include_args(self, path, is_system) def get_module_incdir_args(self): return ('-I', ) def get_module_outdir_args(self, path): return ['-module', path] def compute_parameters_with_absolute_paths(self, parameter_list, build_dir): for idx, i in enumerate(parameter_list): if i[:2] == '-I' or i[:2] == '-L': parameter_list[idx] = i[:2] + os.path.normpath(os.path.join(build_dir, i[2:])) return parameter_list def module_name_to_filename(self, module_name: str) -> str: if '_' in module_name: # submodule s = module_name.lower() if self.id in ('gcc', 'intel'): filename = s.replace('_', '@') + '.smod' elif self.id in ('pgi', 'flang'): filename = s.replace('_', '-') + '.mod' else: filename = s + '.mod' else: # module filename = module_name.lower() + '.mod' return filename def get_std_shared_lib_link_args(self): return CCompiler.get_std_shared_lib_link_args(self) def _get_search_dirs(self, *args, **kwargs): return CCompiler._get_search_dirs(self, *args, **kwargs) def get_compiler_dirs(self, *args, **kwargs): return CCompiler.get_compiler_dirs(self, *args, **kwargs) def get_library_dirs(self, *args, **kwargs): return CCompiler.get_library_dirs(self, *args, **kwargs) def get_pic_args(self): return CCompiler.get_pic_args(self) def name_string(self): return CCompiler.name_string(self) def get_linker_search_args(self, dirname): return CCompiler.get_linker_search_args(self, dirname) def get_default_include_dirs(self): return CCompiler.get_default_include_dirs(self) def gen_export_dynamic_link_args(self, env): return CCompiler.gen_export_dynamic_link_args(self, env) def gen_import_library_args(self, implibname): return CCompiler.gen_import_library_args(self, implibname) def _get_basic_compiler_args(self, env, mode): return CCompiler._get_basic_compiler_args(self, env, mode) def _get_compiler_check_args(self, env, extra_args, dependencies, mode='compile'): return CCompiler._get_compiler_check_args(self, env, extra_args, dependencies, mode=mode) def compiles(self, code, env, *, extra_args=None, dependencies=None, mode='compile', disable_cache=False): return CCompiler.compiles(self, code, env, extra_args=extra_args, dependencies=dependencies, mode=mode, disable_cache=disable_cache) def _build_wrapper(self, code, env, extra_args, dependencies=None, mode='compile', want_output=False, disable_cache=False): return CCompiler._build_wrapper(self, code, env, extra_args, dependencies, mode, want_output, disable_cache=disable_cache) def links(self, code, env, *, extra_args=None, dependencies=None, disable_cache=False): return CCompiler.links(self, code, env, extra_args=extra_args, dependencies=dependencies, disable_cache=disable_cache) def run(self, code, env, *, extra_args=None, dependencies=None): return CCompiler.run(self, code, env, extra_args=extra_args, dependencies=dependencies) def _get_patterns(self, *args, **kwargs): return CCompiler._get_patterns(self, *args, **kwargs) def get_library_naming(self, *args, **kwargs): return CCompiler.get_library_naming(self, *args, **kwargs) def find_library_real(self, *args): return CCompiler.find_library_real(self, *args) def find_library_impl(self, *args): return CCompiler.find_library_impl(self, *args) def find_library(self, libname, env, extra_dirs, libtype: LibType = LibType.PREFER_SHARED): code = '''program main call exit(0) end program main''' return self.find_library_impl(libname, env, extra_dirs, code, libtype) def thread_flags(self, env): return CCompiler.thread_flags(self, env) def thread_link_flags(self, env): return CCompiler.thread_link_flags(self, env) def linker_to_compiler_args(self, args): return CCompiler.linker_to_compiler_args(self, args) def has_arguments(self, args, env, code, mode): return CCompiler.has_arguments(self, args, env, code, mode) def has_multi_arguments(self, args, env): return CCompiler.has_multi_arguments(self, args, env) def has_header(self, hname, prefix, env, *, extra_args=None, dependencies=None, disable_cache=False): return CCompiler.has_header(self, hname, prefix, env, extra_args=extra_args, dependencies=dependencies, disable_cache=disable_cache) def get_define(self, dname, prefix, env, extra_args, dependencies, disable_cache=False): return CCompiler.get_define(self, dname, prefix, env, extra_args, dependencies, disable_cache=disable_cache) @classmethod def _get_trials_from_pattern(cls, pattern, directory, libname): return CCompiler._get_trials_from_pattern(pattern, directory, libname) @staticmethod def _get_file_from_list(env, files: List[str]) -> Path: return CCompiler._get_file_from_list(env, files) class GnuFortranCompiler(GnuCompiler, FortranCompiler): def __init__(self, exelist, version, compiler_type, is_cross, exe_wrapper=None, defines=None, **kwargs): FortranCompiler.__init__(self, exelist, version, is_cross, exe_wrapper, **kwargs) GnuCompiler.__init__(self, compiler_type, defines) default_warn_args = ['-Wall'] self.warn_args = {'0': [], '1': default_warn_args, '2': default_warn_args + ['-Wextra'], '3': default_warn_args + ['-Wextra', '-Wpedantic']} def get_dependency_gen_args(self, outtarget, outfile): # Disabled until this is fixed: # https://gcc.gnu.org/bugzilla/show_bug.cgi?id=62162 # return ['-cpp', '-MD', '-MQ', outtarget] return [] def get_module_outdir_args(self, path): return ['-J' + path] def language_stdlib_only_link_flags(self): return ['-lgfortran', '-lm'] class ElbrusFortranCompiler(GnuFortranCompiler, ElbrusCompiler): def __init__(self, exelist, version, compiler_type, is_cross, exe_wrapper=None, defines=None, **kwargs): GnuFortranCompiler.__init__(self, exelist, version, compiler_type, is_cross, exe_wrapper, defines, **kwargs) ElbrusCompiler.__init__(self, compiler_type, defines) class G95FortranCompiler(FortranCompiler): def __init__(self, exelist, version, is_cross, exe_wrapper=None, **kwags): FortranCompiler.__init__(self, exelist, version, is_cross, exe_wrapper, **kwags) self.id = 'g95' default_warn_args = ['-Wall'] self.warn_args = {'0': [], '1': default_warn_args, '2': default_warn_args + ['-Wextra'], '3': default_warn_args + ['-Wextra', '-pedantic']} def get_module_outdir_args(self, path): return ['-fmod=' + path] def get_no_warn_args(self): # FIXME: Confirm that there's no compiler option to disable all warnings return [] class SunFortranCompiler(FortranCompiler): def __init__(self, exelist, version, is_cross, exe_wrapper=None, **kwags): FortranCompiler.__init__(self, exelist, version, is_cross, exe_wrapper, **kwags) self.id = 'sun' def get_dependency_gen_args(self, outtarget, outfile): return ['-fpp'] def get_always_args(self): return [] def get_warn_args(self, level): return [] def get_module_incdir_args(self): return ('-M', ) def get_module_outdir_args(self, path): return ['-moddir=' + path] def openmp_flags(self): return ['-xopenmp'] class IntelFortranCompiler(IntelCompiler, FortranCompiler): def __init__(self, exelist, version, is_cross, exe_wrapper=None, **kwags): self.file_suffixes = ('f90', 'f', 'for', 'ftn', 'fpp') FortranCompiler.__init__(self, exelist, version, is_cross, exe_wrapper, **kwags) # FIXME: Add support for OS X and Windows in detect_fortran_compiler so # we are sent the type of compiler IntelCompiler.__init__(self, CompilerType.ICC_STANDARD) self.id = 'intel' default_warn_args = ['-warn', 'general', '-warn', 'truncated_source'] self.warn_args = {'0': [], '1': default_warn_args, '2': default_warn_args + ['-warn', 'unused'], '3': ['-warn', 'all']} def get_preprocess_only_args(self): return ['-cpp', '-EP'] def get_always_args(self): """Ifort doesn't have -pipe.""" val = super().get_always_args() val.remove('-pipe') return val def language_stdlib_only_link_flags(self): return ['-lifcore', '-limf'] class PathScaleFortranCompiler(FortranCompiler): def __init__(self, exelist, version, is_cross, exe_wrapper=None, **kwags): FortranCompiler.__init__(self, exelist, version, is_cross, exe_wrapper, **kwags) self.id = 'pathscale' default_warn_args = ['-fullwarn'] self.warn_args = {'0': [], '1': default_warn_args, '2': default_warn_args, '3': default_warn_args} def openmp_flags(self): return ['-mp'] class PGIFortranCompiler(PGICompiler, FortranCompiler): def __init__(self, exelist, version, is_cross, exe_wrapper=None, **kwags): FortranCompiler.__init__(self, exelist, version, is_cross, exe_wrapper, **kwags) PGICompiler.__init__(self, CompilerType.PGI_STANDARD) class FlangFortranCompiler(ClangCompiler, FortranCompiler): def __init__(self, exelist, version, is_cross, exe_wrapper=None, **kwags): FortranCompiler.__init__(self, exelist, version, is_cross, exe_wrapper, **kwags) ClangCompiler.__init__(self, CompilerType.CLANG_STANDARD) self.id = 'flang' default_warn_args = ['-Minform=inform'] self.warn_args = {'0': [], '1': default_warn_args, '2': default_warn_args, '3': default_warn_args} class Open64FortranCompiler(FortranCompiler): def __init__(self, exelist, version, is_cross, exe_wrapper=None, **kwags): FortranCompiler.__init__(self, exelist, version, is_cross, exe_wrapper, **kwags) self.id = 'open64' default_warn_args = ['-fullwarn'] self.warn_args = {'0': [], '1': default_warn_args, '2': default_warn_args, '3': default_warn_args} def openmp_flags(self): return ['-mp'] class NAGFortranCompiler(FortranCompiler): def __init__(self, exelist, version, is_cross, exe_wrapper=None, **kwags): FortranCompiler.__init__(self, exelist, version, is_cross, exe_wrapper, **kwags) self.id = 'nagfor' def get_warn_args(self, level): return [] def get_module_outdir_args(self, path): return ['-mdir', path] def openmp_flags(self): return ['-openmp']