| // 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); |
| } |
| } |