// Copyright 2016 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.MoreObjects; import com.google.common.collect.ImmutableSet; import com.google.devtools.build.lib.util.GroupedList; import com.google.devtools.build.lib.util.Preconditions; import com.google.devtools.build.skyframe.NodeEntry.DirtyState; import java.util.Collection; import java.util.Set; /** * A {@link BuildingState} for a node that has been dirtied, and will be checked to see if it needs * re-evaluation, and either marked clean or re-evaluated. See {@link BuildingState} for more. * *

This class is public only for the benefit of alternative graph implementations outside of the * package. */ public abstract class DirtyBuildingState extends BuildingState { /** * The state of a dirty node. A node is marked dirty in the DirtyBuildingState constructor, and * goes into either the state {@link DirtyState#CHECK_DEPENDENCIES} or {@link * DirtyState#NEEDS_REBUILDING}, depending on whether the caller specified that the node was * itself changed or not. Never null. */ private DirtyState dirtyState; /** * The dependencies requested (with group markers) last time the node was built (and below, the * value last time the node was built). They will be compared to dependencies requested on this * build to check whether this node has changed in {@link NodeEntry#setValue}. If they are null, * it means that this node is being built for the first time. See {@link * InMemoryNodeEntry#directDeps} for more on dependency group storage. */ protected final GroupedList lastBuildDirectDeps; /** The value of the node the last time it was built. */ protected abstract SkyValue getLastBuildValue() throws InterruptedException; /** * Group of children to be checked next in the process of determining if this entry needs to be * re-evaluated. Used by {@link DirtyBuildingState#getNextDirtyDirectDeps} and {@link #signalDep}. */ private int dirtyDirectDepIndex = -1; protected DirtyBuildingState(boolean isChanged, GroupedList lastBuildDirectDeps) { this.lastBuildDirectDeps = lastBuildDirectDeps; Preconditions.checkState( isChanged || !this.lastBuildDirectDeps.isEmpty(), "%s is being marked dirty, not changed, but has no children that could have dirtied it", this); dirtyState = isChanged ? DirtyState.NEEDS_REBUILDING : DirtyState.CHECK_DEPENDENCIES; // We need to iterate through the deps to see if they have changed, or to remove them if one // has. Initialize the iterating index. dirtyDirectDepIndex = 0; } static BuildingState create( boolean isChanged, GroupedList lastBuildDirectDeps, SkyValue lastBuildValue) { return new DirtyBuildingStateWithValue(isChanged, lastBuildDirectDeps, lastBuildValue); } final void markChanged() { Preconditions.checkState(isDirty(), this); Preconditions.checkState(!isChanged(), this); Preconditions.checkState(!isEvaluating(), this); dirtyState = DirtyState.NEEDS_REBUILDING; } final void forceChanged() { Preconditions.checkState(isDirty(), this); Preconditions.checkState(!isChanged(), this); Preconditions.checkState(isEvaluating(), this); Preconditions.checkState(lastBuildDirectDeps.listSize() == dirtyDirectDepIndex, this); dirtyState = DirtyState.REBUILDING; } final int getSignaledDeps() { return signaledDeps; } @Override final boolean isDirty() { return true; } @Override final boolean isChanged() { return dirtyState == DirtyState.NEEDS_REBUILDING || dirtyState == DirtyState.REBUILDING; } @Override protected final void checkFinishedBuildingWhenAboutToSetValue() { Preconditions.checkState(isEvaluating(), "not started building %s", this); Preconditions.checkState( !isDirty() || dirtyState == DirtyState.VERIFIED_CLEAN || dirtyState == DirtyState.REBUILDING, "not done building %s", this); } /** * If this node is not yet known to need rebuilding, sets {@link #dirtyState} to {@link * DirtyState#NEEDS_REBUILDING} if the child has changed, and {@link DirtyState#VERIFIED_CLEAN} if * the child has not changed and this was the last child to be checked (as determined by {@link * #isReady} and comparing {@link #dirtyDirectDepIndex} and {@link * DirtyBuildingState#lastBuildDirectDeps#listSize}. */ @Override final void signalDepInternal(boolean childChanged, int numDirectDeps) { if (!isChanged()) { // Synchronization isn't needed here because the only caller is NodeEntry, which does it // through the synchronized method signalDep(Version). if (childChanged) { dirtyState = DirtyState.NEEDS_REBUILDING; } else if (dirtyState == DirtyState.CHECK_DEPENDENCIES && isReady(numDirectDeps) && lastBuildDirectDeps.listSize() == dirtyDirectDepIndex) { // No other dep already marked this as NEEDS_REBUILDING, no deps outstanding, and this was // the last block of deps to be checked. dirtyState = DirtyState.VERIFIED_CLEAN; } } } /** * Returns true if {@code newValue}.equals the value from the last time this node was built. * Should only be used by {@link NodeEntry#setValue}. * *

Changes in direct deps do not force this to return false. Only the value is * considered. */ final boolean unchangedFromLastBuild(SkyValue newValue) throws InterruptedException { checkFinishedBuildingWhenAboutToSetValue(); return !(newValue instanceof NotComparableSkyValue) && getLastBuildValue().equals(newValue); } /** * Returns true if the deps requested during this evaluation ({@code directDeps}) are exactly * those requested the last time this node was built, in the same order. */ final boolean depsUnchangedFromLastBuild(GroupedList directDeps) { checkFinishedBuildingWhenAboutToSetValue(); return lastBuildDirectDeps.equals(directDeps); } final boolean noDepsLastBuild() { return lastBuildDirectDeps.isEmpty(); } /** * Gets the current state of checking this dirty entry to see if it must be re-evaluated. Must be * called each time evaluation of a dirty entry starts to find the proper action to perform next, * as enumerated by {@link DirtyState}. * * @see NodeEntry#getDirtyState() */ final DirtyState getDirtyState() { // Entry may not be ready if being built just for its errors. Preconditions.checkState(isEvaluating(), "must be evaluating to get dirty state %s", this); return dirtyState; } /** * Gets the next children to be re-evaluated to see if this dirty node needs to be re-evaluated. * *

See {@link NodeEntry#getNextDirtyDirectDeps}. */ final Collection getNextDirtyDirectDeps() { Preconditions.checkState(isDirty(), this); Preconditions.checkState(dirtyState == DirtyState.CHECK_DEPENDENCIES, this); Preconditions.checkState(isEvaluating(), this); Preconditions.checkState(dirtyDirectDepIndex < lastBuildDirectDeps.listSize(), this); return lastBuildDirectDeps.get(dirtyDirectDepIndex++); } /** * Returns the remaining direct deps that have not been checked. If {@code preservePosition} is * true, this method is non-mutating. If {@code preservePosition} is false, the caller must * process the returned set, and so subsequent calls to this method will return the empty set. */ Set getAllRemainingDirtyDirectDeps(boolean preservePosition) { Preconditions.checkState(isDirty(), this); ImmutableSet.Builder result = ImmutableSet.builder(); for (int ind = dirtyDirectDepIndex; ind < lastBuildDirectDeps.listSize(); ind++) { result.addAll(lastBuildDirectDeps.get(ind)); } if (!preservePosition) { dirtyDirectDepIndex = lastBuildDirectDeps.listSize(); } return result.build(); } protected void markRebuilding() { Preconditions.checkState(dirtyState == DirtyState.NEEDS_REBUILDING, this); dirtyState = DirtyState.REBUILDING; } @Override protected MoreObjects.ToStringHelper getStringHelper() { return super.getStringHelper() .add("dirtyState", dirtyState) .add("lastBuildDirectDeps", lastBuildDirectDeps) .add("dirtyDirectDepIndex", dirtyDirectDepIndex); } private static class DirtyBuildingStateWithValue extends DirtyBuildingState { private final SkyValue lastBuildValue; private DirtyBuildingStateWithValue( boolean isChanged, GroupedList lastBuildDirectDeps, SkyValue lastBuildValue) { super(isChanged, lastBuildDirectDeps); this.lastBuildValue = lastBuildValue; } @Override protected SkyValue getLastBuildValue() { return lastBuildValue; } @Override protected MoreObjects.ToStringHelper getStringHelper() { return super.getStringHelper().add("lastBuildValue", lastBuildValue); } } }