GN: Look at all target outputs when detecting duplicates
BUG=554660
Review URL: https://codereview.chromium.org/1828113002
Cr-Original-Commit-Position: refs/heads/master@{#384357}
Cr-Mirrored-From: https://chromium.googlesource.com/chromium/src
Cr-Mirrored-Commit: 83671be906e98ac69bbfaf2c2206977806ff5448
diff --git a/tools/gn/BUILD.gn b/tools/gn/BUILD.gn
index 37bce8d..bb36b1f 100644
--- a/tools/gn/BUILD.gn
+++ b/tools/gn/BUILD.gn
@@ -285,6 +285,7 @@
"loader_unittest.cc",
"ninja_action_target_writer_unittest.cc",
"ninja_binary_target_writer_unittest.cc",
+ "ninja_build_writer_unittest.cc",
"ninja_copy_target_writer_unittest.cc",
"ninja_create_bundle_target_writer_unittest.cc",
"ninja_group_target_writer_unittest.cc",
diff --git a/tools/gn/ninja_build_writer.cc b/tools/gn/ninja_build_writer.cc
index d27ce2d..d289690 100644
--- a/tools/gn/ninja_build_writer.cc
+++ b/tools/gn/ninja_build_writer.cc
@@ -114,8 +114,12 @@
const OutputFile& bad_output) {
std::vector<const Target*> matches;
for (const Target* target : all_targets) {
- if (GetTargetOutputFile(target) == bad_output)
- matches.push_back(target);
+ for (const auto& output : target->computed_outputs()) {
+ if (output == bad_output) {
+ matches.push_back(target);
+ break;
+ }
+ }
}
// There should always be at least two targets generating this file for this
@@ -127,11 +131,10 @@
Err result(matches[0]->defined_from(), "Duplicate output file.",
"Two or more targets generate the same output:\n " +
- bad_output.value() + "\n"
- "This is normally the result of either overriding the output name or\n"
- "having two shared libraries or executables in different directories\n"
- "with the same name (since all such targets will be written to the root\n"
- "output directory).\n\nCollisions:\n" + matches_string);
+ bad_output.value() + "\n\n"
+ "This is can often be fixed by changing one of the target names, or by \n"
+ "setting an output_name on one of them.\n"
+ "\nCollisions:\n" + matches_string);
for (size_t i = 1; i < matches.size(); i++)
result.AppendSubErr(Err(matches[i]->defined_from(), "Collision."));
return result;
@@ -294,12 +297,14 @@
for (const auto& target : default_toolchain_targets_) {
const Label& label = target->label();
- OutputFile target_file = GetTargetOutputFile(target);
- if (!target_files.insert(target_file.value()).second) {
- *err = GetDuplicateOutputError(default_toolchain_targets_, target_file);
- return false;
+ for (const auto& output : target->computed_outputs()) {
+ if (!target_files.insert(output.value()).second) {
+ *err = GetDuplicateOutputError(default_toolchain_targets_, output);
+ return false;
+ }
}
+ OutputFile target_file = GetTargetOutputFile(target);
// Write the long name "foo/bar:baz" for the target "//foo/bar:baz".
std::string long_name = label.GetUserVisibleName(false);
base::TrimString(long_name, "/", &long_name);
diff --git a/tools/gn/ninja_build_writer.h b/tools/gn/ninja_build_writer.h
index 2df3365..0d5b40e 100644
--- a/tools/gn/ninja_build_writer.h
+++ b/tools/gn/ninja_build_writer.h
@@ -30,7 +30,6 @@
const std::vector<const Target*>& default_toolchain_targets,
Err* err);
- private:
NinjaBuildWriter(const BuildSettings* settings,
const std::vector<const Settings*>& all_settings,
const Toolchain* default_toolchain,
@@ -41,6 +40,7 @@
bool Run(Err* err);
+ private:
void WriteNinjaRules();
void WriteLinkPool();
void WriteSubninjas();
diff --git a/tools/gn/ninja_build_writer_unittest.cc b/tools/gn/ninja_build_writer_unittest.cc
new file mode 100644
index 0000000..f82dc83
--- /dev/null
+++ b/tools/gn/ninja_build_writer_unittest.cc
@@ -0,0 +1,119 @@
+// Copyright 2016 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 <sstream>
+
+#include "testing/gtest/include/gtest/gtest.h"
+#include "tools/gn/ninja_build_writer.h"
+#include "tools/gn/scheduler.h"
+#include "tools/gn/target.h"
+#include "tools/gn/test_with_scope.h"
+
+TEST(NinjaBuildWriter, TwoTargets) {
+ Scheduler scheduler;
+ TestWithScope setup;
+ Err err;
+
+ Target target_foo(setup.settings(), Label(SourceDir("//foo/"), "bar"));
+ target_foo.set_output_type(Target::ACTION);
+ target_foo.action_values().set_script(SourceFile("//foo/script.py"));
+ target_foo.action_values().outputs() = SubstitutionList::MakeForTest(
+ "//out/Debug/out1.out", "//out/Debug/out2.out");
+ target_foo.SetToolchain(setup.toolchain());
+ ASSERT_TRUE(target_foo.OnResolved(&err));
+
+ Target target_bar(setup.settings(), Label(SourceDir("//bar/"), "bar"));
+ target_bar.set_output_type(Target::ACTION);
+ target_bar.action_values().set_script(SourceFile("//bar/script.py"));
+ target_bar.action_values().outputs() = SubstitutionList::MakeForTest(
+ "//out/Debug/out3.out", "//out/Debug/out4.out");
+ target_bar.SetToolchain(setup.toolchain());
+ ASSERT_TRUE(target_bar.OnResolved(&err));
+
+ std::ostringstream ninja_out;
+ std::ostringstream depfile_out;
+ std::vector<const Settings*> all_settings = {setup.settings()};
+ std::vector<const Target*> targets = {&target_foo, &target_bar};
+ NinjaBuildWriter writer(setup.build_settings(), all_settings,
+ setup.toolchain(), targets, ninja_out, depfile_out);
+ ASSERT_TRUE(writer.Run(&err));
+
+ const char expected_rule_gn[] = "rule gn\n";
+ const char expected_build_ninja[] =
+ "build build.ninja: gn\n"
+ " generator = 1\n"
+ " depfile = build.ninja.d\n"
+ "\n";
+ const char expected_link_pool[] =
+ "pool link_pool\n"
+ " depth = 0\n"
+ "\n";
+ const char expected_toolchain[] =
+ "subninja toolchain.ninja\n"
+ "\n";
+ const char expected_targets[] =
+ "build foo$:bar: phony obj/foo/bar.stamp\n"
+ "build bar$:bar: phony obj/bar/bar.stamp\n"
+ "build bar: phony obj/bar/bar.stamp\n"
+ "\n";
+ const char expected_root_target[] =
+ "build all: phony obj/foo/bar.stamp $\n"
+ " obj/bar/bar.stamp\n"
+ "default all\n";
+ std::string out_str = ninja_out.str();
+#define EXPECT_SNIPPET(expected) \
+ EXPECT_NE(std::string::npos, out_str.find(expected)) << \
+ "Expected to find: " << expected << std::endl << \
+ "Within: " << out_str
+ EXPECT_SNIPPET(expected_rule_gn);
+ EXPECT_SNIPPET(expected_build_ninja);
+ EXPECT_SNIPPET(expected_link_pool);
+ EXPECT_SNIPPET(expected_toolchain);
+ EXPECT_SNIPPET(expected_targets);
+ EXPECT_SNIPPET(expected_root_target);
+#undef EXPECT_SNIPPET
+}
+
+TEST(NinjaBuildWriter, DuplicateOutputs) {
+ Scheduler scheduler;
+ TestWithScope setup;
+ Err err;
+
+ Target target_foo(setup.settings(), Label(SourceDir("//foo/"), "bar"));
+ target_foo.set_output_type(Target::ACTION);
+ target_foo.action_values().set_script(SourceFile("//foo/script.py"));
+ target_foo.action_values().outputs() = SubstitutionList::MakeForTest(
+ "//out/Debug/out1.out", "//out/Debug/out2.out");
+ target_foo.SetToolchain(setup.toolchain());
+ ASSERT_TRUE(target_foo.OnResolved(&err));
+
+ Target target_bar(setup.settings(), Label(SourceDir("//bar/"), "bar"));
+ target_bar.set_output_type(Target::ACTION);
+ target_bar.action_values().set_script(SourceFile("//bar/script.py"));
+ target_bar.action_values().outputs() = SubstitutionList::MakeForTest(
+ "//out/Debug/out3.out", "//out/Debug/out2.out");
+ target_bar.SetToolchain(setup.toolchain());
+ ASSERT_TRUE(target_bar.OnResolved(&err));
+
+ std::ostringstream ninja_out;
+ std::ostringstream depfile_out;
+ std::vector<const Settings*> all_settings = { setup.settings() };
+ std::vector<const Target*> targets = { &target_foo, &target_bar };
+ NinjaBuildWriter writer(setup.build_settings(), all_settings,
+ setup.toolchain(), targets, ninja_out, depfile_out);
+ ASSERT_FALSE(writer.Run(&err));
+
+ const char expected_help_test[] =
+ "Two or more targets generate the same output:\n"
+ " out2.out\n"
+ "\n"
+ "This is can often be fixed by changing one of the target names, or by \n"
+ "setting an output_name on one of them.\n"
+ "\n"
+ "Collisions:\n"
+ " //foo:bar\n"
+ " //bar:bar\n";
+
+ EXPECT_EQ(expected_help_test, err.help_text());
+}