|  | // Copyright 2019 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.lib.skyframe; | 
|  |  | 
|  | import com.google.common.base.Preconditions; | 
|  | import com.google.devtools.build.lib.analysis.config.BuildOptions; | 
|  | import com.google.devtools.build.lib.cmdline.Label; | 
|  | import com.google.devtools.build.lib.skyframe.trimming.TrimmedConfigurationCache; | 
|  | import com.google.devtools.build.skyframe.EvaluationProgressReceiver.EvaluationState; | 
|  | import com.google.devtools.build.skyframe.EvaluationProgressReceiver.EvaluationSuccessState; | 
|  | import com.google.devtools.build.skyframe.EvaluationProgressReceiver.InvalidationState; | 
|  | import com.google.devtools.build.skyframe.EvaluationProgressReceiver.NullEvaluationProgressReceiver; | 
|  | import com.google.devtools.build.skyframe.SkyKey; | 
|  | import com.google.devtools.build.skyframe.SkyValue; | 
|  | import java.util.function.Supplier; | 
|  | import javax.annotation.Nullable; | 
|  |  | 
|  | /** | 
|  | * Skyframe progress receiver which keeps a {@link TrimmedConfigurationCache} in sync with Skyframe | 
|  | * invalidations and revalidations. | 
|  | */ | 
|  | public final class TrimmedConfigurationProgressReceiver extends NullEvaluationProgressReceiver { | 
|  |  | 
|  | private final TrimmedConfigurationCache<SkyKey, Label, BuildOptions.OptionsDiffForReconstruction> | 
|  | cache; | 
|  |  | 
|  | private boolean enabled = true; | 
|  |  | 
|  | public TrimmedConfigurationProgressReceiver( | 
|  | TrimmedConfigurationCache<SkyKey, Label, BuildOptions.OptionsDiffForReconstruction> cache) { | 
|  | this.cache = cache; | 
|  | } | 
|  |  | 
|  | public static TrimmedConfigurationCache<SkyKey, Label, BuildOptions.OptionsDiffForReconstruction> | 
|  | buildCache() { | 
|  | return new TrimmedConfigurationCache<>( | 
|  | TrimmedConfigurationProgressReceiver::extractLabel, | 
|  | TrimmedConfigurationProgressReceiver::extractOptionsDiff, | 
|  | BuildOptions.OptionsDiffForReconstruction::compareFragments); | 
|  | } | 
|  |  | 
|  | public TrimmedConfigurationCache<SkyKey, Label, BuildOptions.OptionsDiffForReconstruction> | 
|  | getCache() { | 
|  | return this.cache; | 
|  | } | 
|  |  | 
|  | public static Label extractLabel(SkyKey key) { | 
|  | Preconditions.checkArgument(isKeyCacheable(key)); | 
|  | ConfiguredTargetKey configuredTargetKey = (ConfiguredTargetKey) key; | 
|  | return configuredTargetKey.getLabel(); | 
|  | } | 
|  |  | 
|  | public static BuildOptions.OptionsDiffForReconstruction extractOptionsDiff(SkyKey key) { | 
|  | Preconditions.checkArgument(isKeyCacheable(key)); | 
|  | ConfiguredTargetKey configuredTargetKey = (ConfiguredTargetKey) key; | 
|  | BuildConfigurationValue.Key configurationKey = configuredTargetKey.getConfigurationKey(); | 
|  | return configurationKey.getOptionsDiff(); | 
|  | } | 
|  |  | 
|  | public static boolean isKeyCacheable(SkyKey key) { | 
|  | if (!(key instanceof ConfiguredTargetKey)) { | 
|  | // Only configured targets go in the cache. | 
|  | // TODO(b/129286648): add aspect support | 
|  | return false; | 
|  | } | 
|  | ConfiguredTargetKey configuredTargetKey = (ConfiguredTargetKey) key; | 
|  | if (configuredTargetKey.getConfigurationKey() == null) { | 
|  | // Null-configured targets do not go in the cache. | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | public void activate() { | 
|  | if (this.enabled) { | 
|  | return; | 
|  | } | 
|  | this.enabled = true; | 
|  | } | 
|  |  | 
|  | public void deactivate() { | 
|  | if (!this.enabled) { | 
|  | return; | 
|  | } | 
|  | this.enabled = false; | 
|  | this.cache.clear(); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public void invalidated(SkyKey key, InvalidationState state) { | 
|  | if (!enabled || !isKeyCacheable(key)) { | 
|  | return; | 
|  | } | 
|  | switch (state) { | 
|  | case DIRTY: | 
|  | cache.invalidate(key); | 
|  | break; | 
|  | case DELETED: | 
|  | cache.remove(key); | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public void evaluated( | 
|  | SkyKey key, | 
|  | @Nullable SkyValue value, | 
|  | Supplier<EvaluationSuccessState> evaluationSuccessState, | 
|  | EvaluationState state) { | 
|  | if (!enabled || !isKeyCacheable(key)) { | 
|  | return; | 
|  | } | 
|  | switch (state) { | 
|  | case BUILT: | 
|  | // Do nothing; the evaluation would have handled putting itself (back) in the cache. | 
|  | break; | 
|  | case CLEAN: | 
|  | cache.revalidate(key); | 
|  | break; | 
|  | } | 
|  | } | 
|  | } |