The Meson Build System
http://mesonbuild.com/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
455 lines
17 KiB
455 lines
17 KiB
# 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, environment): |
|
source_name = os.path.join(work_dir, 'sanitycheckf.f90') |
|
binary_name = os.path.join(work_dir, 'sanitycheckf') |
|
with open(source_name, 'w') as ofile: |
|
ofile.write('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) |
|
pc = subprocess.Popen(self.exelist + extra_flags + [source_name, '-o', 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 + [binary_name] |
|
else: |
|
cmdlist = [binary_name] |
|
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()) |
|
|
|
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='compile') |
|
|
|
def compiles(self, code, env, *, extra_args=None, dependencies=None, mode='compile'): |
|
return CCompiler.compiles(self, code, env, extra_args=extra_args, |
|
dependencies=dependencies, mode=mode) |
|
|
|
def _build_wrapper(self, code, env, extra_args, dependencies=None, mode='compile', want_output=False): |
|
return CCompiler._build_wrapper(self, code, env, extra_args, dependencies, mode, want_output) |
|
|
|
def links(self, code, env, *, extra_args=None, dependencies=None): |
|
return CCompiler.links(self, code, env, extra_args=extra_args, |
|
dependencies=dependencies) |
|
|
|
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): |
|
return CCompiler.has_header(self, hname, prefix, env, extra_args=extra_args, dependencies=dependencies) |
|
|
|
def get_define(self, dname, prefix, env, extra_args, dependencies): |
|
return CCompiler.get_define(self, dname, prefix, env, extra_args, dependencies) |
|
|
|
@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']
|
|
|