blob: 70f126d3f3e71e434568a332227be9000580c3d0 [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 java.util.Set;
import java.util.concurrent.atomic.AtomicReference;
import java.util.function.Supplier;
import javax.annotation.Nullable;
/**
* A testing utility to keep track of evaluation.
*/
public class TrackingProgressReceiver
extends EvaluationProgressReceiver.NullEvaluationProgressReceiver {
private final boolean checkEvaluationResults;
/**
* Callback to be executed on a next {@link #invalidated} 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 invalidated(SkyKey skyKey, InvalidationState state) {
final Runnable invalidateCallback = nextInvalidationCallback.getAndSet(null);
if (invalidateCallback != null) {
invalidateCallback.run();
}
switch (state) {
case DELETED:
dirty.remove(skyKey);
deleted.add(skyKey);
break;
case DIRTY:
dirty.add(skyKey);
Preconditions.checkState(!deleted.contains(skyKey));
break;
default:
throw new IllegalStateException();
}
}
@Override
public void enqueueing(SkyKey skyKey) {
enqueued.add(skyKey);
}
@Override
public void evaluated(
SkyKey skyKey,
@Nullable SkyValue value,
Supplier<EvaluationSuccessState> evaluationSuccessState,
EvaluationState state) {
evaluated.add(skyKey);
if (checkEvaluationResults && evaluationSuccessState.get().succeeded()) {
deleted.remove(skyKey);
if (state.equals(EvaluationState.CLEAN)) {
dirty.remove(skyKey);
}
}
}
public void clear() {
dirty.clear();
deleted.clear();
enqueued.clear();
evaluated.clear();
}
void setNextInvalidationCallback(Runnable runnable) {
final Runnable oldCallback = nextInvalidationCallback.getAndSet(runnable);
Preconditions.checkState(
oldCallback == null, "Overwriting a left-over callback: %s", oldCallback);
}
}