blob: d6554fddbd0254024e6ffd138f6eee123e0becf9 [file] [log] [blame]
// Copyright (c) 2013 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "gn/target.h"
#include <memory>
#include <utility>
#include "gn/build_settings.h"
#include "gn/config.h"
#include "gn/resolved_target_data.h"
#include "gn/scheduler.h"
#include "gn/settings.h"
#include "gn/test_with_scheduler.h"
#include "gn/test_with_scope.h"
#include "gn/toolchain.h"
#include "util/test/test.h"
namespace {
// Asserts that the current global scheduler has a single unknown generated
// file with the given name from the given target.
void AssertSchedulerHasOneUnknownFileMatching(const Target* target,
const SourceFile& file) {
auto unknown = g_scheduler->GetUnknownGeneratedInputs();
ASSERT_EQ(1u, unknown.size()); // Should be one unknown file.
auto found = unknown.find(file);
ASSERT_TRUE(found != unknown.end()) << file.value();
EXPECT_TRUE(target == found->second)
<< "Target doesn't match. Expected\n "
<< target->label().GetUserVisibleName(false) << "\nBut got\n "
<< found->second->label().GetUserVisibleName(false);
}
} // namespace
using TargetTest = TestWithScheduler;
// Test all_dependent_configs and public_config inheritance.
TEST_F(TargetTest, DependentConfigs) {
TestWithScope setup;
Err err;
// Set up a dependency chain of a -> b -> c
TestTarget a(setup, "//foo:a", Target::EXECUTABLE);
TestTarget b(setup, "//foo:b", Target::STATIC_LIBRARY);
TestTarget c(setup, "//foo:c", Target::STATIC_LIBRARY);
a.private_deps().push_back(LabelTargetPair(&b));
b.private_deps().push_back(LabelTargetPair(&c));
// Normal non-inherited config.
Config config(setup.settings(), Label(SourceDir("//foo/"), "config"));
config.visibility().SetPublic();
ASSERT_TRUE(config.OnResolved(&err));
c.configs().push_back(LabelConfigPair(&config));
// All dependent config.
Config all(setup.settings(), Label(SourceDir("//foo/"), "all"));
all.visibility().SetPublic();
ASSERT_TRUE(all.OnResolved(&err));
c.all_dependent_configs().push_back(LabelConfigPair(&all));
// Direct dependent config.
Config direct(setup.settings(), Label(SourceDir("//foo/"), "direct"));
direct.visibility().SetPublic();
ASSERT_TRUE(direct.OnResolved(&err));
c.public_configs().push_back(LabelConfigPair(&direct));
ASSERT_TRUE(c.OnResolved(&err));
ASSERT_TRUE(b.OnResolved(&err));
ASSERT_TRUE(a.OnResolved(&err));
// B should have gotten both dependent configs from C.
ASSERT_EQ(2u, b.configs().size());
EXPECT_EQ(&all, b.configs()[0].ptr);
EXPECT_EQ(&direct, b.configs()[1].ptr);
ASSERT_EQ(1u, b.all_dependent_configs().size());
EXPECT_EQ(&all, b.all_dependent_configs()[0].ptr);
// A should have just gotten the "all" dependent config from C.
ASSERT_EQ(1u, a.configs().size());
EXPECT_EQ(&all, a.configs()[0].ptr);
EXPECT_EQ(&all, a.all_dependent_configs()[0].ptr);
// Making an an alternate A and B with B forwarding the direct dependents.
TestTarget a_fwd(setup, "//foo:a_fwd", Target::EXECUTABLE);
TestTarget b_fwd(setup, "//foo:b_fwd", Target::STATIC_LIBRARY);
a_fwd.private_deps().push_back(LabelTargetPair(&b_fwd));
b_fwd.private_deps().push_back(LabelTargetPair(&c));
ASSERT_TRUE(b_fwd.OnResolved(&err));
ASSERT_TRUE(a_fwd.OnResolved(&err));
// A_fwd should now have both configs.
ASSERT_EQ(1u, a_fwd.configs().size());
EXPECT_EQ(&all, a_fwd.configs()[0].ptr);
ASSERT_EQ(1u, a_fwd.all_dependent_configs().size());
EXPECT_EQ(&all, a_fwd.all_dependent_configs()[0].ptr);
}
// Tests that dependent configs don't propagate between toolchains.
TEST_F(TargetTest, NoDependentConfigsBetweenToolchains) {
TestWithScope setup;
Err err;
// Create another toolchain.
Toolchain other_toolchain(setup.settings(),
Label(SourceDir("//other/"), "toolchain"));
TestWithScope::SetupToolchain(&other_toolchain);
// Set up a dependency chain of |a| -> |b| -> |c| where |a| has a different
// toolchain.
Target a(setup.settings(),
Label(SourceDir("//foo/"), "a", other_toolchain.label().dir(),
other_toolchain.label().name()));
a.set_output_type(Target::EXECUTABLE);
EXPECT_TRUE(a.SetToolchain(&other_toolchain, &err));
TestTarget b(setup, "//foo:b", Target::EXECUTABLE);
TestTarget c(setup, "//foo:c", Target::SOURCE_SET);
a.private_deps().push_back(LabelTargetPair(&b));
b.private_deps().push_back(LabelTargetPair(&c));
// All dependent config.
Config all_dependent(setup.settings(), Label(SourceDir("//foo/"), "all"));
all_dependent.visibility().SetPublic();
ASSERT_TRUE(all_dependent.OnResolved(&err));
c.all_dependent_configs().push_back(LabelConfigPair(&all_dependent));
// Public config.
Config public_config(setup.settings(), Label(SourceDir("//foo/"), "public"));
public_config.visibility().SetPublic();
ASSERT_TRUE(public_config.OnResolved(&err));
c.public_configs().push_back(LabelConfigPair(&public_config));
// Another public config.
Config public_config2(setup.settings(),
Label(SourceDir("//foo/"), "public2"));
public_config2.visibility().SetPublic();
ASSERT_TRUE(public_config2.OnResolved(&err));
b.public_configs().push_back(LabelConfigPair(&public_config2));
ASSERT_TRUE(c.OnResolved(&err));
ASSERT_TRUE(b.OnResolved(&err));
ASSERT_TRUE(a.OnResolved(&err));
// B should have gotten the configs from C.
ASSERT_EQ(3u, b.configs().size());
EXPECT_EQ(&public_config2, b.configs()[0].ptr);
EXPECT_EQ(&all_dependent, b.configs()[1].ptr);
EXPECT_EQ(&public_config, b.configs()[2].ptr);
ASSERT_EQ(1u, b.all_dependent_configs().size());
EXPECT_EQ(&all_dependent, b.all_dependent_configs()[0].ptr);
// A should not have gotten any configs from B or C.
ASSERT_EQ(0u, a.configs().size());
ASSERT_EQ(0u, a.all_dependent_configs().size());
}
// Tests that dependent configs propagate between toolchains if
// propagates_configs is set.
TEST_F(TargetTest, DependentConfigsBetweenToolchainsWhenSet) {
TestWithScope setup;
Err err;
// Create another toolchain.
Toolchain other_toolchain(setup.settings(),
Label(SourceDir("//other/"), "toolchain"));
TestWithScope::SetupToolchain(&other_toolchain);
other_toolchain.set_propagates_configs(true);
// Set up a dependency chain of |a| -> |b| where |b| has a different
// toolchain (with propagate_configs set).
TestTarget a(setup, "//foo:a", Target::EXECUTABLE);
Target b(setup.settings(),
Label(SourceDir("//foo/"), "b", other_toolchain.label().dir(),
other_toolchain.label().name()));
b.visibility().SetPublic();
b.set_output_type(Target::SHARED_LIBRARY);
EXPECT_TRUE(b.SetToolchain(&other_toolchain, &err));
a.private_deps().push_back(LabelTargetPair(&b));
// All dependent config.
Config all_dependent(setup.settings(), Label(SourceDir("//foo/"), "all"));
all_dependent.visibility().SetPublic();
ASSERT_TRUE(all_dependent.OnResolved(&err));
b.all_dependent_configs().push_back(LabelConfigPair(&all_dependent));
// Public config.
Config public_config(setup.settings(), Label(SourceDir("//foo/"), "public"));
public_config.visibility().SetPublic();
ASSERT_TRUE(public_config.OnResolved(&err));
b.public_configs().push_back(LabelConfigPair(&public_config));
ASSERT_TRUE(b.OnResolved(&err));
ASSERT_TRUE(a.OnResolved(&err));
// A should have gotten the configs from B.
ASSERT_EQ(2u, a.configs().size());
EXPECT_EQ(&all_dependent, a.configs()[0].ptr);
EXPECT_EQ(&public_config, a.configs()[1].ptr);
ASSERT_EQ(1u, a.all_dependent_configs().size());
EXPECT_EQ(&all_dependent, a.all_dependent_configs()[0].ptr);
}
TEST_F(TargetTest, GetComputedOutputName) {
TestWithScope setup;
Err err;
// Basic target with no prefix (executable type tool in the TestWithScope has
// no prefix) or output name.
TestTarget basic(setup, "//foo:bar", Target::EXECUTABLE);
ASSERT_TRUE(basic.OnResolved(&err));
EXPECT_EQ("bar", basic.GetComputedOutputName());
// Target with no prefix but an output name.
TestTarget with_name(setup, "//foo:bar", Target::EXECUTABLE);
with_name.set_output_name("myoutput");
ASSERT_TRUE(with_name.OnResolved(&err));
EXPECT_EQ("myoutput", with_name.GetComputedOutputName());
// Target with a "lib" prefix (the static library tool in the TestWithScope
// should specify a "lib" output prefix).
TestTarget with_prefix(setup, "//foo:bar", Target::STATIC_LIBRARY);
ASSERT_TRUE(with_prefix.OnResolved(&err));
EXPECT_EQ("libbar", with_prefix.GetComputedOutputName());
// Target with a "lib" prefix that already has it applied. The prefix should
// not duplicate something already in the target name.
TestTarget dup_prefix(setup, "//foo:bar", Target::STATIC_LIBRARY);
dup_prefix.set_output_name("libbar");
ASSERT_TRUE(dup_prefix.OnResolved(&err));
EXPECT_EQ("libbar", dup_prefix.GetComputedOutputName());
// Target with an output prefix override should not have a prefix.
TestTarget override_prefix(setup, "//foo:bar", Target::SHARED_LIBRARY);
override_prefix.set_output_prefix_override(true);
ASSERT_TRUE(dup_prefix.OnResolved(&err));
EXPECT_EQ("bar", override_prefix.GetComputedOutputName());
}
// Test visibility failure case.
TEST_F(TargetTest, VisibilityFails) {
TestWithScope setup;
Err err;
TestTarget b(setup, "//private:b", Target::STATIC_LIBRARY);
b.visibility().SetPrivate(b.label().dir());
ASSERT_TRUE(b.OnResolved(&err));
// Make a target depending on "b". The dependency must have an origin to mark
// it as user-set so we check visibility. This check should fail.
TestTarget a(setup, "//app:a", Target::EXECUTABLE);
a.private_deps().push_back(LabelTargetPair(&b));
IdentifierNode origin; // Dummy origin.
a.private_deps()[0].origin = &origin;
ASSERT_FALSE(a.OnResolved(&err));
}
// Test config visibility failure cases.
TEST_F(TargetTest, VisibilityConfigFails) {
TestWithScope setup;
Err err;
Label config_label(SourceDir("//a/"), "config");
Config config(setup.settings(), config_label);
config.visibility().SetPrivate(config.label().dir());
ASSERT_TRUE(config.OnResolved(&err));
// Make a target using configs. This should fail.
TestTarget a(setup, "//app:a", Target::EXECUTABLE);
a.configs().push_back(LabelConfigPair(&config));
ASSERT_FALSE(a.OnResolved(&err));
// A target using public_configs should also fail.
TestTarget b(setup, "//app:b", Target::EXECUTABLE);
b.public_configs().push_back(LabelConfigPair(&config));
ASSERT_FALSE(b.OnResolved(&err));
// A target using all_dependent_configs should fail as well.
TestTarget c(setup, "//app:c", Target::EXECUTABLE);
c.all_dependent_configs().push_back(LabelConfigPair(&config));
ASSERT_FALSE(c.OnResolved(&err));
}
// Test Config -> Group -> A where the config is group is visible from A but
// the config isn't, and the config is visible from the group.
TEST_F(TargetTest, VisibilityConfigGroup) {
TestWithScope setup;
Err err;
Label config_label(SourceDir("//a/"), "config");
Config config(setup.settings(), config_label);
config.visibility().SetPrivate(config.label().dir());
ASSERT_TRUE(config.OnResolved(&err));
// Make a target using the config in the same directory.
TestTarget a(setup, "//a:a", Target::GROUP);
a.public_configs().push_back(LabelConfigPair(&config));
ASSERT_TRUE(a.OnResolved(&err));
// A target depending on a should be okay.
TestTarget b(setup, "//app:b", Target::EXECUTABLE);
b.private_deps().push_back(LabelTargetPair(&a));
ASSERT_TRUE(b.OnResolved(&err));
}
// Test visibility with a single data_dep.
TEST_F(TargetTest, VisibilityDatadeps) {
TestWithScope setup;
Err err;
TestTarget b(setup, "//public:b", Target::STATIC_LIBRARY);
ASSERT_TRUE(b.OnResolved(&err));
// Make a target depending on "b". The dependency must have an origin to mark
// it as user-set so we check visibility. This check should fail.
TestTarget a(setup, "//app:a", Target::EXECUTABLE);
a.data_deps().push_back(LabelTargetPair(&b));
IdentifierNode origin; // Dummy origin.
a.data_deps()[0].origin = &origin;
ASSERT_TRUE(a.OnResolved(&err)) << err.help_text();
}
// Tests that A -> Group -> B where the group is visible from A but B isn't,
// passes visibility even though the group's deps get expanded into A.
TEST_F(TargetTest, VisibilityGroup) {
TestWithScope setup;
Err err;
IdentifierNode origin; // Dummy origin.
// B has private visibility. This lets the group see it since the group is in
// the same directory.
TestTarget b(setup, "//private:b", Target::STATIC_LIBRARY);
b.visibility().SetPrivate(b.label().dir());
ASSERT_TRUE(b.OnResolved(&err));
// The group has public visibility and depends on b.
TestTarget g(setup, "//public:g", Target::GROUP);
g.private_deps().push_back(LabelTargetPair(&b));
g.private_deps()[0].origin = &origin;
ASSERT_TRUE(b.OnResolved(&err));
// Make a target depending on "g". This should succeed.
TestTarget a(setup, "//app:a", Target::EXECUTABLE);
a.private_deps().push_back(LabelTargetPair(&g));
a.private_deps()[0].origin = &origin;
ASSERT_TRUE(a.OnResolved(&err));
}
// Verifies that only testonly targets can depend on other testonly targets.
// Many of the above dependency checking cases covered the non-testonly
// case.
TEST_F(TargetTest, Testonly) {
TestWithScope setup;
Err err;
// "testlib" is a test-only library.
TestTarget testlib(setup, "//test:testlib", Target::STATIC_LIBRARY);
testlib.set_testonly(true);
ASSERT_TRUE(testlib.OnResolved(&err));
// "test" is a test-only executable depending on testlib, this is OK.
TestTarget test(setup, "//test:test", Target::EXECUTABLE);
test.set_testonly(true);
test.private_deps().push_back(LabelTargetPair(&testlib));
ASSERT_TRUE(test.OnResolved(&err));
// "product" is a non-test depending on testlib. This should fail.
TestTarget product(setup, "//app:product", Target::EXECUTABLE);
product.set_testonly(false);
product.private_deps().push_back(LabelTargetPair(&testlib));
ASSERT_FALSE(product.OnResolved(&err));
}
// Configs can be testonly too.
// Repeat the testonly test with a config.
TEST_F(TargetTest, TestonlyConfig) {
TestWithScope setup;
Err err;
// "testconfig" is a test-only config.
Config testconfig(setup.settings(), Label(SourceDir("//test/"), "config"));
testconfig.set_testonly(true);
testconfig.visibility().SetPublic();
ASSERT_TRUE(testconfig.OnResolved(&err));
// "test" is a test-only executable that uses testconfig, this is OK.
TestTarget test(setup, "//test:test", Target::EXECUTABLE);
test.set_testonly(true);
test.configs().push_back(LabelConfigPair(&testconfig));
ASSERT_TRUE(test.OnResolved(&err));
// "product" is a non-test that uses testconfig. This should fail.
TestTarget product(setup, "//app:product", Target::EXECUTABLE);
product.set_testonly(false);
product.configs().push_back(LabelConfigPair(&testconfig));
ASSERT_FALSE(product.OnResolved(&err));
}
TEST_F(TargetTest, PublicConfigs) {
TestWithScope setup;
Err err;
Label pub_config_label(SourceDir("//a/"), "pubconfig");
Config pub_config(setup.settings(), pub_config_label);
pub_config.visibility().SetPublic();
LibFile lib_name("testlib");
pub_config.own_values().libs().push_back(lib_name);
ASSERT_TRUE(pub_config.OnResolved(&err));
// This is the destination target that has a public config.
TestTarget dest(setup, "//a:a", Target::SOURCE_SET);
dest.public_configs().push_back(LabelConfigPair(&pub_config));
ASSERT_TRUE(dest.OnResolved(&err));
// This target has a public dependency on dest.
TestTarget pub(setup, "//a:pub", Target::SOURCE_SET);
pub.public_deps().push_back(LabelTargetPair(&dest));
ASSERT_TRUE(pub.OnResolved(&err));
// Depending on the target with the public dependency should forward dest's
// to the current target.
TestTarget dep_on_pub(setup, "//a:dop", Target::SOURCE_SET);
dep_on_pub.private_deps().push_back(LabelTargetPair(&pub));
ASSERT_TRUE(dep_on_pub.OnResolved(&err));
ASSERT_EQ(1u, dep_on_pub.configs().size());
EXPECT_EQ(&pub_config, dep_on_pub.configs()[0].ptr);
// Libs have special handling, check that they were forwarded from the
// public config to all_libs.
ResolvedTargetData resolved;
const auto& dep_on_pub_all_libs = resolved.GetLinkedLibraries(&dep_on_pub);
ASSERT_EQ(1u, dep_on_pub_all_libs.size());
ASSERT_EQ(lib_name, dep_on_pub_all_libs[0]);
// This target has a private dependency on dest for forwards configs.
TestTarget forward(setup, "//a:f", Target::SOURCE_SET);
forward.private_deps().push_back(LabelTargetPair(&dest));
ASSERT_TRUE(forward.OnResolved(&err));
}
// Tests that configs are ordered properly between local and pulled ones.
TEST_F(TargetTest, ConfigOrdering) {
TestWithScope setup;
Err err;
// Make Dep1. It has all_dependent_configs and public_configs.
TestTarget dep1(setup, "//:dep1", Target::SOURCE_SET);
Label dep1_all_config_label(SourceDir("//"), "dep1_all_config");
Config dep1_all_config(setup.settings(), dep1_all_config_label);
dep1_all_config.visibility().SetPublic();
ASSERT_TRUE(dep1_all_config.OnResolved(&err));
dep1.all_dependent_configs().push_back(LabelConfigPair(&dep1_all_config));
Label dep1_public_config_label(SourceDir("//"), "dep1_public_config");
Config dep1_public_config(setup.settings(), dep1_public_config_label);
dep1_public_config.visibility().SetPublic();
ASSERT_TRUE(dep1_public_config.OnResolved(&err));
dep1.public_configs().push_back(LabelConfigPair(&dep1_public_config));
ASSERT_TRUE(dep1.OnResolved(&err));
// Make Dep2 with the same structure.
TestTarget dep2(setup, "//:dep2", Target::SOURCE_SET);
Label dep2_all_config_label(SourceDir("//"), "dep2_all_config");
Config dep2_all_config(setup.settings(), dep2_all_config_label);
dep2_all_config.visibility().SetPublic();
ASSERT_TRUE(dep2_all_config.OnResolved(&err));
dep2.all_dependent_configs().push_back(LabelConfigPair(&dep2_all_config));
Label dep2_public_config_label(SourceDir("//"), "dep2_public_config");
Config dep2_public_config(setup.settings(), dep2_public_config_label);
dep2_public_config.visibility().SetPublic();
ASSERT_TRUE(dep2_public_config.OnResolved(&err));
dep2.public_configs().push_back(LabelConfigPair(&dep2_public_config));
ASSERT_TRUE(dep2.OnResolved(&err));
// This target depends on both previous targets.
TestTarget target(setup, "//:foo", Target::SOURCE_SET);
target.private_deps().push_back(LabelTargetPair(&dep1));
target.private_deps().push_back(LabelTargetPair(&dep2));
// It also has a private and public config.
Label public_config_label(SourceDir("//"), "public");
Config public_config(setup.settings(), public_config_label);
public_config.visibility().SetPublic();
ASSERT_TRUE(public_config.OnResolved(&err));
target.public_configs().push_back(LabelConfigPair(&public_config));
Label private_config_label(SourceDir("//"), "private");
Config private_config(setup.settings(), private_config_label);
private_config.visibility().SetPublic();
ASSERT_TRUE(private_config.OnResolved(&err));
target.configs().push_back(LabelConfigPair(&private_config));
// Resolve to get the computed list of configs applying.
ASSERT_TRUE(target.OnResolved(&err));
const auto& computed = target.configs();
// Order should be:
// 1. local private
// 2. local public
// 3. inherited all dependent
// 4. inherited public
ASSERT_EQ(6u, computed.size());
EXPECT_EQ(private_config_label, computed[0].label);
EXPECT_EQ(public_config_label, computed[1].label);
EXPECT_EQ(dep1_all_config_label, computed[2].label);
EXPECT_EQ(dep2_all_config_label, computed[3].label);
EXPECT_EQ(dep1_public_config_label, computed[4].label);
EXPECT_EQ(dep2_public_config_label, computed[5].label);
}
// Tests that different link/depend outputs work for solink tools.
TEST_F(TargetTest, LinkAndDepOutputs) {
TestWithScope setup;
Err err;
Toolchain toolchain(setup.settings(), Label(SourceDir("//tc/"), "tc"));
std::unique_ptr<Tool> solink = Tool::CreateTool(CTool::kCToolSolink);
CTool* solink_tool = solink->AsC();
solink_tool->set_output_prefix("lib");
solink_tool->set_default_output_extension(".so");
const char kLinkPattern[] =
"{{root_out_dir}}/{{target_output_name}}{{output_extension}}";
SubstitutionPattern link_output =
SubstitutionPattern::MakeForTest(kLinkPattern);
solink_tool->set_link_output(link_output);
const char kDependPattern[] =
"{{root_out_dir}}/{{target_output_name}}{{output_extension}}.TOC";
SubstitutionPattern depend_output =
SubstitutionPattern::MakeForTest(kDependPattern);
solink_tool->set_depend_output(depend_output);
solink_tool->set_outputs(
SubstitutionList::MakeForTest(kLinkPattern, kDependPattern));
toolchain.SetTool(std::move(solink));
Target target(setup.settings(), Label(SourceDir("//a/"), "a"));
target.set_output_type(Target::SHARED_LIBRARY);
target.SetToolchain(&toolchain);
ASSERT_TRUE(target.OnResolved(&err));
EXPECT_EQ("./liba.so", target.link_output_file().value());
EXPECT_EQ("./liba.so.TOC", target.dependency_output_file().value());
ASSERT_EQ(1u, target.runtime_outputs().size());
EXPECT_EQ("./liba.so", target.runtime_outputs()[0].value());
}
// Tests that runtime_outputs works without an explicit link_output for
// solink tools.
//
// Also tests GetOutputsAsSourceFiles() for binaries (the setup is the same).
TEST_F(TargetTest, RuntimeOuputs) {
TestWithScope setup;
Err err;
Toolchain toolchain(setup.settings(), Label(SourceDir("//tc/"), "tc"));
std::unique_ptr<Tool> solink = Tool::CreateTool(CTool::kCToolSolink);
CTool* solink_tool = solink->AsC();
solink_tool->set_output_prefix("");
solink_tool->set_default_output_extension(".dll");
// Say the linker makes a DLL< an import library, and a symbol file we want
// to treat as a runtime output.
const char kLibPattern[] =
"{{root_out_dir}}/{{target_output_name}}{{output_extension}}.lib";
const char kDllPattern[] =
"{{root_out_dir}}/{{target_output_name}}{{output_extension}}";
const char kPdbPattern[] = "{{root_out_dir}}/{{target_output_name}}.pdb";
SubstitutionPattern pdb_pattern =
SubstitutionPattern::MakeForTest(kPdbPattern);
solink_tool->set_outputs(
SubstitutionList::MakeForTest(kLibPattern, kDllPattern, kPdbPattern));
// Say we only want the DLL and symbol file treaded as runtime outputs.
solink_tool->set_runtime_outputs(
SubstitutionList::MakeForTest(kDllPattern, kPdbPattern));
toolchain.SetTool(std::move(solink));
Target target(setup.settings(), Label(SourceDir("//a/"), "a"));
target.set_output_type(Target::SHARED_LIBRARY);
target.SetToolchain(&toolchain);
ASSERT_TRUE(target.OnResolved(&err));
EXPECT_EQ("./a.dll.lib", target.link_output_file().value());
EXPECT_EQ("./a.dll.lib", target.dependency_output_file().value());
ASSERT_EQ(2u, target.runtime_outputs().size());
EXPECT_EQ("./a.dll", target.runtime_outputs()[0].value());
EXPECT_EQ("./a.pdb", target.runtime_outputs()[1].value());
// Test GetOutputsAsSourceFiles().
std::vector<SourceFile> computed_outputs;
EXPECT_TRUE(target.GetOutputsAsSourceFiles(LocationRange(), true,
&computed_outputs, &err));
ASSERT_EQ(3u, computed_outputs.size());
EXPECT_EQ("//out/Debug/a.dll.lib", computed_outputs[0].value());
EXPECT_EQ("//out/Debug/a.dll", computed_outputs[1].value());
EXPECT_EQ("//out/Debug/a.pdb", computed_outputs[2].value());
}
// Tests Target::GetOutputFilesForSource for binary targets (these require a
// tool definition). Also tests GetOutputsAsSourceFiles() for source sets.
TEST_F(TargetTest, GetOutputFilesForSource_Binary) {
TestWithScope setup;
Toolchain toolchain(setup.settings(), Label(SourceDir("//tc/"), "tc"));
std::unique_ptr<Tool> tool = Tool::CreateTool(CTool::kCToolCxx);
CTool* cxx = tool->AsC();
cxx->set_outputs(SubstitutionList::MakeForTest("{{source_file_part}}.o"));
toolchain.SetTool(std::move(tool));
Target target(setup.settings(), Label(SourceDir("//a/"), "a"));
target.set_output_type(Target::SOURCE_SET);
target.SetToolchain(&toolchain);
Err err;
ASSERT_TRUE(target.OnResolved(&err));
const char* computed_tool_type = nullptr;
std::vector<OutputFile> output;
bool result = target.GetOutputFilesForSource(SourceFile("//source/input.cc"),
&computed_tool_type, &output);
ASSERT_TRUE(result);
EXPECT_EQ(std::string("cxx"), std::string(computed_tool_type));
// Outputs are relative to the build directory "//out/Debug/".
ASSERT_EQ(1u, output.size());
EXPECT_EQ("input.cc.o", output[0].value()) << output[0].value();
// Test GetOutputsAsSourceFiles(). Since this is a source set it should give a
// stamp file.
std::vector<SourceFile> computed_outputs;
EXPECT_TRUE(target.GetOutputsAsSourceFiles(LocationRange(), true,
&computed_outputs, &err));
ASSERT_EQ(1u, computed_outputs.size());
EXPECT_EQ("//out/Debug/obj/a/a.stamp", computed_outputs[0].value());
}
TEST_F(TargetTest, CheckStampFileName) {
TestWithScope setup;
Toolchain toolchain(setup.settings(), Label(SourceDir("//tc/"), "tc"));
std::unique_ptr<Tool> tool = Tool::CreateTool(CTool::kCToolCxx);
CTool* cxx = tool->AsC();
cxx->set_outputs(SubstitutionList::MakeForTest("{{source_file_part}}.o"));
toolchain.SetTool(std::move(tool));
Target target(setup.settings(), Label(SourceDir("//a/"), "a"));
target.set_output_type(Target::SOURCE_SET);
target.SetToolchain(&toolchain);
// Change the output artifact name on purpose.
target.set_output_name("b");
Err err;
ASSERT_TRUE(target.OnResolved(&err));
// Test GetOutputsAsSourceFiles(). Since this is a source set it should give a
// stamp file.
std::vector<SourceFile> computed_outputs;
EXPECT_TRUE(target.GetOutputsAsSourceFiles(LocationRange(), true,
&computed_outputs, &err));
ASSERT_EQ(1u, computed_outputs.size());
EXPECT_EQ("//out/Debug/obj/a/a.stamp", computed_outputs[0].value())
<< "was instead: " << computed_outputs[0].value();
}
// Tests Target::GetOutputFilesForSource for action_foreach targets (these, like
// copy targets, apply a pattern to the source file). Also tests
// GetOutputsAsSourceFiles() for action_foreach().
TEST_F(TargetTest, GetOutputFilesForSource_ActionForEach) {
TestWithScope setup;
TestTarget target(setup, "//a:a", Target::ACTION_FOREACH);
target.sources().push_back(SourceFile("//a/source_file1.txt"));
target.sources().push_back(SourceFile("//a/source_file2.txt"));
target.action_values().outputs() =
SubstitutionList::MakeForTest("//out/Debug/{{source_file_part}}.one",
"//out/Debug/{{source_file_part}}.two");
Err err;
ASSERT_TRUE(target.OnResolved(&err));
const char* computed_tool_type = nullptr;
std::vector<OutputFile> output;
bool result = target.GetOutputFilesForSource(SourceFile("//source/input.txt"),
&computed_tool_type, &output);
ASSERT_TRUE(result);
// Outputs are relative to the build directory "//out/Debug/".
ASSERT_EQ(2u, output.size());
EXPECT_EQ("input.txt.one", output[0].value());
EXPECT_EQ("input.txt.two", output[1].value());
// Test GetOutputsAsSourceFiles(). It should give both outputs for each of the
// two inputs.
std::vector<SourceFile> computed_outputs;
EXPECT_TRUE(target.GetOutputsAsSourceFiles(LocationRange(), true,
&computed_outputs, &err));
ASSERT_EQ(4u, computed_outputs.size());
EXPECT_EQ("//out/Debug/source_file1.txt.one", computed_outputs[0].value());
EXPECT_EQ("//out/Debug/source_file1.txt.two", computed_outputs[1].value());
EXPECT_EQ("//out/Debug/source_file2.txt.one", computed_outputs[2].value());
EXPECT_EQ("//out/Debug/source_file2.txt.two", computed_outputs[3].value());
}
// Tests Target::GetOutputFilesForSource for action targets (these just list the
// output of the action as the result of all possible inputs). This should also
// cover everything other than binary, action_foreach, and copy targets.
TEST_F(TargetTest, GetOutputFilesForSource_Action) {
TestWithScope setup;
TestTarget target(setup, "//a:a", Target::ACTION);
target.sources().push_back(SourceFile("//a/source_file1.txt"));
target.action_values().outputs() =
SubstitutionList::MakeForTest("//out/Debug/one", "//out/Debug/two");
Err err;
ASSERT_TRUE(target.OnResolved(&err));
const char* computed_tool_type = nullptr;
std::vector<OutputFile> output;
bool result = target.GetOutputFilesForSource(SourceFile("//source/input.txt"),
&computed_tool_type, &output);
ASSERT_TRUE(result);
// Outputs are relative to the build directory "//out/Debug/".
ASSERT_EQ(2u, output.size());
EXPECT_EQ("one", output[0].value());
EXPECT_EQ("two", output[1].value());
// Test GetOutputsAsSourceFiles(). It should give the listed outputs.
std::vector<SourceFile> computed_outputs;
EXPECT_TRUE(target.GetOutputsAsSourceFiles(LocationRange(), true,
&computed_outputs, &err));
ASSERT_EQ(2u, computed_outputs.size());
EXPECT_EQ("//out/Debug/one", computed_outputs[0].value());
EXPECT_EQ("//out/Debug/two", computed_outputs[1].value());
// Test that the copy target type behaves the same. This target requires only
// one output.
target.action_values().outputs() =
SubstitutionList::MakeForTest("//out/Debug/one");
target.set_output_type(Target::COPY_FILES);
// Outputs are relative to the build directory "//out/Debug/".
result = target.GetOutputFilesForSource(SourceFile("//source/input.txt"),
&computed_tool_type, &output);
ASSERT_TRUE(result);
ASSERT_EQ(1u, output.size());
EXPECT_EQ("one", output[0].value());
// Test GetOutputsAsSourceFiles() for the copy case.
EXPECT_TRUE(target.GetOutputsAsSourceFiles(LocationRange(), true,
&computed_outputs, &err));
ASSERT_EQ(1u, computed_outputs.size()) << computed_outputs.size();
EXPECT_EQ("//out/Debug/one", computed_outputs[0].value());
}
TEST_F(TargetTest, GeneratedInputs) {
TestWithScope setup;
Err err;
SourceFile generated_file("//out/Debug/generated.cc");
// This target has a generated input and no dependency makes it.
TestTarget non_existent_generator(setup, "//foo:non_existent_generator",
Target::EXECUTABLE);
non_existent_generator.sources().push_back(generated_file);
EXPECT_TRUE(non_existent_generator.OnResolved(&err)) << err.message();
AssertSchedulerHasOneUnknownFileMatching(&non_existent_generator,
generated_file);
scheduler().ClearUnknownGeneratedInputsAndWrittenFiles();
// Make a target that generates the file.
TestTarget generator(setup, "//foo:generator", Target::ACTION);
generator.action_values().outputs() =
SubstitutionList::MakeForTest(generated_file.value().c_str());
err = Err();
EXPECT_TRUE(generator.OnResolved(&err)) << err.message();
// A target that depends on the generator that uses the file as a source
// should be OK. This uses a private dep (will be used later).
TestTarget existent_generator(setup, "//foo:existent_generator",
Target::SHARED_LIBRARY);
existent_generator.sources().push_back(generated_file);
existent_generator.private_deps().push_back(LabelTargetPair(&generator));
EXPECT_TRUE(existent_generator.OnResolved(&err)) << err.message();
EXPECT_TRUE(scheduler().GetUnknownGeneratedInputs().empty());
// A target that depends on the previous one should *not* be allowed to
// use the generated file, because existent_generator used private deps.
// This is:
// indirect_private --> existent_generator --[private]--> generator
TestTarget indirect_private(setup, "//foo:indirect_private",
Target::EXECUTABLE);
indirect_private.sources().push_back(generated_file);
indirect_private.public_deps().push_back(
LabelTargetPair(&existent_generator));
EXPECT_TRUE(indirect_private.OnResolved(&err));
AssertSchedulerHasOneUnknownFileMatching(&indirect_private, generated_file);
scheduler().ClearUnknownGeneratedInputsAndWrittenFiles();
// Now make a chain like the above but with all public deps, it should be OK.
TestTarget existent_public(setup, "//foo:existent_public",
Target::SHARED_LIBRARY);
existent_public.public_deps().push_back(LabelTargetPair(&generator));
EXPECT_TRUE(existent_public.OnResolved(&err)) << err.message();
TestTarget indirect_public(setup, "//foo:indirect_public",
Target::EXECUTABLE);
indirect_public.sources().push_back(generated_file);
indirect_public.public_deps().push_back(LabelTargetPair(&existent_public));
EXPECT_TRUE(indirect_public.OnResolved(&err)) << err.message();
EXPECT_TRUE(scheduler().GetUnknownGeneratedInputs().empty());
}
// This is sort of a Scheduler test, but is related to the above test more.
TEST_F(TargetTest, WriteFileGeneratedInputs) {
TestWithScope setup;
Err err;
SourceFile generated_file("//out/Debug/generated.data");
// This target has a generated input and no dependency makes it.
TestTarget non_existent_generator(setup, "//foo:non_existent_generator",
Target::EXECUTABLE);
non_existent_generator.sources().push_back(generated_file);
EXPECT_TRUE(non_existent_generator.OnResolved(&err));
AssertSchedulerHasOneUnknownFileMatching(&non_existent_generator,
generated_file);
scheduler().ClearUnknownGeneratedInputsAndWrittenFiles();
// This target has a generated file and we've decared we write it.
TestTarget existent_generator(setup, "//foo:existent_generator",
Target::EXECUTABLE);
existent_generator.sources().push_back(generated_file);
EXPECT_TRUE(existent_generator.OnResolved(&err));
scheduler().AddWrittenFile(generated_file);
// Should be OK.
EXPECT_TRUE(scheduler().GetUnknownGeneratedInputs().empty());
}
TEST_F(TargetTest, WriteRuntimeDepsGeneratedInputs) {
TestWithScope setup;
Err err;
SourceFile source_file("//out/Debug/generated.runtime_deps");
OutputFile output_file(setup.build_settings(), source_file);
TestTarget generator(setup, "//foo:generator", Target::EXECUTABLE);
generator.set_write_runtime_deps_output(output_file);
g_scheduler->AddWriteRuntimeDepsTarget(&generator);
TestTarget middle_data_dep(setup, "//foo:middle", Target::EXECUTABLE);
middle_data_dep.data_deps().push_back(LabelTargetPair(&generator));
// This target has a generated input and no dependency makes it.
TestTarget dep_missing(setup, "//foo:no_dep", Target::EXECUTABLE);
dep_missing.sources().push_back(source_file);
EXPECT_TRUE(dep_missing.OnResolved(&err));
AssertSchedulerHasOneUnknownFileMatching(&dep_missing, source_file);
scheduler().ClearUnknownGeneratedInputsAndWrittenFiles();
// This target has a generated file and we've directly dependended on it.
TestTarget dep_present(setup, "//foo:with_dep", Target::EXECUTABLE);
dep_present.sources().push_back(source_file);
dep_present.private_deps().push_back(LabelTargetPair(&generator));
EXPECT_TRUE(dep_present.OnResolved(&err));
EXPECT_TRUE(scheduler().GetUnknownGeneratedInputs().empty());
// This target has a generated file and we've indirectly dependended on it
// via data_deps.
TestTarget dep_indirect(setup, "//foo:with_dep", Target::EXECUTABLE);
dep_indirect.sources().push_back(source_file);
dep_indirect.data_deps().push_back(LabelTargetPair(&middle_data_dep));
EXPECT_TRUE(dep_indirect.OnResolved(&err));
AssertSchedulerHasOneUnknownFileMatching(&dep_indirect, source_file);
scheduler().ClearUnknownGeneratedInputsAndWrittenFiles();
// This target has a generated file and we've directly dependended on it
// via data_deps.
TestTarget data_dep_present(setup, "//foo:with_dep", Target::EXECUTABLE);
data_dep_present.sources().push_back(source_file);
data_dep_present.data_deps().push_back(LabelTargetPair(&generator));
EXPECT_TRUE(data_dep_present.OnResolved(&err));
EXPECT_TRUE(scheduler().GetUnknownGeneratedInputs().empty());
}
// Tests that intermediate object files generated by binary targets are also
// considered generated for the purposes of input checking. Above, we tested
// the failure cases for generated inputs, so here only test .o files that are
// present.
TEST_F(TargetTest, ObjectGeneratedInputs) {
TestWithScope setup;
Err err;
// This target compiles the source.
SourceFile source_file("//source.cc");
TestTarget source_generator(setup, "//:source_target", Target::SOURCE_SET);
source_generator.sources().push_back(source_file);
EXPECT_TRUE(source_generator.OnResolved(&err));
// This is the object file that the test toolchain generates for the source.
SourceFile object_file("//out/Debug/obj/source_target.source.o");
TestTarget final_target(setup, "//:final", Target::ACTION);
final_target.config_values().inputs().push_back(object_file);
EXPECT_TRUE(final_target.OnResolved(&err));
AssertSchedulerHasOneUnknownFileMatching(&final_target, object_file);
}
TEST_F(TargetTest, ResolvePrecompiledHeaders) {
TestWithScope setup;
Err err;
Target target(setup.settings(), Label(SourceDir("//foo/"), "bar",
SourceDir("//toolchain/"), "default"));
// Target with no settings, no configs, should be a no-op.
EXPECT_TRUE(target.ResolvePrecompiledHeaders(&err));
// Config with PCH values.
Config config_1(
setup.settings(),
Label(SourceDir("//foo/"), "c1", SourceDir("//toolchain/"), "default"));
std::string pch_1("pch.h");
SourceFile pcs_1("//pcs.cc");
config_1.own_values().set_precompiled_header(pch_1);
config_1.own_values().set_precompiled_source(pcs_1);
ASSERT_TRUE(config_1.OnResolved(&err));
target.configs().push_back(LabelConfigPair(&config_1));
// No PCH info specified on TargetTest, but the config specifies one, the
// values should get copied to the target.
EXPECT_TRUE(target.ResolvePrecompiledHeaders(&err));
EXPECT_EQ(pch_1, target.config_values().precompiled_header());
EXPECT_TRUE(target.config_values().precompiled_source() == pcs_1);
// Now both target and config have matching PCH values. Resolving again
// should be a no-op since they all match.
EXPECT_TRUE(target.ResolvePrecompiledHeaders(&err));
EXPECT_TRUE(target.config_values().precompiled_header() == pch_1);
EXPECT_TRUE(target.config_values().precompiled_source() == pcs_1);
// Second config with different PCH values.
Config config_2(
setup.settings(),
Label(SourceDir("//foo/"), "c2", SourceDir("//toolchain/"), "default"));
std::string pch_2("pch2.h");
SourceFile pcs_2("//pcs2.cc");
config_2.own_values().set_precompiled_header(pch_2);
config_2.own_values().set_precompiled_source(pcs_2);
ASSERT_TRUE(config_2.OnResolved(&err));
target.configs().push_back(LabelConfigPair(&config_2));
// This should be an error since they don't match.
EXPECT_FALSE(target.ResolvePrecompiledHeaders(&err));
// Make sure the proper labels are blamed.
EXPECT_EQ(
"The target //foo:bar\n"
"has conflicting precompiled header settings.\n"
"\n"
"From //foo:bar\n"
" header: pch.h\n"
" source: //pcs.cc\n"
"\n"
"From //foo:c2\n"
" header: pch2.h\n"
" source: //pcs2.cc",
err.help_text());
}
TEST_F(TargetTest, AssertNoDeps) {
TestWithScope setup;
Err err;
// A target.
TestTarget a(setup, "//a", Target::SHARED_LIBRARY);
ASSERT_TRUE(a.OnResolved(&err));
// B depends on A and has an assert_no_deps for a random dir.
TestTarget b(setup, "//b", Target::SHARED_LIBRARY);
b.private_deps().push_back(LabelTargetPair(&a));
b.assert_no_deps().push_back(LabelPattern(LabelPattern::RECURSIVE_DIRECTORY,
SourceDir("//disallowed/"),
std::string(), Label()));
ASSERT_TRUE(b.OnResolved(&err));
LabelPattern disallow_a(LabelPattern::RECURSIVE_DIRECTORY, SourceDir("//a/"),
std::string(), Label());
// C depends on B and disallows depending on A. This should fail.
TestTarget c(setup, "//c", Target::EXECUTABLE);
c.private_deps().push_back(LabelTargetPair(&b));
c.assert_no_deps().push_back(disallow_a);
ASSERT_FALSE(c.OnResolved(&err));
// Validate the error message has the proper path.
EXPECT_EQ(
"//c:c has an assert_no_deps entry:\n"
" //a/*\n"
"which fails for the dependency path:\n"
" //c:c ->\n"
" //b:b ->\n"
" //a:a",
err.help_text());
err = Err();
// Add an intermediate executable with: exe -> b -> a
TestTarget exe(setup, "//exe", Target::EXECUTABLE);
exe.private_deps().push_back(LabelTargetPair(&b));
ASSERT_TRUE(exe.OnResolved(&err));
// D depends on the executable and disallows depending on A. Since there is
// an intermediate executable, this should be OK.
TestTarget d(setup, "//d", Target::EXECUTABLE);
d.private_deps().push_back(LabelTargetPair(&exe));
d.assert_no_deps().push_back(disallow_a);
ASSERT_TRUE(d.OnResolved(&err));
// A2 disallows depending on anything in its own directory, but the
// assertions should not match the target itself so this should be OK.
TestTarget a2(setup, "//a:a2", Target::EXECUTABLE);
a2.assert_no_deps().push_back(disallow_a);
ASSERT_TRUE(a2.OnResolved(&err));
}
TEST_F(TargetTest, PullRecursiveBundleData) {
TestWithScope setup;
Err err;
// We have the following dependency graph:
// A (create_bundle) -> B (bundle_data)
// \-> C (create_bundle) -> D (bundle_data)
// \-> E (group) -> F (bundle_data)
// \-> B (bundle_data)
TestTarget a(setup, "//foo:a", Target::CREATE_BUNDLE);
TestTarget b(setup, "//foo:b", Target::BUNDLE_DATA);
TestTarget c(setup, "//foo:c", Target::CREATE_BUNDLE);
TestTarget d(setup, "//foo:d", Target::BUNDLE_DATA);
TestTarget e(setup, "//foo:e", Target::GROUP);
TestTarget f(setup, "//foo:f", Target::BUNDLE_DATA);
a.public_deps().push_back(LabelTargetPair(&b));
a.public_deps().push_back(LabelTargetPair(&c));
a.public_deps().push_back(LabelTargetPair(&e));
c.public_deps().push_back(LabelTargetPair(&d));
e.public_deps().push_back(LabelTargetPair(&f));
e.public_deps().push_back(LabelTargetPair(&b));
a.bundle_data().root_dir() = SourceDir("//out/foo_a.bundle");
a.bundle_data().resources_dir() = SourceDir("//out/foo_a.bundle/Resources");
b.sources().push_back(SourceFile("//foo/b1.txt"));
b.sources().push_back(SourceFile("//foo/b2.txt"));
b.action_values().outputs() = SubstitutionList::MakeForTest(
"{{bundle_resources_dir}}/{{source_file_part}}");
ASSERT_TRUE(b.OnResolved(&err));
c.bundle_data().root_dir() = SourceDir("//out/foo_c.bundle");
c.bundle_data().resources_dir() = SourceDir("//out/foo_c.bundle/Resources");
d.sources().push_back(SourceFile("//foo/d.txt"));
d.action_values().outputs() = SubstitutionList::MakeForTest(
"{{bundle_resources_dir}}/{{source_file_part}}");
ASSERT_TRUE(d.OnResolved(&err));
f.sources().push_back(SourceFile("//foo/f1.txt"));
f.sources().push_back(SourceFile("//foo/f2.txt"));
f.sources().push_back(SourceFile("//foo/f3.txt"));
f.sources().push_back(
SourceFile("//foo/Foo.xcassets/foo.imageset/Contents.json"));
f.sources().push_back(
SourceFile("//foo/Foo.xcassets/foo.imageset/FooEmpty-29.png"));
f.sources().push_back(
SourceFile("//foo/Foo.xcassets/foo.imageset/FooEmpty-29@2x.png"));
f.sources().push_back(
SourceFile("//foo/Foo.xcassets/foo.imageset/FooEmpty-29@3x.png"));
f.action_values().outputs() = SubstitutionList::MakeForTest(
"{{bundle_resources_dir}}/{{source_file_part}}");
ASSERT_TRUE(f.OnResolved(&err));
ASSERT_TRUE(e.OnResolved(&err));
ASSERT_TRUE(c.OnResolved(&err));
ASSERT_TRUE(a.OnResolved(&err));
// A gets its data from B and F.
ASSERT_EQ(a.bundle_data().file_rules().size(), 2u);
ASSERT_EQ(a.bundle_data().file_rules()[0].sources().size(), 2u);
ASSERT_EQ(a.bundle_data().file_rules()[1].sources().size(), 3u);
ASSERT_EQ(a.bundle_data().assets_catalog_sources().size(), 1u);
ASSERT_EQ(a.bundle_data().bundle_deps().size(), 2u);
// C gets its data from D.
ASSERT_EQ(c.bundle_data().file_rules().size(), 1u);
ASSERT_EQ(c.bundle_data().file_rules()[0].sources().size(), 1u);
ASSERT_EQ(c.bundle_data().bundle_deps().size(), 1u);
// E does not have any bundle_data information but gets a list of
// bundle_deps to propagate them during target resolution.
ASSERT_TRUE(e.bundle_data().file_rules().empty());
ASSERT_TRUE(e.bundle_data().assets_catalog_sources().empty());
ASSERT_EQ(e.bundle_data().bundle_deps().size(), 2u);
}
TEST(TargetTest, CollectMetadataNoRecurse) {
TestWithScope setup;
TestTarget one(setup, "//foo:one", Target::SOURCE_SET);
Value a_expected(nullptr, Value::LIST);
a_expected.list_value().push_back(Value(nullptr, "foo"));
one.metadata().contents().insert(
std::pair<std::string_view, Value>("a", a_expected));
Value b_expected(nullptr, Value::LIST);
b_expected.list_value().push_back(Value(nullptr, true));
one.metadata().contents().insert(
std::pair<std::string_view, Value>("b", b_expected));
one.metadata().set_source_dir(SourceDir("/usr/home/files/"));
std::vector<std::string> data_keys;
data_keys.push_back("a");
data_keys.push_back("b");
std::vector<std::string> walk_keys;
Err err;
std::vector<Value> result;
TargetSet targets;
one.GetMetadata(data_keys, walk_keys, SourceDir(), false, &result, &targets,
&err);
EXPECT_FALSE(err.has_error());
std::vector<Value> expected;
expected.push_back(Value(nullptr, "foo"));
expected.push_back(Value(nullptr, true));
EXPECT_EQ(result, expected);
}
TEST(TargetTest, CollectMetadataWithRecurse) {
TestWithScope setup;
TestTarget one(setup, "//foo:one", Target::SOURCE_SET);
Value a_expected(nullptr, Value::LIST);
a_expected.list_value().push_back(Value(nullptr, "foo"));
one.metadata().contents().insert(
std::pair<std::string_view, Value>("a", a_expected));
Value b_expected(nullptr, Value::LIST);
b_expected.list_value().push_back(Value(nullptr, true));
one.metadata().contents().insert(
std::pair<std::string_view, Value>("b", b_expected));
TestTarget two(setup, "//foo:two", Target::SOURCE_SET);
Value a_2_expected(nullptr, Value::LIST);
a_2_expected.list_value().push_back(Value(nullptr, "bar"));
two.metadata().contents().insert(
std::pair<std::string_view, Value>("a", a_2_expected));
one.public_deps().push_back(LabelTargetPair(&two));
std::vector<std::string> data_keys;
data_keys.push_back("a");
data_keys.push_back("b");
std::vector<std::string> walk_keys;
Err err;
std::vector<Value> result;
TargetSet targets;
one.GetMetadata(data_keys, walk_keys, SourceDir(), false, &result, &targets,
&err);
EXPECT_FALSE(err.has_error());
std::vector<Value> expected;
expected.push_back(Value(nullptr, "bar"));
expected.push_back(Value(nullptr, "foo"));
expected.push_back(Value(nullptr, true));
EXPECT_EQ(result, expected);
}
TEST(TargetTest, CollectMetadataWithRecurseHole) {
TestWithScope setup;
TestTarget one(setup, "//foo:one", Target::SOURCE_SET);
Value a_expected(nullptr, Value::LIST);
a_expected.list_value().push_back(Value(nullptr, "foo"));
one.metadata().contents().insert(
std::pair<std::string_view, Value>("a", a_expected));
Value b_expected(nullptr, Value::LIST);
b_expected.list_value().push_back(Value(nullptr, true));
one.metadata().contents().insert(
std::pair<std::string_view, Value>("b", b_expected));
// Target two does not have metadata but depends on three
// which does.
TestTarget two(setup, "//foo:two", Target::SOURCE_SET);
TestTarget three(setup, "//foo:three", Target::SOURCE_SET);
Value a_3_expected(nullptr, Value::LIST);
a_3_expected.list_value().push_back(Value(nullptr, "bar"));
three.metadata().contents().insert(
std::pair<std::string_view, Value>("a", a_3_expected));
one.public_deps().push_back(LabelTargetPair(&two));
two.public_deps().push_back(LabelTargetPair(&three));
std::vector<std::string> data_keys;
data_keys.push_back("a");
data_keys.push_back("b");
std::vector<std::string> walk_keys;
Err err;
std::vector<Value> result;
TargetSet targets;
one.GetMetadata(data_keys, walk_keys, SourceDir(), false, &result, &targets,
&err);
EXPECT_FALSE(err.has_error());
std::vector<Value> expected;
expected.push_back(Value(nullptr, "bar"));
expected.push_back(Value(nullptr, "foo"));
expected.push_back(Value(nullptr, true));
EXPECT_EQ(result, expected);
}
TEST(TargetTest, CollectMetadataWithBarrier) {
TestWithScope setup;
TestTarget one(setup, "//foo:one", Target::SOURCE_SET);
Value a_expected(nullptr, Value::LIST);
a_expected.list_value().push_back(Value(nullptr, "foo"));
one.metadata().contents().insert(
std::pair<std::string_view, Value>("a", a_expected));
Value walk_expected(nullptr, Value::LIST);
walk_expected.list_value().push_back(Value(nullptr, "two"));
one.metadata().contents().insert(
std::pair<std::string_view, Value>("walk", walk_expected));
TestTarget two(setup, "//foo/two:two", Target::SOURCE_SET);
Value a_2_expected(nullptr, Value::LIST);
a_2_expected.list_value().push_back(Value(nullptr, "bar"));
two.metadata().contents().insert(
std::pair<std::string_view, Value>("a", a_2_expected));
TestTarget three(setup, "//foo:three", Target::SOURCE_SET);
Value a_3_expected(nullptr, Value::LIST);
a_3_expected.list_value().push_back(Value(nullptr, "baz"));
three.metadata().contents().insert(
std::pair<std::string_view, Value>("a", a_3_expected));
one.private_deps().push_back(LabelTargetPair(&two));
one.public_deps().push_back(LabelTargetPair(&three));
std::vector<std::string> data_keys;
data_keys.push_back("a");
std::vector<std::string> walk_keys;
walk_keys.push_back("walk");
Err err;
std::vector<Value> result;
TargetSet targets;
one.GetMetadata(data_keys, walk_keys, SourceDir(), false, &result, &targets,
&err);
EXPECT_FALSE(err.has_error()) << err.message();
std::vector<Value> expected;
expected.push_back(Value(nullptr, "bar"));
expected.push_back(Value(nullptr, "foo"));
EXPECT_EQ(result, expected) << result.size();
}
TEST(TargetTest, CollectMetadataWithError) {
TestWithScope setup;
TestTarget one(setup, "//foo:one", Target::SOURCE_SET);
Value a_expected(nullptr, Value::LIST);
a_expected.list_value().push_back(Value(nullptr, "foo"));
one.metadata().contents().insert(
std::pair<std::string_view, Value>("a", a_expected));
Value walk_expected(nullptr, Value::LIST);
walk_expected.list_value().push_back(Value(nullptr, "//foo:missing"));
one.metadata().contents().insert(
std::pair<std::string_view, Value>("walk", walk_expected));
std::vector<std::string> data_keys;
data_keys.push_back("a");
std::vector<std::string> walk_keys;
walk_keys.push_back("walk");
Err err;
std::vector<Value> result;
TargetSet targets;
one.GetMetadata(data_keys, walk_keys, SourceDir(), false, &result, &targets,
&err);
EXPECT_TRUE(err.has_error());
EXPECT_EQ(err.message(),
"I was expecting //foo:missing(//toolchain:default) to be a "
"dependency of //foo:one(//toolchain:default). "
"Make sure it's included in the deps or data_deps, and that you've "
"specified the appropriate toolchain.")
<< err.message();
}
TEST_F(TargetTest, WriteMetadataCollection) {
TestWithScope setup;
Err err;
SourceFile source_file("//out/Debug/metadata.json");
OutputFile output_file(setup.build_settings(), source_file);
TestTarget generator(setup, "//foo:write", Target::GENERATED_FILE);
generator.action_values().outputs() =
SubstitutionList::MakeForTest("//out/Debug/metadata.json");
EXPECT_TRUE(generator.OnResolved(&err));
TestTarget middle_data_dep(setup, "//foo:middle", Target::EXECUTABLE);
middle_data_dep.data_deps().push_back(LabelTargetPair(&generator));
EXPECT_TRUE(middle_data_dep.OnResolved(&err));
// This target has a generated metadata input and no dependency makes it.
TestTarget dep_missing(setup, "//foo:no_dep", Target::EXECUTABLE);
dep_missing.sources().push_back(source_file);
EXPECT_TRUE(dep_missing.OnResolved(&err));
AssertSchedulerHasOneUnknownFileMatching(&dep_missing, source_file);
scheduler().ClearUnknownGeneratedInputsAndWrittenFiles();
// This target has a generated file and we've directly dependended on it.
TestTarget dep_present(setup, "//foo:with_dep", Target::EXECUTABLE);
dep_present.sources().push_back(source_file);
dep_present.private_deps().push_back(LabelTargetPair(&generator));
EXPECT_TRUE(dep_present.OnResolved(&err));
EXPECT_TRUE(scheduler().GetUnknownGeneratedInputs().empty());
// This target has a generated file and we've indirectly dependended on it
// via data_deps.
TestTarget dep_indirect(setup, "//foo:indirect_dep", Target::EXECUTABLE);
dep_indirect.sources().push_back(source_file);
dep_indirect.data_deps().push_back(LabelTargetPair(&middle_data_dep));
EXPECT_TRUE(dep_indirect.OnResolved(&err));
AssertSchedulerHasOneUnknownFileMatching(&dep_indirect, source_file);
scheduler().ClearUnknownGeneratedInputsAndWrittenFiles();
// This target has a generated file and we've directly dependended on it
// via data_deps.
TestTarget data_dep_present(setup, "//foo:with_data_dep", Target::EXECUTABLE);
data_dep_present.sources().push_back(source_file);
data_dep_present.data_deps().push_back(LabelTargetPair(&generator));
EXPECT_TRUE(data_dep_present.OnResolved(&err));
EXPECT_TRUE(scheduler().GetUnknownGeneratedInputs().empty());
}
// Tests that modulemap files use the cxx_module tool.
TEST_F(TargetTest, ModuleMap) {
TestWithScope setup;
Toolchain toolchain(setup.settings(), Label(SourceDir("//tc/"), "tc"));
std::unique_ptr<Tool> tool = Tool::CreateTool(CTool::kCToolCxxModule);
CTool* cxx_module = tool->AsC();
cxx_module->set_outputs(
SubstitutionList::MakeForTest("{{source_file_part}}.pcm"));
toolchain.SetTool(std::move(tool));
Target target(setup.settings(), Label(SourceDir("//a/"), "a"));
target.set_output_type(Target::SOURCE_SET);
target.SetToolchain(&toolchain);
Err err;
ASSERT_TRUE(target.OnResolved(&err));
const char* computed_tool_type = nullptr;
std::vector<OutputFile> output;
bool result = target.GetOutputFilesForSource(
SourceFile("//source/input.modulemap"), &computed_tool_type, &output);
ASSERT_TRUE(result);
EXPECT_EQ(std::string("cxx_module"), std::string(computed_tool_type));
// Outputs are relative to the build directory "//out/Debug/".
ASSERT_EQ(1u, output.size());
EXPECT_EQ("input.modulemap.pcm", output[0].value()) << output[0].value();
}