blob: cffbc22895aabf9451907eeb4328fb4ee812338e [file] [log] [blame]
// 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;
}
}
}