| # SPDX-License-Identifier: Apache-2.0 |
| # Copyright 2012-2017 The Meson development team |
| |
| from __future__ import annotations |
| |
| import typing as T |
| import functools |
| import os |
| |
| from .. import options |
| from .. import mesonlib |
| from .compilers import ( |
| clike_debug_args, |
| Compiler, |
| CompileCheckMode, |
| ) |
| from .mixins.clike import CLikeCompiler |
| from .mixins.gnu import GnuCompiler, gnu_optimization_args |
| from .mixins.intel import IntelGnuLikeCompiler, IntelVisualStudioLikeCompiler |
| from .mixins.clang import ClangCompiler |
| from .mixins.elbrus import ElbrusCompiler |
| from .mixins.pgi import PGICompiler |
| |
| from mesonbuild.mesonlib import ( |
| version_compare, MesonException, |
| LibType, |
| ) |
| |
| if T.TYPE_CHECKING: |
| from ..coredata import MutableKeyedOptionDictType, KeyedOptionDictType |
| from ..dependencies import Dependency |
| from ..envconfig import MachineInfo |
| from ..environment import Environment |
| from ..linkers.linkers import DynamicLinker |
| from ..mesonlib import MachineChoice |
| |
| |
| class FortranCompiler(CLikeCompiler, Compiler): |
| |
| language = 'fortran' |
| |
| def __init__(self, exelist: T.List[str], version: str, for_machine: MachineChoice, is_cross: bool, |
| info: 'MachineInfo', |
| linker: T.Optional['DynamicLinker'] = None, |
| full_version: T.Optional[str] = None): |
| Compiler.__init__(self, [], exelist, version, for_machine, info, |
| is_cross=is_cross, full_version=full_version, linker=linker) |
| CLikeCompiler.__init__(self) |
| |
| def has_function(self, funcname: str, prefix: str, env: 'Environment', *, |
| extra_args: T.Optional[T.List[str]] = None, |
| dependencies: T.Optional[T.List['Dependency']] = None) -> T.Tuple[bool, bool]: |
| raise MesonException('Fortran does not have "has_function" capability.\n' |
| 'It is better to test if a Fortran capability is working like:\n\n' |
| "meson.get_compiler('fortran').links('block; end block; end program')\n\n" |
| 'that example is to see if the compiler has Fortran 2008 Block element.') |
| |
| def _get_basic_compiler_args(self, env: 'Environment', mode: CompileCheckMode) -> T.Tuple[T.List[str], T.List[str]]: |
| cargs = env.coredata.get_external_args(self.for_machine, self.language) |
| largs = env.coredata.get_external_link_args(self.for_machine, self.language) |
| return cargs, largs |
| |
| def sanity_check(self, work_dir: str, environment: 'Environment') -> None: |
| source_name = 'sanitycheckf.f' |
| code = ' PROGRAM MAIN\n PRINT *, "Fortran compilation is working."\n END\n' |
| return self._sanity_check_impl(work_dir, environment, source_name, code) |
| |
| def get_optimization_args(self, optimization_level: str) -> T.List[str]: |
| return gnu_optimization_args[optimization_level] |
| |
| def get_debug_args(self, is_debug: bool) -> T.List[str]: |
| return clike_debug_args[is_debug] |
| |
| def get_preprocess_only_args(self) -> T.List[str]: |
| return ['-cpp'] + super().get_preprocess_only_args() |
| |
| def get_module_incdir_args(self) -> T.Tuple[str, ...]: |
| return ('-I', ) |
| |
| def get_module_outdir_args(self, path: str) -> T.List[str]: |
| return ['-module', path] |
| |
| def compute_parameters_with_absolute_paths(self, parameter_list: T.List[str], |
| build_dir: str) -> T.List[str]: |
| 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', 'intel-cl'}: |
| 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 find_library(self, libname: str, env: 'Environment', extra_dirs: T.List[str], |
| libtype: LibType = LibType.PREFER_SHARED, lib_prefix_warning: bool = True) -> T.Optional[T.List[str]]: |
| code = 'stop; end program' |
| return self._find_library_impl(libname, env, extra_dirs, code, libtype, lib_prefix_warning) |
| |
| def has_multi_arguments(self, args: T.List[str], env: 'Environment') -> T.Tuple[bool, bool]: |
| return self._has_multi_arguments(args, env, 'stop; end program') |
| |
| def has_multi_link_arguments(self, args: T.List[str], env: 'Environment') -> T.Tuple[bool, bool]: |
| return self._has_multi_link_arguments(args, env, 'stop; end program') |
| |
| def get_options(self) -> 'MutableKeyedOptionDictType': |
| return self.update_options( |
| super().get_options(), |
| self.create_option(options.UserComboOption, |
| self.form_compileropt_key('std'), |
| 'Fortran language standard to use', |
| ['none'], |
| 'none'), |
| ) |
| |
| def _compile_int(self, expression: str, prefix: str, env: 'Environment', |
| extra_args: T.Union[None, T.List[str], T.Callable[[CompileCheckMode], T.List[str]]], |
| dependencies: T.Optional[T.List['Dependency']]) -> bool: |
| # Use a trick for emulating a static assert |
| # Taken from https://github.com/j3-fortran/fortran_proposals/issues/70 |
| t = f'''program test |
| {prefix} |
| real(merge(kind(1.),-1,({expression}))), parameter :: fail = 1. |
| end program test''' |
| return self.compiles(t, env, extra_args=extra_args, |
| dependencies=dependencies)[0] |
| |
| def cross_compute_int(self, expression: str, low: T.Optional[int], high: T.Optional[int], |
| guess: T.Optional[int], prefix: str, env: 'Environment', |
| extra_args: T.Union[None, T.List[str], T.Callable[[CompileCheckMode], T.List[str]]] = None, |
| dependencies: T.Optional[T.List['Dependency']] = None) -> int: |
| # This only difference between this implementation and that of CLikeCompiler |
| # is a change in logical conjunction operator (.and. instead of &&) |
| |
| # Try user's guess first |
| if isinstance(guess, int): |
| if self._compile_int(f'{expression} == {guess}', prefix, env, extra_args, dependencies): |
| return guess |
| |
| # If no bounds are given, compute them in the limit of int32 |
| maxint = 0x7fffffff |
| minint = -0x80000000 |
| if not isinstance(low, int) or not isinstance(high, int): |
| if self._compile_int(f'{expression} >= 0', prefix, env, extra_args, dependencies): |
| low = cur = 0 |
| while self._compile_int(f'{expression} > {cur}', prefix, env, extra_args, dependencies): |
| low = cur + 1 |
| if low > maxint: |
| raise mesonlib.EnvironmentException('Cross-compile check overflowed') |
| cur = min(cur * 2 + 1, maxint) |
| high = cur |
| else: |
| high = cur = -1 |
| while self._compile_int(f'{expression} < {cur}', prefix, env, extra_args, dependencies): |
| high = cur - 1 |
| if high < minint: |
| raise mesonlib.EnvironmentException('Cross-compile check overflowed') |
| cur = max(cur * 2, minint) |
| low = cur |
| else: |
| # Sanity check limits given by user |
| if high < low: |
| raise mesonlib.EnvironmentException('high limit smaller than low limit') |
| condition = f'{expression} <= {high} .and. {expression} >= {low}' |
| if not self._compile_int(condition, prefix, env, extra_args, dependencies): |
| raise mesonlib.EnvironmentException('Value out of given range') |
| |
| # Binary search |
| while low != high: |
| cur = low + int((high - low) / 2) |
| if self._compile_int(f'{expression} <= {cur}', prefix, env, extra_args, dependencies): |
| high = cur |
| else: |
| low = cur + 1 |
| |
| return low |
| |
| def compute_int(self, expression: str, low: T.Optional[int], high: T.Optional[int], |
| guess: T.Optional[int], prefix: str, env: 'Environment', *, |
| extra_args: T.Union[None, T.List[str], T.Callable[[CompileCheckMode], T.List[str]]], |
| dependencies: T.Optional[T.List['Dependency']] = None) -> int: |
| if extra_args is None: |
| extra_args = [] |
| if self.is_cross: |
| return self.cross_compute_int(expression, low, high, guess, prefix, env, extra_args, dependencies) |
| t = f'''program test |
| {prefix} |
| print '(i0)', {expression} |
| end program test |
| ''' |
| res = self.run(t, env, extra_args=extra_args, |
| dependencies=dependencies) |
| if not res.compiled: |
| return -1 |
| if res.returncode != 0: |
| raise mesonlib.EnvironmentException('Could not run compute_int test binary.') |
| return int(res.stdout) |
| |
| def cross_sizeof(self, typename: str, prefix: str, env: 'Environment', *, |
| extra_args: T.Union[None, T.List[str], T.Callable[[CompileCheckMode], T.List[str]]] = None, |
| dependencies: T.Optional[T.List['Dependency']] = None) -> int: |
| if extra_args is None: |
| extra_args = [] |
| t = f'''program test |
| use iso_c_binding |
| {prefix} |
| {typename} :: something |
| end program test |
| ''' |
| if not self.compiles(t, env, extra_args=extra_args, |
| dependencies=dependencies)[0]: |
| return -1 |
| return self.cross_compute_int('c_sizeof(x)', None, None, None, prefix + '\nuse iso_c_binding\n' + typename + ' :: x', env, extra_args, dependencies) |
| |
| def sizeof(self, typename: str, prefix: str, env: 'Environment', *, |
| extra_args: T.Union[None, T.List[str], T.Callable[[CompileCheckMode], T.List[str]]] = None, |
| dependencies: T.Optional[T.List['Dependency']] = None) -> T.Tuple[int, bool]: |
| if extra_args is None: |
| extra_args = [] |
| if self.is_cross: |
| r = self.cross_sizeof(typename, prefix, env, extra_args=extra_args, |
| dependencies=dependencies) |
| return r, False |
| t = f'''program test |
| use iso_c_binding |
| {prefix} |
| {typename} :: x |
| print '(i0)', c_sizeof(x) |
| end program test |
| ''' |
| res = self.cached_run(t, env, extra_args=extra_args, |
| dependencies=dependencies) |
| if not res.compiled: |
| return -1, False |
| if res.returncode != 0: |
| raise mesonlib.EnvironmentException('Could not run sizeof test binary.') |
| return int(res.stdout), res.cached |
| |
| @functools.lru_cache() |
| def output_is_64bit(self, env: 'Environment') -> bool: |
| ''' |
| returns true if the output produced is 64-bit, false if 32-bit |
| ''' |
| return self.sizeof('type(c_ptr)', '', env)[0] == 8 |
| |
| |
| class GnuFortranCompiler(GnuCompiler, FortranCompiler): |
| |
| def __init__(self, exelist: T.List[str], version: str, for_machine: MachineChoice, is_cross: bool, |
| info: 'MachineInfo', |
| defines: T.Optional[T.Dict[str, str]] = None, |
| linker: T.Optional['DynamicLinker'] = None, |
| full_version: T.Optional[str] = None): |
| FortranCompiler.__init__(self, exelist, version, for_machine, |
| is_cross, info, linker=linker, |
| full_version=full_version) |
| GnuCompiler.__init__(self, defines) |
| default_warn_args = ['-Wall'] |
| self.warn_args = {'0': [], |
| '1': default_warn_args, |
| '2': default_warn_args + ['-Wextra'], |
| '3': default_warn_args + ['-Wextra', '-Wpedantic', '-fimplicit-none'], |
| 'everything': default_warn_args + ['-Wextra', '-Wpedantic', '-fimplicit-none']} |
| |
| def get_options(self) -> 'MutableKeyedOptionDictType': |
| opts = FortranCompiler.get_options(self) |
| fortran_stds = ['legacy', 'f95', 'f2003'] |
| if version_compare(self.version, '>=4.4.0'): |
| fortran_stds += ['f2008'] |
| if version_compare(self.version, '>=8.0.0'): |
| fortran_stds += ['f2018'] |
| key = self.form_compileropt_key('std') |
| opts[key].choices = ['none'] + fortran_stds |
| return opts |
| |
| def get_option_compile_args(self, options: 'KeyedOptionDictType') -> T.List[str]: |
| args: T.List[str] = [] |
| key = self.form_compileropt_key('std') |
| std = options.get_value(key) |
| if std != 'none': |
| args.append('-std=' + std) |
| return args |
| |
| def get_dependency_gen_args(self, outtarget: str, outfile: str) -> T.List[str]: |
| # 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: str) -> T.List[str]: |
| return ['-J' + path] |
| |
| def language_stdlib_only_link_flags(self, env: 'Environment') -> T.List[str]: |
| # We need to apply the search prefix here, as these link arguments may |
| # be passed to a different compiler with a different set of default |
| # search paths, such as when using Clang for C/C++ and gfortran for |
| # fortran, |
| search_dirs: T.List[str] = [] |
| for d in self.get_compiler_dirs(env, 'libraries'): |
| search_dirs.append(f'-L{d}') |
| return search_dirs + ['-lgfortran', '-lm'] |
| |
| def has_header(self, hname: str, prefix: str, env: 'Environment', *, |
| extra_args: T.Union[None, T.List[str], T.Callable[['CompileCheckMode'], T.List[str]]] = None, |
| dependencies: T.Optional[T.List['Dependency']] = None, |
| disable_cache: bool = False) -> T.Tuple[bool, bool]: |
| ''' |
| Derived from mixins/clike.py:has_header, but without C-style usage of |
| __has_include which breaks with GCC-Fortran 10: |
| https://github.com/mesonbuild/meson/issues/7017 |
| ''' |
| code = f'{prefix}\n#include <{hname}>' |
| return self.compiles(code, env, extra_args=extra_args, |
| dependencies=dependencies, mode=CompileCheckMode.PREPROCESS, disable_cache=disable_cache) |
| |
| |
| class ElbrusFortranCompiler(ElbrusCompiler, FortranCompiler): |
| def __init__(self, exelist: T.List[str], version: str, for_machine: MachineChoice, is_cross: bool, |
| info: 'MachineInfo', |
| defines: T.Optional[T.Dict[str, str]] = None, |
| linker: T.Optional['DynamicLinker'] = None, |
| full_version: T.Optional[str] = None): |
| FortranCompiler.__init__(self, exelist, version, for_machine, is_cross, |
| info, linker=linker, full_version=full_version) |
| ElbrusCompiler.__init__(self) |
| |
| def get_options(self) -> 'MutableKeyedOptionDictType': |
| opts = FortranCompiler.get_options(self) |
| fortran_stds = ['f95', 'f2003', 'f2008', 'gnu', 'legacy', 'f2008ts'] |
| key = self.form_compileropt_key('std') |
| opts[key].choices = ['none'] + fortran_stds |
| return opts |
| |
| def get_module_outdir_args(self, path: str) -> T.List[str]: |
| return ['-J' + path] |
| |
| |
| class G95FortranCompiler(FortranCompiler): |
| |
| LINKER_PREFIX = '-Wl,' |
| id = 'g95' |
| |
| def __init__(self, exelist: T.List[str], version: str, for_machine: MachineChoice, is_cross: bool, |
| info: 'MachineInfo', |
| linker: T.Optional['DynamicLinker'] = None, |
| full_version: T.Optional[str] = None): |
| FortranCompiler.__init__(self, exelist, version, for_machine, |
| is_cross, info, linker=linker, |
| full_version=full_version) |
| default_warn_args = ['-Wall'] |
| self.warn_args = {'0': [], |
| '1': default_warn_args, |
| '2': default_warn_args + ['-Wextra'], |
| '3': default_warn_args + ['-Wextra', '-pedantic'], |
| 'everything': default_warn_args + ['-Wextra', '-pedantic']} |
| |
| def get_module_outdir_args(self, path: str) -> T.List[str]: |
| return ['-fmod=' + path] |
| |
| |
| class SunFortranCompiler(FortranCompiler): |
| |
| LINKER_PREFIX = '-Wl,' |
| id = 'sun' |
| |
| def get_dependency_gen_args(self, outtarget: str, outfile: str) -> T.List[str]: |
| return ['-fpp'] |
| |
| def get_always_args(self) -> T.List[str]: |
| return [] |
| |
| def get_warn_args(self, level: str) -> T.List[str]: |
| return [] |
| |
| def get_module_incdir_args(self) -> T.Tuple[str, ...]: |
| return ('-M', ) |
| |
| def get_module_outdir_args(self, path: str) -> T.List[str]: |
| return ['-moddir=' + path] |
| |
| def openmp_flags(self, env: Environment) -> T.List[str]: |
| return ['-xopenmp'] |
| |
| |
| class IntelFortranCompiler(IntelGnuLikeCompiler, FortranCompiler): |
| |
| id = 'intel' |
| |
| def __init__(self, exelist: T.List[str], version: str, for_machine: MachineChoice, is_cross: bool, |
| info: 'MachineInfo', |
| linker: T.Optional['DynamicLinker'] = None, |
| full_version: T.Optional[str] = None): |
| FortranCompiler.__init__(self, exelist, version, for_machine, |
| is_cross, info, linker=linker, |
| full_version=full_version) |
| # FIXME: Add support for OS X and Windows in detect_fortran_compiler so |
| # we are sent the type of compiler |
| IntelGnuLikeCompiler.__init__(self) |
| self.file_suffixes = ('f90', 'f', 'for', 'ftn', 'fpp', ) |
| 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'], |
| 'everything': ['-warn', 'all']} |
| |
| def get_options(self) -> 'MutableKeyedOptionDictType': |
| opts = FortranCompiler.get_options(self) |
| key = self.form_compileropt_key('std') |
| opts[key].choices = ['none', 'legacy', 'f95', 'f2003', 'f2008', 'f2018'] |
| return opts |
| |
| def get_option_compile_args(self, options: 'KeyedOptionDictType') -> T.List[str]: |
| args: T.List[str] = [] |
| key = self.form_compileropt_key('std') |
| std = options.get_value(key) |
| stds = {'legacy': 'none', 'f95': 'f95', 'f2003': 'f03', 'f2008': 'f08', 'f2018': 'f18'} |
| if std != 'none': |
| args.append('-stand=' + stds[std]) |
| return args |
| |
| def get_preprocess_only_args(self) -> T.List[str]: |
| return ['-cpp', '-EP'] |
| |
| def get_werror_args(self) -> T.List[str]: |
| return ['-warn', 'errors'] |
| |
| def language_stdlib_only_link_flags(self, env: 'Environment') -> T.List[str]: |
| # TODO: needs default search path added |
| return ['-lifcore', '-limf'] |
| |
| def get_dependency_gen_args(self, outtarget: str, outfile: str) -> T.List[str]: |
| return ['-gen-dep=' + outtarget, '-gen-depformat=make'] |
| |
| |
| class IntelLLVMFortranCompiler(IntelFortranCompiler): |
| |
| id = 'intel-llvm' |
| |
| |
| class IntelClFortranCompiler(IntelVisualStudioLikeCompiler, FortranCompiler): |
| |
| always_args = ['/nologo'] |
| |
| def __init__(self, exelist: T.List[str], version: str, for_machine: MachineChoice, |
| is_cross: bool, info: 'MachineInfo', target: str, |
| linker: T.Optional['DynamicLinker'] = None, |
| full_version: T.Optional[str] = None): |
| FortranCompiler.__init__(self, exelist, version, for_machine, |
| is_cross, info, linker=linker, |
| full_version=full_version) |
| IntelVisualStudioLikeCompiler.__init__(self, target) |
| self.file_suffixes = ('f90', 'f', 'for', 'ftn', 'fpp', ) |
| |
| 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'], |
| 'everything': ['/warn:all']} |
| |
| def get_options(self) -> 'MutableKeyedOptionDictType': |
| opts = FortranCompiler.get_options(self) |
| key = self.form_compileropt_key('std') |
| opts[key].choices = ['none', 'legacy', 'f95', 'f2003', 'f2008', 'f2018'] |
| return opts |
| |
| def get_option_compile_args(self, options: 'KeyedOptionDictType') -> T.List[str]: |
| args: T.List[str] = [] |
| key = self.form_compileropt_key('std') |
| std = options.get_value(key) |
| stds = {'legacy': 'none', 'f95': 'f95', 'f2003': 'f03', 'f2008': 'f08', 'f2018': 'f18'} |
| if std != 'none': |
| args.append('/stand:' + stds[std]) |
| return args |
| |
| def get_werror_args(self) -> T.List[str]: |
| return ['/warn:errors'] |
| |
| def get_module_outdir_args(self, path: str) -> T.List[str]: |
| return ['/module:' + path] |
| |
| |
| class IntelLLVMClFortranCompiler(IntelClFortranCompiler): |
| |
| id = 'intel-llvm-cl' |
| |
| class PathScaleFortranCompiler(FortranCompiler): |
| |
| id = 'pathscale' |
| |
| def __init__(self, exelist: T.List[str], version: str, for_machine: MachineChoice, is_cross: bool, |
| info: 'MachineInfo', |
| linker: T.Optional['DynamicLinker'] = None, |
| full_version: T.Optional[str] = None): |
| FortranCompiler.__init__(self, exelist, version, for_machine, |
| is_cross, info, linker=linker, |
| full_version=full_version) |
| default_warn_args = ['-fullwarn'] |
| self.warn_args = {'0': [], |
| '1': default_warn_args, |
| '2': default_warn_args, |
| '3': default_warn_args, |
| 'everything': default_warn_args} |
| |
| def openmp_flags(self, env: Environment) -> T.List[str]: |
| return ['-mp'] |
| |
| |
| class PGIFortranCompiler(PGICompiler, FortranCompiler): |
| |
| def __init__(self, exelist: T.List[str], version: str, for_machine: MachineChoice, is_cross: bool, |
| info: 'MachineInfo', |
| linker: T.Optional['DynamicLinker'] = None, |
| full_version: T.Optional[str] = None): |
| FortranCompiler.__init__(self, exelist, version, for_machine, |
| is_cross, info, linker=linker, |
| full_version=full_version) |
| PGICompiler.__init__(self) |
| |
| default_warn_args = ['-Minform=inform'] |
| self.warn_args = {'0': [], |
| '1': default_warn_args, |
| '2': default_warn_args, |
| '3': default_warn_args + ['-Mdclchk'], |
| 'everything': default_warn_args + ['-Mdclchk']} |
| |
| def language_stdlib_only_link_flags(self, env: 'Environment') -> T.List[str]: |
| # TODO: needs default search path added |
| return ['-lpgf90rtl', '-lpgf90', '-lpgf90_rpm1', '-lpgf902', |
| '-lpgf90rtl', '-lpgftnrtl', '-lrt'] |
| |
| |
| class NvidiaHPC_FortranCompiler(PGICompiler, FortranCompiler): |
| |
| id = 'nvidia_hpc' |
| |
| def __init__(self, exelist: T.List[str], version: str, for_machine: MachineChoice, is_cross: bool, |
| info: 'MachineInfo', |
| linker: T.Optional['DynamicLinker'] = None, |
| full_version: T.Optional[str] = None): |
| FortranCompiler.__init__(self, exelist, version, for_machine, |
| is_cross, info, linker=linker, |
| full_version=full_version) |
| PGICompiler.__init__(self) |
| |
| default_warn_args = ['-Minform=inform'] |
| self.warn_args = {'0': [], |
| '1': default_warn_args, |
| '2': default_warn_args, |
| '3': default_warn_args + ['-Mdclchk'], |
| 'everything': default_warn_args + ['-Mdclchk']} |
| |
| |
| class ClassicFlangFortranCompiler(ClangCompiler, FortranCompiler): |
| |
| id = 'flang' |
| |
| def __init__(self, exelist: T.List[str], version: str, for_machine: MachineChoice, is_cross: bool, |
| info: 'MachineInfo', |
| linker: T.Optional['DynamicLinker'] = None, |
| full_version: T.Optional[str] = None): |
| FortranCompiler.__init__(self, exelist, version, for_machine, |
| is_cross, info, linker=linker, |
| full_version=full_version) |
| ClangCompiler.__init__(self, {}) |
| default_warn_args = ['-Minform=inform'] |
| self.warn_args = {'0': [], |
| '1': default_warn_args, |
| '2': default_warn_args, |
| '3': default_warn_args, |
| 'everything': default_warn_args} |
| |
| def language_stdlib_only_link_flags(self, env: 'Environment') -> T.List[str]: |
| # We need to apply the search prefix here, as these link arguments may |
| # be passed to a different compiler with a different set of default |
| # search paths, such as when using Clang for C/C++ and gfortran for |
| # fortran, |
| # XXX: Untested.... |
| search_dirs: T.List[str] = [] |
| for d in self.get_compiler_dirs(env, 'libraries'): |
| search_dirs.append(f'-L{d}') |
| return search_dirs + ['-lflang', '-lpgmath'] |
| |
| |
| class ArmLtdFlangFortranCompiler(ClassicFlangFortranCompiler): |
| |
| id = 'armltdflang' |
| |
| |
| class LlvmFlangFortranCompiler(ClangCompiler, FortranCompiler): |
| |
| id = 'llvm-flang' |
| |
| def __init__(self, exelist: T.List[str], version: str, for_machine: MachineChoice, is_cross: bool, |
| info: 'MachineInfo', linker: T.Optional['DynamicLinker'] = None, |
| full_version: T.Optional[str] = None): |
| FortranCompiler.__init__(self, exelist, version, for_machine, |
| is_cross, info, linker=linker, |
| full_version=full_version) |
| ClangCompiler.__init__(self, {}) |
| default_warn_args = ['-Wall'] |
| self.warn_args = {'0': [], |
| '1': default_warn_args, |
| '2': default_warn_args, |
| '3': default_warn_args, |
| 'everything': default_warn_args} |
| |
| def get_colorout_args(self, colortype: str) -> T.List[str]: |
| # not yet supported, see https://github.com/llvm/llvm-project/issues/89888 |
| return [] |
| |
| def get_dependency_gen_args(self, outtarget: str, outfile: str) -> T.List[str]: |
| # not yet supported, see https://github.com/llvm/llvm-project/issues/89888 |
| return [] |
| |
| def get_module_outdir_args(self, path: str) -> T.List[str]: |
| # different syntax from classic flang (which supported `-module`), see |
| # https://github.com/llvm/llvm-project/issues/66969 |
| return ['-module-dir', path] |
| |
| def gnu_symbol_visibility_args(self, vistype: str) -> T.List[str]: |
| # flang doesn't support symbol visibility flag yet, see |
| # https://github.com/llvm/llvm-project/issues/92459 |
| return [] |
| |
| def language_stdlib_only_link_flags(self, env: 'Environment') -> T.List[str]: |
| # matching setup from ClassicFlangFortranCompiler |
| search_dirs: T.List[str] = [] |
| for d in self.get_compiler_dirs(env, 'libraries'): |
| search_dirs.append(f'-L{d}') |
| # does not automatically link to Fortran_main anymore after |
| # https://github.com/llvm/llvm-project/commit/9d6837d595719904720e5ff68ec1f1a2665bdc2f |
| # note that this changed again in flang 19 with |
| # https://github.com/llvm/llvm-project/commit/8d5386669ed63548daf1bee415596582d6d78d7d; |
| # it seems flang 18 doesn't work if something accidentally includes a program unit, see |
| # https://github.com/llvm/llvm-project/issues/92496 |
| return search_dirs + ['-lFortranRuntime', '-lFortranDecimal'] |
| |
| |
| class Open64FortranCompiler(FortranCompiler): |
| |
| id = 'open64' |
| |
| def __init__(self, exelist: T.List[str], version: str, for_machine: MachineChoice, is_cross: bool, |
| info: 'MachineInfo', |
| linker: T.Optional['DynamicLinker'] = None, |
| full_version: T.Optional[str] = None): |
| FortranCompiler.__init__(self, exelist, version, for_machine, |
| is_cross, info, linker=linker, |
| full_version=full_version) |
| default_warn_args = ['-fullwarn'] |
| self.warn_args = {'0': [], |
| '1': default_warn_args, |
| '2': default_warn_args, |
| '3': default_warn_args, |
| 'everything': default_warn_args} |
| |
| def openmp_flags(self, env: Environment) -> T.List[str]: |
| return ['-mp'] |
| |
| |
| class NAGFortranCompiler(FortranCompiler): |
| |
| id = 'nagfor' |
| |
| def __init__(self, exelist: T.List[str], version: str, for_machine: MachineChoice, is_cross: bool, |
| info: 'MachineInfo', |
| linker: T.Optional['DynamicLinker'] = None, |
| full_version: T.Optional[str] = None): |
| FortranCompiler.__init__(self, exelist, version, for_machine, |
| is_cross, info, linker=linker, |
| full_version=full_version) |
| # Warnings are on by default; -w disables (by category): |
| self.warn_args = { |
| '0': ['-w=all'], |
| '1': [], |
| '2': [], |
| '3': [], |
| 'everything': [], |
| } |
| |
| def get_always_args(self) -> T.List[str]: |
| return self.get_nagfor_quiet(self.version) |
| |
| def get_module_outdir_args(self, path: str) -> T.List[str]: |
| return ['-mdir', path] |
| |
| @staticmethod |
| def get_nagfor_quiet(version: str) -> T.List[str]: |
| return ['-quiet'] if version_compare(version, '>=7100') else [] |
| |
| def get_pic_args(self) -> T.List[str]: |
| return ['-PIC'] |
| |
| def get_preprocess_only_args(self) -> T.List[str]: |
| return ['-fpp'] |
| |
| def get_std_exe_link_args(self) -> T.List[str]: |
| return self.get_always_args() |
| |
| def openmp_flags(self, env: Environment) -> T.List[str]: |
| return ['-openmp'] |