| // 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.skyframe; | 
 |  | 
 | import com.google.common.base.Preconditions; | 
 | import com.google.devtools.build.lib.concurrent.ThreadSafety.ThreadSafe; | 
 | import com.google.devtools.build.skyframe.NodeEntry.DirtyState; | 
 | import javax.annotation.Nullable; | 
 |  | 
 | /** | 
 |  * A node in the graph without the means to access its value. All operations on this class are | 
 |  * thread-safe (note, however, the warning on the return value of {@link #markDirty}). | 
 |  * | 
 |  * <p>This interface is public only for the benefit of alternative graph implementations outside of | 
 |  * the package. | 
 |  */ | 
 | public interface ThinNodeEntry { | 
 |  | 
 |   /** Returns whether the entry has been built and is finished evaluating. */ | 
 |   @ThreadSafe | 
 |   boolean isDone(); | 
 |  | 
 |   /** | 
 |    * Returns true if the entry is new or marked as dirty. This includes the case where its deps are | 
 |    * still being checked for up-to-dateness. | 
 |    */ | 
 |   @ThreadSafe | 
 |   boolean isDirty(); | 
 |  | 
 |   /** | 
 |    * Returns true if the entry is marked changed, meaning that it must be re-evaluated even if its | 
 |    * dependencies' values have not changed. | 
 |    */ | 
 |   @ThreadSafe | 
 |   boolean isChanged(); | 
 |  | 
 |   /** Ways that a node may be dirtied. */ | 
 |   enum DirtyType { | 
 |     /** | 
 |      * A node P dirtied with DIRTY is re-evaluated during the evaluation phase if it's requested and | 
 |      * directly depends on some node C whose value changed since the last evaluation of P. If it's | 
 |      * requested and there is no such node C, P is marked clean. | 
 |      */ | 
 |     DIRTY(DirtyState.CHECK_DEPENDENCIES), | 
 |  | 
 |     /** | 
 |      * A node dirtied with CHANGE is re-evaluated during the evaluation phase if it's requested | 
 |      * (regardless of the state of its dependencies). Such a node is expected to evaluate to the | 
 |      * same value if evaluated at the same graph version. | 
 |      */ | 
 |     CHANGE(DirtyState.NEEDS_REBUILDING), | 
 |  | 
 |     /** | 
 |      * A node dirtied with FORCE_REBUILD behaves like a {@link #CHANGE}d node, except that it may | 
 |      * evaluate to a different value even if evaluated at the same graph version. | 
 |      */ | 
 |     FORCE_REBUILD(DirtyState.NEEDS_FORCED_REBUILDING); | 
 |  | 
 |     private final DirtyState initialDirtyState; | 
 |  | 
 |     DirtyType(DirtyState initialDirtyState) { | 
 |       this.initialDirtyState = initialDirtyState; | 
 |     } | 
 |  | 
 |     DirtyState getInitialDirtyState() { | 
 |       return initialDirtyState; | 
 |     } | 
 |   } | 
 |  | 
 |   /** | 
 |    * Marks this node dirty as specified by the provided {@link DirtyType}. | 
 |    * | 
 |    * <p>{@code markDirty(DirtyType.DIRTY)} may only be called on a node P for which {@code | 
 |    * P.isDone() || P.isChanged()} (the latter is permitted but has no effect). Similarly, {@code | 
 |    * markDirty(DirtyType.CHANGE)} may only be called on a node P for which {@code P.isDone() || | 
 |    * !P.isChanged()}. Otherwise, this will throw {@link IllegalStateException}. | 
 |    * | 
 |    * <p>{@code markDirty(DirtyType.FORCE_REBUILD)} may be called multiple times; only the first has | 
 |    * any effect. | 
 |    * | 
 |    * @return if the node was done, a {@link MarkedDirtyResult} which may include the node's reverse | 
 |    *     deps; otherwise {@code null} | 
 |    */ | 
 |   @Nullable | 
 |   @ThreadSafe | 
 |   MarkedDirtyResult markDirty(DirtyType dirtyType) throws InterruptedException; | 
 |  | 
 |   /** | 
 |    * Returned by {@link #markDirty} if that call changed the node from done to dirty. Contains an | 
 |    * iterable of the node's reverse deps for efficiency, because an important use case for {@link | 
 |    * #markDirty} is during invalidation, and the invalidator must immediately afterwards schedule | 
 |    * the invalidation of a node's reverse deps if the invalidator successfully dirties that node. | 
 |    * | 
 |    * <p>Warning: {@link #getReverseDepsUnsafe()} may return a live view of the reverse deps | 
 |    * collection of the marked-dirty node. The consumer of this data must be careful only to iterate | 
 |    * over and consume its values while that collection is guaranteed not to change. This is true | 
 |    * during invalidation, because reverse deps don't change during invalidation. | 
 |    */ | 
 |   class MarkedDirtyResult { | 
 |     private final Iterable<SkyKey> reverseDepsUnsafe; | 
 |  | 
 |     public MarkedDirtyResult(Iterable<SkyKey> reverseDepsUnsafe) { | 
 |       this.reverseDepsUnsafe = Preconditions.checkNotNull(reverseDepsUnsafe); | 
 |     } | 
 |  | 
 |     public Iterable<SkyKey> getReverseDepsUnsafe() { | 
 |       return reverseDepsUnsafe; | 
 |     } | 
 |   } | 
 | } |