blob: 8a17b980d2db70d068e66cf0df6e055fafe8ce25 [file] [log] [blame]
// 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.auto.value.AutoValue;
import com.google.common.base.Throwables;
import com.google.common.cache.CacheBuilder;
import com.google.common.cache.CacheLoader;
import com.google.common.cache.LoadingCache;
import com.google.common.collect.ClassToInstanceMap;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.ImmutableSortedSet;
import com.google.common.collect.MutableClassToInstanceMap;
import com.google.devtools.build.lib.actions.ActionEnvironment;
import com.google.devtools.build.lib.analysis.BlazeDirectories;
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.BuildOptions;
import com.google.devtools.build.lib.analysis.config.ConfigurationFragmentFactory;
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 java.util.Set;
import java.util.concurrent.ExecutionException;
/**
* A builder for {@link BuildConfigurationValue} instances.
*/
public class BuildConfigurationFunction implements SkyFunction {
/** Cache with weak values can't have null values. */
private static final Fragment NULL_MARKER = new Fragment() {};
private final BlazeDirectories directories;
private final ConfiguredRuleClassProvider ruleClassProvider;
private final BuildOptions defaultBuildOptions;
private final LoadingCache<FragmentKey, Fragment> fragmentCache =
CacheBuilder.newBuilder()
.weakValues()
.build(
new CacheLoader<FragmentKey, Fragment>() {
@Override
public Fragment load(FragmentKey key) throws InvalidConfigurationException {
return makeFragment(key);
}
});
public BuildConfigurationFunction(
BlazeDirectories directories,
RuleClassProvider ruleClassProvider,
BuildOptions defaultBuildOptions) {
this.directories = directories;
this.ruleClassProvider = (ConfiguredRuleClassProvider) ruleClassProvider;
this.defaultBuildOptions = defaultBuildOptions;
}
@Override
public SkyValue compute(SkyKey skyKey, Environment env)
throws InterruptedException, BuildConfigurationFunctionException {
WorkspaceNameValue workspaceNameValue = (WorkspaceNameValue) env
.getValue(WorkspaceNameValue.key());
if (workspaceNameValue == null) {
return null;
}
BuildConfigurationValue.Key key = (BuildConfigurationValue.Key) skyKey.argument();
Set<Fragment> fragments;
try {
fragments = getConfigurationFragments(key);
} 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);
}
BuildOptions options = defaultBuildOptions.applyDiff(key.getOptionsDiff());
ActionEnvironment actionEnvironment =
ruleClassProvider.getActionEnvironmentProvider().getActionEnvironment(options);
BuildConfiguration config =
new BuildConfiguration(
directories,
fragmentsMap,
options,
key.getOptionsDiff(),
ruleClassProvider.getReservedActionMnemonics(),
actionEnvironment,
workspaceNameValue.getName());
return new BuildConfigurationValue(config);
}
private Set<Fragment> getConfigurationFragments(BuildConfigurationValue.Key key)
throws InvalidConfigurationException {
BuildOptions options = defaultBuildOptions.applyDiff(key.getOptionsDiff());
ImmutableSortedSet<Class<? extends Fragment>> fragmentClasses = key.getFragments();
ImmutableSet.Builder<Fragment> fragments =
ImmutableSet.builderWithExpectedSize(fragmentClasses.size());
for (Class<? extends BuildConfiguration.Fragment> fragmentClass : fragmentClasses) {
BuildOptions trimmedOptions =
options.trim(
BuildConfiguration.getOptionsClasses(
ImmutableList.of(fragmentClass), ruleClassProvider));
Fragment fragment;
FragmentKey fragmentKey = FragmentKey.create(trimmedOptions, fragmentClass);
try {
fragment = fragmentCache.get(fragmentKey);
} catch (ExecutionException e) {
Throwables.propagateIfPossible(e.getCause(), InvalidConfigurationException.class);
throw new IllegalStateException(e);
}
if (fragment != NULL_MARKER) {
fragments.add(fragment);
} else {
// NULL_MARKER is never GC'ed, so this entry will stay in cache forever unless we delete it
// ourselves. Since it's a cheap computation we don't care about recomputing it.
fragmentCache.invalidate(fragmentKey);
}
}
return fragments.build();
}
@AutoValue
abstract static class FragmentKey {
abstract BuildOptions getBuildOptions();
abstract Class<? extends BuildConfiguration.Fragment> getFragmentClass();
private static FragmentKey create(
BuildOptions buildOptions, Class<? extends BuildConfiguration.Fragment> fragmentClass) {
return new AutoValue_BuildConfigurationFunction_FragmentKey(buildOptions, fragmentClass);
}
}
private Fragment makeFragment(FragmentKey fragmentKey) throws InvalidConfigurationException {
BuildOptions buildOptions = fragmentKey.getBuildOptions();
ConfigurationFragmentFactory factory = getFactory(fragmentKey.getFragmentClass());
Fragment fragment = factory.create(buildOptions);
return fragment != null ? fragment : NULL_MARKER;
}
private ConfigurationFragmentFactory getFactory(Class<? extends Fragment> fragmentType) {
for (ConfigurationFragmentFactory factory : ruleClassProvider.getConfigurationFragments()) {
if (factory.creates().equals(fragmentType)) {
return factory;
}
}
throw new IllegalStateException(
"There is no factory for fragment: " + fragmentType.getSimpleName());
}
@Override
public String extractTag(SkyKey skyKey) {
return null;
}
private static final class BuildConfigurationFunctionException extends SkyFunctionException {
public BuildConfigurationFunctionException(InvalidConfigurationException e) {
super(e, Transience.PERSISTENT);
}
}
}