| # 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. |
| |
| import os.path |
| import typing |
| |
| from .. import coredata |
| from ..mesonlib import MachineChoice, MesonException, mlog, version_compare |
| from .c_function_attributes import C_FUNC_ATTRIBUTES |
| from .mixins.clike import CLikeCompiler |
| from .mixins.ccrx import CcrxCompiler |
| from .mixins.arm import ArmCompiler, ArmclangCompiler |
| from .mixins.visualstudio import VisualStudioLikeCompiler |
| from .mixins.gnu import GnuCompiler |
| from .mixins.intel import IntelGnuLikeCompiler, IntelVisualStudioLikeCompiler |
| from .mixins.clang import ClangCompiler |
| from .mixins.elbrus import ElbrusCompiler |
| from .mixins.pgi import PGICompiler |
| from .mixins.islinker import BasicLinkerIsCompilerMixin, LinkerEnvVarsMixin |
| from .compilers import ( |
| gnu_winlibs, |
| msvc_winlibs, |
| Compiler, |
| CompilerType, |
| ) |
| |
| |
| class CCompiler(CLikeCompiler, Compiler): |
| |
| @staticmethod |
| def attribute_check_func(name): |
| try: |
| return C_FUNC_ATTRIBUTES[name] |
| except KeyError: |
| raise MesonException('Unknown function attribute "{}"'.format(name)) |
| |
| def __init__(self, exelist, version, for_machine: MachineChoice, is_cross: bool, |
| exe_wrapper: typing.Optional[str] = None, **kwargs): |
| # If a child ObjC or CPP class has already set it, don't set it ourselves |
| self.language = 'c' |
| Compiler.__init__(self, exelist, version, for_machine, **kwargs) |
| CLikeCompiler.__init__(self, is_cross, exe_wrapper) |
| |
| def get_no_stdinc_args(self): |
| return ['-nostdinc'] |
| |
| def sanity_check(self, work_dir, environment): |
| code = 'int main() { int class=0; return class; }\n' |
| return self.sanity_check_impl(work_dir, environment, 'sanitycheckc.c', code) |
| |
| def has_header_symbol(self, hname, symbol, prefix, env, *, extra_args=None, dependencies=None): |
| fargs = {'prefix': prefix, 'header': hname, 'symbol': symbol} |
| t = '''{prefix} |
| #include <{header}> |
| int main () {{ |
| /* If it's not defined as a macro, try to use as a symbol */ |
| #ifndef {symbol} |
| {symbol}; |
| #endif |
| return 0; |
| }}''' |
| return self.compiles(t.format(**fargs), env, extra_args=extra_args, |
| dependencies=dependencies) |
| |
| |
| class ClangCCompiler(ClangCompiler, CCompiler): |
| def __init__(self, exelist, version, compiler_type, for_machine: MachineChoice, is_cross, exe_wrapper=None, **kwargs): |
| CCompiler.__init__(self, exelist, version, for_machine, is_cross, exe_wrapper, **kwargs) |
| ClangCompiler.__init__(self, compiler_type) |
| default_warn_args = ['-Wall', '-Winvalid-pch'] |
| self.warn_args = {'0': [], |
| '1': default_warn_args, |
| '2': default_warn_args + ['-Wextra'], |
| '3': default_warn_args + ['-Wextra', '-Wpedantic']} |
| |
| def get_options(self): |
| opts = CCompiler.get_options(self) |
| c_stds = ['c89', 'c99', 'c11'] |
| g_stds = ['gnu89', 'gnu99', 'gnu11'] |
| # https://releases.llvm.org/6.0.0/tools/clang/docs/ReleaseNotes.html |
| # https://en.wikipedia.org/wiki/Xcode#Latest_versions |
| v = '>=10.0.0' if self.compiler_type is CompilerType.CLANG_OSX else '>=6.0.0' |
| if version_compare(self.version, v): |
| c_stds += ['c17'] |
| g_stds += ['gnu17'] |
| v = '>=11.0.0' if self.compiler_type is CompilerType.CLANG_OSX else '>=8.0.0' |
| if version_compare(self.version, v): |
| c_stds += ['c18'] |
| g_stds += ['gnu18'] |
| opts.update({'c_std': coredata.UserComboOption('C language standard to use', |
| ['none'] + c_stds + g_stds, |
| 'none')}) |
| return opts |
| |
| def get_option_compile_args(self, options): |
| args = [] |
| std = options['c_std'] |
| if std.value != 'none': |
| args.append('-std=' + std.value) |
| return args |
| |
| def get_option_link_args(self, options): |
| return [] |
| |
| |
| class EmscriptenCCompiler(LinkerEnvVarsMixin, BasicLinkerIsCompilerMixin, ClangCCompiler): |
| def __init__(self, exelist, version, compiler_type, for_machine: MachineChoice, is_cross, exe_wrapper=None, **kwargs): |
| if not is_cross: |
| raise MesonException('Emscripten compiler can only be used for cross compilation.') |
| ClangCCompiler.__init__(self, exelist, version, compiler_type, for_machine, is_cross, exe_wrapper, **kwargs) |
| self.id = 'emscripten' |
| |
| def get_option_link_args(self, options): |
| return [] |
| |
| def get_soname_args(self, *args, **kwargs): |
| raise MesonException('Emscripten does not support shared libraries.') |
| |
| class ArmclangCCompiler(ArmclangCompiler, CCompiler): |
| def __init__(self, exelist, version, compiler_type, for_machine: MachineChoice, is_cross, exe_wrapper=None, **kwargs): |
| CCompiler.__init__(self, exelist, version, for_machine, is_cross, exe_wrapper, **kwargs) |
| ArmclangCompiler.__init__(self, compiler_type) |
| default_warn_args = ['-Wall', '-Winvalid-pch'] |
| self.warn_args = {'0': [], |
| '1': default_warn_args, |
| '2': default_warn_args + ['-Wextra'], |
| '3': default_warn_args + ['-Wextra', '-Wpedantic']} |
| |
| def get_options(self): |
| opts = CCompiler.get_options(self) |
| opts.update({'c_std': coredata.UserComboOption('C language standard to use', |
| ['none', 'c90', 'c99', 'c11', |
| 'gnu90', 'gnu99', 'gnu11'], |
| 'none')}) |
| return opts |
| |
| def get_option_compile_args(self, options): |
| args = [] |
| std = options['c_std'] |
| if std.value != 'none': |
| args.append('-std=' + std.value) |
| return args |
| |
| def get_option_link_args(self, options): |
| return [] |
| |
| |
| class GnuCCompiler(GnuCompiler, CCompiler): |
| def __init__(self, exelist, version, compiler_type, for_machine: MachineChoice, is_cross, exe_wrapper=None, defines=None, **kwargs): |
| CCompiler.__init__(self, exelist, version, for_machine, is_cross, exe_wrapper, **kwargs) |
| GnuCompiler.__init__(self, compiler_type, defines) |
| default_warn_args = ['-Wall', '-Winvalid-pch'] |
| self.warn_args = {'0': [], |
| '1': default_warn_args, |
| '2': default_warn_args + ['-Wextra'], |
| '3': default_warn_args + ['-Wextra', '-Wpedantic']} |
| |
| def get_options(self): |
| opts = CCompiler.get_options(self) |
| c_stds = ['c89', 'c99', 'c11'] |
| g_stds = ['gnu89', 'gnu99', 'gnu11'] |
| v = '>=8.0.0' |
| if version_compare(self.version, v): |
| c_stds += ['c17', 'c18'] |
| g_stds += ['gnu17', 'gnu18'] |
| opts.update({'c_std': coredata.UserComboOption('C language standard to use', |
| ['none'] + c_stds + g_stds, |
| 'none')}) |
| if self.compiler_type.is_windows_compiler: |
| opts.update({ |
| 'c_winlibs': coredata.UserArrayOption('Standard Win libraries to link against', |
| gnu_winlibs), }) |
| return opts |
| |
| def get_option_compile_args(self, options): |
| args = [] |
| std = options['c_std'] |
| if std.value != 'none': |
| args.append('-std=' + std.value) |
| return args |
| |
| def get_option_link_args(self, options): |
| if self.compiler_type.is_windows_compiler: |
| return options['c_winlibs'].value[:] |
| return [] |
| |
| def get_pch_use_args(self, pch_dir, header): |
| return ['-fpch-preprocess', '-include', os.path.basename(header)] |
| |
| |
| class PGICCompiler(PGICompiler, CCompiler): |
| def __init__(self, exelist, version, compiler_type, for_machine: MachineChoice, is_cross, exe_wrapper=None, **kwargs): |
| CCompiler.__init__(self, exelist, version, for_machine, is_cross, exe_wrapper, **kwargs) |
| PGICompiler.__init__(self, compiler_type) |
| |
| |
| class ElbrusCCompiler(GnuCCompiler, ElbrusCompiler): |
| def __init__(self, exelist, version, compiler_type, for_machine: MachineChoice, is_cross, exe_wrapper=None, defines=None, **kwargs): |
| GnuCCompiler.__init__(self, exelist, version, compiler_type, for_machine, is_cross, exe_wrapper, defines, **kwargs) |
| ElbrusCompiler.__init__(self, compiler_type, defines) |
| |
| # It does support some various ISO standards and c/gnu 90, 9x, 1x in addition to those which GNU CC supports. |
| def get_options(self): |
| opts = CCompiler.get_options(self) |
| opts.update({'c_std': coredata.UserComboOption('C language standard to use', |
| ['none', 'c89', 'c90', 'c9x', 'c99', 'c1x', 'c11', |
| 'gnu89', 'gnu90', 'gnu9x', 'gnu99', 'gnu1x', 'gnu11', |
| 'iso9899:2011', 'iso9899:1990', 'iso9899:199409', 'iso9899:1999'], |
| 'none')}) |
| return opts |
| |
| # Elbrus C compiler does not have lchmod, but there is only linker warning, not compiler error. |
| # So we should explicitly fail at this case. |
| def has_function(self, funcname, prefix, env, *, extra_args=None, dependencies=None): |
| if funcname == 'lchmod': |
| return False, False |
| else: |
| return super().has_function(funcname, prefix, env, |
| extra_args=extra_args, |
| dependencies=dependencies) |
| |
| |
| class IntelCCompiler(IntelGnuLikeCompiler, CCompiler): |
| def __init__(self, exelist, version, compiler_type, for_machine: MachineChoice, is_cross, exe_wrapper=None, **kwargs): |
| CCompiler.__init__(self, exelist, version, for_machine, is_cross, exe_wrapper, **kwargs) |
| IntelGnuLikeCompiler.__init__(self, compiler_type) |
| self.lang_header = 'c-header' |
| default_warn_args = ['-Wall', '-w3', '-diag-disable:remark'] |
| self.warn_args = {'0': [], |
| '1': default_warn_args, |
| '2': default_warn_args + ['-Wextra'], |
| '3': default_warn_args + ['-Wextra']} |
| |
| def get_options(self): |
| opts = CCompiler.get_options(self) |
| c_stds = ['c89', 'c99'] |
| g_stds = ['gnu89', 'gnu99'] |
| if version_compare(self.version, '>=16.0.0'): |
| c_stds += ['c11'] |
| opts.update({'c_std': coredata.UserComboOption('C language standard to use', |
| ['none'] + c_stds + g_stds, |
| 'none')}) |
| return opts |
| |
| def get_option_compile_args(self, options): |
| args = [] |
| std = options['c_std'] |
| if std.value != 'none': |
| args.append('-std=' + std.value) |
| return args |
| |
| |
| class VisualStudioLikeCCompilerMixin: |
| |
| """Shared methods that apply to MSVC-like C compilers.""" |
| |
| def get_options(self): |
| opts = super().get_options() |
| opts.update({'c_winlibs': coredata.UserArrayOption('Windows libs to link against.', |
| msvc_winlibs)}) |
| return opts |
| |
| def get_option_link_args(self, options): |
| return options['c_winlibs'].value[:] |
| |
| class VisualStudioCCompiler(VisualStudioLikeCompiler, VisualStudioLikeCCompilerMixin, CCompiler): |
| |
| def __init__(self, exelist, version, for_machine: MachineChoice, is_cross, exe_wrap, target: str, **kwargs): |
| CCompiler.__init__(self, exelist, version, for_machine, is_cross, exe_wrap, **kwargs) |
| VisualStudioLikeCompiler.__init__(self, target) |
| self.id = 'msvc' |
| |
| class ClangClCCompiler(VisualStudioLikeCompiler, VisualStudioLikeCCompilerMixin, CCompiler): |
| def __init__(self, exelist, version, for_machine: MachineChoice, is_cross, exe_wrap, target, **kwargs): |
| CCompiler.__init__(self, exelist, version, for_machine, is_cross, exe_wrap, **kwargs) |
| VisualStudioLikeCompiler.__init__(self, target) |
| self.id = 'clang-cl' |
| |
| def sanitizer_compile_args(self, value: str) -> typing.List[str]: |
| if value == 'none': |
| return [] |
| args = ['-fsanitize=' + value] |
| return args |
| |
| class IntelClCCompiler(IntelVisualStudioLikeCompiler, VisualStudioLikeCCompilerMixin, CCompiler): |
| |
| """Intel "ICL" compiler abstraction.""" |
| |
| __have_warned = False |
| |
| def __init__(self, exelist, version, for_machine: MachineChoice, is_cross, exe_wrap, target, **kwargs): |
| CCompiler.__init__(self, exelist, version, for_machine, is_cross, exe_wrap, **kwargs) |
| IntelVisualStudioLikeCompiler.__init__(self, target) |
| |
| def get_options(self): |
| opts = super().get_options() |
| c_stds = ['none', 'c89', 'c99', 'c11'] |
| opts.update({'c_std': coredata.UserComboOption('C language standard to use', |
| c_stds, |
| 'none')}) |
| return opts |
| |
| def get_option_compile_args(self, options): |
| args = [] |
| std = options['c_std'] |
| if std.value == 'c89': |
| if not self.__have_warned: |
| self.__have_warned = True |
| mlog.warning("ICL doesn't explicitly implement c89, setting the standard to 'none', which is close.") |
| elif std.value != 'none': |
| args.append('/Qstd:' + std.value) |
| return args |
| |
| |
| class ArmCCompiler(ArmCompiler, CCompiler): |
| def __init__(self, exelist, version, compiler_type, for_machine: MachineChoice, is_cross, exe_wrapper=None, **kwargs): |
| CCompiler.__init__(self, exelist, version, for_machine, is_cross, exe_wrapper, **kwargs) |
| ArmCompiler.__init__(self, compiler_type) |
| |
| def get_options(self): |
| opts = CCompiler.get_options(self) |
| opts.update({'c_std': coredata.UserComboOption('C language standard to use', |
| ['none', 'c90', 'c99'], |
| 'none')}) |
| return opts |
| |
| def get_option_compile_args(self, options): |
| args = [] |
| std = options['c_std'] |
| if std.value != 'none': |
| args.append('--' + std.value) |
| return args |
| |
| class CcrxCCompiler(CcrxCompiler, CCompiler): |
| def __init__(self, exelist, version, compiler_type, for_machine: MachineChoice, is_cross, exe_wrapper=None, **kwargs): |
| CCompiler.__init__(self, exelist, version, for_machine, is_cross, exe_wrapper, **kwargs) |
| CcrxCompiler.__init__(self, compiler_type) |
| |
| # Override CCompiler.get_always_args |
| def get_always_args(self): |
| return ['-nologo'] |
| |
| def get_options(self): |
| opts = CCompiler.get_options(self) |
| opts.update({'c_std': coredata.UserComboOption('C language standard to use', |
| ['none', 'c89', 'c99'], |
| 'none')}) |
| return opts |
| |
| def get_option_compile_args(self, options): |
| args = [] |
| std = options['c_std'] |
| if std.value == 'c89': |
| args.append('-lang=c') |
| elif std.value == 'c99': |
| args.append('-lang=c99') |
| return args |
| |
| def get_compile_only_args(self): |
| return [] |
| |
| def get_no_optimization_args(self): |
| return ['-optimize=0'] |
| |
| def get_output_args(self, target): |
| return ['-output=obj=%s' % target] |
| |
| def get_werror_args(self): |
| return ['-change_message=error'] |
| |
| def get_include_args(self, path, is_system): |
| if path == '': |
| path = '.' |
| return ['-include=' + path] |