// Copyright 2015 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.base.Preconditions; import com.google.common.collect.Collections2; import com.google.common.collect.ImmutableSet; import com.google.common.collect.Lists; import com.google.common.collect.Multimap; import com.google.common.collect.Sets; import com.google.devtools.build.lib.cmdline.Label; import com.google.devtools.build.lib.events.EventHandler; import com.google.devtools.build.lib.packages.Attribute; import com.google.devtools.build.lib.packages.BuildFileContainsErrorsException; import com.google.devtools.build.lib.packages.DependencyFilter; import com.google.devtools.build.lib.packages.InputFile; import com.google.devtools.build.lib.packages.NoSuchPackageException; import com.google.devtools.build.lib.packages.NoSuchTargetException; import com.google.devtools.build.lib.packages.OutputFile; import com.google.devtools.build.lib.packages.Package; import com.google.devtools.build.lib.packages.PackageGroup; import com.google.devtools.build.lib.packages.Rule; import com.google.devtools.build.lib.packages.Target; 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 com.google.devtools.build.skyframe.ValueOrException2; import java.util.Collection; import java.util.HashMap; import java.util.HashSet; import java.util.List; import java.util.Map; import java.util.Set; import javax.annotation.Nullable; /** * This class can be extended to define {@link SkyFunction}s that traverse a target and its * transitive dependencies and return values based on that traversal. * *

The {@code TProcessedTargets} type parameter represents the result of processing a target and * its transitive dependencies. * *

{@code TransitiveBaseTraversalFunction} asks for one to be constructed via {@link * #processTarget}, and then asks for it to be updated based on the current target's * attributes' dependencies via {@link #processDeps}, and then asks for it to be updated based * on the current target' aspects' dependencies via {@link #processDeps}. Finally, it calls * {@link #computeSkyValue} with the {#code ProcessedTargets} to get the {@link SkyValue} to * return. */ abstract class TransitiveBaseTraversalFunction implements SkyFunction { /** * Returns a {@link SkyKey} corresponding to the traversal of a target specified by {@code label} * and its transitive dependencies. * *

Extenders of this class should implement this function to return a key with their * specialized {@link SkyFunction}'s name. * *

{@link TransitiveBaseTraversalFunction} calls this for each dependency of a target, and * then gets their values from the environment. * *

The key's {@link SkyFunction} may throw at most {@link NoSuchPackageException} and * {@link NoSuchTargetException}. Other exception types are not handled by {@link * TransitiveBaseTraversalFunction}. */ abstract SkyKey getKey(Label label); abstract TProcessedTargets processTarget(Label label, TargetAndErrorIfAny targetAndErrorIfAny); abstract void processDeps( TProcessedTargets processedTargets, EventHandler eventHandler, TargetAndErrorIfAny targetAndErrorIfAny, Iterable>> depEntries) throws InterruptedException; /** * Returns a {@link SkyValue} based on the target and any errors it has, and the values * accumulated across it and a traversal of its transitive dependencies. */ abstract SkyValue computeSkyValue(TargetAndErrorIfAny targetAndErrorIfAny, TProcessedTargets processedTargets); /** * Returns a {@link TargetMarkerValue} corresponding to the {@param targetMarkerKey} or {@code * null} if the value isn't ready. */ @Nullable abstract TargetMarkerValue getTargetMarkerValue(SkyKey targetMarkerKey, Environment env) throws NoSuchTargetException, NoSuchPackageException, InterruptedException; abstract Label argumentFromKey(SkyKey key); @Override public SkyValue compute(SkyKey key, Environment env) throws TransitiveBaseTraversalFunctionException, InterruptedException { Label label = argumentFromKey(key); LoadTargetResults loadTargetResults; try { loadTargetResults = loadTarget(env, label); } catch (NoSuchTargetException e) { throw new TransitiveBaseTraversalFunctionException(e); } catch (NoSuchPackageException e) { throw new TransitiveBaseTraversalFunctionException(e); } LoadTargetResultsType loadTargetResultsType = loadTargetResults.getType(); if (loadTargetResultsType.equals(LoadTargetResultsType.VALUES_MISSING)) { return null; } Preconditions.checkState( loadTargetResultsType.equals(LoadTargetResultsType.TARGET_AND_ERROR_IF_ANY), loadTargetResultsType); TargetAndErrorIfAny targetAndErrorIfAny = (TargetAndErrorIfAny) loadTargetResults; // Process deps from attributes. It is essential that the last getValue(s) call we made to // skyframe for building this node was for the corresponding PackageValue. Collection labelDepKeys = getLabelDepKeys(env, targetAndErrorIfAny); Map> depMap = env.getValuesOrThrow(labelDepKeys, NoSuchPackageException.class, NoSuchTargetException.class); if (env.valuesMissing()) { return null; } // Process deps from attributes. It is essential that the second-to-last getValue(s) call we // made to skyframe for building this node was for the corresponding PackageValue. Iterable labelAspectKeys = getStrictLabelAspectDepKeys(env, depMap, targetAndErrorIfAny); Set>> labelAspectEntries = env.getValuesOrThrow( labelAspectKeys, NoSuchPackageException.class, NoSuchTargetException.class) .entrySet(); if (env.valuesMissing()) { return null; } TProcessedTargets processedTargets = processTarget(label, targetAndErrorIfAny); processDeps(processedTargets, env.getListener(), targetAndErrorIfAny, depMap.entrySet()); processDeps(processedTargets, env.getListener(), targetAndErrorIfAny, labelAspectEntries); return computeSkyValue(targetAndErrorIfAny, processedTargets); } Collection getLabelDepKeys( SkyFunction.Environment env, TargetAndErrorIfAny targetAndErrorIfAny) throws InterruptedException { return Collections2.transform(getLabelDeps(targetAndErrorIfAny.getTarget()), this::getKey); } Iterable getStrictLabelAspectDepKeys( SkyFunction.Environment env, Map> depMap, TargetAndErrorIfAny targetAndErrorIfAny) throws InterruptedException { return getStrictLabelAspectKeys(targetAndErrorIfAny.getTarget(), depMap, env); } @Override public String extractTag(SkyKey skyKey) { return Label.print(argumentFromKey(skyKey)); } /** * Return an Iterable of SkyKeys corresponding to the Aspect-related dependencies of target. * *

This method may return a precise set of aspect keys, but may need to request additional * dependencies from the env to do so. */ private Iterable getStrictLabelAspectKeys( Target target, Map> depMap, Environment env) throws InterruptedException { List depKeys = Lists.newArrayList(); if (target instanceof Rule) { Map> labelDepMap = new HashMap<>(depMap.size()); for (Map.Entry> entry : depMap.entrySet()) { labelDepMap.put(argumentFromKey(entry.getKey()), entry.getValue()); } Multimap transitions = ((Rule) target).getTransitions(DependencyFilter.NO_NODEP_ATTRIBUTES); for (Map.Entry entry : transitions.entries()) { ValueOrException2 value = labelDepMap.get(entry.getValue()); for (Label label : getAspectLabels((Rule) target, entry.getKey(), entry.getValue(), value, env)) { depKeys.add(getKey(label)); } } } return depKeys; } /** Get the Aspect-related Label deps for the given edge. */ protected abstract Collection

In keep_going mode, used the same way, but only for targets that could not be loaded at * all (we proceed with transitive loading on targets that contain errors).

*/ public TransitiveBaseTraversalFunctionException(NoSuchTargetException e) { super(e, Transience.PERSISTENT); } } }