blob: c2fc43e7410ed7ac7222ea97d87c32ca8fdbc639 [file] [log] [blame]
// Copyright 2018 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.analysis.starlark;
import static com.google.devtools.build.lib.analysis.starlark.FunctionTransitionUtil.applyAndValidate;
import static com.google.devtools.build.lib.analysis.starlark.StarlarkAttributesCollection.ERROR_MESSAGE_FOR_NO_ATTR;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Preconditions;
import com.google.common.collect.ImmutableMap;
import com.google.devtools.build.lib.analysis.config.BuildOptions;
import com.google.devtools.build.lib.analysis.config.BuildOptionsView;
import com.google.devtools.build.lib.analysis.config.StarlarkDefinedConfigTransition;
import com.google.devtools.build.lib.analysis.config.transitions.SplitTransition;
import com.google.devtools.build.lib.analysis.config.transitions.TransitionFactory;
import com.google.devtools.build.lib.events.EventHandler;
import com.google.devtools.build.lib.packages.Attribute;
import com.google.devtools.build.lib.packages.AttributeMap;
import com.google.devtools.build.lib.packages.AttributeTransitionData;
import com.google.devtools.build.lib.packages.ConfiguredAttributeMapper;
import com.google.devtools.build.lib.packages.StructImpl;
import com.google.devtools.build.lib.packages.StructProvider;
import com.google.devtools.build.lib.starlarkbuildapi.SplitTransitionProviderApi;
import java.util.LinkedHashMap;
import java.util.Objects;
import net.starlark.java.eval.Printer;
import net.starlark.java.eval.Starlark.InvalidStarlarkValueException;
/**
* This class implements {@link TransitionFactory} to provide a starlark-defined transition that
* rules can apply to their dependencies' configurations. This transition has access to (1) the map
* of the current configuration's build settings and (2) the configured attributes of the given rule
* (not its dependencies').
*
* <p>For starlark defined rule class transitions, see {@link StarlarkRuleTransitionProvider}.
*
* <p>TODO(bazel-team): Consider allowing dependency-typed attributes to actually return providers
* instead of just labels (see {@link StarlarkAttributesCollection#addAttribute}).
*/
public class StarlarkAttributeTransitionProvider
implements TransitionFactory<AttributeTransitionData>, SplitTransitionProviderApi {
private final StarlarkDefinedConfigTransition starlarkDefinedConfigTransition;
public StarlarkAttributeTransitionProvider(
StarlarkDefinedConfigTransition starlarkDefinedConfigTransition) {
this.starlarkDefinedConfigTransition = starlarkDefinedConfigTransition;
}
@VisibleForTesting
public StarlarkDefinedConfigTransition getStarlarkDefinedConfigTransitionForTesting() {
return starlarkDefinedConfigTransition;
}
@Override
public int hashCode() {
return starlarkDefinedConfigTransition.hashCode();
}
@Override
public SplitTransition create(AttributeTransitionData data) {
AttributeMap attributeMap = data.attributes();
Preconditions.checkArgument(
attributeMap == null || attributeMap instanceof ConfiguredAttributeMapper);
// TODO(bazel-team): consider caching transition instances to save CPU time, similar to what's
// done in StarlarkRuleTransitionProvider. This could benefit builds that apply transitions over
// many build graph edges.
return new FunctionSplitTransition(
starlarkDefinedConfigTransition, (ConfiguredAttributeMapper) attributeMap);
}
public boolean allowImmutableFlagChanges() {
return false;
}
@Override
public TransitionType transitionType() {
return TransitionType.ATTRIBUTE;
}
@Override
public boolean isSplit() {
return true;
}
@Override
public void repr(Printer printer) {
printer.append("<transition object>");
}
final class FunctionSplitTransition extends StarlarkTransition implements SplitTransition {
private final StructImpl attrObject;
private final int hashCode;
private FunctionSplitTransition(
StarlarkDefinedConfigTransition starlarkDefinedConfigTransition,
ConfiguredAttributeMapper attributeMap) {
super(starlarkDefinedConfigTransition);
LinkedHashMap<String, Object> attributes = new LinkedHashMap<>();
if (attributeMap != null) {
for (String attribute : attributeMap.getAttributeNames()) {
Object val = attributeMap.get(attribute, attributeMap.getAttributeType(attribute));
try {
Object starlarkVal = Attribute.valueToStarlark(val);
attributes.put(Attribute.getStarlarkName(attribute), starlarkVal);
} catch (InvalidStarlarkValueException e) {
// This is only possible for native targets, since Starlark targets by definition have
// Starlark-readable attributes. The only Starlark transition that can apply to native
// targets is the exec transition (ExecutionTransitionFactory). Since that's
// experimental
// we don't need to do anything further.
// TODO(b/288258583): encode this more cleanly than a universally swallowed exception.
}
}
}
attrObject = StructProvider.STRUCT.create(attributes, ERROR_MESSAGE_FOR_NO_ATTR);
this.hashCode = Objects.hash(attrObject, super.hashCode());
}
/**
* @return the post-transition build options or a clone of the original build options if an
* error was encountered during transition application/validation.
*/
@Override
public final ImmutableMap<String, BuildOptions> split(
BuildOptionsView buildOptionsView, EventHandler eventHandler) throws InterruptedException {
// Starlark transitions already have logic to enforce they only access declared inputs and
// outputs. Rather than complicate BuildOptionsView with more access points to BuildOptions,
// we just use the original BuildOptions and trust the transition's enforcement logic.
BuildOptions buildOptions = buildOptionsView.underlying();
ImmutableMap<String, BuildOptions> res =
applyAndValidate(
buildOptions,
starlarkDefinedConfigTransition,
allowImmutableFlagChanges(),
attrObject,
eventHandler);
if (res == null) {
return ImmutableMap.of("error", buildOptions.clone());
}
return res;
}
@Override
public boolean equals(Object object) {
if (object == this) {
return true;
}
if (!(object instanceof FunctionSplitTransition other)) {
return false;
}
return Objects.equals(attrObject, other.attrObject) && super.equals(other);
}
@Override
public int hashCode() {
return hashCode;
}
}
}