| // Copyright 2015 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 static com.google.devtools.build.lib.analysis.config.BuildConfiguration.Fragment; |
| |
| import com.google.common.collect.ClassToInstanceMap; |
| import com.google.common.collect.ImmutableSet; |
| import com.google.common.collect.MutableClassToInstanceMap; |
| import com.google.devtools.build.lib.analysis.BlazeDirectories; |
| import com.google.devtools.build.lib.analysis.ConfigurationCollectionFactory; |
| import com.google.devtools.build.lib.analysis.ConfiguredRuleClassProvider; |
| import com.google.devtools.build.lib.analysis.config.BuildConfiguration; |
| import com.google.devtools.build.lib.analysis.config.InvalidConfigurationException; |
| import com.google.devtools.build.lib.packages.RuleClassProvider; |
| import com.google.devtools.build.skyframe.SkyFunction; |
| import com.google.devtools.build.skyframe.SkyFunctionException; |
| import com.google.devtools.build.skyframe.SkyKey; |
| import com.google.devtools.build.skyframe.SkyValue; |
| import com.google.devtools.build.skyframe.ValueOrException; |
| |
| import java.util.LinkedHashSet; |
| import java.util.Map; |
| import java.util.Set; |
| |
| /** |
| * A builder for {@link BuildConfigurationValue} instances. |
| */ |
| public class BuildConfigurationFunction implements SkyFunction { |
| |
| private final BlazeDirectories directories; |
| private final ConfigurationCollectionFactory collectionFactory; |
| |
| public BuildConfigurationFunction(BlazeDirectories directories, |
| RuleClassProvider ruleClassProvider) { |
| this.directories = directories; |
| collectionFactory = |
| ((ConfiguredRuleClassProvider) ruleClassProvider).getConfigurationCollectionFactory(); |
| } |
| |
| @Override |
| public SkyValue compute(SkyKey skyKey, Environment env) throws InterruptedException, |
| SkyFunctionException { |
| BuildConfigurationValue.Key key = (BuildConfigurationValue.Key) skyKey.argument(); |
| Set<Fragment> fragments; |
| try { |
| fragments = getConfigurationFragments(key, env); |
| } catch (InvalidConfigurationException e) { |
| throw new BuildConfigurationFunctionException(e); |
| } |
| if (fragments == null) { |
| return null; |
| } |
| |
| ClassToInstanceMap<Fragment> fragmentsMap = MutableClassToInstanceMap.create(); |
| for (Fragment fragment : fragments) { |
| fragmentsMap.put(fragment.getClass(), fragment); |
| } |
| |
| BuildConfiguration config = new BuildConfiguration(directories, fragmentsMap, |
| key.getBuildOptions(), !key.actionsEnabled()); |
| // Unlike static configurations, dynamic configurations don't need to embed transition logic |
| // within the configuration itself. However we still use this interface to provide a mapping |
| // between Transition types (e.g. HOST) and the dynamic transitions that apply those |
| // transitions. Once static configurations are cleaned out we won't need this interface |
| // any more (all the centralized logic that maintains the transition logic can be distributed |
| // to the actual rule code that uses it). |
| config.setConfigurationTransitions(collectionFactory.getDynamicTransitionLogic(config)); |
| |
| return new BuildConfigurationValue(config); |
| } |
| |
| private Set<Fragment> getConfigurationFragments(BuildConfigurationValue.Key key, Environment env) |
| throws InvalidConfigurationException { |
| |
| // Get SkyKeys for the fragments we need to load. |
| Set<SkyKey> fragmentKeys = new LinkedHashSet<>(); |
| for (Class<? extends BuildConfiguration.Fragment> fragmentClass : key.getFragments()) { |
| fragmentKeys.add(ConfigurationFragmentValue.key(key.getBuildOptions(), fragmentClass)); |
| } |
| |
| // Load them as Skyframe deps. |
| Map<SkyKey, ValueOrException<InvalidConfigurationException>> fragmentDeps = |
| env.getValuesOrThrow(fragmentKeys, InvalidConfigurationException.class); |
| if (env.valuesMissing()) { |
| return null; |
| } |
| |
| // Collect and return the results. |
| ImmutableSet.Builder<Fragment> fragments = ImmutableSet.builder(); |
| for (ValueOrException<InvalidConfigurationException> value : fragmentDeps.values()) { |
| BuildConfiguration.Fragment fragment = |
| ((ConfigurationFragmentValue) value.get()).getFragment(); |
| if (fragment != null) { |
| fragments.add(fragment); |
| } |
| } |
| return fragments.build(); |
| } |
| |
| @Override |
| public String extractTag(SkyKey skyKey) { |
| return null; |
| } |
| |
| private static final class BuildConfigurationFunctionException extends SkyFunctionException { |
| public BuildConfigurationFunctionException(InvalidConfigurationException e) { |
| super(e, Transience.PERSISTENT); |
| } |
| } |
| } |