blob: 59245139d867ca107a18a7bceaa0d48cca322336 [file] [log] [blame]
// 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.Sets;
import com.google.devtools.build.skyframe.NodeEntry.DirtyType;
import java.util.Set;
import java.util.concurrent.atomic.AtomicReference;
import javax.annotation.Nullable;
/** A testing utility to keep track of evaluation. */
public class TrackingProgressReceiver implements EvaluationProgressReceiver {
private final boolean checkEvaluationResults;
/**
* Callback to be executed on a next {@link #dirtied} or {@link #deleted(SkyKey)} call. It will be
* run once and is expected to be run if set.
*/
private final AtomicReference<Runnable> nextInvalidationCallback = new AtomicReference<>();
public final Set<SkyKey> dirty = Sets.newConcurrentHashSet();
public final Set<SkyKey> deleted = Sets.newConcurrentHashSet();
public final Set<SkyKey> enqueued = Sets.newConcurrentHashSet();
public final Set<SkyKey> evaluated = Sets.newConcurrentHashSet();
public TrackingProgressReceiver(boolean checkEvaluationResults) {
this.checkEvaluationResults = checkEvaluationResults;
}
@Override
public void dirtied(SkyKey skyKey, DirtyType dirtyType) {
runInvalidationCallbackIfPresent();
dirty.add(skyKey);
Preconditions.checkState(!deleted.contains(skyKey), skyKey);
}
@Override
public void deleted(SkyKey skyKey) {
runInvalidationCallbackIfPresent();
dirty.remove(skyKey);
deleted.add(skyKey);
}
private void runInvalidationCallbackIfPresent() {
Runnable callback = nextInvalidationCallback.getAndSet(null);
if (callback != null) {
callback.run();
}
}
@Override
public void enqueueing(SkyKey skyKey) {
enqueued.add(skyKey);
}
@Override
public void evaluated(
SkyKey skyKey,
EvaluationState state,
@Nullable SkyValue value,
@Nullable ErrorInfo error,
@Nullable GroupedDeps directDeps) {
evaluated.add(skyKey);
if (checkEvaluationResults && state.succeeded()) {
deleted.remove(skyKey);
if (!state.versionChanged()) {
dirty.remove(skyKey);
}
}
}
public void clear() {
dirty.clear();
deleted.clear();
enqueued.clear();
evaluated.clear();
}
void setNextInvalidationCallback(Runnable runnable) {
Runnable oldCallback = nextInvalidationCallback.getAndSet(runnable);
Preconditions.checkState(
oldCallback == null, "Overwriting a left-over callback: %s", oldCallback);
}
}