blob: 8c0193f189018588f70cec8992f4439697e96eed [file] [log] [blame]
// 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.standalone;
import com.google.common.collect.ImmutableList;
import com.google.devtools.build.lib.actions.SpawnStrategy;
import com.google.devtools.build.lib.analysis.actions.FileWriteActionContext;
import com.google.devtools.build.lib.analysis.actions.LocalTemplateExpansionStrategy;
import com.google.devtools.build.lib.analysis.actions.TemplateExpansionContext;
import com.google.devtools.build.lib.analysis.test.TestActionContext;
import com.google.devtools.build.lib.buildtool.BuildRequest;
import com.google.devtools.build.lib.dynamic.DynamicExecutionOptions;
import com.google.devtools.build.lib.exec.ExecutionOptions;
import com.google.devtools.build.lib.exec.ExecutorBuilder;
import com.google.devtools.build.lib.exec.FileWriteStrategy;
import com.google.devtools.build.lib.exec.RunfilesTreeUpdater;
import com.google.devtools.build.lib.exec.SpawnRunner;
import com.google.devtools.build.lib.exec.StandaloneTestStrategy;
import com.google.devtools.build.lib.exec.TestStrategy;
import com.google.devtools.build.lib.exec.local.LocalEnvProvider;
import com.google.devtools.build.lib.exec.local.LocalExecutionOptions;
import com.google.devtools.build.lib.exec.local.LocalSpawnRunner;
import com.google.devtools.build.lib.rules.cpp.CppIncludeExtractionContext;
import com.google.devtools.build.lib.rules.cpp.CppIncludeScanningContext;
import com.google.devtools.build.lib.rules.test.ExclusiveTestStrategy;
import com.google.devtools.build.lib.runtime.BlazeModule;
import com.google.devtools.build.lib.runtime.CommandEnvironment;
import com.google.devtools.build.lib.util.AbruptExitException;
import com.google.devtools.build.lib.util.ExitCode;
import com.google.devtools.build.lib.vfs.Path;
/**
* StandaloneModule provides pluggable functionality for blaze.
*/
public class StandaloneModule extends BlazeModule {
@Override
public void beforeCommand(CommandEnvironment env) throws AbruptExitException {
LocalExecutionOptions localOptions = env.getOptions().getOptions(LocalExecutionOptions.class);
if (localOptions == null) {
// This module doesn't make sense in non-build commands (which don't register these options).
return;
}
DynamicExecutionOptions dynamicOptions =
env.getOptions().getOptions(DynamicExecutionOptions.class);
if (dynamicOptions != null) { // Guard against tests that don't pull this module in.
if (localOptions.localLockfreeOutput && dynamicOptions.legacySpawnScheduler) {
throw new AbruptExitException(
"--experimental_local_lockfree_output requires --nolegacy_spawn_scheduler",
ExitCode.COMMAND_LINE_ERROR);
}
}
}
@Override
public void executorInit(CommandEnvironment env, BuildRequest request, ExecutorBuilder builder) {
// TODO(ulfjack): Move this to another module.
builder.addActionContext(
CppIncludeExtractionContext.class, new DummyCppIncludeExtractionContext(env));
builder.addActionContext(CppIncludeScanningContext.class, new DummyCppIncludeScanningContext());
ExecutionOptions executionOptions = env.getOptions().getOptions(ExecutionOptions.class);
Path testTmpRoot =
TestStrategy.getTmpRoot(env.getWorkspace(), env.getExecRoot(), executionOptions);
TestActionContext testStrategy =
new StandaloneTestStrategy(
executionOptions,
env.getBlazeWorkspace().getBinTools(),
testTmpRoot);
SpawnRunner localSpawnRunner =
new LocalSpawnRunner(
env.getExecRoot(),
env.getOptions().getOptions(LocalExecutionOptions.class),
env.getLocalResourceManager(),
LocalEnvProvider.forCurrentOs(env.getClientEnv()),
env.getBlazeWorkspace().getBinTools(),
// TODO(buchgr): Replace singleton by a command-scoped RunfilesTreeUpdater
RunfilesTreeUpdater.INSTANCE);
// Order of strategies passed to builder is significant - when there are many strategies that
// could potentially be used and a spawnActionContext doesn't specify which one it wants, the
// last one from strategies list will be used
builder.addActionContext(
SpawnStrategy.class,
new StandaloneSpawnStrategy(env.getExecRoot(), localSpawnRunner),
"standalone",
"local");
builder.addActionContext(TestActionContext.class, testStrategy, "standalone");
builder.addActionContext(
TestActionContext.class, new ExclusiveTestStrategy(testStrategy), "exclusive");
builder.addActionContext(FileWriteActionContext.class, new FileWriteStrategy(), "local");
builder.addActionContext(
TemplateExpansionContext.class, new LocalTemplateExpansionStrategy(), "local");
// This makes the "sandboxed" strategy the default Spawn strategy, unless it is overridden by a
// later BlazeModule.
builder.addStrategyByMnemonic("", ImmutableList.of("standalone"));
// This makes the "standalone" strategy available via --spawn_strategy=standalone, but it is not
// necessarily the default.
builder.addStrategyByContext(SpawnStrategy.class, "standalone");
}
}