// 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.lib.analysis; import com.google.common.collect.ImmutableMap; import com.google.devtools.build.lib.analysis.config.BuildConfiguration; import com.google.devtools.build.lib.cmdline.Label; import com.google.devtools.build.lib.packages.AspectDescriptor; import com.google.devtools.build.lib.packages.Attribute; import com.google.devtools.build.lib.util.Preconditions; import java.util.Map; import java.util.Objects; import javax.annotation.Nullable; /** * A dependency of a configured target through a label. * * <p>For static configurations: includes the target and the configuration of the dependency * configured target and any aspects that may be required, as well as the configurations for * these aspects. * * <p>For dynamic configurations: includes the target and the desired configuration transitions * that should be applied to produce the dependency's configuration. It's the caller's * responsibility to construct an actual configuration out of that. A set of aspects is also * included; the caller must also construct configurations for each of these. * * <p>Note that the presence of an aspect here does not necessarily mean that it will be created. * They will be filtered based on the {@link TransitiveInfoProvider} instances their associated * configured targets have (we cannot do that here because the configured targets are not * available yet). No error or warning is reported in this case, because it is expected that rules * sometimes over-approximate the providers they supply in their definitions. */ public abstract class Dependency { /** * Creates a new {@link Dependency} with a null configuration, suitable for edges with no * configuration in static configuration builds. */ public static Dependency withNullConfiguration(Label label) { return new NullConfigurationDependency(label); } /** * Creates a new {@link Dependency} with the given configuration, suitable for static * configuration builds. * * <p>The configuration must not be {@code null}. * * <p>A {@link Dependency} created this way will have no associated aspects. */ public static Dependency withConfiguration(Label label, BuildConfiguration configuration) { return new StaticConfigurationDependency( label, configuration, AspectCollection.EMPTY, ImmutableMap.<AspectDescriptor, BuildConfiguration>of()); } /** * Creates a new {@link Dependency} with the given configuration and aspects, suitable for * static configuration builds. The configuration is also applied to all aspects. * * <p>The configuration and aspects must not be {@code null}. */ public static Dependency withConfigurationAndAspects( Label label, BuildConfiguration configuration, AspectCollection aspects) { ImmutableMap.Builder<AspectDescriptor, BuildConfiguration> aspectBuilder = new ImmutableMap.Builder<>(); for (AspectDescriptor aspect : aspects.getAllAspects()) { aspectBuilder.put(aspect, configuration); } return new StaticConfigurationDependency(label, configuration, aspects, aspectBuilder.build()); } /** * Creates a new {@link Dependency} with the given configuration and aspects, suitable for * storing the output of a dynamic configuration trimming step. The aspects each have their own * configuration. * * <p>The aspects and configurations must not be {@code null}. */ public static Dependency withConfiguredAspects( Label label, BuildConfiguration configuration, AspectCollection aspects, Map<AspectDescriptor, BuildConfiguration> aspectConfigurations) { return new StaticConfigurationDependency( label, configuration, aspects, ImmutableMap.copyOf(aspectConfigurations)); } /** * Creates a new {@link Dependency} with the given transition and aspects, suitable for dynamic * configuration builds. */ public static Dependency withTransitionAndAspects( Label label, Attribute.Transition transition, AspectCollection aspects) { return new DynamicConfigurationDependency(label, transition, aspects); } protected final Label label; /** * Only the implementations below should extend this class. Use the factory methods above to * create new Dependencies. */ private Dependency(Label label) { this.label = Preconditions.checkNotNull(label); } /** Returns the label of the target this dependency points to. */ public Label getLabel() { return label; } /** * Returns true if this dependency specifies a static configuration, false if it specifies * a dynamic transition. */ public abstract boolean hasStaticConfiguration(); /** * Returns the static configuration for the target this dependency points to. * * @throws IllegalStateException if {@link #hasStaticConfiguration} returns false. */ @Nullable public abstract BuildConfiguration getConfiguration(); /** * Returns the dynamic transition to be applied to reach the target this dependency points to. * * @throws IllegalStateException if {@link #hasStaticConfiguration} returns true. */ public abstract Attribute.Transition getTransition(); /** * Returns the set of aspects which should be evaluated and combined with the configured target * pointed to by this dependency. * * @see #getAspectConfiguration(AspectDescriptor) */ public abstract AspectCollection getAspects(); /** * Returns the the static configuration an aspect should be evaluated with ** * @throws IllegalStateException if {@link #hasStaticConfiguration()} returns false. */ public abstract BuildConfiguration getAspectConfiguration(AspectDescriptor aspect); /** * Implementation of a dependency with no configuration, suitable for static configuration * builds of edges to source files or e.g. for visibility. */ private static final class NullConfigurationDependency extends Dependency { public NullConfigurationDependency(Label label) { super(label); } @Override public boolean hasStaticConfiguration() { return true; } @Nullable @Override public BuildConfiguration getConfiguration() { return null; } @Override public Attribute.Transition getTransition() { throw new IllegalStateException( "A dependency with a static configuration does not have a transition."); } @Override public AspectCollection getAspects() { return AspectCollection.EMPTY; } @Override public BuildConfiguration getAspectConfiguration(AspectDescriptor aspect) { return null; } @Override public int hashCode() { return label.hashCode(); } @Override public boolean equals(Object other) { if (!(other instanceof NullConfigurationDependency)) { return false; } NullConfigurationDependency otherDep = (NullConfigurationDependency) other; return label.equals(otherDep.label); } @Override public String toString() { return String.format("NullConfigurationDependency{label=%s}", label); } } /** * Implementation of a dependency with static configurations, suitable for static configuration * builds. */ private static final class StaticConfigurationDependency extends Dependency { private final BuildConfiguration configuration; private final AspectCollection aspects; private final ImmutableMap<AspectDescriptor, BuildConfiguration> aspectConfigurations; public StaticConfigurationDependency( Label label, BuildConfiguration configuration, AspectCollection aspects, ImmutableMap<AspectDescriptor, BuildConfiguration> aspectConfigurations) { super(label); this.configuration = Preconditions.checkNotNull(configuration); this.aspects = Preconditions.checkNotNull(aspects); this.aspectConfigurations = Preconditions.checkNotNull(aspectConfigurations); } @Override public boolean hasStaticConfiguration() { return true; } @Override public BuildConfiguration getConfiguration() { return configuration; } @Override public Attribute.Transition getTransition() { throw new IllegalStateException( "A dependency with a static configuration does not have a transition."); } @Override public AspectCollection getAspects() { return aspects; } @Override public BuildConfiguration getAspectConfiguration(AspectDescriptor aspect) { return aspectConfigurations.get(aspect); } @Override public int hashCode() { return Objects.hash(label, configuration, aspectConfigurations); } @Override public boolean equals(Object other) { if (!(other instanceof StaticConfigurationDependency)) { return false; } StaticConfigurationDependency otherDep = (StaticConfigurationDependency) other; return label.equals(otherDep.label) && configuration.equals(otherDep.configuration) && aspects.equals(otherDep.aspects) && aspectConfigurations.equals(otherDep.aspectConfigurations); } @Override public String toString() { return String.format( "StaticConfigurationDependency{label=%s, configuration=%s, aspectConfigurations=%s}", label, configuration, aspectConfigurations); } } /** * Implementation of a dependency with a given configuration transition, suitable for dynamic * configuration builds. */ private static final class DynamicConfigurationDependency extends Dependency { private final Attribute.Transition transition; private final AspectCollection aspects; public DynamicConfigurationDependency( Label label, Attribute.Transition transition, AspectCollection aspects) { super(label); this.transition = Preconditions.checkNotNull(transition); this.aspects = Preconditions.checkNotNull(aspects); } @Override public boolean hasStaticConfiguration() { return false; } @Override public BuildConfiguration getConfiguration() { throw new IllegalStateException( "A dependency with a dynamic configuration transition does not have a configuration."); } @Override public Attribute.Transition getTransition() { return transition; } @Override public AspectCollection getAspects() { return aspects; } @Override public BuildConfiguration getAspectConfiguration(AspectDescriptor aspect) { throw new IllegalStateException( "A dependency with a dynamic configuration transition does not have aspect " + "configurations."); } @Override public int hashCode() { return Objects.hash(label, transition, aspects); } @Override public boolean equals(Object other) { if (!(other instanceof DynamicConfigurationDependency)) { return false; } DynamicConfigurationDependency otherDep = (DynamicConfigurationDependency) other; return label.equals(otherDep.label) && transition.equals(otherDep.transition) && aspects.equals(otherDep.aspects); } @Override public String toString() { return String.format( "DynamicConfigurationDependency{label=%s, transition=%s, aspects=%s}", label, transition, aspects); } } }