|  | // Copyright 2014 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.skyframe; | 
|  |  | 
|  | import com.google.common.base.Preconditions; | 
|  | import com.google.common.collect.ImmutableList; | 
|  | import com.google.common.collect.ImmutableMap; | 
|  | import com.google.common.collect.ImmutableSet; | 
|  | import com.google.common.collect.Iterables; | 
|  | import com.google.devtools.build.lib.collect.nestedset.NestedSet; | 
|  | import com.google.devtools.build.lib.concurrent.ThreadSafety.ThreadCompatible; | 
|  | import com.google.devtools.build.lib.events.ExtendedEventHandler; | 
|  | import com.google.devtools.build.lib.events.ExtendedEventHandler.Postable; | 
|  | import com.google.devtools.build.lib.profiler.Profiler; | 
|  | import com.google.devtools.build.lib.profiler.ProfilerTask; | 
|  | import com.google.devtools.build.lib.profiler.SilentCloseable; | 
|  | import com.google.devtools.build.lib.util.Pair; | 
|  | import com.google.devtools.build.skyframe.EvaluationProgressReceiver.EvaluationState; | 
|  | import com.google.devtools.build.skyframe.EvaluationProgressReceiver.EvaluationSuccessState; | 
|  | import com.google.devtools.build.skyframe.MemoizingEvaluator.EmittedEventState; | 
|  | import com.google.devtools.build.skyframe.NodeEntry.DependencyState; | 
|  | import com.google.devtools.build.skyframe.QueryableGraph.Reason; | 
|  | import com.google.devtools.build.skyframe.SkyFunctionException.ReifiedSkyFunctionException; | 
|  | import java.util.ArrayList; | 
|  | import java.util.HashMap; | 
|  | import java.util.HashSet; | 
|  | import java.util.List; | 
|  | import java.util.Map; | 
|  | import java.util.Set; | 
|  | import java.util.concurrent.ExecutorService; | 
|  | import java.util.function.Supplier; | 
|  | import java.util.logging.Logger; | 
|  | import javax.annotation.Nullable; | 
|  |  | 
|  | /** | 
|  | * Evaluates a set of given functions ({@code SkyFunction}s) with arguments ({@code SkyKey}s). | 
|  | * Cycles are not allowed and are detected during the traversal. | 
|  | * | 
|  | * <p>This class implements multi-threaded evaluation. This is a fairly complex process that has | 
|  | * strong consistency requirements between the {@link ProcessableGraph}, the nodes in the graph of | 
|  | * type {@link NodeEntry}, the work queue, and the set of in-flight nodes. | 
|  | * | 
|  | * <p>The basic invariants are: | 
|  | * | 
|  | * <p>A node can be in one of three states: ready, waiting, and done. A node is ready if and only if | 
|  | * all of its dependencies have been signaled. A node is done if it has a value. It is waiting if | 
|  | * not all of its dependencies have been signaled. | 
|  | * | 
|  | * <p>A node must be in the work queue if and only if it is ready. It is an error for a node to be | 
|  | * in the work queue twice at the same time. | 
|  | * | 
|  | * <p>A node is considered in-flight if it has been created, and is not done yet. In case of an | 
|  | * interrupt, the work queue is discarded, and the in-flight set is used to remove partially | 
|  | * computed values. | 
|  | * | 
|  | * <p>Each evaluation of the graph takes place at a "version," which is currently given by a | 
|  | * non-negative {@code long}. The version can also be thought of as an "mtime." Each node in the | 
|  | * graph has a version, which is the last version at which its value changed. This version data is | 
|  | * used to avoid unnecessary re-evaluation of values. If a node is re-evaluated and found to have | 
|  | * the same data as before, its version (mtime) remains the same. If all of a node's children's have | 
|  | * the same version as before, its re-evaluation can be skipped. | 
|  | * | 
|  | * <p>Through its generic exception type parameter, this class supports a notion of an | 
|  | * "evaluation-wide" exception, to be implemented and detected by the classes that extend this one. | 
|  | * Those classes have the opportunity to check their conditions and throw instances of their | 
|  | * exception type in their {@link #bubbleErrorUpExceptionally} and {@link | 
|  | * #constructResultExceptionally} method implementations. | 
|  | */ | 
|  | public abstract class AbstractExceptionalParallelEvaluator<E extends Exception> | 
|  | extends AbstractParallelEvaluator { | 
|  | private static final Logger logger = | 
|  | Logger.getLogger(AbstractExceptionalParallelEvaluator.class.getName()); | 
|  |  | 
|  | AbstractExceptionalParallelEvaluator( | 
|  | ProcessableGraph graph, | 
|  | Version graphVersion, | 
|  | ImmutableMap<SkyFunctionName, ? extends SkyFunction> skyFunctions, | 
|  | final ExtendedEventHandler reporter, | 
|  | EmittedEventState emittedEventState, | 
|  | EventFilter storedEventFilter, | 
|  | ErrorInfoManager errorInfoManager, | 
|  | boolean keepGoing, | 
|  | DirtyTrackingProgressReceiver progressReceiver, | 
|  | GraphInconsistencyReceiver graphInconsistencyReceiver, | 
|  | Supplier<ExecutorService> executorService, | 
|  | CycleDetector cycleDetector, | 
|  | EvaluationVersionBehavior evaluationVersionBehavior) { | 
|  | super( | 
|  | graph, | 
|  | graphVersion, | 
|  | skyFunctions, | 
|  | reporter, | 
|  | emittedEventState, | 
|  | storedEventFilter, | 
|  | errorInfoManager, | 
|  | keepGoing, | 
|  | progressReceiver, | 
|  | graphInconsistencyReceiver, | 
|  | executorService, | 
|  | cycleDetector, | 
|  | evaluationVersionBehavior); | 
|  | } | 
|  |  | 
|  | private void informProgressReceiverThatValueIsDone(SkyKey key, NodeEntry entry) | 
|  | throws InterruptedException { | 
|  | if (evaluatorContext.getProgressReceiver() != null) { | 
|  | Preconditions.checkState(entry.isDone(), entry); | 
|  | SkyValue value = entry.getValue(); | 
|  | Version valueVersion = entry.getVersion(); | 
|  | Preconditions.checkState( | 
|  | valueVersion.atMost(evaluatorContext.getGraphVersion()), | 
|  | "%s should be at most %s in the version partial ordering", | 
|  | valueVersion, | 
|  | evaluatorContext.getGraphVersion()); | 
|  |  | 
|  | if (value != null) { | 
|  | ValueWithMetadata valueWithMetadata = | 
|  | ValueWithMetadata.wrapWithMetadata(entry.getValueMaybeWithMetadata()); | 
|  | replay(valueWithMetadata); | 
|  | } | 
|  |  | 
|  | // For most nodes we do not inform the progress receiver if they were already done when we | 
|  | // retrieve them, but top-level nodes are presumably of more interest. | 
|  | // If valueVersion is not equal to graphVersion, it must be less than it (by the | 
|  | // Preconditions check above), and so the node is clean. | 
|  | EvaluationState evaluationState = | 
|  | valueVersion.equals(evaluatorContext.getGraphVersion()) | 
|  | ? EvaluationState.BUILT | 
|  | : EvaluationState.CLEAN; | 
|  | evaluatorContext | 
|  | .getProgressReceiver() | 
|  | .evaluated( | 
|  | key, | 
|  | evaluationState == EvaluationState.BUILT ? value : null, | 
|  | value != null | 
|  | ? EvaluationSuccessState.SUCCESS.supplier() | 
|  | : EvaluationSuccessState.FAILURE.supplier(), | 
|  | evaluationState); | 
|  | } | 
|  | } | 
|  |  | 
|  | <T extends SkyValue> EvaluationResult<T> evalExceptionally(Iterable<? extends SkyKey> skyKeys) | 
|  | throws InterruptedException, E { | 
|  | ImmutableSet<SkyKey> skyKeySet = ImmutableSet.copyOf(skyKeys); | 
|  |  | 
|  | // Optimization: if all required node values are already present in the cache, return them | 
|  | // directly without launching the heavy machinery, spawning threads, etc. | 
|  | // Inform progressReceiver that these nodes are done to be consistent with the main code path. | 
|  | boolean allAreDone = true; | 
|  | Map<SkyKey, ? extends NodeEntry> batch = | 
|  | evaluatorContext.getBatchValues(null, Reason.PRE_OR_POST_EVALUATION, skyKeySet); | 
|  | for (SkyKey key : skyKeySet) { | 
|  | if (!isDoneForBuild(batch.get(key))) { | 
|  | allAreDone = false; | 
|  | break; | 
|  | } | 
|  | } | 
|  | if (allAreDone) { | 
|  | for (SkyKey skyKey : skyKeySet) { | 
|  | informProgressReceiverThatValueIsDone(skyKey, batch.get(skyKey)); | 
|  | } | 
|  | // Note that the 'catastrophe' parameter doesn't really matter here (it's only used for | 
|  | // sanity checking). | 
|  | return constructResultExceptionally(skyKeySet, null, /*catastrophe=*/ false); | 
|  | } | 
|  |  | 
|  | if (!evaluatorContext.keepGoing()) { | 
|  | Set<SkyKey> cachedErrorKeys = new HashSet<>(); | 
|  | for (SkyKey skyKey : skyKeySet) { | 
|  | NodeEntry entry = graph.get(null, Reason.PRE_OR_POST_EVALUATION, skyKey); | 
|  | if (entry == null) { | 
|  | continue; | 
|  | } | 
|  | if (entry.isDone() && entry.getErrorInfo() != null) { | 
|  | informProgressReceiverThatValueIsDone(skyKey, entry); | 
|  | cachedErrorKeys.add(skyKey); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Errors, even cached ones, should halt evaluations not in keepGoing mode. | 
|  | if (!cachedErrorKeys.isEmpty()) { | 
|  | // Note that the 'catastrophe' parameter doesn't really matter here (it's only used for | 
|  | // sanity checking). | 
|  | return constructResultExceptionally(cachedErrorKeys, null, /*catastrophe=*/ false); | 
|  | } | 
|  | } | 
|  |  | 
|  | try (SilentCloseable c = | 
|  | Profiler.instance().profile(ProfilerTask.SKYFRAME_EVAL, "Parallel Evaluator evaluation")) { | 
|  | return doMutatingEvaluation(skyKeySet); | 
|  | } | 
|  | } | 
|  |  | 
|  | @ThreadCompatible | 
|  | private <T extends SkyValue> EvaluationResult<T> doMutatingEvaluation( | 
|  | ImmutableSet<SkyKey> skyKeys) throws InterruptedException, E { | 
|  | // We unconditionally add the ErrorTransienceValue here, to ensure that it will be created, and | 
|  | // in the graph, by the time that it is needed. Creating it on demand in a parallel context sets | 
|  | // up a race condition, because there is no way to atomically create a node and set its value. | 
|  | NodeEntry errorTransienceEntry = | 
|  | Iterables.getOnlyElement( | 
|  | graph | 
|  | .createIfAbsentBatch( | 
|  | null, Reason.PRE_OR_POST_EVALUATION, ImmutableList.of(ErrorTransienceValue.KEY)) | 
|  | .values()); | 
|  | if (!errorTransienceEntry.isDone()) { | 
|  | injectValues( | 
|  | ImmutableMap.of(ErrorTransienceValue.KEY, ErrorTransienceValue.INSTANCE), | 
|  | evaluatorContext.getGraphVersion(), | 
|  | graph, | 
|  | evaluatorContext.getProgressReceiver()); | 
|  | } | 
|  | try { | 
|  | for (Map.Entry<SkyKey, ? extends NodeEntry> e : | 
|  | graph.createIfAbsentBatch(null, Reason.PRE_OR_POST_EVALUATION, skyKeys).entrySet()) { | 
|  | SkyKey skyKey = e.getKey(); | 
|  | NodeEntry entry = e.getValue(); | 
|  | // This must be equivalent to the code in enqueueChild above, in order to be thread-safe. | 
|  | switch (entry.addReverseDepAndCheckIfDone(null)) { | 
|  | case NEEDS_SCHEDULING: | 
|  | // Low priority because this node is not needed by any other currently evaluating node. | 
|  | // So keep it at the back of the queue as long as there's other useful work to be done. | 
|  | evaluatorContext.getVisitor().enqueueEvaluation(skyKey, Integer.MIN_VALUE); | 
|  | break; | 
|  | case DONE: | 
|  | informProgressReceiverThatValueIsDone(skyKey, entry); | 
|  | break; | 
|  | case ALREADY_EVALUATING: | 
|  | break; | 
|  | default: | 
|  | throw new IllegalStateException(entry + " for " + skyKey + " in unknown state"); | 
|  | } | 
|  | } | 
|  | } catch (InterruptedException ie) { | 
|  | // When multiple keys are being evaluated, it's possible that a key may get queued before | 
|  | // an InterruptedException is thrown from either #addReverseDepAndCheckIfDone or | 
|  | // #informProgressReceiverThatValueIsDone on a different key. Therefore we have to make sure | 
|  | // all evaluation threads are properly interrupted and shut down, if main thread (current | 
|  | // thread) is interrupted. | 
|  | Thread.currentThread().interrupt(); | 
|  | try { | 
|  | evaluatorContext.getVisitor().waitForCompletion(); | 
|  | } catch (SchedulerException se) { | 
|  | // A SchedulerException due to a SkyFunction observing the interrupt is completely expected. | 
|  | if (!(se.getCause() instanceof InterruptedException)) { | 
|  | throw se; | 
|  | } | 
|  | } | 
|  |  | 
|  | // Rethrow the InterruptedException to avoid proceeding to construct the result. | 
|  | throw ie; | 
|  | } | 
|  |  | 
|  | return waitForCompletionAndConstructResult(skyKeys); | 
|  | } | 
|  |  | 
|  | private <T extends SkyValue> EvaluationResult<T> waitForCompletionAndConstructResult( | 
|  | Iterable<SkyKey> skyKeys) throws InterruptedException, E { | 
|  | Map<SkyKey, ValueWithMetadata> bubbleErrorInfo = null; | 
|  | boolean catastrophe = false; | 
|  | try { | 
|  | evaluatorContext.getVisitor().waitForCompletion(); | 
|  | } catch (final SchedulerException e) { | 
|  | propagateEvaluatorContextCrashIfAny(); | 
|  | propagateInterruption(e); | 
|  | SkyKey errorKey = Preconditions.checkNotNull(e.getFailedValue(), e); | 
|  | // ErrorInfo could only be null if SchedulerException wrapped an InterruptedException, but | 
|  | // that should have been propagated. | 
|  | ErrorInfo errorInfo = Preconditions.checkNotNull(e.getErrorInfo(), errorKey); | 
|  | bubbleErrorInfo = | 
|  | bubbleErrorUpExceptionally(errorInfo, errorKey, skyKeys, e.getRdepsToBubbleUpTo()); | 
|  | if (evaluatorContext.keepGoing()) { | 
|  | Preconditions.checkState( | 
|  | errorInfo.isCatastrophic(), | 
|  | "Scheduler exception only thrown for catastrophe in keep_going evaluation: %s", | 
|  | e); | 
|  | catastrophe = true; | 
|  | } | 
|  | } | 
|  | Preconditions.checkState( | 
|  | evaluatorContext.getVisitor().getCrashes().isEmpty(), | 
|  | evaluatorContext.getVisitor().getCrashes()); | 
|  |  | 
|  | // Successful evaluation, barring evaluation-wide exceptions, either because keepGoing or | 
|  | // because we actually did succeed. | 
|  | // TODO(bazel-team): Maybe report root causes during the build for lower latency. | 
|  | return constructResultExceptionally(skyKeys, bubbleErrorInfo, catastrophe); | 
|  | } | 
|  |  | 
|  | abstract Map<SkyKey, ValueWithMetadata> bubbleErrorUpExceptionally( | 
|  | final ErrorInfo leafFailure, | 
|  | SkyKey errorKey, | 
|  | Iterable<SkyKey> roots, | 
|  | Set<SkyKey> rdepsToBubbleUpTo) | 
|  | throws InterruptedException, E; | 
|  |  | 
|  | /** | 
|  | * Walk up graph to find a top-level node (without parents) that wanted this failure. Store the | 
|  | * failed nodes along the way in a map, with ErrorInfos that are appropriate for that layer. | 
|  | * Example: | 
|  | * | 
|  | * <pre> | 
|  | *                      foo   bar | 
|  | *                        \   / | 
|  | *           unrequested   baz | 
|  | *                     \    | | 
|  | *                      failed-node | 
|  | * </pre> | 
|  | * | 
|  | * User requests foo, bar. When failed-node fails, we look at its parents. unrequested is not | 
|  | * in-flight, so we replace failed-node by baz and repeat. We look at baz's parents. foo is | 
|  | * in-flight, so we replace baz by foo. Since foo is a top-level node and doesn't have parents, we | 
|  | * then break, since we know a top-level node, foo, that depended on the failed node. | 
|  | * | 
|  | * <p>There's the potential for a weird "track jump" here in the case: | 
|  | * | 
|  | * <pre> | 
|  | *                        foo | 
|  | *                       / \ | 
|  | *                   fail1 fail2 | 
|  | * </pre> | 
|  | * | 
|  | * If fail1 and fail2 fail simultaneously, fail2 may start propagating up in the loop below. | 
|  | * However, foo requests fail1 first, and then throws an exception based on that. This is not | 
|  | * incorrect, but may be unexpected. | 
|  | * | 
|  | * <p>Returns a map of errors that have been constructed during the bubbling up, so that the | 
|  | * appropriate error can be returned to the caller, even though that error was not written to the | 
|  | * graph. If a cycle is detected during the bubbling, this method aborts and returns null so that | 
|  | * the normal cycle detection can handle the cycle. | 
|  | * | 
|  | * <p>Note that we are not propagating error to the first top-level node but to the highest one, | 
|  | * because during this process we can add useful information about error from other nodes. | 
|  | * | 
|  | * <p>Every node on this walk but the leaf node is not done, by the following argument: the leaf | 
|  | * node is done, but the parents of it that we consider are in {@code rdepsToBubbleUpTo}. Each | 
|  | * parent is either (1) a parent that requested the leaf node and found it to be in error, meaning | 
|  | * it is not done, or (2) a parent that had registered a dependency on this leaf node before it | 
|  | * finished building. In the second case, that parent would not have been enqueued, since we | 
|  | * failed fast and prevented all new evaluations. Thus, we will only visit unfinished parents of | 
|  | * the leaf node. For the inductive argument, the only parents we consider are those that were | 
|  | * registered during this build (via {@link NodeEntry#getInProgressReverseDeps}. Since we don't | 
|  | * allow a node to build with unfinished deps, those parents cannot have built. | 
|  | */ | 
|  | Map<SkyKey, ValueWithMetadata> bubbleErrorUp( | 
|  | final ErrorInfo leafFailure, | 
|  | SkyKey errorKey, | 
|  | Iterable<SkyKey> roots, | 
|  | Set<SkyKey> rdepsToBubbleUpTo) | 
|  | throws InterruptedException { | 
|  | Set<SkyKey> rootValues = ImmutableSet.copyOf(roots); | 
|  | ErrorInfo error = leafFailure; | 
|  | Map<SkyKey, ValueWithMetadata> bubbleErrorInfo = new HashMap<>(); | 
|  | boolean externalInterrupt = false; | 
|  | boolean firstIteration = true; | 
|  | while (true) { | 
|  | NodeEntry errorEntry = | 
|  | Preconditions.checkNotNull(graph.get(null, Reason.ERROR_BUBBLING, errorKey), errorKey); | 
|  | Iterable<SkyKey> reverseDeps; | 
|  | if (errorEntry.isDone()) { | 
|  | Preconditions.checkState( | 
|  | firstIteration, | 
|  | "Non-leaf done node reached: %s %s %s %s %s", | 
|  | errorKey, | 
|  | leafFailure, | 
|  | roots, | 
|  | rdepsToBubbleUpTo, | 
|  | bubbleErrorInfo); | 
|  | reverseDeps = rdepsToBubbleUpTo; | 
|  | } else { | 
|  | Preconditions.checkState( | 
|  | !firstIteration, | 
|  | "undone first iteration: %s %s %s %s %s %s", | 
|  | errorKey, | 
|  | errorEntry, | 
|  | leafFailure, | 
|  | roots, | 
|  | rdepsToBubbleUpTo, | 
|  | bubbleErrorInfo); | 
|  | reverseDeps = errorEntry.getInProgressReverseDeps(); | 
|  | } | 
|  | firstIteration = false; | 
|  | // We should break from loop only when node doesn't have any parents. | 
|  | if (Iterables.isEmpty(reverseDeps)) { | 
|  | Preconditions.checkState( | 
|  | rootValues.contains(errorKey), | 
|  | "Current key %s has to be a top-level key: %s", | 
|  | errorKey, | 
|  | rootValues); | 
|  | break; | 
|  | } | 
|  | SkyKey parent = Preconditions.checkNotNull(Iterables.getFirst(reverseDeps, null)); | 
|  | if (bubbleErrorInfo.containsKey(parent)) { | 
|  | logger.info( | 
|  | "Bubbled into a cycle. Don't try to bubble anything up. Cycle detection will kick in. " | 
|  | + parent | 
|  | + ": " | 
|  | + errorEntry | 
|  | + ", " | 
|  | + bubbleErrorInfo | 
|  | + ", " | 
|  | + leafFailure | 
|  | + ", " | 
|  | + roots | 
|  | + ", " | 
|  | + rdepsToBubbleUpTo); | 
|  | return null; | 
|  | } | 
|  | NodeEntry parentEntry = | 
|  | Preconditions.checkNotNull( | 
|  | graph.get(errorKey, Reason.ERROR_BUBBLING, parent), | 
|  | "parent %s of %s not in graph", | 
|  | parent, | 
|  | errorKey); | 
|  | Preconditions.checkState( | 
|  | !parentEntry.isDone(), | 
|  | "We cannot bubble into a done node entry: a done node cannot depend on a not-done node," | 
|  | + " and the first errorParent was not done: %s %s %s %s %s %s %s %s", | 
|  | errorKey, | 
|  | errorEntry, | 
|  | parent, | 
|  | parentEntry, | 
|  | leafFailure, | 
|  | roots, | 
|  | rdepsToBubbleUpTo, | 
|  | bubbleErrorInfo); | 
|  | Preconditions.checkState( | 
|  | evaluatorContext.getProgressReceiver().isInflight(parent), | 
|  | "In-progress reverse deps can only include in-flight nodes: " + "%s %s %s %s %s %s", | 
|  | errorKey, | 
|  | errorEntry, | 
|  | parent, | 
|  | parentEntry, | 
|  | leafFailure, | 
|  | roots, | 
|  | rdepsToBubbleUpTo, | 
|  | bubbleErrorInfo); | 
|  | Preconditions.checkState( | 
|  | parentEntry.getTemporaryDirectDeps().expensiveContains(errorKey), | 
|  | "In-progress reverse deps can only include nodes that have declared a dep: " | 
|  | + "%s %s %s %s %s %s", | 
|  | errorKey, | 
|  | errorEntry, | 
|  | parent, | 
|  | parentEntry, | 
|  | leafFailure, | 
|  | roots, | 
|  | rdepsToBubbleUpTo, | 
|  | bubbleErrorInfo); | 
|  | Preconditions.checkNotNull(parentEntry, "%s %s", errorKey, parent); | 
|  | SkyFunction factory = evaluatorContext.getSkyFunctions().get(parent.functionName()); | 
|  | if (parentEntry.isDirty()) { | 
|  | switch (parentEntry.getDirtyState()) { | 
|  | case CHECK_DEPENDENCIES: | 
|  | // If this value's child was bubbled up to, it did not signal this value, and so we must | 
|  | // manually make it ready to build. | 
|  | parentEntry.signalDep(evaluatorContext.getGraphVersion(), errorKey); | 
|  | // Fall through to NEEDS_REBUILDING, since state is now NEEDS_REBUILDING. | 
|  | case NEEDS_REBUILDING: | 
|  | maybeMarkRebuilding(parentEntry); | 
|  | break; | 
|  | case NEEDS_FORCED_REBUILDING: | 
|  | parentEntry.forceRebuild(); | 
|  | break; | 
|  | case REBUILDING: | 
|  | case FORCED_REBUILDING: | 
|  | break; | 
|  | default: | 
|  | throw new AssertionError(parent + " not in valid dirty state: " + parentEntry); | 
|  | } | 
|  | } | 
|  | errorKey = parent; | 
|  | SkyFunctionEnvironment env = | 
|  | new SkyFunctionEnvironment( | 
|  | parent, | 
|  | parentEntry.getTemporaryDirectDeps(), | 
|  | bubbleErrorInfo, | 
|  | ImmutableSet.<SkyKey>of(), | 
|  | evaluatorContext); | 
|  | externalInterrupt = externalInterrupt || Thread.currentThread().isInterrupted(); | 
|  | try { | 
|  | // This build is only to check if the parent node can give us a better error. We don't | 
|  | // care about a return value. | 
|  | factory.compute(parent, env); | 
|  | } catch (InterruptedException interruptedException) { | 
|  | // Do nothing. | 
|  | // This throw happens if the builder requested the failed node, and then checked the | 
|  | // interrupted state later -- getValueOrThrow sets the interrupted bit after the failed | 
|  | // value is requested, to prevent the builder from doing too much work. | 
|  | } catch (SkyFunctionException builderException) { | 
|  | // Clear interrupted status. We're not listening to interrupts here. | 
|  | Thread.interrupted(); | 
|  | ReifiedSkyFunctionException reifiedBuilderException = | 
|  | new ReifiedSkyFunctionException(builderException, parent); | 
|  | if (reifiedBuilderException.getRootCauseSkyKey().equals(parent)) { | 
|  | error = | 
|  | ErrorInfo.fromException(reifiedBuilderException, /*isTransitivelyTransient=*/ false); | 
|  | Pair<NestedSet<TaggedEvents>, NestedSet<Postable>> eventsAndPostables = | 
|  | env.buildAndReportEventsAndPostables(parentEntry, /*expectDoneDeps=*/ false); | 
|  | bubbleErrorInfo.put( | 
|  | errorKey, | 
|  | ValueWithMetadata.error( | 
|  | ErrorInfo.fromChildErrors(errorKey, ImmutableSet.of(error)), | 
|  | eventsAndPostables.first, | 
|  | eventsAndPostables.second)); | 
|  | continue; | 
|  | } | 
|  | } finally { | 
|  | // Clear interrupted status. We're not listening to interrupts here. | 
|  | Thread.interrupted(); | 
|  | } | 
|  | // Builder didn't throw an exception, so just propagate this one up. | 
|  | Pair<NestedSet<TaggedEvents>, NestedSet<Postable>> eventsAndPostables = | 
|  | env.buildAndReportEventsAndPostables(parentEntry, /*expectDoneDeps=*/ false); | 
|  | bubbleErrorInfo.put( | 
|  | errorKey, | 
|  | ValueWithMetadata.error( | 
|  | ErrorInfo.fromChildErrors(errorKey, ImmutableSet.of(error)), | 
|  | eventsAndPostables.first, | 
|  | eventsAndPostables.second)); | 
|  | } | 
|  |  | 
|  | // Reset the interrupt bit if there was an interrupt from outside this evaluator interrupt. | 
|  | // Note that there are internal interrupts set in the node builder environment if an error | 
|  | // bubbling node calls getValueOrThrow() on a node in error. | 
|  | if (externalInterrupt) { | 
|  | Thread.currentThread().interrupt(); | 
|  | } | 
|  | return bubbleErrorInfo; | 
|  | } | 
|  |  | 
|  | private void replay(ValueWithMetadata valueWithMetadata) { | 
|  | // Replaying actions is done on a small number of nodes, but potentially over a large dependency | 
|  | // graph. Under those conditions, using the regular NestedSet flattening with .toList() is more | 
|  | // efficient than using NestedSetVisitor's custom traversal logic. | 
|  | evaluatorContext | 
|  | .getReplayingNestedSetPostableVisitor() | 
|  | .visit(valueWithMetadata.getTransitivePostables().toList()); | 
|  | evaluatorContext | 
|  | .getReplayingNestedSetEventVisitor() | 
|  | .visit(valueWithMetadata.getTransitiveEvents().toList()); | 
|  | } | 
|  |  | 
|  | abstract <T extends SkyValue> EvaluationResult<T> constructResultExceptionally( | 
|  | Iterable<SkyKey> skyKeys, | 
|  | @Nullable Map<SkyKey, ValueWithMetadata> bubbleErrorInfo, | 
|  | boolean catastrophe) | 
|  | throws InterruptedException, E; | 
|  |  | 
|  | /** | 
|  | * Constructs an {@link EvaluationResult} from the {@link #graph}. Looks for cycles if there are | 
|  | * unfinished nodes but no error was already found through bubbling up (as indicated by {@code | 
|  | * bubbleErrorInfo} being null). | 
|  | * | 
|  | * <p>{@code visitor} may be null, but only in the case where all graph entries corresponding to | 
|  | * {@code skyKeys} are known to be in the DONE state ({@code entry.isDone()} returns true). | 
|  | */ | 
|  | <T extends SkyValue> EvaluationResult<T> constructResult( | 
|  | Iterable<SkyKey> skyKeys, | 
|  | @Nullable Map<SkyKey, ValueWithMetadata> bubbleErrorInfo, | 
|  | boolean catastrophe) | 
|  | throws InterruptedException { | 
|  | Preconditions.checkState( | 
|  | !catastrophe || evaluatorContext.keepGoing(), | 
|  | "Catastrophe not consistent with keepGoing mode: %s %s %s", | 
|  | skyKeys, | 
|  | catastrophe, | 
|  | bubbleErrorInfo); | 
|  | EvaluationResult.Builder<T> result = EvaluationResult.builder(); | 
|  | List<SkyKey> cycleRoots = new ArrayList<>(); | 
|  | for (SkyKey skyKey : skyKeys) { | 
|  | SkyValue unwrappedValue = | 
|  | maybeGetValueFromError( | 
|  | skyKey, graph.get(null, Reason.PRE_OR_POST_EVALUATION, skyKey), bubbleErrorInfo); | 
|  | ValueWithMetadata valueWithMetadata = | 
|  | unwrappedValue == null ? null : ValueWithMetadata.wrapWithMetadata(unwrappedValue); | 
|  | // Cycle checking: if there is a cycle, evaluation cannot progress, therefore, | 
|  | // the final values will not be in DONE state when the work runs out. | 
|  | if (valueWithMetadata == null) { | 
|  | // Don't look for cycles if the build failed for a known reason. | 
|  | if (bubbleErrorInfo == null) { | 
|  | cycleRoots.add(skyKey); | 
|  | } | 
|  | continue; | 
|  | } | 
|  | // Replaying here is necessary for error bubbling and other cases. | 
|  | replay(valueWithMetadata); | 
|  | SkyValue value = valueWithMetadata.getValue(); | 
|  | ErrorInfo errorInfo = valueWithMetadata.getErrorInfo(); | 
|  | Preconditions.checkState(value != null || errorInfo != null, skyKey); | 
|  | if (!evaluatorContext.keepGoing() && errorInfo != null) { | 
|  | // value will be null here unless the value was already built on a prior keepGoing build. | 
|  | result.addError(skyKey, errorInfo); | 
|  | continue; | 
|  | } | 
|  | if (value == null) { | 
|  | // Note that we must be in the keepGoing case. Only make this value an error if it doesn't | 
|  | // have a value. The error shouldn't matter to the caller since the value succeeded after a | 
|  | // fashion. | 
|  | result.addError(skyKey, errorInfo); | 
|  | } else { | 
|  | result.addResult(skyKey, value); | 
|  | } | 
|  | } | 
|  | if (!cycleRoots.isEmpty()) { | 
|  | cycleDetector.checkForCycles(cycleRoots, result, evaluatorContext); | 
|  | } | 
|  | if (catastrophe && bubbleErrorInfo != null) { | 
|  | // We may not have a top-level node completed. Inform the caller of at least one catastrophic | 
|  | // exception that shut down the evaluation so that it has some context. | 
|  | for (ValueWithMetadata valueWithMetadata : bubbleErrorInfo.values()) { | 
|  | ErrorInfo errorInfo = | 
|  | Preconditions.checkNotNull( | 
|  | valueWithMetadata.getErrorInfo(), | 
|  | "bubbleErrorInfo should have contained element with errorInfo: %s", | 
|  | bubbleErrorInfo); | 
|  | Preconditions.checkState( | 
|  | errorInfo.isCatastrophic(), | 
|  | "bubbleErrorInfo should have contained element with catastrophe: %s", | 
|  | bubbleErrorInfo); | 
|  | result.setCatastrophe(errorInfo.getException()); | 
|  | } | 
|  | } | 
|  | EvaluationResult<T> builtResult = result.build(); | 
|  | Preconditions.checkState( | 
|  | bubbleErrorInfo == null || builtResult.hasError(), | 
|  | "If an error bubbled up, some top-level node must be in error: %s %s %s", | 
|  | bubbleErrorInfo, | 
|  | skyKeys, | 
|  | builtResult); | 
|  | return builtResult; | 
|  | } | 
|  |  | 
|  | @Nullable | 
|  | static SkyValue maybeGetValueFromError( | 
|  | SkyKey key, | 
|  | @Nullable NodeEntry entry, | 
|  | @Nullable Map<SkyKey, ValueWithMetadata> bubbleErrorInfo) | 
|  | throws InterruptedException { | 
|  | SkyValue value = bubbleErrorInfo == null ? null : bubbleErrorInfo.get(key); | 
|  | if (value != null) { | 
|  | return value; | 
|  | } | 
|  | return isDoneForBuild(entry) ? entry.getValueMaybeWithMetadata() : null; | 
|  | } | 
|  |  | 
|  | static void injectValues( | 
|  | Map<SkyKey, SkyValue> injectionMap, | 
|  | Version version, | 
|  | EvaluableGraph graph, | 
|  | DirtyTrackingProgressReceiver progressReceiver) | 
|  | throws InterruptedException { | 
|  | Map<SkyKey, ? extends NodeEntry> prevNodeEntries = | 
|  | graph.createIfAbsentBatch(null, Reason.OTHER, injectionMap.keySet()); | 
|  | for (Map.Entry<SkyKey, SkyValue> injectionEntry : injectionMap.entrySet()) { | 
|  | SkyKey key = injectionEntry.getKey(); | 
|  | SkyValue value = injectionEntry.getValue(); | 
|  | NodeEntry prevEntry = prevNodeEntries.get(key); | 
|  | DependencyState newState = prevEntry.addReverseDepAndCheckIfDone(null); | 
|  | Preconditions.checkState( | 
|  | newState != DependencyState.ALREADY_EVALUATING, "%s %s", key, prevEntry); | 
|  | if (prevEntry.isDirty()) { | 
|  | // Get the node in the state where it is able to accept a value. | 
|  | Preconditions.checkState( | 
|  | newState == DependencyState.NEEDS_SCHEDULING, "%s %s", key, prevEntry); | 
|  | // If there was a node in the graph before, check that the previous node has no | 
|  | // dependencies. Overwriting a value with deps with an injected value (which is by | 
|  | // definition deps-free) needs a little additional bookkeeping (removing reverse deps from | 
|  | // the dependencies), but more importantly it's something that we want to avoid, because it | 
|  | // indicates confusion of input values and derived values. | 
|  | Preconditions.checkState( | 
|  | prevEntry.noDepsLastBuild(), "existing entry for %s has deps: %s", key, prevEntry); | 
|  | } | 
|  | prevEntry.markRebuilding(); | 
|  | prevEntry.setValue( | 
|  | value, | 
|  | version, | 
|  | prevEntry.canPruneDepsByFingerprint() ? new DepFingerprintList.Builder(0).build() : null); | 
|  | // Now that this key's injected value is set, it is no longer dirty. | 
|  | progressReceiver.injected(key); | 
|  | } | 
|  | } | 
|  | } |