// 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.skyframe; import com.google.devtools.build.lib.analysis.config.BuildConfiguration; import com.google.devtools.build.lib.cmdline.Label; import com.google.devtools.build.lib.cmdline.PackageIdentifier; import com.google.devtools.build.lib.collect.nestedset.NestedSet; import com.google.devtools.build.lib.collect.nestedset.NestedSetBuilder; import com.google.devtools.build.lib.collect.nestedset.Order; import com.google.devtools.build.lib.concurrent.ThreadSafety.Immutable; import com.google.devtools.build.lib.concurrent.ThreadSafety.ThreadSafe; import com.google.devtools.build.lib.packages.NoSuchTargetException; import com.google.devtools.build.lib.util.Preconditions; import com.google.devtools.build.skyframe.LegacySkyKey; import com.google.devtools.build.skyframe.SkyKey; import com.google.devtools.build.skyframe.SkyValue; import java.io.IOException; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; import java.util.Set; import javax.annotation.Nullable; /** * A <i>transitive</i> target reference that, when built in skyframe, loads the entire * transitive closure of a target. * * This will probably be unnecessary once other refactorings occur throughout the codebase * which make loading/analysis interleaving more feasible, or we will migrate "blaze query" to * use this to evaluate its Target graph. */ @Immutable @ThreadSafe public class TransitiveTargetValue implements SkyValue { // Non-final for serialization purposes. private NestedSet<PackageIdentifier> transitiveSuccessfulPkgs; private NestedSet<PackageIdentifier> transitiveUnsuccessfulPkgs; private NestedSet<Label> transitiveTargets; @Nullable private NestedSet<Label> transitiveRootCauses; @Nullable private NoSuchTargetException errorLoadingTarget; private NestedSet<Class<? extends BuildConfiguration.Fragment>> transitiveConfigFragments; private TransitiveTargetValue(NestedSet<PackageIdentifier> transitiveSuccessfulPkgs, NestedSet<PackageIdentifier> transitiveUnsuccessfulPkgs, NestedSet<Label> transitiveTargets, @Nullable NestedSet<Label> transitiveRootCauses, @Nullable NoSuchTargetException errorLoadingTarget, NestedSet<Class<? extends BuildConfiguration.Fragment>> transitiveConfigFragments) { this.transitiveSuccessfulPkgs = transitiveSuccessfulPkgs; this.transitiveUnsuccessfulPkgs = transitiveUnsuccessfulPkgs; this.transitiveTargets = transitiveTargets; this.transitiveRootCauses = transitiveRootCauses; this.errorLoadingTarget = errorLoadingTarget; this.transitiveConfigFragments = transitiveConfigFragments; } private void writeObject(ObjectOutputStream out) throws IOException { // It helps to flatten the transitiveSuccessfulPkgs nested set as it has lots of duplicates. Set<PackageIdentifier> successfulPkgs = transitiveSuccessfulPkgs.toSet(); out.writeInt(successfulPkgs.size()); for (PackageIdentifier pkg : successfulPkgs) { out.writeObject(pkg); } out.writeObject(transitiveUnsuccessfulPkgs); // Deliberately do not write out transitiveTargets. There is a lot of those and they drive // serialization costs through the roof, both in terms of space and of time. // TODO(bazel-team): Deal with this properly once we have efficient serialization of NestedSets. out.writeObject(transitiveRootCauses); out.writeObject(errorLoadingTarget); out.writeObject(transitiveConfigFragments); } @SuppressWarnings("unchecked") private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException { int successfulPkgCount = in.readInt(); NestedSetBuilder<PackageIdentifier> pkgs = NestedSetBuilder.stableOrder(); for (int i = 0; i < successfulPkgCount; i++) { pkgs.add((PackageIdentifier) in.readObject()); } transitiveSuccessfulPkgs = pkgs.build(); transitiveUnsuccessfulPkgs = (NestedSet<PackageIdentifier>) in.readObject(); // TODO(bazel-team): Deal with transitiveTargets properly. transitiveTargets = NestedSetBuilder.emptySet(Order.STABLE_ORDER); transitiveRootCauses = (NestedSet<Label>) in.readObject(); errorLoadingTarget = (NoSuchTargetException) in.readObject(); transitiveConfigFragments = (NestedSet<Class<? extends BuildConfiguration.Fragment>>) in.readObject(); } static TransitiveTargetValue unsuccessfulTransitiveLoading( NestedSet<PackageIdentifier> transitiveSuccessfulPkgs, NestedSet<PackageIdentifier> transitiveUnsuccessfulPkgs, NestedSet<Label> transitiveTargets, NestedSet<Label> rootCauses, @Nullable NoSuchTargetException errorLoadingTarget, NestedSet<Class<? extends BuildConfiguration.Fragment>> transitiveConfigFragments) { return new TransitiveTargetValue(transitiveSuccessfulPkgs, transitiveUnsuccessfulPkgs, transitiveTargets, rootCauses, errorLoadingTarget, transitiveConfigFragments); } static TransitiveTargetValue successfulTransitiveLoading( NestedSet<PackageIdentifier> transitiveSuccessfulPkgs, NestedSet<PackageIdentifier> transitiveUnsuccessfulPkgs, NestedSet<Label> transitiveTargets, NestedSet<Class<? extends BuildConfiguration.Fragment>> transitiveConfigFragments) { return new TransitiveTargetValue(transitiveSuccessfulPkgs, transitiveUnsuccessfulPkgs, transitiveTargets, null, null, transitiveConfigFragments); } /** Returns the error, if any, from loading the target. */ @Nullable public NoSuchTargetException getErrorLoadingTarget() { return errorLoadingTarget; } /** Returns the packages that were transitively successfully loaded. */ public NestedSet<PackageIdentifier> getTransitiveSuccessfulPackages() { return transitiveSuccessfulPkgs; } /** Returns the packages that were transitively successfully loaded. */ public NestedSet<PackageIdentifier> getTransitiveUnsuccessfulPackages() { return transitiveUnsuccessfulPkgs; } /** Returns the targets that were transitively loaded. */ public NestedSet<Label> getTransitiveTargets() { return transitiveTargets; } /** Returns the root causes, if any, of why targets weren't loaded. */ @Nullable public NestedSet<Label> getTransitiveRootCauses() { return transitiveRootCauses; } /** * Returns the set of {@link * com.google.devtools.build.lib.analysis.config.BuildConfiguration.Fragment} classes required to * configure a rule's transitive closure. These are used to instantiate the right {@link * ConfigurationFragmentValue} instances for a rule's {@link BuildConfigurationValue}. * * <p>This provides the basis for rule-scoped configurations. For example, Java-related build * flags have nothing to do with C++. So changing a Java flag shouldn't invalidate a C++ rule * (unless it has transitive dependencies on other Java rules). Likewise, a C++ rule shouldn't * fail because the Java configuration doesn't recognize the chosen architecture. * * <p>The general principle is that a rule can be influenced by the configuration parameters it * directly uses and the configuration parameters its transitive dependencies use (since it * reads its dependencies as part of analysis). So we need to 1) determine which configuration * fragments provide these parameters, 2) load those fragments, then 3) create a configuration * from them to feed the rule's configured target. This provides the first step. * * <p>See * {@link com.google.devtools.build.lib.packages.RuleClass.Builder#requiresConfigurationFragments} */ NestedSet<Class<? extends BuildConfiguration.Fragment>> getTransitiveConfigFragments() { return transitiveConfigFragments; } @ThreadSafe public static SkyKey key(Label label) { Preconditions.checkArgument(!label.getPackageIdentifier().getRepository().isDefault()); return LegacySkyKey.create(SkyFunctions.TRANSITIVE_TARGET, label); } }