| // 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.common.collect.ImmutableList; |
| import com.google.devtools.build.lib.analysis.DependencyResolver; |
| import com.google.devtools.build.lib.analysis.TargetAndConfiguration; |
| import com.google.devtools.build.lib.analysis.config.BuildConfiguration; |
| import com.google.devtools.build.lib.analysis.config.BuildOptions; |
| import com.google.devtools.build.lib.analysis.config.InvalidConfigurationException; |
| import com.google.devtools.build.lib.cmdline.Label; |
| import com.google.devtools.build.lib.collect.nestedset.NestedSetBuilder; |
| import com.google.devtools.build.lib.events.Event; |
| import com.google.devtools.build.lib.packages.NoSuchPackageException; |
| import com.google.devtools.build.lib.packages.NoSuchTargetException; |
| import com.google.devtools.build.lib.packages.NoSuchThingException; |
| import com.google.devtools.build.lib.packages.Package; |
| import com.google.devtools.build.lib.packages.Target; |
| import com.google.devtools.build.lib.packages.TargetUtils; |
| import com.google.devtools.build.skyframe.SkyFunction.Environment; |
| import com.google.devtools.build.skyframe.SkyKey; |
| import com.google.devtools.build.skyframe.ValueOrException; |
| import java.util.ArrayList; |
| import java.util.List; |
| import java.util.Map; |
| import java.util.Set; |
| import javax.annotation.Nullable; |
| |
| /** |
| * A dependency resolver for use within Skyframe. Loads packages lazily when possible. |
| */ |
| public final class SkyframeDependencyResolver extends DependencyResolver { |
| |
| private final Environment env; |
| |
| public SkyframeDependencyResolver(Environment env) { |
| this.env = env; |
| } |
| |
| @Override |
| protected void invalidVisibilityReferenceHook(TargetAndConfiguration value, Label label) { |
| env.getListener().handle( |
| Event.error(TargetUtils.getLocationMaybe(value.getTarget()), String.format( |
| "Label '%s' in visibility attribute does not refer to a package group", label))); |
| } |
| |
| @Override |
| protected void invalidPackageGroupReferenceHook(TargetAndConfiguration value, Label label) { |
| env.getListener().handle( |
| Event.error(TargetUtils.getLocationMaybe(value.getTarget()), String.format( |
| "label '%s' does not refer to a package group", label))); |
| } |
| |
| @Override |
| protected void missingEdgeHook(Target from, Label to, NoSuchThingException e) |
| throws InterruptedException { |
| if (e instanceof NoSuchTargetException) { |
| NoSuchTargetException nste = (NoSuchTargetException) e; |
| if (to.equals(nste.getLabel())) { |
| env.getListener().handle( |
| Event.error( |
| TargetUtils.getLocationMaybe(from), |
| TargetUtils.formatMissingEdge(from, to, e))); |
| } |
| } else if (e instanceof NoSuchPackageException) { |
| NoSuchPackageException nspe = (NoSuchPackageException) e; |
| if (nspe.getPackageId().equals(to.getPackageIdentifier())) { |
| env.getListener().handle( |
| Event.error( |
| TargetUtils.getLocationMaybe(from), |
| TargetUtils.formatMissingEdge(from, to, e))); |
| } |
| } |
| } |
| |
| @Nullable |
| @Override |
| protected Target getTarget(Target from, Label label, NestedSetBuilder<Label> rootCauses) |
| throws InterruptedException { |
| SkyKey key = PackageValue.key(label.getPackageIdentifier()); |
| PackageValue packageValue; |
| try { |
| packageValue = (PackageValue) env.getValueOrThrow(key, NoSuchPackageException.class); |
| } catch (NoSuchPackageException e) { |
| rootCauses.add(label); |
| missingEdgeHook(from, label, e); |
| return null; |
| } |
| if (packageValue == null) { |
| return null; |
| } |
| Package pkg = packageValue.getPackage(); |
| try { |
| Target target = pkg.getTarget(label.getName()); |
| if (pkg.containsErrors()) { |
| NoSuchTargetException e = new NoSuchTargetException(target); |
| missingEdgeHook(from, label, e); |
| if (target != null) { |
| rootCauses.add(label); |
| return target; |
| } else { |
| return null; |
| } |
| } |
| return target; |
| } catch (NoSuchTargetException e) { |
| rootCauses.add(label); |
| missingEdgeHook(from, label, e); |
| return null; |
| } |
| } |
| |
| @Nullable |
| @Override |
| protected List<BuildConfiguration> getConfigurations( |
| Set<Class<? extends BuildConfiguration.Fragment>> fragments, |
| Iterable<BuildOptions> buildOptions) |
| throws InvalidConfigurationException, InterruptedException { |
| List<SkyKey> keys = new ArrayList<>(); |
| for (BuildOptions options : buildOptions) { |
| keys.add(BuildConfigurationValue.key(fragments, options)); |
| } |
| Map<SkyKey, ValueOrException<InvalidConfigurationException>> configValues = |
| env.getValuesOrThrow(keys, InvalidConfigurationException.class); |
| if (env.valuesMissing()) { |
| return null; |
| } |
| ImmutableList.Builder<BuildConfiguration> result = ImmutableList.builder(); |
| for (SkyKey key : keys) { |
| result.add(((BuildConfigurationValue) configValues.get(key).get()).getConfiguration()); |
| } |
| return result.build(); |
| } |
| } |