|
|
|
# SPDX-License-Identifier: Apache-2.0
|
|
|
|
# Copyright 2020 The Meson development team
|
|
|
|
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
"""Entrypoint script for backend agnostic compile."""
|
|
|
|
|
|
|
|
import os
|
|
|
|
import json
|
|
|
|
import re
|
|
|
|
import sys
|
|
|
|
import shutil
|
|
|
|
import typing as T
|
|
|
|
from collections import defaultdict
|
|
|
|
from pathlib import Path
|
|
|
|
|
|
|
|
from . import mlog
|
|
|
|
from . import mesonlib
|
|
|
|
from .options import OptionKey
|
|
|
|
from .mesonlib import MesonException, RealPathAction, join_args, listify_array_value, setup_vsenv
|
|
|
|
from mesonbuild.environment import detect_ninja
|
|
|
|
from mesonbuild import build
|
|
|
|
|
|
|
|
if T.TYPE_CHECKING:
|
|
|
|
import argparse
|
|
|
|
|
|
|
|
def array_arg(value: str) -> T.List[str]:
|
|
|
|
return listify_array_value(value)
|
|
|
|
|
|
|
|
def validate_builddir(builddir: Path) -> None:
|
|
|
|
if not (builddir / 'meson-private' / 'coredata.dat').is_file():
|
|
|
|
raise MesonException(f'Current directory is not a meson build directory: `{builddir}`.\n'
|
|
|
|
'Please specify a valid build dir or change the working directory to it.\n'
|
|
|
|
'It is also possible that the build directory was generated with an old\n'
|
|
|
|
'meson version. Please regenerate it in this case.')
|
|
|
|
|
|
|
|
def parse_introspect_data(builddir: Path) -> T.Dict[str, T.List[dict]]:
|
|
|
|
"""
|
|
|
|
Converts a List of name-to-dict to a dict of name-to-dicts (since names are not unique)
|
|
|
|
"""
|
|
|
|
path_to_intro = builddir / 'meson-info' / 'intro-targets.json'
|
|
|
|
if not path_to_intro.exists():
|
|
|
|
raise MesonException(f'`{path_to_intro.name}` is missing! Directory is not configured yet?')
|
|
|
|
with path_to_intro.open(encoding='utf-8') as f:
|
|
|
|
schema = json.load(f)
|
|
|
|
|
|
|
|
parsed_data: T.Dict[str, T.List[dict]] = defaultdict(list)
|
|
|
|
for target in schema:
|
|
|
|
parsed_data[target['name']] += [target]
|
|
|
|
return parsed_data
|
|
|
|
|
|
|
|
class ParsedTargetName:
|
|
|
|
full_name = ''
|
|
|
|
base_name = ''
|
|
|
|
name = ''
|
|
|
|
type = ''
|
|
|
|
path = ''
|
|
|
|
suffix = ''
|
|
|
|
|
|
|
|
def __init__(self, target: str):
|
|
|
|
self.full_name = target
|
|
|
|
split = target.rsplit(':', 1)
|
|
|
|
if len(split) > 1:
|
|
|
|
self.type = split[1]
|
|
|
|
if not self._is_valid_type(self.type):
|
|
|
|
raise MesonException(f'Can\'t invoke target `{target}`: unknown target type: `{self.type}`')
|
|
|
|
|
|
|
|
split = split[0].rsplit('/', 1)
|
|
|
|
if len(split) > 1:
|
|
|
|
self.path = split[0]
|
|
|
|
self.name = split[1]
|
|
|
|
else:
|
|
|
|
self.name = split[0]
|
|
|
|
|
|
|
|
split = self.name.rsplit('.', 1)
|
|
|
|
if len(split) > 1:
|
|
|
|
self.base_name = split[0]
|
|
|
|
self.suffix = split[1]
|
|
|
|
else:
|
|
|
|
self.base_name = split[0]
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def _is_valid_type(type: str) -> bool:
|
|
|
|
# Amend docs in Commands.md when editing this list
|
|
|
|
allowed_types = {
|
|
|
|
'executable',
|
|
|
|
'static_library',
|
|
|
|
'shared_library',
|
|
|
|
'shared_module',
|
|
|
|
'custom',
|
|
|
|
'alias',
|
|
|
|
'run',
|
|
|
|
'jar',
|
|
|
|
}
|
|
|
|
return type in allowed_types
|
|
|
|
|
|
|
|
def get_target_from_intro_data(target: ParsedTargetName, builddir: Path, introspect_data: T.Dict[str, T.Any]) -> T.Dict[str, T.Any]:
|
|
|
|
if target.name not in introspect_data and target.base_name not in introspect_data:
|
|
|
|
raise MesonException(f'Can\'t invoke target `{target.full_name}`: target not found')
|
|
|
|
|
|
|
|
intro_targets = introspect_data[target.name]
|
|
|
|
# if target.name doesn't find anything, try just the base name
|
|
|
|
if not intro_targets:
|
|
|
|
intro_targets = introspect_data[target.base_name]
|
|
|
|
found_targets: T.List[T.Dict[str, T.Any]] = []
|
|
|
|
|
|
|
|
resolved_bdir = builddir.resolve()
|
|
|
|
|
|
|
|
if not target.type and not target.path and not target.suffix:
|
|
|
|
found_targets = intro_targets
|
|
|
|
else:
|
|
|
|
for intro_target in intro_targets:
|
|
|
|
# Parse out the name from the id if needed
|
|
|
|
intro_target_name = intro_target['name']
|
|
|
|
split = intro_target['id'].rsplit('@', 1)
|
|
|
|
if len(split) > 1:
|
|
|
|
split = split[0].split('@@', 1)
|
|
|
|
if len(split) > 1:
|
|
|
|
intro_target_name = split[1]
|
|
|
|
else:
|
|
|
|
intro_target_name = split[0]
|
|
|
|
if ((target.type and target.type != intro_target['type'].replace(' ', '_')) or
|
|
|
|
(target.name != intro_target_name) or
|
|
|
|
(target.path and intro_target['filename'] != 'no_name' and
|
|
|
|
Path(target.path) != Path(intro_target['filename'][0]).relative_to(resolved_bdir).parent)):
|
|
|
|
continue
|
|
|
|
found_targets += [intro_target]
|
|
|
|
|
|
|
|
if not found_targets:
|
|
|
|
raise MesonException(f'Can\'t invoke target `{target.full_name}`: target not found')
|
|
|
|
elif len(found_targets) > 1:
|
|
|
|
suggestions: T.List[str] = []
|
|
|
|
for i in found_targets:
|
|
|
|
i_name = i['name']
|
|
|
|
split = i['id'].rsplit('@', 1)
|
|
|
|
if len(split) > 1:
|
|
|
|
split = split[0].split('@@', 1)
|
|
|
|
if len(split) > 1:
|
|
|
|
i_name = split[1]
|
|
|
|
else:
|
|
|
|
i_name = split[0]
|
|
|
|
p = Path(i['filename'][0]).relative_to(resolved_bdir).parent / i_name
|
|
|
|
t = i['type'].replace(' ', '_')
|
|
|
|
suggestions.append(f'- ./{p}:{t}')
|
|
|
|
suggestions_str = '\n'.join(suggestions)
|
|
|
|
raise MesonException(f'Can\'t invoke target `{target.full_name}`: ambiguous name.'
|
|
|
|
f' Add target type and/or path:\n{suggestions_str}')
|
|
|
|
|
|
|
|
return found_targets[0]
|
|
|
|
|
|
|
|
def generate_target_names_ninja(target: ParsedTargetName, builddir: Path, introspect_data: dict) -> T.List[str]:
|
|
|
|
intro_target = get_target_from_intro_data(target, builddir, introspect_data)
|
|
|
|
|
|
|
|
if intro_target['type'] in {'alias', 'run'}:
|
|
|
|
return [target.name]
|
|
|
|
else:
|
|
|
|
return [str(Path(out_file).relative_to(builddir.resolve())) for out_file in intro_target['filename']]
|
|
|
|
|
|
|
|
def get_parsed_args_ninja(options: 'argparse.Namespace', builddir: Path) -> T.Tuple[T.List[str], T.Optional[T.Dict[str, str]]]:
|
|
|
|
runner = detect_ninja()
|
|
|
|
if runner is None:
|
|
|
|
raise MesonException('Cannot find ninja.')
|
|
|
|
|
|
|
|
cmd = runner
|
|
|
|
if not builddir.samefile('.'):
|
|
|
|
cmd.extend(['-C', builddir.as_posix()])
|
|
|
|
|
|
|
|
# If the value is set to < 1 then don't set anything, which let's
|
|
|
|
# ninja/samu decide what to do.
|
|
|
|
if options.jobs > 0:
|
|
|
|
cmd.extend(['-j', str(options.jobs)])
|
|
|
|
if options.load_average > 0:
|
|
|
|
cmd.extend(['-l', str(options.load_average)])
|
|
|
|
|
|
|
|
if options.verbose:
|
|
|
|
cmd.append('-v')
|
|
|
|
|
|
|
|
cmd += options.ninja_args
|
|
|
|
|
|
|
|
# operands must be processed after options/option-arguments
|
|
|
|
if options.targets:
|
|
|
|
intro_data = parse_introspect_data(builddir)
|
|
|
|
for t in options.targets:
|
|
|
|
cmd.extend(generate_target_names_ninja(ParsedTargetName(t), builddir, intro_data))
|
|
|
|
if options.clean:
|
|
|
|
cmd.append('clean')
|
|
|
|
|
|
|
|
return cmd, None
|
|
|
|
|
|
|
|
def generate_target_name_vs(target: ParsedTargetName, builddir: Path, introspect_data: dict) -> str:
|
|
|
|
intro_target = get_target_from_intro_data(target, builddir, introspect_data)
|
|
|
|
|
|
|
|
assert intro_target['type'] not in {'alias', 'run'}, 'Should not reach here: `run` targets must be handle above'
|
|
|
|
|
|
|
|
# Normalize project name
|
|
|
|
# Source: https://docs.microsoft.com/en-us/visualstudio/msbuild/how-to-build-specific-targets-in-solutions-by-using-msbuild-exe
|
|
|
|
target_name = re.sub(r"[\%\$\@\;\.\(\)']", '_', intro_target['id'])
|
|
|
|
rel_path = Path(intro_target['filename'][0]).relative_to(builddir.resolve()).parent
|
|
|
|
if rel_path != Path('.'):
|
|
|
|
target_name = str(rel_path / target_name)
|
|
|
|
return target_name
|
|
|
|
|
|
|
|
def get_parsed_args_vs(options: 'argparse.Namespace', builddir: Path) -> T.Tuple[T.List[str], T.Optional[T.Dict[str, str]]]:
|
|
|
|
slns = list(builddir.glob('*.sln'))
|
|
|
|
assert len(slns) == 1, 'More than one solution in a project?'
|
|
|
|
sln = slns[0]
|
|
|
|
|
|
|
|
cmd = ['msbuild']
|
|
|
|
|
|
|
|
if options.targets:
|
|
|
|
intro_data = parse_introspect_data(builddir)
|
|
|
|
has_run_target = any(
|
|
|
|
get_target_from_intro_data(ParsedTargetName(t), builddir, intro_data)['type'] in {'alias', 'run'}
|
|
|
|
for t in options.targets)
|
|
|
|
|
|
|
|
if has_run_target:
|
|
|
|
# `run` target can't be used the same way as other targets on `vs` backend.
|
|
|
|
# They are defined as disabled projects, which can't be invoked as `.sln`
|
|
|
|
# target and have to be invoked directly as project instead.
|
|
|
|
# Issue: https://github.com/microsoft/msbuild/issues/4772
|
|
|
|
|
|
|
|
if len(options.targets) > 1:
|
|
|
|
raise MesonException('Only one target may be specified when `run` target type is used on this backend.')
|
|
|
|
intro_target = get_target_from_intro_data(ParsedTargetName(options.targets[0]), builddir, intro_data)
|
|
|
|
proj_dir = Path(intro_target['filename'][0]).parent
|
|
|
|
proj = proj_dir/'{}.vcxproj'.format(intro_target['id'])
|
|
|
|
cmd += [str(proj.resolve())]
|
|
|
|
else:
|
|
|
|
cmd += [str(sln.resolve())]
|
|
|
|
cmd.extend(['-target:{}'.format(generate_target_name_vs(ParsedTargetName(t), builddir, intro_data)) for t in options.targets])
|
|
|
|
else:
|
|
|
|
cmd += [str(sln.resolve())]
|
|
|
|
|
|
|
|
if options.clean:
|
|
|
|
cmd.extend(['-target:Clean'])
|
|
|
|
|
|
|
|
# In msbuild `-maxCpuCount` with no number means "detect cpus", the default is `-maxCpuCount:1`
|
|
|
|
if options.jobs > 0:
|
|
|
|
cmd.append(f'-maxCpuCount:{options.jobs}')
|
|
|
|
else:
|
|
|
|
cmd.append('-maxCpuCount')
|
|
|
|
|
|
|
|
if options.load_average:
|
|
|
|
mlog.warning('Msbuild does not have a load-average switch, ignoring.')
|
|
|
|
|
|
|
|
if not options.verbose:
|
|
|
|
cmd.append('-verbosity:minimal')
|
|
|
|
|
|
|
|
cmd += options.vs_args
|
|
|
|
|
|
|
|
# Remove platform from env if set so that msbuild does not
|
|
|
|
# pick x86 platform when solution platform is Win32
|
|
|
|
env = os.environ.copy()
|
|
|
|
env.pop('PLATFORM', None)
|
|
|
|
|
|
|
|
return cmd, env
|
|
|
|
|
|
|
|
def get_parsed_args_xcode(options: 'argparse.Namespace', builddir: Path) -> T.Tuple[T.List[str], T.Optional[T.Dict[str, str]]]:
|
|
|
|
runner = 'xcodebuild'
|
|
|
|
if not shutil.which(runner):
|
|
|
|
raise MesonException('Cannot find xcodebuild, did you install XCode?')
|
|
|
|
|
|
|
|
# No argument to switch directory
|
|
|
|
os.chdir(str(builddir))
|
|
|
|
|
|
|
|
cmd = [runner, '-parallelizeTargets']
|
|
|
|
|
|
|
|
if options.targets:
|
|
|
|
for t in options.targets:
|
|
|
|
cmd += ['-target', t]
|
|
|
|
|
|
|
|
if options.clean:
|
|
|
|
if options.targets:
|
|
|
|
cmd += ['clean']
|
|
|
|
else:
|
|
|
|
cmd += ['-alltargets', 'clean']
|
|
|
|
# Otherwise xcodebuild tries to delete the builddir and fails
|
|
|
|
cmd += ['-UseNewBuildSystem=FALSE']
|
|
|
|
|
|
|
|
if options.jobs > 0:
|
|
|
|
cmd.extend(['-jobs', str(options.jobs)])
|
|
|
|
|
|
|
|
if options.load_average > 0:
|
|
|
|
mlog.warning('xcodebuild does not have a load-average switch, ignoring')
|
|
|
|
|
|
|
|
if options.verbose:
|
|
|
|
# xcodebuild is already quite verbose, and -quiet doesn't print any
|
|
|
|
# status messages
|
|
|
|
pass
|
|
|
|
|
|
|
|
cmd += options.xcode_args
|
|
|
|
return cmd, None
|
|
|
|
|
|
|
|
# Note: when adding arguments, please also add them to the completion
|
|
|
|
# scripts in $MESONSRC/data/shell-completions/
|
|
|
|
def add_arguments(parser: 'argparse.ArgumentParser') -> None:
|
|
|
|
"""Add compile specific arguments."""
|
|
|
|
parser.add_argument(
|
|
|
|
'targets',
|
|
|
|
metavar='TARGET',
|
|
|
|
nargs='*',
|
|
|
|
default=None,
|
|
|
|
help='Targets to build. Target has the following format: [PATH_TO_TARGET/]TARGET_NAME.TARGET_SUFFIX[:TARGET_TYPE].')
|
|
|
|
parser.add_argument(
|
|
|
|
'--clean',
|
|
|
|
action='store_true',
|
|
|
|
help='Clean the build directory.'
|
|
|
|
)
|
|
|
|
parser.add_argument('-C', dest='wd', action=RealPathAction,
|
|
|
|
help='directory to cd into before running')
|
|
|
|
|
|
|
|
parser.add_argument(
|
|
|
|
'-j', '--jobs',
|
|
|
|
action='store',
|
|
|
|
default=0,
|
|
|
|
type=int,
|
|
|
|
help='The number of worker jobs to run (if supported). If the value is less than 1 the build program will guess.'
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'-l', '--load-average',
|
|
|
|
action='store',
|
|
|
|
default=0,
|
|
|
|
type=float,
|
|
|
|
help='The system load average to try to maintain (if supported).'
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'-v', '--verbose',
|
|
|
|
action='store_true',
|
|
|
|
help='Show more verbose output.'
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--ninja-args',
|
|
|
|
type=array_arg,
|
|
|
|
default=[],
|
|
|
|
help='Arguments to pass to `ninja` (applied only on `ninja` backend).'
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--vs-args',
|
|
|
|
type=array_arg,
|
|
|
|
default=[],
|
|
|
|
help='Arguments to pass to `msbuild` (applied only on `vs` backend).'
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--xcode-args',
|
|
|
|
type=array_arg,
|
|
|
|
default=[],
|
|
|
|
help='Arguments to pass to `xcodebuild` (applied only on `xcode` backend).'
|
|
|
|
)
|
|
|
|
|
|
|
|
def run(options: 'argparse.Namespace') -> int:
|
|
|
|
bdir = Path(options.wd)
|
|
|
|
validate_builddir(bdir)
|
|
|
|
if options.targets and options.clean:
|
|
|
|
raise MesonException('`TARGET` and `--clean` can\'t be used simultaneously')
|
|
|
|
|
|
|
|
b = build.load(options.wd)
|
|
|
|
cdata = b.environment.coredata
|
|
|
|
need_vsenv = T.cast('bool', cdata.get_option(OptionKey('vsenv')))
|
|
|
|
if setup_vsenv(need_vsenv):
|
|
|
|
mlog.log(mlog.green('INFO:'), 'automatically activated MSVC compiler environment')
|
|
|
|
|
|
|
|
cmd: T.List[str] = []
|
|
|
|
env: T.Optional[T.Dict[str, str]] = None
|
|
|
|
|
|
|
|
backend = cdata.get_option(OptionKey('backend'))
|
|
|
|
assert isinstance(backend, str)
|
|
|
|
mlog.log(mlog.green('INFO:'), 'autodetecting backend as', backend)
|
|
|
|
if backend == 'ninja':
|
|
|
|
cmd, env = get_parsed_args_ninja(options, bdir)
|
|
|
|
elif backend.startswith('vs'):
|
|
|
|
cmd, env = get_parsed_args_vs(options, bdir)
|
|
|
|
elif backend == 'xcode':
|
|
|
|
cmd, env = get_parsed_args_xcode(options, bdir)
|
|
|
|
else:
|
|
|
|
raise MesonException(
|
|
|
|
f'Backend `{backend}` is not yet supported by `compile`. Use generated project files directly instead.')
|
|
|
|
|
|
|
|
mlog.log(mlog.green('INFO:'), 'calculating backend command to run:', join_args(cmd))
|
|
|
|
p, *_ = mesonlib.Popen_safe(cmd, stdout=sys.stdout.buffer, stderr=sys.stderr.buffer, env=env)
|
|
|
|
|
|
|
|
return p.returncode
|