Skip to content

Commit a0ff145

Browse files
tristan957dcbaker
authored andcommitted
Add required kwarg to compiler.{compiles,links,run}
This is a similar commit to the one that added required to all the compiler.has* functions.
1 parent f1f2481 commit a0ff145

File tree

7 files changed

+124
-8
lines changed

7 files changed

+124
-8
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
## Required kwarg on more `compiler` methods
2+
3+
The following `compiler` methods now support the `required` keyword argument:
4+
5+
- `compiler.compiles()`
6+
- `compiler.links()`
7+
- `compiler.runs()`
8+
9+
```meson
10+
cc.compiles(valid, name: 'valid', required : true)
11+
cc.links(valid, name: 'valid', required : true)
12+
cc.run(valid, name: 'valid', required : true)
13+
14+
assert(not cc.compiles(valid, name: 'valid', required : opt))
15+
assert(not cc.links(valid, name: 'valid', required : opt))
16+
res = cc.run(valid, name: 'valid', required : opt)
17+
assert(res.compiled())
18+
assert(res.returncode() == 0)
19+
assert(res.stdout() == '')
20+
assert(res.stderr() == '')
21+
```

docs/yaml/objects/compiler.yaml

+10
Original file line numberDiff line numberDiff line change
@@ -121,6 +121,16 @@ methods:
121121
- name: _compiles
122122
returns: void
123123
description: You have found a bug if you can see this!
124+
kwargs:
125+
required:
126+
type: bool | feature
127+
default: false
128+
since: 1.5.0
129+
description:
130+
When set to `true`, Meson will halt if the check fails.
131+
132+
When set to a [`feature`](Build-options.md#features) option, the feature
133+
will control if it is searched and whether to fail if not found.
124134
kwargs_inherit:
125135
- compiler._args
126136
- compiler._include_directories

mesonbuild/interpreter/compiler.py

+39-8
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@
1515
from .. import dependencies
1616
from .. import mesonlib
1717
from .. import mlog
18-
from ..compilers import SUFFIX_TO_LANG
18+
from ..compilers import SUFFIX_TO_LANG, RunResult
1919
from ..compilers.compilers import CompileCheckMode
2020
from ..interpreterbase import (ObjectHolder, noPosargs, noKwargs,
2121
FeatureNew, FeatureNewKwargs, disablerIfNotFound,
@@ -27,7 +27,7 @@
2727

2828
if T.TYPE_CHECKING:
2929
from ..interpreter import Interpreter
30-
from ..compilers import Compiler, RunResult
30+
from ..compilers import Compiler
3131
from ..interpreterbase import TYPE_var, TYPE_kwargs
3232
from .kwargs import ExtractRequired, ExtractSearchDirs
3333
from .interpreter import SourceOutputs
@@ -50,7 +50,7 @@ class BaseCompileKW(TypedDict):
5050
include_directories: T.List[build.IncludeDirs]
5151
args: T.List[str]
5252

53-
class CompileKW(BaseCompileKW):
53+
class CompileKW(BaseCompileKW, ExtractRequired):
5454

5555
name: str
5656
dependencies: T.List[dependencies.Dependency]
@@ -178,7 +178,8 @@ def stderr_method(self, args: T.List['TYPE_var'], kwargs: 'TYPE_kwargs') -> str:
178178

179179
# Common methods of compiles, links, runs, and similar
180180
_COMPILES_KWS: T.List[KwargInfo] = [_NAME_KW, _ARGS_KW, _DEPENDENCIES_KW, _INCLUDE_DIRS_KW, _NO_BUILTIN_ARGS_KW,
181-
_WERROR_KW]
181+
_WERROR_KW,
182+
REQUIRED_KW.evolve(since='1.5.0', default=False)]
182183

183184
_HEADER_KWS: T.List[KwargInfo] = [REQUIRED_KW.evolve(since='0.50.0', default=False), *_COMMON_KWS]
184185
_HAS_REQUIRED_KW = REQUIRED_KW.evolve(since='1.3.0', default=False)
@@ -306,15 +307,25 @@ def run_method(self, args: T.Tuple['mesonlib.FileOrString'], kwargs: 'CompileKW'
306307
FeatureNew.single_use(f'compiler.run for {self.compiler.get_display_language()} language',
307308
'1.5.0', self.subproject, location=self.current_node)
308309
code = args[0]
310+
testname = kwargs['name']
311+
312+
disabled, required, feature = extract_required_kwarg(kwargs, self.subproject, default=False)
313+
if disabled:
314+
if testname:
315+
mlog.log('Checking if', mlog.bold(testname, True), 'runs:', 'skipped: feature', mlog.bold(feature), 'disabled')
316+
return RunResult(compiled=True, returncode=0, stdout='', stderr='', cached=False)
317+
309318
if isinstance(code, mesonlib.File):
310319
self.interpreter.add_build_def_file(code)
311320
code = mesonlib.File.from_absolute_file(
312321
code.rel_to_builddir(self.environment.source_dir))
313-
testname = kwargs['name']
314322
extra_args = functools.partial(self._determine_args, kwargs)
315323
deps, msg = self._determine_dependencies(kwargs['dependencies'], compile_only=False, endl=None)
316324
result = self.compiler.run(code, self.environment, extra_args=extra_args,
317325
dependencies=deps)
326+
if required and result.returncode != 0:
327+
raise InterpreterException(f'Could not run {testname if testname else "code"}')
328+
318329
if testname:
319330
if not result.compiled:
320331
h = mlog.red('DID NOT COMPILE')
@@ -510,19 +521,29 @@ def has_define_method(self, args: T.Tuple[str], kwargs: 'CommonKW') -> bool:
510521
@typed_kwargs('compiler.compiles', *_COMPILES_KWS)
511522
def compiles_method(self, args: T.Tuple['mesonlib.FileOrString'], kwargs: 'CompileKW') -> bool:
512523
code = args[0]
524+
testname = kwargs['name']
525+
526+
disabled, required, feature = extract_required_kwarg(kwargs, self.subproject, default=False)
527+
if disabled:
528+
if testname:
529+
mlog.log('Checking if', mlog.bold(testname, True), 'compiles:', 'skipped: feature', mlog.bold(feature), 'disabled')
530+
return False
531+
513532
if isinstance(code, mesonlib.File):
514533
if code.is_built:
515534
FeatureNew.single_use('compiler.compiles with file created at setup time', '1.2.0', self.subproject,
516535
'It was broken and either errored or returned false.', self.current_node)
517536
self.interpreter.add_build_def_file(code)
518537
code = mesonlib.File.from_absolute_file(
519538
code.absolute_path(self.environment.source_dir, self.environment.build_dir))
520-
testname = kwargs['name']
521539
extra_args = functools.partial(self._determine_args, kwargs)
522540
deps, msg = self._determine_dependencies(kwargs['dependencies'], endl=None)
523541
result, cached = self.compiler.compiles(code, self.environment,
524542
extra_args=extra_args,
525543
dependencies=deps)
544+
if required and not result:
545+
raise InterpreterException(f'Could not compile {testname}')
546+
526547
if testname:
527548
if result:
528549
h = mlog.green('YES')
@@ -536,6 +557,14 @@ def compiles_method(self, args: T.Tuple['mesonlib.FileOrString'], kwargs: 'Compi
536557
@typed_kwargs('compiler.links', *_COMPILES_KWS)
537558
def links_method(self, args: T.Tuple['mesonlib.FileOrString'], kwargs: 'CompileKW') -> bool:
538559
code = args[0]
560+
testname = kwargs['name']
561+
562+
disabled, required, feature = extract_required_kwarg(kwargs, self.subproject, default=False)
563+
if disabled:
564+
if testname:
565+
mlog.log('Checking if', mlog.bold(testname, True), 'links:', 'skipped: feature', mlog.bold(feature), 'disabled')
566+
return False
567+
539568
compiler = None
540569
if isinstance(code, mesonlib.File):
541570
if code.is_built:
@@ -556,19 +585,21 @@ def links_method(self, args: T.Tuple['mesonlib.FileOrString'], kwargs: 'CompileK
556585
else:
557586
compiler = clist[SUFFIX_TO_LANG[suffix]]
558587

559-
testname = kwargs['name']
560588
extra_args = functools.partial(self._determine_args, kwargs)
561589
deps, msg = self._determine_dependencies(kwargs['dependencies'], compile_only=False)
562590
result, cached = self.compiler.links(code, self.environment,
563591
compiler=compiler,
564592
extra_args=extra_args,
565593
dependencies=deps)
566-
cached_msg = mlog.blue('(cached)') if cached else ''
594+
if required and not result:
595+
raise InterpreterException(f'Could not link {testname if testname else "code"}')
596+
567597
if testname:
568598
if result:
569599
h = mlog.green('YES')
570600
else:
571601
h = mlog.red('NO')
602+
cached_msg = mlog.blue('(cached)') if cached else ''
572603
mlog.log('Checking if', mlog.bold(testname, True), msg, 'links:', h, cached_msg)
573604
return result
574605

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
// The error in this file is an homage to the xz incident :)
2+
//
3+
int
4+
main(void)
5+
{
6+
. return 0;
7+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
project('required keyword in compiles functions', 'c')
2+
3+
cc = meson.get_compiler('c')
4+
opt = get_option('opt')
5+
6+
valid = files('valid.c')
7+
invalid = files('invalid.c')
8+
9+
cc.compiles(valid, name: 'valid', required : true)
10+
cc.links(valid, name: 'valid', required : true)
11+
if meson.can_run_host_binaries()
12+
cc.run(valid, name: 'valid', required : true)
13+
endif
14+
15+
assert(not cc.compiles(valid, name: 'valid', required : opt))
16+
assert(not cc.links(valid, name: 'valid', required : opt))
17+
if meson.can_run_host_binaries()
18+
res = cc.run(valid, name: 'valid', required : opt)
19+
assert(res.compiled())
20+
assert(res.returncode() == 0)
21+
assert(res.stdout() == '')
22+
assert(res.stderr() == '')
23+
endif
24+
25+
testcase expect_error('''compiler.compiles keyword argument 'required' was of type str but should have been one of: bool, UserFeatureOption''')
26+
cc.compiles(valid, name: 'valid', required : 'not a bool')
27+
endtestcase
28+
29+
testcase expect_error('''Could not compile invalid''')
30+
cc.compiles(invalid, name: 'invalid', required : true)
31+
endtestcase
32+
33+
testcase expect_error('''Could not link invalid''')
34+
cc.links(invalid, name: 'invalid', required : true)
35+
endtestcase
36+
37+
if meson.can_run_host_binaries()
38+
testcase expect_error('''Could not run invalid''')
39+
cc.run(invalid, name: 'invalid', required : true)
40+
endtestcase
41+
endif
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
option('opt', type: 'feature', value: 'disabled')
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
int
2+
main(void)
3+
{
4+
return 0;
5+
}

0 commit comments

Comments
 (0)