blob: 671f9f906d676e2d4f0526ab9416c8f7257caeab [file] [log] [blame]
// Copyright 2018 The Bazel Authors. All rights reserved.
//
// 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.
package com.google.devtools.build.lib.skyframe.toolchains;
import static com.google.common.collect.ImmutableList.toImmutableList;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.collect.ImmutableList;
import com.google.devtools.build.lib.analysis.ConfiguredTarget;
import com.google.devtools.build.lib.analysis.ConfiguredTargetValue;
import com.google.devtools.build.lib.analysis.PlatformConfiguration;
import com.google.devtools.build.lib.analysis.config.BuildConfigurationValue;
import com.google.devtools.build.lib.analysis.platform.PlatformInfo;
import com.google.devtools.build.lib.analysis.platform.PlatformProviderUtils;
import com.google.devtools.build.lib.bazel.bzlmod.BazelDepGraphValue;
import com.google.devtools.build.lib.bazel.bzlmod.Module;
import com.google.devtools.build.lib.bazel.bzlmod.ModuleKey;
import com.google.devtools.build.lib.cmdline.Label;
import com.google.devtools.build.lib.cmdline.LabelConstants;
import com.google.devtools.build.lib.cmdline.RepositoryName;
import com.google.devtools.build.lib.cmdline.SignedTargetPattern;
import com.google.devtools.build.lib.cmdline.TargetParsingException;
import com.google.devtools.build.lib.cmdline.TargetPattern;
import com.google.devtools.build.lib.packages.Package;
import com.google.devtools.build.lib.packages.Target;
import com.google.devtools.build.lib.packages.semantics.BuildLanguageOptions;
import com.google.devtools.build.lib.pkgcache.FilteringPolicy;
import com.google.devtools.build.lib.server.FailureDetails.Analysis;
import com.google.devtools.build.lib.server.FailureDetails.Analysis.Code;
import com.google.devtools.build.lib.server.FailureDetails.FailureDetail;
import com.google.devtools.build.lib.server.FailureDetails.Toolchain;
import com.google.devtools.build.lib.skyframe.ConfiguredTargetKey;
import com.google.devtools.build.lib.skyframe.ConfiguredValueCreationException;
import com.google.devtools.build.lib.skyframe.PackageValue;
import com.google.devtools.build.lib.skyframe.PrecomputedValue;
import com.google.devtools.build.lib.skyframe.RepositoryMappingValue;
import com.google.devtools.build.lib.skyframe.SaneAnalysisException;
import com.google.devtools.build.lib.skyframe.TargetPatternUtil;
import com.google.devtools.build.lib.skyframe.TargetPatternUtil.InvalidTargetPatternException;
import com.google.devtools.build.lib.skyframe.serialization.autocodec.SerializationConstant;
import com.google.devtools.build.lib.skyframe.toolchains.PlatformLookupUtil.InvalidPlatformException;
import com.google.devtools.build.lib.util.DetailedExitCode;
import com.google.devtools.build.lib.vfs.PathFragment;
import com.google.devtools.build.skyframe.SkyFunction;
import com.google.devtools.build.skyframe.SkyFunctionException;
import com.google.devtools.build.skyframe.SkyFunctionException.Transience;
import com.google.devtools.build.skyframe.SkyKey;
import com.google.devtools.build.skyframe.SkyValue;
import com.google.devtools.build.skyframe.SkyframeLookupResult;
import java.util.List;
import javax.annotation.Nullable;
import net.starlark.java.eval.StarlarkSemantics;
/** {@link SkyFunction} that returns all registered execution platforms available. */
public class RegisteredExecutionPlatformsFunction implements SkyFunction {
@SerializationConstant
static final FilteringPolicy HAS_PLATFORM_INFO =
(Target target, boolean explicit) -> explicit || PlatformLookupUtil.hasPlatformInfo(target);
@Nullable
@Override
public SkyValue compute(SkyKey skyKey, Environment env)
throws RegisteredExecutionPlatformsFunctionException, InterruptedException {
StarlarkSemantics starlarkSemantics = PrecomputedValue.STARLARK_SEMANTICS.get(env);
BuildConfigurationValue configuration =
(BuildConfigurationValue)
env.getValue(((RegisteredExecutionPlatformsValue.Key) skyKey).getConfigurationKey());
RepositoryMappingValue mainRepoMapping =
(RepositoryMappingValue) env.getValue(RepositoryMappingValue.key(RepositoryName.MAIN));
if (env.valuesMissing()) {
return null;
}
TargetPattern.Parser mainRepoParser =
new TargetPattern.Parser(
PathFragment.EMPTY_FRAGMENT,
RepositoryName.MAIN,
mainRepoMapping.getRepositoryMapping());
ImmutableList.Builder<SignedTargetPattern> targetPatternBuilder = new ImmutableList.Builder<>();
// Get the execution platforms from the configuration.
PlatformConfiguration platformConfiguration =
configuration.getFragment(PlatformConfiguration.class);
if (platformConfiguration != null) {
try {
targetPatternBuilder.addAll(
TargetPatternUtil.parseAllSigned(
platformConfiguration.getExtraExecutionPlatforms(), mainRepoParser));
} catch (InvalidTargetPatternException e) {
throw new RegisteredExecutionPlatformsFunctionException(
new InvalidExecutionPlatformLabelException(e), Transience.PERSISTENT);
}
}
// Get registered execution platforms from the root Bazel module.
ImmutableList<TargetPattern> bzlmodRootModuleExecutionPlatforms =
getBzlmodExecutionPlatforms(starlarkSemantics, env, /* forRootModule= */ true);
if (bzlmodRootModuleExecutionPlatforms == null) {
return null;
}
targetPatternBuilder.addAll(TargetPatternUtil.toSigned(bzlmodRootModuleExecutionPlatforms));
// Get the registered execution platforms from the WORKSPACE.
// The WORKSPACE suffixes don't register any execution platforms, so we can register all
// platforms in WORKSPACE before those in non-root Bazel modules.
ImmutableList<TargetPattern> workspaceExecutionPlatforms = getWorkspaceExecutionPlatforms(env);
if (workspaceExecutionPlatforms == null) {
return null;
}
targetPatternBuilder.addAll(TargetPatternUtil.toSigned(workspaceExecutionPlatforms));
// Get registered execution platforms from the non-root Bazel modules.
ImmutableList<TargetPattern> bzlmodNonRootModuleExecutionPlatforms =
getBzlmodExecutionPlatforms(starlarkSemantics, env, /* forRootModule= */ false);
if (bzlmodNonRootModuleExecutionPlatforms == null) {
return null;
}
targetPatternBuilder.addAll(TargetPatternUtil.toSigned(bzlmodNonRootModuleExecutionPlatforms));
// Expand target patterns.
ImmutableList<Label> platformLabels;
try {
platformLabels =
TargetPatternUtil.expandTargetPatterns(
env, targetPatternBuilder.build(), HAS_PLATFORM_INFO);
if (env.valuesMissing()) {
return null;
}
} catch (TargetPatternUtil.InvalidTargetPatternException e) {
throw new RegisteredExecutionPlatformsFunctionException(
new InvalidExecutionPlatformLabelException(e), Transience.PERSISTENT);
}
// Load the configured target for each, and get the declared execution platforms providers.
ImmutableList<ConfiguredTargetKey> registeredExecutionPlatformKeys =
configureRegisteredExecutionPlatforms(env, configuration, platformLabels);
if (env.valuesMissing()) {
return null;
}
return RegisteredExecutionPlatformsValue.create(registeredExecutionPlatformKeys);
}
/**
* Loads the external package and then returns the registered execution platform labels.
*
* @param env the environment to use for lookups
*/
@Nullable
@VisibleForTesting
public static ImmutableList<TargetPattern> getWorkspaceExecutionPlatforms(Environment env)
throws InterruptedException {
PackageValue externalPackageValue =
(PackageValue) env.getValue(LabelConstants.EXTERNAL_PACKAGE_IDENTIFIER);
if (externalPackageValue == null) {
return null;
}
Package externalPackage = externalPackageValue.getPackage();
return externalPackage.getRegisteredExecutionPlatforms();
}
@Nullable
private static ImmutableList<TargetPattern> getBzlmodExecutionPlatforms(
StarlarkSemantics semantics, Environment env, boolean forRootModule)
throws InterruptedException, RegisteredExecutionPlatformsFunctionException {
if (!semantics.getBool(BuildLanguageOptions.ENABLE_BZLMOD)) {
return ImmutableList.of();
}
BazelDepGraphValue bazelDepGraphValue =
(BazelDepGraphValue) env.getValue(BazelDepGraphValue.KEY);
if (bazelDepGraphValue == null) {
return null;
}
ImmutableList.Builder<TargetPattern> executionPlatforms = ImmutableList.builder();
for (Module module : bazelDepGraphValue.getDepGraph().values()) {
if (forRootModule != module.getKey().equals(ModuleKey.ROOT)) {
continue;
}
TargetPattern.Parser parser =
new TargetPattern.Parser(
PathFragment.EMPTY_FRAGMENT,
module.getCanonicalRepoName(),
bazelDepGraphValue.getFullRepoMapping(module.getKey()));
for (String pattern : module.getExecutionPlatformsToRegister()) {
try {
executionPlatforms.add(parser.parse(pattern));
} catch (TargetParsingException e) {
throw new RegisteredExecutionPlatformsFunctionException(
new InvalidExecutionPlatformLabelException(pattern, e), Transience.PERSISTENT);
}
}
}
return executionPlatforms.build();
}
@Nullable
private static ImmutableList<ConfiguredTargetKey> configureRegisteredExecutionPlatforms(
Environment env, BuildConfigurationValue configuration, List<Label> labels)
throws InterruptedException, RegisteredExecutionPlatformsFunctionException {
ImmutableList<ConfiguredTargetKey> keys =
labels.stream()
.map(
label ->
ConfiguredTargetKey.builder()
.setLabel(label)
.setConfiguration(configuration)
.build())
.collect(toImmutableList());
SkyframeLookupResult values = env.getValuesAndExceptions(keys);
ImmutableList.Builder<ConfiguredTargetKey> validPlatformKeys = new ImmutableList.Builder<>();
boolean valuesMissing = false;
for (ConfiguredTargetKey platformKey : keys) {
Label platformLabel = platformKey.getLabel();
try {
SkyValue value = values.getOrThrow(platformKey, ConfiguredValueCreationException.class);
if (value == null) {
valuesMissing = true;
continue;
}
ConfiguredTarget target = ((ConfiguredTargetValue) value).getConfiguredTarget();
PlatformInfo platformInfo = PlatformProviderUtils.platform(target);
if (platformInfo == null) {
throw new RegisteredExecutionPlatformsFunctionException(
new InvalidPlatformException(platformLabel), Transience.PERSISTENT);
}
validPlatformKeys.add(platformKey);
} catch (ConfiguredValueCreationException e) {
throw new RegisteredExecutionPlatformsFunctionException(
new InvalidPlatformException(platformLabel, e), Transience.PERSISTENT);
}
}
if (valuesMissing) {
return null;
}
return validPlatformKeys.build();
}
/**
* Used to indicate that the given {@link Label} represents a {@link ConfiguredTarget} which is
* not a valid {@link PlatformInfo} provider.
*/
static final class InvalidExecutionPlatformLabelException extends ToolchainException
implements SaneAnalysisException {
InvalidExecutionPlatformLabelException(TargetPatternUtil.InvalidTargetPatternException e) {
this(e.getInvalidPattern(), e.getTpe());
}
InvalidExecutionPlatformLabelException(String invalidPattern, TargetParsingException e) {
super(
String.format(
"invalid registered execution platform '%s': %s", invalidPattern, e.getMessage()),
e);
}
@Override
protected Toolchain.Code getDetailedCode() {
return Toolchain.Code.INVALID_PLATFORM_VALUE;
}
@Override
public DetailedExitCode getDetailedExitCode() {
return DetailedExitCode.of(
FailureDetail.newBuilder()
.setMessage(getMessage())
.setAnalysis(Analysis.newBuilder().setCode(Code.INVALID_EXECUTION_PLATFORM))
.build());
}
}
/**
* Used to declare all the exception types that can be wrapped in the exception thrown by {@link
* #compute}.
*/
private static class RegisteredExecutionPlatformsFunctionException extends SkyFunctionException {
private RegisteredExecutionPlatformsFunctionException(
InvalidExecutionPlatformLabelException cause, Transience transience) {
super(cause, transience);
}
private RegisteredExecutionPlatformsFunctionException(
InvalidPlatformException cause, Transience transience) {
super(cause, transience);
}
}
}