blob: dad0929812c306530e10376339ae9c60e25dbbd7 [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.cpp;
import static com.google.common.truth.Truth.assertThat;
import com.google.common.base.Predicate;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Iterables;
import com.google.devtools.build.lib.actions.Artifact;
import com.google.devtools.build.lib.analysis.util.AnalysisMock;
import com.google.devtools.build.lib.analysis.util.BuildViewTestCase;
import com.google.devtools.build.lib.packages.util.MockCcSupport;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
/** Tests that {@code CppCompileAction} is populated with the correct build variables. */
@RunWith(JUnit4.class)
public class CompileBuildVariablesTest extends BuildViewTestCase {
private CppCompileAction getCppCompileAction(final String label, final String name) throws
Exception {
return (CppCompileAction)
getGeneratingAction(
Iterables.find(
getGeneratingAction(
Iterables.getOnlyElement(getFilesToBuild(getConfiguredTarget(label))))
.getInputs(),
new Predicate<Artifact>() {
@Override
public boolean apply(Artifact artifact) {
return artifact.getExecPath().getBaseName().startsWith(name);
}
}));
}
/** Returns active build variables for a compile action of given type for given target. */
protected CcToolchainVariables getCompileBuildVariables(String label, String name)
throws Exception {
return getCppCompileAction(label, name).getCompileCommandLine().getVariables();
}
@Test
public void testPresenceOfBasicVariables() throws Exception {
scratch.file("x/BUILD", "cc_binary(name = 'bin', srcs = ['bin.cc'])");
scratch.file("x/bin.cc");
CcToolchainVariables variables = getCompileBuildVariables("//x:bin", "bin");
assertThat(variables.getStringVariable(CompileBuildVariables.SOURCE_FILE.getVariableName()))
.contains("x/bin.cc");
assertThat(variables.getStringVariable(CompileBuildVariables.OUTPUT_FILE.getVariableName()))
.contains("_objs/bin/bin");
}
@Test
public void testPresenceOfLegacyCompileFlags() throws Exception {
AnalysisMock.get().ccSupport().setupCrosstool(mockToolsConfig, "cxx_flag: '-foo'");
useConfiguration();
scratch.file("x/BUILD", "cc_binary(name = 'bin', srcs = ['bin.cc'])");
scratch.file("x/bin.cc");
CcToolchainVariables variables = getCompileBuildVariables("//x:bin", "bin");
ImmutableList<String> copts =
CcToolchainVariables.toStringList(
variables, CompileBuildVariables.LEGACY_COMPILE_FLAGS.getVariableName());
assertThat(copts).contains("-foo");
}
@Test
public void testPresenceOfConfigurationCompileFlags() throws Exception {
AnalysisMock.get().ccSupport().setupCrosstool(mockToolsConfig);
useConfiguration("--copt=-foo");
scratch.file("x/BUILD", "cc_binary(name = 'bin', srcs = ['bin.cc'], copts = ['-bar'],)");
scratch.file("x/bin.cc");
CcToolchainVariables variables = getCompileBuildVariables("//x:bin", "bin");
ImmutableList<String> userCopts =
CcToolchainVariables.toStringList(
variables, CompileBuildVariables.USER_COMPILE_FLAGS.getVariableName());
assertThat(userCopts).containsAllIn(ImmutableList.<String>of("-foo", "-bar")).inOrder();
ImmutableList<String> legacyCopts =
CcToolchainVariables.toStringList(
variables, CompileBuildVariables.LEGACY_COMPILE_FLAGS.getVariableName());
assertThat(legacyCopts).doesNotContain("-foo");
}
@Test
public void testPresenceOfUserCompileFlags() throws Exception {
AnalysisMock.get().ccSupport().setupCrosstool(mockToolsConfig);
useConfiguration();
scratch.file("x/BUILD", "cc_binary(name = 'bin', srcs = ['bin.cc'], copts = ['-foo'])");
scratch.file("x/bin.cc");
CcToolchainVariables variables = getCompileBuildVariables("//x:bin", "bin");
ImmutableList<String> copts =
CcToolchainVariables.toStringList(
variables, CompileBuildVariables.USER_COMPILE_FLAGS.getVariableName());
assertThat(copts).contains("-foo");
}
@Test
public void testPresenceOfUnfilteredCompileFlags() throws Exception {
AnalysisMock.get()
.ccSupport()
.setupCrosstool(mockToolsConfig, "unfiltered_cxx_flag: '--i_ll_live_forever'");
useConfiguration();
scratch.file("x/BUILD", "cc_binary(name = 'bin', srcs = ['bin.cc'])");
scratch.file("x/bin.cc");
CcToolchainVariables variables = getCompileBuildVariables("//x:bin", "bin");
ImmutableList<String> unfilteredCompileFlags =
CcToolchainVariables.toStringList(
variables, CompileBuildVariables.UNFILTERED_COMPILE_FLAGS.getVariableName());
assertThat(unfilteredCompileFlags).contains("--i_ll_live_forever");
}
@Test
public void testPerFileCoptsAreInUserCompileFlags() throws Exception {
scratch.file("x/BUILD", "cc_binary(name = 'bin', srcs = ['bin.cc'])");
scratch.file("x/bin.cc");
useConfiguration("--per_file_copt=//x:bin@-foo", "--per_file_copt=//x:bar\\.cc@-bar");
CcToolchainVariables variables = getCompileBuildVariables("//x:bin", "bin");
ImmutableList<String> copts =
CcToolchainVariables.toStringList(
variables, CompileBuildVariables.USER_COMPILE_FLAGS.getVariableName());
assertThat(copts).containsExactly("-foo").inOrder();
}
@Test
public void testPresenceOfSysrootBuildVariable() throws Exception {
AnalysisMock.get()
.ccSupport()
.setupCrosstool(mockToolsConfig, "builtin_sysroot: '/usr/local/custom-sysroot'");
useConfiguration();
scratch.file("x/BUILD", "cc_binary(name = 'bin', srcs = ['bin.cc'])");
scratch.file("x/bin.cc");
CcToolchainVariables variables = getCompileBuildVariables("//x:bin", "bin");
assertThat(variables.getStringVariable(CcCommon.SYSROOT_VARIABLE_NAME))
.isEqualTo("/usr/local/custom-sysroot");
}
@Test
public void testPresenceOfPerObjectDebugFileBuildVariable() throws Exception {
AnalysisMock.get()
.ccSupport()
.setupCrosstool(mockToolsConfig, MockCcSupport.PER_OBJECT_DEBUG_INFO_CONFIGURATION);
useConfiguration("--fission=yes");
scratch.file("x/BUILD", "cc_binary(name = 'bin', srcs = ['bin.cc'])");
scratch.file("x/bin.cc");
CcToolchainVariables variables = getCompileBuildVariables("//x:bin", "bin");
assertThat(
variables.getStringVariable(
CompileBuildVariables.PER_OBJECT_DEBUG_INFO_FILE.getVariableName()))
.isNotNull();
}
@Test
public void testPresenceOfPerObjectDebugFileBuildVariableUsingLegacyFields() throws Exception {
AnalysisMock.get().ccSupport().setupCrosstool(mockToolsConfig, "supports_fission: true");
useConfiguration("--fission=yes");
scratch.file("x/BUILD", "cc_binary(name = 'bin', srcs = ['bin.cc'])");
scratch.file("x/bin.cc");
CcToolchainVariables variables = getCompileBuildVariables("//x:bin", "bin");
assertThat(
variables.getStringVariable(
CompileBuildVariables.PER_OBJECT_DEBUG_INFO_FILE.getVariableName()))
.isNotNull();
}
@Test
public void testPresenceOfMinOsVersionBuildVariable() throws Exception {
AnalysisMock.get()
.ccSupport()
.setupCrosstool(
mockToolsConfig,
"feature {"
+ " name: 'min_os_version_flag'"
+ " flag_set {"
+ " action: 'c++-compile'"
+ " flag_group {"
+ " expand_if_all_available: 'minimum_os_version'"
+ " flag: '-DMIN_OS=%{minimum_os_version}'"
+ " }"
+ " }"
+ "}");
useConfiguration("--minimum_os_version=6");
scratch.file("x/BUILD", "cc_binary(name = 'bin', srcs = ['bin.cc'])");
scratch.file("x/bin.cc");
CcToolchainVariables variables = getCompileBuildVariables("//x:bin", "bin");
assertThat(variables.getStringVariable(CcCommon.MINIMUM_OS_VERSION_VARIABLE_NAME))
.isEqualTo("6");
}
}