// 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.annotations.VisibleForTesting; import com.google.common.base.MoreObjects; import com.google.common.base.Preconditions; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableSet; import com.google.common.collect.Iterables; import com.google.devtools.build.lib.util.GroupedList; import com.google.devtools.build.lib.util.GroupedList.GroupedListHelper; import java.util.Collection; import java.util.List; import java.util.Set; import javax.annotation.Nullable; /** * In-memory implementation of {@link NodeEntry}. All operations on this class are thread-safe. * *

Care was taken to provide certain compound operations to avoid certain check-then-act races. * That means this class is somewhat closely tied to the exact Evaluator implementation. * *

Consider the example with two threads working on two nodes, where one depends on the other, * say b depends on a. If a completes first, it's done. If it completes second, it needs to signal * b, and potentially re-schedule it. If b completes first, it must exit, because it will be * signaled (and re-scheduled) by a. If it completes second, it must signal (and re-schedule) * itself. However, if the Evaluator supported re-entrancy for a node, then this wouldn't have to * be so strict, because duplicate scheduling would be less problematic. * *

The transient state of an {@code InMemoryNodeEntry} is kept in a {@link BuildingState} object. * Many of the methods of {@code InMemoryNodeEntry} are just wrappers around the corresponding * {@link BuildingState} methods. * *

