// 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.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.packages.NoSuchThingException; import com.google.devtools.build.lib.skyframe.ConfiguredTargetFunction.ConfiguredValueCreationException; import com.google.devtools.build.lib.skyframe.RegisteredToolchainsFunction.InvalidTargetException; 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(); // Get all toolchains. RegisteredToolchainsValue toolchains; try { toolchains = (RegisteredToolchainsValue) env.getValueOrThrow( RegisteredToolchainsValue.key(key.configuration()), ConfiguredValueCreationException.class, InvalidTargetException.class, EvalException.class); if (toolchains == null) { return null; } } catch (ConfiguredValueCreationException e) { throw new ToolchainResolutionFunctionException(e); } catch (InvalidTargetException e) { throw new ToolchainResolutionFunctionException(e); } catch (EvalException e) { throw new ToolchainResolutionFunctionException(e); } // Find the right one. DeclaredToolchainInfo toolchain = resolveConstraints( key.toolchainType(), key.execPlatform(), key.targetPlatform(), toolchains.registeredToolchains()); 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 toolchains) { for (DeclaredToolchainInfo toolchain : toolchains) { // Make sure the type matches. if (!toolchain.toolchainType().equals(toolchainType)) { continue; } if (!checkConstraints(toolchain.execConstraints(), execPlatform)) { continue; } if (!checkConstraints(toolchain.targetConstraints(), targetPlatform)) { continue; } return toolchain; } return null; } /** * Returns {@code true} iff all constraints set by the toolchain are present in the {@link * PlatformInfo}. */ private static boolean checkConstraints( Iterable toolchainConstraints, PlatformInfo platform) { for (ConstraintValueInfo constraint : toolchainConstraints) { ConstraintValueInfo found = platform.getConstraint(constraint.constraint()); if (!constraint.equals(found)) { 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(InvalidTargetException e) { super(e, Transience.PERSISTENT); } public ToolchainResolutionFunctionException(EvalException e) { super(e, Transience.PERSISTENT); } } }