|  | // 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 "tools/gn/ninja_build_writer.h" | 
|  | #include "tools/gn/pool.h" | 
|  | #include "tools/gn/scheduler.h" | 
|  | #include "tools/gn/target.h" | 
|  | #include "tools/gn/test_with_scheduler.h" | 
|  | #include "tools/gn/test_with_scope.h" | 
|  | #include "util/test/test.h" | 
|  |  | 
|  | using NinjaBuildWriterTest = TestWithScheduler; | 
|  |  | 
|  | TEST_F(NinjaBuildWriterTest, TwoTargets) { | 
|  | 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)); | 
|  |  | 
|  | // Make a secondary toolchain that references two pools. | 
|  | Label other_toolchain_label(SourceDir("//other/"), "toolchain"); | 
|  | Toolchain other_toolchain(setup.settings(), other_toolchain_label); | 
|  | TestWithScope::SetupToolchain(&other_toolchain); | 
|  |  | 
|  | Pool other_regular_pool( | 
|  | setup.settings(), | 
|  | Label(SourceDir("//other/"), "depth_pool", other_toolchain_label.dir(), | 
|  | other_toolchain_label.name())); | 
|  | other_regular_pool.set_depth(42); | 
|  | other_toolchain.GetTool(Toolchain::TYPE_LINK) | 
|  | ->set_pool(LabelPtrPair<Pool>(&other_regular_pool)); | 
|  |  | 
|  | // The console pool must be in the default toolchain. | 
|  | Pool console_pool(setup.settings(), Label(SourceDir("//"), "console", | 
|  | setup.toolchain()->label().dir(), | 
|  | setup.toolchain()->label().name())); | 
|  | console_pool.set_depth(1); | 
|  | other_toolchain.GetTool(Toolchain::TYPE_STAMP) | 
|  | ->set_pool(LabelPtrPair<Pool>(&console_pool)); | 
|  |  | 
|  | // Settings to go with the other toolchain. | 
|  | Settings other_settings(setup.build_settings(), "toolchain/"); | 
|  | other_settings.set_toolchain_label(other_toolchain_label); | 
|  |  | 
|  | std::unordered_map<const Settings*, const Toolchain*> used_toolchains; | 
|  | used_toolchains[setup.settings()] = setup.toolchain(); | 
|  | used_toolchains[&other_settings] = &other_toolchain; | 
|  |  | 
|  | std::vector<const Target*> targets = {&target_foo, &target_bar}; | 
|  |  | 
|  | std::ostringstream ninja_out; | 
|  | std::ostringstream depfile_out; | 
|  |  | 
|  | NinjaBuildWriter writer(setup.build_settings(), used_toolchains, | 
|  | 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"; | 
|  | const char expected_other_pool[] = | 
|  | "pool other_toolchain_other_depth_pool\n" | 
|  | "  depth = 42\n"; | 
|  | const char expected_toolchain[] = "subninja toolchain.ninja\n"; | 
|  | const char expected_targets[] = | 
|  | "build bar: phony obj/bar/bar.stamp\n" | 
|  | "build foo$:bar: phony obj/foo/bar.stamp\n" | 
|  | "build bar$:bar: phony obj/bar/bar.stamp\n"; | 
|  | const char expected_root_target[] = | 
|  | "build all: phony $\n" | 
|  | "    obj/foo/bar.stamp $\n" | 
|  | "    obj/bar/bar.stamp\n"; | 
|  | const char expected_default[] = "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 << "\n"      \ | 
|  | << "Within: " << out_str | 
|  | EXPECT_SNIPPET(expected_rule_gn); | 
|  | EXPECT_SNIPPET(expected_build_ninja); | 
|  | EXPECT_SNIPPET(expected_other_pool); | 
|  | EXPECT_SNIPPET(expected_toolchain); | 
|  | EXPECT_SNIPPET(expected_targets); | 
|  | EXPECT_SNIPPET(expected_root_target); | 
|  | EXPECT_SNIPPET(expected_default); | 
|  | #undef EXPECT_SNIPPET | 
|  |  | 
|  | // A pool definition for ninja's built-in console pool must not be written. | 
|  | EXPECT_EQ(std::string::npos, out_str.find("pool console")); | 
|  | } | 
|  |  | 
|  | TEST_F(NinjaBuildWriterTest, DuplicateOutputs) { | 
|  | 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::unordered_map<const Settings*, const Toolchain*> used_toolchains; | 
|  | used_toolchains[setup.settings()] = setup.toolchain(); | 
|  | std::vector<const Target*> targets = {&target_foo, &target_bar}; | 
|  | std::ostringstream ninja_out; | 
|  | std::ostringstream depfile_out; | 
|  | NinjaBuildWriter writer(setup.build_settings(), used_toolchains, | 
|  | 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()); | 
|  | } |