| // Copyright 2017 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; |
| |
| import com.google.common.annotations.VisibleForTesting; |
| import com.google.common.collect.ImmutableList; |
| import com.google.devtools.build.lib.analysis.PlatformConfiguration; |
| import com.google.devtools.build.lib.analysis.PlatformOptions; |
| import com.google.devtools.build.lib.analysis.config.BuildConfiguration; |
| import com.google.devtools.build.lib.analysis.platform.ConstraintValueInfo; |
| import com.google.devtools.build.lib.analysis.platform.DeclaredToolchainInfo; |
| import com.google.devtools.build.lib.analysis.platform.PlatformInfo; |
| import com.google.devtools.build.lib.cmdline.Label; |
| import com.google.devtools.build.lib.events.Event; |
| import com.google.devtools.build.lib.events.EventHandler; |
| import com.google.devtools.build.lib.packages.NoSuchThingException; |
| import com.google.devtools.build.lib.skyframe.ConfiguredTargetFunction.ConfiguredValueCreationException; |
| import com.google.devtools.build.lib.skyframe.RegisteredToolchainsFunction.InvalidToolchainLabelException; |
| import com.google.devtools.build.lib.skyframe.ToolchainResolutionValue.ToolchainResolutionKey; |
| import com.google.devtools.build.lib.syntax.EvalException; |
| import com.google.devtools.build.skyframe.SkyFunction; |
| import com.google.devtools.build.skyframe.SkyFunctionException; |
| import com.google.devtools.build.skyframe.SkyKey; |
| import com.google.devtools.build.skyframe.SkyValue; |
| import javax.annotation.Nullable; |
| |
| /** {@link SkyFunction} which performs toolchain resolution for a class of rules. */ |
| public class ToolchainResolutionFunction implements SkyFunction { |
| |
| @Nullable |
| @Override |
| public SkyValue compute(SkyKey skyKey, Environment env) |
| throws SkyFunctionException, InterruptedException { |
| ToolchainResolutionKey key = (ToolchainResolutionKey) skyKey.argument(); |
| |
| BuildConfiguration configuration = key.configuration(); |
| PlatformConfiguration platformConfiguration = |
| configuration.getFragment(PlatformConfiguration.class); |
| |
| if (platformConfiguration.hasToolchainOverride(key.toolchainType())) { |
| // Short circuit everything and just return the override. |
| return ToolchainResolutionValue.create( |
| platformConfiguration.getToolchainOverride(key.toolchainType())); |
| } |
| |
| // Get all toolchains. |
| RegisteredToolchainsValue toolchains; |
| try { |
| toolchains = |
| (RegisteredToolchainsValue) |
| env.getValueOrThrow( |
| RegisteredToolchainsValue.key(key.configuration()), |
| InvalidToolchainLabelException.class, |
| EvalException.class); |
| if (toolchains == null) { |
| return null; |
| } |
| } catch (InvalidToolchainLabelException e) { |
| throw new ToolchainResolutionFunctionException(e); |
| } catch (EvalException e) { |
| throw new ToolchainResolutionFunctionException(e); |
| } |
| |
| // Find the right one. |
| boolean debug = configuration.getOptions().get(PlatformOptions.class).toolchainResolutionDebug; |
| DeclaredToolchainInfo toolchain = |
| resolveConstraints( |
| key.toolchainType(), |
| key.execPlatform(), |
| key.targetPlatform(), |
| toolchains.registeredToolchains(), |
| debug ? env.getListener() : null); |
| |
| if (toolchain == null) { |
| throw new ToolchainResolutionFunctionException( |
| new NoToolchainFoundException(key.toolchainType())); |
| } |
| return ToolchainResolutionValue.create(toolchain.toolchainLabel()); |
| } |
| |
| @VisibleForTesting |
| static DeclaredToolchainInfo resolveConstraints( |
| Label toolchainType, |
| PlatformInfo execPlatform, |
| PlatformInfo targetPlatform, |
| ImmutableList<DeclaredToolchainInfo> toolchains, |
| @Nullable EventHandler eventHandler) { |
| |
| debugMessage(eventHandler, "Looking for toolchain of type %s...", toolchainType); |
| for (DeclaredToolchainInfo toolchain : toolchains) { |
| // Make sure the type matches. |
| if (!toolchain.toolchainType().equals(toolchainType)) { |
| continue; |
| } |
| debugMessage(eventHandler, " Considering toolchain %s...", toolchain.toolchainLabel()); |
| if (!checkConstraints(eventHandler, toolchain.execConstraints(), "execution", execPlatform)) { |
| debugMessage( |
| eventHandler, |
| " Rejected toolchain %s, because of execution platform mismatch", |
| toolchain.toolchainLabel()); |
| continue; |
| } |
| if (!checkConstraints( |
| eventHandler, toolchain.targetConstraints(), "target", targetPlatform)) { |
| debugMessage( |
| eventHandler, |
| " Rejected toolchain %s, because of target platform mismatch", |
| toolchain.toolchainLabel()); |
| continue; |
| } |
| |
| debugMessage(eventHandler, " Selected toolchain %s", toolchain.toolchainLabel()); |
| return toolchain; |
| } |
| |
| debugMessage(eventHandler, " No toolchain found"); |
| return null; |
| } |
| |
| /** |
| * Helper method to print a debugging message, if the given {@link EventHandler} is not {@code |
| * null}. |
| */ |
| private static void debugMessage( |
| @Nullable EventHandler eventHandler, String template, Object... args) { |
| if (eventHandler == null) { |
| return; |
| } |
| |
| eventHandler.handle(Event.info("ToolchainResolution: " + String.format(template, args))); |
| } |
| |
| /** |
| * Returns {@code true} iff all constraints set by the toolchain are present in the {@link |
| * PlatformInfo}. |
| */ |
| private static boolean checkConstraints( |
| @Nullable EventHandler eventHandler, |
| Iterable<ConstraintValueInfo> toolchainConstraints, |
| String platformType, |
| PlatformInfo platform) { |
| |
| for (ConstraintValueInfo constraint : toolchainConstraints) { |
| ConstraintValueInfo found = platform.getConstraint(constraint.constraint()); |
| if (!constraint.equals(found)) { |
| debugMessage( |
| eventHandler, |
| " Toolchain constraint %s has value %s, " |
| + "which does not match value %s from the %s platform %s", |
| constraint.constraint().label(), |
| constraint.label(), |
| found != null ? found.label() : "<missing>", |
| platformType, |
| platform.label()); |
| return false; |
| } |
| } |
| return true; |
| } |
| |
| @Nullable |
| @Override |
| public String extractTag(SkyKey skyKey) { |
| return null; |
| } |
| |
| /** Used to indicate that a toolchain was not found for the current request. */ |
| public static final class NoToolchainFoundException extends NoSuchThingException { |
| private final Label missingToolchainType; |
| |
| public NoToolchainFoundException(Label missingToolchainType) { |
| super(String.format("no matching toolchain found for %s", missingToolchainType)); |
| this.missingToolchainType = missingToolchainType; |
| } |
| |
| public Label missingToolchainType() { |
| return missingToolchainType; |
| } |
| } |
| |
| /** Used to indicate errors during the computation of an {@link ToolchainResolutionValue}. */ |
| private static final class ToolchainResolutionFunctionException extends SkyFunctionException { |
| public ToolchainResolutionFunctionException(NoToolchainFoundException e) { |
| super(e, Transience.PERSISTENT); |
| } |
| |
| public ToolchainResolutionFunctionException(ConfiguredValueCreationException e) { |
| super(e, Transience.PERSISTENT); |
| } |
| |
| public ToolchainResolutionFunctionException(InvalidToolchainLabelException e) { |
| super(e, Transience.PERSISTENT); |
| } |
| |
| public ToolchainResolutionFunctionException(EvalException e) { |
| super(e, Transience.PERSISTENT); |
| } |
| } |
| } |