blob: c07e510a551a821dd17384b41e07adedd8dc0077 [file] [log] [blame]
// 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.Preconditions;
import com.google.common.collect.Maps;
import com.google.devtools.build.lib.events.ExtendedEventHandler;
import com.google.devtools.build.skyframe.QueryableGraph.Reason;
import java.util.Map;
/**
* A {@link SkyFunction.Environment} backed by a {@link QueryableGraph}. For use when a single
* SkyFunction needs recomputation, and its dependencies do not need to be evaluated. Any missing
* dependencies will be ignored.
*/
public class QueryableGraphBackedSkyFunctionEnvironment extends AbstractSkyFunctionEnvironment {
private final QueryableGraph queryableGraph;
private final ExtendedEventHandler eventHandler;
public QueryableGraphBackedSkyFunctionEnvironment(
QueryableGraph queryableGraph, ExtendedEventHandler eventHandler) {
this.queryableGraph = queryableGraph;
this.eventHandler = eventHandler;
}
private ValueOrUntypedException toUntypedValue(NodeEntry nodeEntry) throws InterruptedException {
if (nodeEntry == null || !nodeEntry.isDone()) {
valuesMissing = true;
return ValueOrUntypedException.ofNull();
}
SkyValue maybeWrappedValue = nodeEntry.getValueMaybeWithMetadata();
SkyValue justValue = ValueWithMetadata.justValue(maybeWrappedValue);
if (justValue != null) {
return ValueOrUntypedException.ofValueUntyped(justValue);
}
errorMightHaveBeenFound = true;
ErrorInfo errorInfo =
Preconditions.checkNotNull(ValueWithMetadata.getMaybeErrorInfo(maybeWrappedValue));
Exception exception = errorInfo.getException();
if (exception != null) {
// Give SkyFunction#compute a chance to handle this exception.
return ValueOrUntypedException.ofExn(exception);
}
// In a cycle.
Preconditions.checkState(
!errorInfo.getCycleInfo().isEmpty(), "%s %s", errorInfo, maybeWrappedValue);
return ValueOrUntypedException.ofNull();
}
@Override
protected Map<SkyKey, ValueOrUntypedException> getValueOrUntypedExceptions(
Iterable<? extends SkyKey> depKeys) throws InterruptedException {
Map<SkyKey, ? extends NodeEntry> resultMap =
queryableGraph.getBatch(null, Reason.DEP_REQUESTED, depKeys);
// resultMap will be smaller than what we actually return if some of depKeys were not found in
// the graph. Pad to a minimum of 16 to avoid excessive resizing.
Map<SkyKey, ValueOrUntypedException> result =
Maps.newHashMapWithExpectedSize(Math.max(16, resultMap.size()));
for (SkyKey dep : depKeys) {
result.put(dep, toUntypedValue(resultMap.get(dep)));
}
return result;
}
@Override
public ExtendedEventHandler getListener() {
return eventHandler;
}
@Override
public boolean inErrorBubblingForTesting() {
return false;
}
@Override
public boolean restartPermitted() {
return false;
}
}