blob: 6a15ac4ce1f565e5398176cb28ed0f00da59b8d0 [file] [log] [blame]
// Copyright 2017 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.rules.java;
import com.google.common.collect.ImmutableMap;
import com.google.devtools.build.lib.actions.Actions;
import com.google.devtools.build.lib.actions.Artifact;
import com.google.devtools.build.lib.actions.MutableActionGraph.ActionConflictException;
import com.google.devtools.build.lib.analysis.CompilationHelper;
import com.google.devtools.build.lib.analysis.ConfiguredTarget;
import com.google.devtools.build.lib.analysis.PrerequisiteArtifacts;
import com.google.devtools.build.lib.analysis.RuleConfiguredTargetBuilder;
import com.google.devtools.build.lib.analysis.RuleConfiguredTargetFactory;
import com.google.devtools.build.lib.analysis.RuleContext;
import com.google.devtools.build.lib.analysis.Runfiles;
import com.google.devtools.build.lib.analysis.RunfilesProvider;
import com.google.devtools.build.lib.analysis.TemplateVariableInfo;
import com.google.devtools.build.lib.analysis.configuredtargets.RuleConfiguredTarget.Mode;
import com.google.devtools.build.lib.cmdline.Label;
import com.google.devtools.build.lib.collect.nestedset.NestedSet;
import com.google.devtools.build.lib.collect.nestedset.NestedSetBuilder;
import com.google.devtools.build.lib.packages.RuleClass.ConfiguredTargetFactory.RuleErrorException;
import com.google.devtools.build.lib.util.OsUtils;
import com.google.devtools.build.lib.vfs.PathFragment;
/** Implementation for the {@code java_runtime} rule. */
public class JavaRuntime implements RuleConfiguredTargetFactory {
// TODO(lberki): This is incorrect but that what the Jvm configuration fragment did. We'd have the
// the ability to do better if we knew what OS the BuildConfiguration refers to.
private static final String BIN_JAVA = "bin/java" + OsUtils.executableExtension();
@Override
public ConfiguredTarget create(RuleContext ruleContext)
throws InterruptedException, RuleErrorException, ActionConflictException {
JavaCommon.checkRuleLoadedThroughMacro(ruleContext);
NestedSetBuilder<Artifact> filesBuilder = NestedSetBuilder.stableOrder();
filesBuilder.addTransitive(PrerequisiteArtifacts.nestedSet(ruleContext, "srcs", Mode.TARGET));
boolean siblingRepositoryLayout =
ruleContext
.getAnalysisEnvironment()
.getSkylarkSemantics()
.experimentalSiblingRepositoryLayout();
PathFragment javaHome = defaultJavaHome(ruleContext.getLabel(), siblingRepositoryLayout);
if (ruleContext.attributes().isAttributeValueExplicitlySpecified("java_home")) {
PathFragment javaHomeAttribute =
PathFragment.create(ruleContext.getExpander().expand("java_home"));
if (!filesBuilder.isEmpty() && javaHomeAttribute.isAbsolute()) {
ruleContext.ruleError("'java_home' with an absolute path requires 'srcs' to be empty.");
}
if (ruleContext.hasErrors()) {
return null;
}
javaHome = javaHome.getRelative(javaHomeAttribute);
}
PathFragment javaBinaryExecPath = javaHome.getRelative(BIN_JAVA);
PathFragment javaBinaryRunfilesPath =
getRunfilesJavaExecutable(javaHome, ruleContext.getLabel());
Artifact java = ruleContext.getPrerequisiteArtifact("java", Mode.TARGET);
if (java != null) {
if (javaHome.isAbsolute()) {
ruleContext.ruleError("'java_home' with an absolute path requires 'java' to be empty.");
}
javaBinaryExecPath = java.getExecPath();
javaBinaryRunfilesPath = java.getRunfilesPath();
if (!isJavaBinary(javaBinaryExecPath)) {
ruleContext.ruleError("the path to 'java' must end in 'bin/java'.");
}
if (ruleContext.hasErrors()) {
return null;
}
javaHome = javaBinaryExecPath.getParentDirectory().getParentDirectory();
filesBuilder.add(java);
}
PathFragment javaHomeRunfilesPath =
javaBinaryRunfilesPath.getParentDirectory().getParentDirectory();
NestedSet<Artifact> filesToBuild = filesBuilder.build();
NestedSet<Artifact> middleman =
CompilationHelper.getAggregatingMiddleman(
ruleContext, Actions.escapeLabel(ruleContext.getLabel()), filesToBuild);
// TODO(cushon): clean up uses of java_runtime in data deps and remove this
Runfiles runfiles =
new Runfiles.Builder(ruleContext.getWorkspaceName())
.addTransitiveArtifacts(filesToBuild)
.build();
JavaRuntimeInfo javaRuntime =
JavaRuntimeInfo.create(
filesToBuild,
middleman,
javaHome,
javaBinaryExecPath,
javaHomeRunfilesPath,
javaBinaryRunfilesPath);
TemplateVariableInfo templateVariableInfo =
new TemplateVariableInfo(
ImmutableMap.of(
"JAVA", javaBinaryExecPath.getPathString(),
"JAVABASE", javaHome.getPathString()),
ruleContext.getRule().getLocation());
return new RuleConfiguredTargetBuilder(ruleContext)
.addProvider(RunfilesProvider.class, RunfilesProvider.simple(runfiles))
.setFilesToBuild(filesToBuild)
.addNativeDeclaredProvider(javaRuntime)
.addNativeDeclaredProvider(templateVariableInfo)
.build();
}
private static boolean isJavaBinary(PathFragment path) {
return path.endsWith(PathFragment.create("bin/java"))
|| path.endsWith(PathFragment.create("bin/java.exe"));
}
static PathFragment defaultJavaHome(Label javabase, boolean siblingRepositoryLayout) {
if (javabase.getPackageIdentifier().getRepository().isDefault()) {
return javabase.getPackageFragment();
}
return javabase.getPackageIdentifier().getExecPath(siblingRepositoryLayout);
}
private static PathFragment getRunfilesJavaExecutable(PathFragment javaHome, Label javabase) {
if (javaHome.isAbsolute() || javabase.getPackageIdentifier().getRepository().isMain()) {
return javaHome.getRelative(BIN_JAVA);
} else {
return javabase
.getPackageIdentifier()
.getRepository()
.getRunfilesPath()
.getRelative(BIN_JAVA);
}
}
}