|
|
|
# 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 .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)
|
|
|
|
|
compilers: Use keyword only arguments for compiler interfaces
Because we need to inherit them in some cases, and python's
keyword-or-positional arguments make this really painful, especially
with inheritance. They do this in two ways:
1) If you want to intercept the arguments you need to check for both a
keyword and a positional argument, because you could get either. Then
you need to make sure that you only pass one of those down to the
next layer.
2) After you do that, if the layer below you decides to do the same
thing, but uses the other form (you used keyword by the lower level
uses positional or vice versa), then you'll get a TypeError since two
layers down got the argument as both a positional and a keyword.
All of this is bad. Fortunately python 3.x provides a mechanism to solve
this, keyword only arguments. These arguments cannot be based
positionally, the interpreter will give us an error in that case.
I have made a best effort to do this correctly, and I've verified it
with GCC, Clang, ICC, and MSVC, but there are other compilers like Arm
and Elbrus that I don't have access to.
6 years ago
|
|
|
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
|
|
|
|
}}'''
|
compilers: Use keyword only arguments for compiler interfaces
Because we need to inherit them in some cases, and python's
keyword-or-positional arguments make this really painful, especially
with inheritance. They do this in two ways:
1) If you want to intercept the arguments you need to check for both a
keyword and a positional argument, because you could get either. Then
you need to make sure that you only pass one of those down to the
next layer.
2) After you do that, if the layer below you decides to do the same
thing, but uses the other form (you used keyword by the lower level
uses positional or vice versa), then you'll get a TypeError since two
layers down got the argument as both a positional and a keyword.
All of this is bad. Fortunately python 3.x provides a mechanism to solve
this, keyword only arguments. These arguments cannot be based
positionally, the interpreter will give us an error in that case.
I have made a best effort to do this correctly, and I've verified it
with GCC, Clang, ICC, and MSVC, but there are other compilers like Arm
and Elbrus that I don't have access to.
6 years ago
|
|
|
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 []
|
|
|
|
|
|
|
|
def get_linker_always_args(self):
|
|
|
|
basic = super().get_linker_always_args()
|
|
|
|
if self.compiler_type.is_osx_compiler:
|
|
|
|
return basic + ['-Wl,-headerpad_max_install_names']
|
|
|
|
return basic
|
|
|
|
|
|
|
|
|
|
|
|
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.
|
compilers: Use keyword only arguments for compiler interfaces
Because we need to inherit them in some cases, and python's
keyword-or-positional arguments make this really painful, especially
with inheritance. They do this in two ways:
1) If you want to intercept the arguments you need to check for both a
keyword and a positional argument, because you could get either. Then
you need to make sure that you only pass one of those down to the
next layer.
2) After you do that, if the layer below you decides to do the same
thing, but uses the other form (you used keyword by the lower level
uses positional or vice versa), then you'll get a TypeError since two
layers down got the argument as both a positional and a keyword.
All of this is bad. Fortunately python 3.x provides a mechanism to solve
this, keyword only arguments. These arguments cannot be based
positionally, the interpreter will give us an error in that case.
I have made a best effort to do this correctly, and I've verified it
with GCC, Clang, ICC, and MSVC, but there are other compilers like Arm
and Elbrus that I don't have access to.
6 years ago
|
|
|
def has_function(self, funcname, prefix, env, *, extra_args=None, dependencies=None):
|
|
|
|
if funcname == 'lchmod':
|
|
|
|
return False, False
|
|
|
|
else:
|
compilers: Use keyword only arguments for compiler interfaces
Because we need to inherit them in some cases, and python's
keyword-or-positional arguments make this really painful, especially
with inheritance. They do this in two ways:
1) If you want to intercept the arguments you need to check for both a
keyword and a positional argument, because you could get either. Then
you need to make sure that you only pass one of those down to the
next layer.
2) After you do that, if the layer below you decides to do the same
thing, but uses the other form (you used keyword by the lower level
uses positional or vice versa), then you'll get a TypeError since two
layers down got the argument as both a positional and a keyword.
All of this is bad. Fortunately python 3.x provides a mechanism to solve
this, keyword only arguments. These arguments cannot be based
positionally, the interpreter will give us an error in that case.
I have made a best effort to do this correctly, and I've verified it
with GCC, Clang, ICC, and MSVC, but there are other compilers like Arm
and Elbrus that I don't have access to.
6 years ago
|
|
|
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):
|
|
|
|
CCompiler.__init__(self, exelist, version, for_machine, is_cross, exe_wrap)
|
|
|
|
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):
|
|
|
|
CCompiler.__init__(self, exelist, version, for_machine, is_cross, exe_wrap)
|
|
|
|
VisualStudioLikeCompiler.__init__(self, target)
|
|
|
|
self.id = 'clang-cl'
|
|
|
|
|
|
|
|
|
|
|
|
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):
|
|
|
|
CCompiler.__init__(self, exelist, version, for_machine, is_cross, exe_wrap)
|
|
|
|
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_linker_output_args(self, outputname):
|
|
|
|
return ['-output=%s' % outputname]
|
|
|
|
|
|
|
|
def get_werror_args(self):
|
|
|
|
return ['-change_message=error']
|
|
|
|
|
|
|
|
def get_include_args(self, path, is_system):
|
|
|
|
if path == '':
|
|
|
|
path = '.'
|
|
|
|
return ['-include=' + path]
|