This class is public only for the benefit of alternative graph implementations outside of the * package. */ public class InMemoryNodeEntry implements NodeEntry { /** Actual data stored in this entry when it is done. */ private SkyValue value = null; /** * The last version of the graph at which this node entry was changed. In {@link #setValue} it * may be determined that the data being written to the graph at a given version is the same as * the already-stored data. In that case, the version will remain the same. The version can be * thought of as the latest timestamp at which this entry was changed. */ protected Version version = MinimalVersion.INSTANCE; /** * This object represents a {@link GroupedList} in a memory-efficient way. It stores the * direct dependencies of this node, in groups if the {@code SkyFunction} requested them that way. */ private Object directDeps = null; /** * This list stores the reverse dependencies of this node that have been declared so far. * *

In case of a single object we store the object unwrapped, without the list, for * memory-efficiency. */ @VisibleForTesting protected Object reverseDeps = ImmutableList.of(); /** * We take advantage of memory alignment to avoid doing a nasty {@code instanceof} for knowing * if {@code reverseDeps} is a single object or a list. */ protected boolean reverseDepIsSingleObject = false; /** * When reverse deps are removed, checked for presence, or possibly added, we store them in this * object instead of directly doing the operation. That is because removals/checks in reverseDeps * are O(N). Originally reverseDeps was a HashSet, but because of memory consumption we switched * to a list. * *

Internally, ReverseDepsUtil consolidates this data periodically, and when the set of reverse * deps is requested. While this operation is not free, it can be done more effectively than * trying to remove/check each dirty reverse dependency individually (O(N) each time). */ private List reverseDepsDataToConsolidate = null; protected static final ReverseDepsUtil REVERSE_DEPS_UTIL = new ReverseDepsUtil() { @Override void setReverseDepsObject(InMemoryNodeEntry container, Object object) { container.reverseDeps = object; } @Override void setSingleReverseDep(InMemoryNodeEntry container, boolean singleObject) { container.reverseDepIsSingleObject = singleObject; } @Override void setDataToConsolidate(InMemoryNodeEntry container, List dataToConsolidate) { container.reverseDepsDataToConsolidate = dataToConsolidate; } @Override Object getReverseDepsObject(InMemoryNodeEntry container) { return container.reverseDeps; } @Override boolean isSingleReverseDep(InMemoryNodeEntry container) { return container.reverseDepIsSingleObject; } @Override List getDataToConsolidate(InMemoryNodeEntry container) { return container.reverseDepsDataToConsolidate; } }; /** * The transient state of this entry, after it has been created but before it is done. It allows * us to keep the current state of the entry across invalidation and successive evaluations. */ @VisibleForTesting @Nullable protected BuildingState buildingState = new BuildingState(); /** * Construct a InMemoryNodeEntry. Use ONLY in Skyframe evaluation and graph implementations. */ public InMemoryNodeEntry() { } @Override public boolean keepEdges() { return true; } @Override public synchronized boolean isDone() { return buildingState == null; } @Override public synchronized SkyValue getValue() { Preconditions.checkState(isDone(), "no value until done. ValueEntry: %s", this); return ValueWithMetadata.justValue(value); } @Override public synchronized SkyValue getValueMaybeWithMetadata() { Preconditions.checkState(isDone(), "no value until done: %s", this); return value; } @Override public synchronized SkyValue toValue() { if (isDone()) { return getErrorInfo() == null ? getValue() : null; } else if (isChanged() || isDirty()) { return (buildingState.getLastBuildValue() == null) ? null : ValueWithMetadata.justValue(buildingState.getLastBuildValue()); } else { // Value has not finished evaluating. It's probably about to be cleaned from the graph. return null; } } @Override public synchronized Iterable getDirectDeps() { assertKeepEdges(); Preconditions.checkState(isDone(), "no deps until done. ValueEntry: %s", this); return GroupedList.create(directDeps).toSet(); } /** * If {@code isDone()}, returns the ordered list of sets of grouped direct dependencies that were * added in {@link #addTemporaryDirectDeps}. */ public synchronized GroupedList getGroupedDirectDeps() { assertKeepEdges(); Preconditions.checkState(isDone(), "no deps until done. ValueEntry: %s", this); return GroupedList.create(directDeps); } @Override @Nullable public synchronized ErrorInfo getErrorInfo() { Preconditions.checkState(isDone(), "no errors until done. ValueEntry: %s", this); return ValueWithMetadata.getMaybeErrorInfo(value); } protected synchronized Set setStateFinishedAndReturnReverseDeps() { // Get reverse deps that need to be signaled. ImmutableSet reverseDepsToSignal = buildingState.getReverseDepsToSignal(); REVERSE_DEPS_UTIL.addReverseDeps(this, reverseDepsToSignal); // Force consistency check. REVERSE_DEPS_UTIL.getReverseDeps(this); this.directDeps = buildingState.getFinishedDirectDeps().compress(); // Set state of entry to done. buildingState = null; if (!keepEdges()) { this.directDeps = null; this.reverseDeps = null; } return reverseDepsToSignal; } @Override public synchronized Set getInProgressReverseDeps() { Preconditions.checkState(!isDone(), this); return buildingState.getReverseDepsToSignal(); } @Override public synchronized Set setValue(SkyValue value, Version version) { Preconditions.checkState(isReady(), "%s %s", this, value); // This check may need to be removed when we move to a non-linear versioning sequence. Preconditions.checkState(this.version.atMost(version), "%s %s %s", this, version, value); if (isDirty() && buildingState.unchangedFromLastBuild(value)) { // If the value is the same as before, just use the old value. Note that we don't use the new // value, because preserving == equality is even better than .equals() equality. this.value = buildingState.getLastBuildValue(); } else { // If this is a new value, or it has changed since the last build, set the version to the // current graph version. this.version = version; this.value = value; } return setStateFinishedAndReturnReverseDeps(); } @Override public synchronized DependencyState addReverseDepAndCheckIfDone(SkyKey reverseDep) { if (reverseDep != null) { if (keepEdges()) { REVERSE_DEPS_UTIL.maybeCheckReverseDepNotPresent(this, reverseDep); } if (isDone()) { if (keepEdges()) { REVERSE_DEPS_UTIL.addReverseDeps(this, ImmutableList.of(reverseDep)); } } else { // Parent should never register itself twice in the same build. buildingState.addReverseDepToSignal(reverseDep); } } if (isDone()) { return DependencyState.DONE; } return buildingState.startEvaluating() ? DependencyState.NEEDS_SCHEDULING : DependencyState.ALREADY_EVALUATING; } @Override public synchronized DependencyState checkIfDoneForDirtyReverseDep(SkyKey reverseDep) { Preconditions.checkNotNull(reverseDep, this); Preconditions.checkState(keepEdges(), "%s %s", reverseDep, this); if (!isDone()) { REVERSE_DEPS_UTIL.removeReverseDep(this, reverseDep); buildingState.addReverseDepToSignal(reverseDep); } else { REVERSE_DEPS_UTIL.checkReverseDep(this, reverseDep); } return addReverseDepAndCheckIfDone(null); } @Override public synchronized void removeReverseDep(SkyKey reverseDep) { if (!keepEdges()) { return; } REVERSE_DEPS_UTIL.removeReverseDep(this, reverseDep); } @Override public synchronized void removeInProgressReverseDep(SkyKey reverseDep) { buildingState.removeReverseDepToSignal(reverseDep); } @Override public synchronized Iterable getReverseDeps() { assertKeepEdges(); Iterable reverseDeps = REVERSE_DEPS_UTIL.getReverseDeps(this); if (isDone()) { return reverseDeps; } else { return Iterables.concat(reverseDeps, buildingState.getReverseDepsToSignal()); } } @Override public synchronized boolean signalDep() { return signalDep(/*childVersion=*/new IntVersion(Long.MAX_VALUE)); } @Override public synchronized boolean signalDep(Version childVersion) { Preconditions.checkState(!isDone(), "Value must not be done in signalDep %s", this); return buildingState.signalDep(/*childChanged=*/!childVersion.atMost(getVersion())); } @Override public synchronized boolean isDirty() { return !isDone() && buildingState.isDirty(); } @Override public synchronized boolean isChanged() { return !isDone() && buildingState.isChanged(); } /** Checks that a caller is not trying to access not-stored graph edges. */ private void assertKeepEdges() { Preconditions.checkState(keepEdges(), "Graph edges not stored. %s", this); } @Override public synchronized boolean markDirty(boolean isChanged) { assertKeepEdges(); if (isDone()) { buildingState = BuildingState.newDirtyState(isChanged, GroupedList.create(directDeps), value); value = null; return true; } // The caller may be simultaneously trying to mark this node dirty and changed, and the dirty // thread may have lost the race, but it is the caller's responsibility not to try to mark // this node changed twice. The end result of racing markers must be a changed node, since one // of the markers is trying to mark the node changed. Preconditions.checkState(isChanged != isChanged(), "Cannot mark node dirty twice or changed twice: %s", this); Preconditions.checkState(value == null, "Value should have been reset already %s", this); if (isChanged) { // If the changed marker lost the race, we just need to mark changed in this method -- all // other work was done by the dirty marker. buildingState.markChanged(); } return false; } @Override public synchronized Set markClean() { this.value = buildingState.getLastBuildValue(); Preconditions.checkState(buildingState.depsUnchangedFromLastBuild(), "Direct deps must be the same as those found last build for node to be marked clean: %s", this); Preconditions.checkState(isDirty(), this); Preconditions.checkState(!buildingState.isChanged(), "shouldn't be changed: %s", this); return setStateFinishedAndReturnReverseDeps(); } @Override public synchronized void forceRebuild() { buildingState.forceChanged(); } @Override public synchronized Version getVersion() { return version; } /** @see BuildingState#getDirtyState() */ @Override public synchronized NodeEntry.DirtyState getDirtyState() { return buildingState.getDirtyState(); } /** @see BuildingState#getNextDirtyDirectDeps() */ @Override public synchronized Collection getNextDirtyDirectDeps() { Preconditions.checkState(isReady(), this); return buildingState.getNextDirtyDirectDeps(); } @Override public synchronized Iterable getAllDirectDepsForIncompleteNode() { Preconditions.checkState(!isDone(), this); if (!isDirty()) { return getTemporaryDirectDeps(); } else { return Iterables.concat( getTemporaryDirectDeps(), buildingState.getAllRemainingDirtyDirectDeps()); } } @Override public synchronized Collection markRebuildingAndGetAllRemainingDirtyDirectDeps() { return buildingState.markRebuildingAndGetAllRemainingDirtyDirectDeps(); } @Override public synchronized Set getTemporaryDirectDeps() { Preconditions.checkState(!isDone(), "temporary shouldn't be done: %s", this); return buildingState.getDirectDepsForBuild(); } @Override public synchronized boolean noDepsLastBuild() { return buildingState.noDepsLastBuild(); } @Override public synchronized void removeUnfinishedDeps(Set unfinishedDeps) { buildingState.removeDirectDeps(unfinishedDeps); } @Override public synchronized void addTemporaryDirectDeps(GroupedListHelper helper) { Preconditions.checkState(!isDone(), "add temp shouldn't be done: %s %s", helper, this); buildingState.addDirectDeps(helper); } @Override public synchronized boolean isReady() { Preconditions.checkState(!isDone(), "can't be ready if done: %s", this); return buildingState.isReady(); } @Override public synchronized String toString() { return MoreObjects.toStringHelper(this) .add("identity", System.identityHashCode(this)) .add("value", value) .add("version", version) .add("directDeps", directDeps == null ? null : GroupedList.create(directDeps)) .add("reverseDeps", REVERSE_DEPS_UTIL.toString(this)) .add("buildingState", buildingState) .toString(); } /** * Do not use except in custom evaluator implementations! Added only temporarily. * *

Clones a InMemoryMutableNodeEntry iff it is a done node. Otherwise it fails. */ public synchronized InMemoryNodeEntry cloneNodeEntry() { // As this is temporary, for now let's limit to done nodes. Preconditions.checkState(isDone(), "Only done nodes can be copied: %s", this); InMemoryNodeEntry nodeEntry = new InMemoryNodeEntry(); nodeEntry.value = value; nodeEntry.version = this.version; REVERSE_DEPS_UTIL.addReverseDeps(nodeEntry, REVERSE_DEPS_UTIL.getReverseDeps(this)); nodeEntry.directDeps = directDeps; nodeEntry.buildingState = null; return nodeEntry; } }