OLD | NEW |
---|---|
1 // Copyright (c) 2013 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2013 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "tools/gn/ninja_target_writer.h" | 5 #include "tools/gn/ninja_target_writer.h" |
6 | 6 |
7 #include <fstream> | 7 #include <fstream> |
8 #include <sstream> | 8 #include <sstream> |
9 | 9 |
10 #include "base/file_util.h" | 10 #include "base/file_util.h" |
11 #include "base/strings/string_util.h" | |
11 #include "tools/gn/err.h" | 12 #include "tools/gn/err.h" |
13 #include "tools/gn/filesystem_utils.h" | |
12 #include "tools/gn/ninja_action_target_writer.h" | 14 #include "tools/gn/ninja_action_target_writer.h" |
13 #include "tools/gn/ninja_binary_target_writer.h" | 15 #include "tools/gn/ninja_binary_target_writer.h" |
14 #include "tools/gn/ninja_copy_target_writer.h" | 16 #include "tools/gn/ninja_copy_target_writer.h" |
15 #include "tools/gn/ninja_group_target_writer.h" | 17 #include "tools/gn/ninja_group_target_writer.h" |
18 #include "tools/gn/ninja_utils.h" | |
16 #include "tools/gn/scheduler.h" | 19 #include "tools/gn/scheduler.h" |
17 #include "tools/gn/string_utils.h" | 20 #include "tools/gn/string_utils.h" |
21 #include "tools/gn/substitution_writer.h" | |
18 #include "tools/gn/target.h" | 22 #include "tools/gn/target.h" |
19 #include "tools/gn/trace.h" | 23 #include "tools/gn/trace.h" |
20 | 24 |
21 NinjaTargetWriter::NinjaTargetWriter(const Target* target, | 25 NinjaTargetWriter::NinjaTargetWriter(const Target* target, |
22 const Toolchain* toolchain, | |
23 std::ostream& out) | 26 std::ostream& out) |
24 : settings_(target->settings()), | 27 : settings_(target->settings()), |
25 target_(target), | 28 target_(target), |
26 toolchain_(toolchain), | |
27 out_(out), | 29 out_(out), |
28 path_output_(settings_->build_settings()->build_dir(), ESCAPE_NINJA), | 30 path_output_(settings_->build_settings()->build_dir(), ESCAPE_NINJA) { |
29 helper_(settings_->build_settings()) { | |
30 } | 31 } |
31 | 32 |
32 NinjaTargetWriter::~NinjaTargetWriter() { | 33 NinjaTargetWriter::~NinjaTargetWriter() { |
33 } | 34 } |
34 | 35 |
35 // static | 36 // static |
36 void NinjaTargetWriter::RunAndWriteFile(const Target* target, | 37 void NinjaTargetWriter::RunAndWriteFile(const Target* target) { |
37 const Toolchain* toolchain) { | |
38 const Settings* settings = target->settings(); | 38 const Settings* settings = target->settings(); |
39 NinjaHelper helper(settings->build_settings()); | |
40 | 39 |
41 ScopedTrace trace(TraceItem::TRACE_FILE_WRITE, | 40 ScopedTrace trace(TraceItem::TRACE_FILE_WRITE, |
42 target->label().GetUserVisibleName(false)); | 41 target->label().GetUserVisibleName(false)); |
43 trace.SetToolchain(settings->toolchain_label()); | 42 trace.SetToolchain(settings->toolchain_label()); |
44 | 43 |
45 base::FilePath ninja_file(settings->build_settings()->GetFullPath( | 44 base::FilePath ninja_file(settings->build_settings()->GetFullPath( |
46 helper.GetNinjaFileForTarget(target).GetSourceFile( | 45 GetNinjaFileForTarget(target))); |
47 settings->build_settings()))); | |
48 | 46 |
49 if (g_scheduler->verbose_logging()) | 47 if (g_scheduler->verbose_logging()) |
50 g_scheduler->Log("Writing", FilePathToUTF8(ninja_file)); | 48 g_scheduler->Log("Writing", FilePathToUTF8(ninja_file)); |
51 | 49 |
52 base::CreateDirectory(ninja_file.DirName()); | 50 base::CreateDirectory(ninja_file.DirName()); |
53 | 51 |
54 // It's rediculously faster to write to a string and then write that to | 52 // It's rediculously faster to write to a string and then write that to |
55 // disk in one operation than to use an fstream here. | 53 // disk in one operation than to use an fstream here. |
56 std::stringstream file; | 54 std::stringstream file; |
57 | 55 |
58 // Call out to the correct sub-type of writer. | 56 // Call out to the correct sub-type of writer. |
59 if (target->output_type() == Target::COPY_FILES) { | 57 if (target->output_type() == Target::COPY_FILES) { |
60 NinjaCopyTargetWriter writer(target, toolchain, file); | 58 NinjaCopyTargetWriter writer(target, file); |
61 writer.Run(); | 59 writer.Run(); |
62 } else if (target->output_type() == Target::ACTION || | 60 } else if (target->output_type() == Target::ACTION || |
63 target->output_type() == Target::ACTION_FOREACH) { | 61 target->output_type() == Target::ACTION_FOREACH) { |
64 NinjaActionTargetWriter writer(target, toolchain, file); | 62 NinjaActionTargetWriter writer(target, file); |
65 writer.Run(); | 63 writer.Run(); |
66 } else if (target->output_type() == Target::GROUP) { | 64 } else if (target->output_type() == Target::GROUP) { |
67 NinjaGroupTargetWriter writer(target, toolchain, file); | 65 NinjaGroupTargetWriter writer(target, file); |
68 writer.Run(); | 66 writer.Run(); |
69 } else if (target->output_type() == Target::EXECUTABLE || | 67 } else if (target->output_type() == Target::EXECUTABLE || |
70 target->output_type() == Target::STATIC_LIBRARY || | 68 target->output_type() == Target::STATIC_LIBRARY || |
71 target->output_type() == Target::SHARED_LIBRARY || | 69 target->output_type() == Target::SHARED_LIBRARY || |
72 target->output_type() == Target::SOURCE_SET) { | 70 target->output_type() == Target::SOURCE_SET) { |
73 NinjaBinaryTargetWriter writer(target, toolchain, file); | 71 NinjaBinaryTargetWriter writer(target, file); |
74 writer.Run(); | 72 writer.Run(); |
75 } else { | 73 } else { |
76 CHECK(0); | 74 CHECK(0); |
77 } | 75 } |
78 | 76 |
79 std::string contents = file.str(); | 77 std::string contents = file.str(); |
80 base::WriteFile(ninja_file, contents.c_str(), | 78 base::WriteFile(ninja_file, contents.c_str(), |
81 static_cast<int>(contents.size())); | 79 static_cast<int>(contents.size())); |
82 } | 80 } |
83 | 81 |
82 void NinjaTargetWriter::WriteSharedVars(const SubstitutionBits& bits) { | |
83 bool written_anything = false; | |
84 | |
85 // Target label. | |
86 if (bits.used[SUBSTITUTION_LABEL]) { | |
87 out_ << kSubstitutionNinjaNames[SUBSTITUTION_LABEL] << " = " | |
88 << SubstitutionWriter::GetTargetSubstitution( | |
89 target_, SUBSTITUTION_LABEL) | |
90 << std::endl; | |
91 written_anything = true; | |
92 } | |
93 | |
94 // Root gen dir. | |
95 if (bits.used[SUBSTITUTION_ROOT_GEN_DIR]) { | |
96 out_ << kSubstitutionNinjaNames[SUBSTITUTION_ROOT_GEN_DIR] << " = " | |
97 << SubstitutionWriter::GetTargetSubstitution( | |
98 target_, SUBSTITUTION_ROOT_GEN_DIR) | |
99 << std::endl; | |
100 written_anything = true; | |
101 } | |
102 | |
103 // Root out dir. | |
104 if (bits.used[SUBSTITUTION_ROOT_OUT_DIR]) { | |
105 out_ << kSubstitutionNinjaNames[SUBSTITUTION_ROOT_OUT_DIR] << " = " | |
106 << SubstitutionWriter::GetTargetSubstitution( | |
107 target_, SUBSTITUTION_ROOT_OUT_DIR) | |
108 << std::endl; | |
109 written_anything = true; | |
110 } | |
111 | |
112 // Target gen dir. | |
113 if (bits.used[SUBSTITUTION_TARGET_GEN_DIR]) { | |
114 out_ << kSubstitutionNinjaNames[SUBSTITUTION_TARGET_GEN_DIR] << " = " | |
115 << SubstitutionWriter::GetTargetSubstitution( | |
116 target_, SUBSTITUTION_TARGET_GEN_DIR) | |
117 << std::endl; | |
118 written_anything = true; | |
119 } | |
120 | |
121 // Target out dir. | |
122 if (bits.used[SUBSTITUTION_TARGET_OUT_DIR]) { | |
123 out_ << kSubstitutionNinjaNames[SUBSTITUTION_TARGET_OUT_DIR] << " = " | |
124 << SubstitutionWriter::GetTargetSubstitution( | |
125 target_, SUBSTITUTION_TARGET_OUT_DIR) | |
126 << std::endl; | |
127 written_anything = true; | |
128 } | |
129 | |
130 // Target output name. | |
131 if (bits.used[SUBSTITUTION_TARGET_OUTPUT_NAME]) { | |
132 out_ << kSubstitutionNinjaNames[SUBSTITUTION_TARGET_OUTPUT_NAME] << " = " | |
133 << SubstitutionWriter::GetTargetSubstitution( | |
134 target_, SUBSTITUTION_TARGET_OUTPUT_NAME) | |
135 << std::endl; | |
136 written_anything = true; | |
137 } | |
138 | |
139 // If we wrote any vars, separate them from the rest of the file that follows | |
140 // with a blank line. | |
141 if (written_anything) | |
142 out_ << std::endl; | |
143 } | |
144 | |
84 std::string NinjaTargetWriter::WriteInputDepsStampAndGetDep( | 145 std::string NinjaTargetWriter::WriteInputDepsStampAndGetDep( |
85 const std::vector<const Target*>& extra_hard_deps) const { | 146 const std::vector<const Target*>& extra_hard_deps) const { |
147 CHECK(target_->toolchain()) | |
148 << "Toolchain not set on target " | |
149 << target_->label().GetUserVisibleName(true); | |
150 | |
86 // For an action (where we run a script only once) the sources are the same | 151 // For an action (where we run a script only once) the sources are the same |
87 // as the source prereqs. | 152 // as the source prereqs. |
88 bool list_sources_as_input_deps = (target_->output_type() == Target::ACTION); | 153 bool list_sources_as_input_deps = (target_->output_type() == Target::ACTION); |
89 | 154 |
90 // Actions get implicit dependencies on the script itself. | 155 // Actions get implicit dependencies on the script itself. |
91 bool add_script_source_as_dep = | 156 bool add_script_source_as_dep = |
92 (target_->output_type() == Target::ACTION) || | 157 (target_->output_type() == Target::ACTION) || |
93 (target_->output_type() == Target::ACTION_FOREACH); | 158 (target_->output_type() == Target::ACTION_FOREACH); |
94 | 159 |
95 if (!add_script_source_as_dep && | 160 if (!add_script_source_as_dep && |
96 extra_hard_deps.empty() && | 161 extra_hard_deps.empty() && |
97 target_->inputs().empty() && | 162 target_->inputs().empty() && |
98 target_->recursive_hard_deps().empty() && | 163 target_->recursive_hard_deps().empty() && |
99 (!list_sources_as_input_deps || target_->sources().empty()) && | 164 (!list_sources_as_input_deps || target_->sources().empty()) && |
100 toolchain_->deps().empty()) | 165 target_->toolchain()->deps().empty()) |
101 return std::string(); // No input/hard deps. | 166 return std::string(); // No input/hard deps. |
102 | 167 |
103 // One potential optimization is if there are few input dependencies (or | 168 // One potential optimization is if there are few input dependencies (or |
104 // potentially few sources that depend on these) it's better to just write | 169 // potentially few sources that depend on these) it's better to just write |
105 // all hard deps on each sources line than have this intermediate stamp. We | 170 // all hard deps on each sources line than have this intermediate stamp. We |
106 // do the stamp file because duplicating all the order-only deps for each | 171 // do the stamp file because duplicating all the order-only deps for each |
107 // source file can really explode the ninja file but this won't be the most | 172 // source file can really explode the ninja file but this won't be the most |
108 // optimal thing in all cases. | 173 // optimal thing in all cases. |
109 | 174 |
110 OutputFile input_stamp_file = helper_.GetTargetOutputDir(target_); | 175 OutputFile input_stamp_file( |
176 RebaseSourceAbsolutePath(GetTargetOutputDir(target_).value(), | |
177 settings_->build_settings()->build_dir())); | |
111 input_stamp_file.value().append(target_->label().name()); | 178 input_stamp_file.value().append(target_->label().name()); |
112 input_stamp_file.value().append(".inputdeps.stamp"); | 179 input_stamp_file.value().append(".inputdeps.stamp"); |
113 | 180 |
114 std::ostringstream stamp_file_stream; | 181 std::ostringstream stamp_file_stream; |
115 path_output_.WriteFile(stamp_file_stream, input_stamp_file); | 182 path_output_.WriteFile(stamp_file_stream, input_stamp_file); |
116 std::string stamp_file_string = stamp_file_stream.str(); | 183 std::string stamp_file_string = stamp_file_stream.str(); |
117 | 184 |
118 out_ << "build " << stamp_file_string << ": " + | 185 out_ << "build " << stamp_file_string << ": " |
119 helper_.GetRulePrefix(settings_) + "stamp"; | 186 << GetNinjaRulePrefixForToolchain(settings_) |
187 << Toolchain::ToolTypeToName(Toolchain::TYPE_STAMP); | |
120 | 188 |
121 // Script file (if applicable). | 189 // Script file (if applicable). |
122 if (add_script_source_as_dep) { | 190 if (add_script_source_as_dep) { |
123 out_ << " "; | 191 out_ << " "; |
124 path_output_.WriteFile(out_, target_->action_values().script()); | 192 path_output_.WriteFile(out_, target_->action_values().script()); |
125 } | 193 } |
126 | 194 |
127 // Input files are order-only deps. | 195 // Input files are order-only deps. |
128 const Target::FileList& prereqs = target_->inputs(); | 196 const Target::FileList& prereqs = target_->inputs(); |
129 for (size_t i = 0; i < prereqs.size(); i++) { | 197 for (size_t i = 0; i < prereqs.size(); i++) { |
130 out_ << " "; | 198 out_ << " "; |
131 path_output_.WriteFile(out_, prereqs[i]); | 199 path_output_.WriteFile(out_, prereqs[i]); |
132 } | 200 } |
133 if (list_sources_as_input_deps) { | 201 if (list_sources_as_input_deps) { |
134 const Target::FileList& sources = target_->sources(); | 202 const Target::FileList& sources = target_->sources(); |
135 for (size_t i = 0; i < sources.size(); i++) { | 203 for (size_t i = 0; i < sources.size(); i++) { |
136 out_ << " "; | 204 out_ << " "; |
137 path_output_.WriteFile(out_, sources[i]); | 205 path_output_.WriteFile(out_, sources[i]); |
138 } | 206 } |
139 } | 207 } |
140 | 208 |
141 // Add on any hard deps that are direct or indirect dependencies. | 209 // The different souces of input deps may duplicate some targets, so uniquify |
210 // them (ordering doesn't matter for this case). | |
211 std::set<const Target*> unique_deps; | |
212 | |
213 // Hard dependencies that are direct or indirect dependencies. | |
142 const std::set<const Target*>& hard_deps = target_->recursive_hard_deps(); | 214 const std::set<const Target*>& hard_deps = target_->recursive_hard_deps(); |
143 for (std::set<const Target*>::const_iterator i = hard_deps.begin(); | 215 for (std::set<const Target*>::const_iterator i = hard_deps.begin(); |
144 i != hard_deps.end(); ++i) { | 216 i != hard_deps.end(); ++i) |
jamesr
2014/08/19 19:30:41
think this gets {} since the for loop is multiline
| |
145 out_ << " "; | 217 unique_deps.insert(*i); |
146 path_output_.WriteFile(out_, helper_.GetTargetOutputFile(*i)); | 218 |
147 } | 219 // Extra hard dependencies passed in. |
220 unique_deps.insert(extra_hard_deps.begin(), extra_hard_deps.end()); | |
148 | 221 |
149 // Toolchain dependencies. These must be resolved before doing anything. | 222 // Toolchain dependencies. These must be resolved before doing anything. |
150 // This just writs all toolchain deps for simplicity. If we find that | 223 // This just writs all toolchain deps for simplicity. If we find that |
151 // toolchains often have more than one dependency, we could consider writing | 224 // toolchains often have more than one dependency, we could consider writing |
152 // a toolchain-specific stamp file and only include the stamp here. | 225 // a toolchain-specific stamp file and only include the stamp here. |
153 const LabelTargetVector& toolchain_deps = toolchain_->deps(); | 226 const LabelTargetVector& toolchain_deps = target_->toolchain()->deps(); |
154 for (size_t i = 0; i < toolchain_deps.size(); i++) { | 227 for (size_t i = 0; i < toolchain_deps.size(); i++) |
228 unique_deps.insert(toolchain_deps[i].ptr); | |
229 | |
230 for (std::set<const Target*>::const_iterator i = unique_deps.begin(); | |
231 i != unique_deps.end(); ++i) { | |
232 DCHECK(!(*i)->dependency_output_file().value().empty()); | |
155 out_ << " "; | 233 out_ << " "; |
156 path_output_.WriteFile(out_, | 234 path_output_.WriteFile(out_, (*i)->dependency_output_file()); |
157 helper_.GetTargetOutputFile(toolchain_deps[i].ptr)); | |
158 } | |
159 | |
160 // Extra hard deps passed in. | |
161 for (size_t i = 0; i < extra_hard_deps.size(); i++) { | |
162 out_ << " "; | |
163 path_output_.WriteFile(out_, | |
164 helper_.GetTargetOutputFile(extra_hard_deps[i])); | |
165 } | 235 } |
166 | 236 |
167 out_ << "\n"; | 237 out_ << "\n"; |
168 return " | " + stamp_file_string; | 238 return " | " + stamp_file_string; |
169 } | 239 } |
240 | |
241 void NinjaTargetWriter::WriteStampForTarget( | |
242 const std::vector<OutputFile>& files, | |
243 const std::vector<OutputFile>& order_only_deps) { | |
244 const OutputFile& stamp_file = target_->dependency_output_file(); | |
245 | |
246 // First validate that the target's dependency is a stamp file. Otherwise, | |
247 // we shouldn't have gotten here! | |
248 CHECK(EndsWith(stamp_file.value(), ".stamp", false)) | |
249 << "Output should end in \".stamp\" for stamp file output. Instead got: " | |
250 << "\"" << stamp_file.value() << "\""; | |
251 | |
252 out_ << "build "; | |
253 path_output_.WriteFile(out_, stamp_file); | |
254 | |
255 out_ << ": " | |
256 << GetNinjaRulePrefixForToolchain(settings_) | |
257 << Toolchain::ToolTypeToName(Toolchain::TYPE_STAMP); | |
258 path_output_.WriteFiles(out_, files); | |
259 | |
260 if (!order_only_deps.empty()) { | |
261 out_ << " ||"; | |
262 path_output_.WriteFiles(out_, order_only_deps); | |
263 } | |
264 out_ << std::endl; | |
265 } | |
OLD | NEW |