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
3 // found in the LICENSE file.
4
5 #include "gn/ninja_action_target_writer.h"
6
7 #include <stddef.h>
8
9 #include "base/strings/string_util.h"
10 #include "gn/deps_iterator.h"
11 #include "gn/err.h"
12 #include "gn/general_tool.h"
13 #include "gn/pool.h"
14 #include "gn/settings.h"
15 #include "gn/string_utils.h"
16 #include "gn/substitution_writer.h"
17 #include "gn/target.h"
18
NinjaActionTargetWriter(const Target * target,std::ostream & out)19 NinjaActionTargetWriter::NinjaActionTargetWriter(const Target* target,
20 std::ostream& out)
21 : NinjaTargetWriter(target, out),
22 path_output_no_escaping_(
23 target->settings()->build_settings()->build_dir(),
24 target->settings()->build_settings()->root_path_utf8(),
25 ESCAPE_NONE) {}
26
27 NinjaActionTargetWriter::~NinjaActionTargetWriter() = default;
28
Run()29 void NinjaActionTargetWriter::Run() {
30 std::string custom_rule_name = WriteRuleDefinition();
31
32 // Collect our deps to pass as additional "hard dependencies" for input deps.
33 // This will force all of the action's dependencies to be completed before
34 // the action is run. Usually, if an action has a dependency, it will be
35 // operating on the result of that previous step, so we need to be sure to
36 // serialize these.
37 std::vector<const Target*> additional_hard_deps;
38 std::vector<OutputFile> data_outs;
39 for (const auto& pair : target_->GetDeps(Target::DEPS_LINKED)) {
40 if (pair.ptr->IsDataOnly()) {
41 data_outs.push_back(pair.ptr->dependency_output_file());
42 } else {
43 additional_hard_deps.push_back(pair.ptr);
44 }
45 }
46
47 // For ACTIONs, the input deps appear only once in the generated ninja
48 // file, so WriteInputDepsStampAndGetDep() won't create a stamp file
49 // and the action will just depend on all the input deps directly.
50 size_t num_stamp_uses =
51 target_->output_type() == Target::ACTION ? 1u : target_->sources().size();
52 std::vector<OutputFile> input_deps =
53 WriteInputDepsStampAndGetDep(additional_hard_deps, num_stamp_uses);
54 out_ << std::endl;
55
56 // Collects all output files for writing below.
57 std::vector<OutputFile> output_files;
58
59 if (target_->output_type() == Target::ACTION_FOREACH) {
60 // Write separate build lines for each input source file.
61 WriteSourceRules(custom_rule_name, input_deps, &output_files);
62 } else {
63 DCHECK(target_->output_type() == Target::ACTION);
64
65 // Write a rule that invokes the script once with the outputs as outputs,
66 // and the data as inputs. It does not depend on the sources.
67 out_ << "build";
68 SubstitutionWriter::GetListAsOutputFiles(
69 settings_, target_->action_values().outputs(), &output_files);
70 path_output_.WriteFiles(out_, output_files);
71
72 out_ << ": " << custom_rule_name;
73 if (!input_deps.empty()) {
74 // As in WriteSourceRules, we want to force this target to rebuild any
75 // time any of its dependencies change.
76 out_ << " |";
77 path_output_.WriteFiles(out_, input_deps);
78 }
79 out_ << std::endl;
80 if (target_->action_values().has_depfile()) {
81 WriteDepfile(SourceFile());
82 }
83
84 WriteNinjaVariablesForAction();
85
86 if (target_->action_values().pool().ptr) {
87 out_ << " pool = ";
88 out_ << target_->action_values().pool().ptr->GetNinjaName(
89 settings_->default_toolchain_label());
90 out_ << std::endl;
91 }
92 }
93 out_ << std::endl;
94
95 // Write the stamp, which also depends on all data deps. These are needed at
96 // runtime and should be compiled when the action is, but don't need to be
97 // done before we run the action.
98 // TODO(thakis): If the action has just a single output, make things depend
99 // on that output directly without writing a stamp file.
100 for (const auto& dep : target_->data_deps())
101 data_outs.push_back(dep.ptr->dependency_output_file());
102 WriteStampForTarget(output_files, data_outs);
103 }
104
WriteRuleDefinition()105 std::string NinjaActionTargetWriter::WriteRuleDefinition() {
106 // Make a unique name for this rule.
107 //
108 // Use a unique name for the response file when there are multiple build
109 // steps so that they don't stomp on each other. When there are no sources,
110 // there will be only one invocation so we can use a simple name.
111 std::string target_label = target_->label().GetUserVisibleName(true);
112 std::string custom_rule_name(target_label);
113 base::ReplaceChars(custom_rule_name, ":/()", "_", &custom_rule_name);
114 custom_rule_name.append("_rule");
115
116 const SubstitutionList& args = target_->action_values().args();
117 EscapeOptions args_escape_options;
118 args_escape_options.mode = ESCAPE_NINJA_COMMAND;
119
120 out_ << "rule " << custom_rule_name << std::endl;
121
122 if (target_->action_values().uses_rsp_file()) {
123 // Needs a response file. The unique_name part is for action_foreach so
124 // each invocation of the rule gets a different response file. This isn't
125 // strictly necessary for regular one-shot actions, but it's easier to
126 // just always define unique_name.
127 std::string rspfile = custom_rule_name;
128 if (!target_->sources().empty())
129 rspfile += ".$unique_name";
130 rspfile += ".rsp";
131 out_ << " rspfile = " << rspfile << std::endl;
132
133 // Response file contents.
134 out_ << " rspfile_content =";
135 for (const auto& arg :
136 target_->action_values().rsp_file_contents().list()) {
137 out_ << " ";
138 SubstitutionWriter::WriteWithNinjaVariables(arg, args_escape_options,
139 out_);
140 }
141 out_ << std::endl;
142 }
143
144 out_ << " command = ";
145 path_output_.WriteFile(out_, settings_->build_settings()->python_path());
146 out_ << " ";
147 path_output_.WriteFile(out_, target_->action_values().script());
148 for (const auto& arg : args.list()) {
149 out_ << " ";
150 SubstitutionWriter::WriteWithNinjaVariables(arg, args_escape_options, out_);
151 }
152 out_ << std::endl;
153 out_ << " description = ACTION " << target_label << std::endl;
154 out_ << " restat = 1" << std::endl;
155 const Tool* tool =
156 target_->toolchain()->GetTool(GeneralTool::kGeneralToolAction);
157 if (tool && tool->pool().ptr) {
158 out_ << " pool = ";
159 out_ << tool->pool().ptr->GetNinjaName(
160 settings_->default_toolchain_label());
161 out_ << std::endl;
162 }
163
164 return custom_rule_name;
165 }
166
WriteSourceRules(const std::string & custom_rule_name,const std::vector<OutputFile> & input_deps,std::vector<OutputFile> * output_files)167 void NinjaActionTargetWriter::WriteSourceRules(
168 const std::string& custom_rule_name,
169 const std::vector<OutputFile>& input_deps,
170 std::vector<OutputFile>* output_files) {
171 EscapeOptions args_escape_options;
172 args_escape_options.mode = ESCAPE_NINJA_COMMAND;
173 // We're writing the substitution values, these should not be quoted since
174 // they will get pasted into the real command line.
175 args_escape_options.inhibit_quoting = true;
176
177 const Target::FileList& sources = target_->sources();
178 for (size_t i = 0; i < sources.size(); i++) {
179 out_ << "build";
180 WriteOutputFilesForBuildLine(sources[i], output_files);
181
182 out_ << ": " << custom_rule_name << " ";
183 path_output_.WriteFile(out_, sources[i]);
184 if (!input_deps.empty()) {
185 // Using "|" for the dependencies forces all implicit dependencies to be
186 // fully up to date before running the action, and will re-run this
187 // action if any input dependencies change. This is important because
188 // this action may consume the outputs of previous steps.
189 out_ << " |";
190 path_output_.WriteFiles(out_, input_deps);
191 }
192 out_ << std::endl;
193
194 // Response files require a unique name be defined.
195 if (target_->action_values().uses_rsp_file())
196 out_ << " unique_name = " << i << std::endl;
197
198 // The required types is the union of the args and response file. This
199 // might theoretically duplicate a definition if the same substitution is
200 // used in both the args and the response file. However, this should be
201 // very unusual (normally the substitutions will go in one place or the
202 // other) and the redundant assignment won't bother Ninja.
203 SubstitutionWriter::WriteNinjaVariablesForSource(
204 target_, settings_, sources[i],
205 target_->action_values().args().required_types(), args_escape_options,
206 out_);
207 SubstitutionWriter::WriteNinjaVariablesForSource(
208 target_, settings_, sources[i],
209 target_->action_values().rsp_file_contents().required_types(),
210 args_escape_options, out_);
211 WriteNinjaVariablesForAction();
212
213 if (target_->action_values().has_depfile()) {
214 WriteDepfile(sources[i]);
215 }
216 if (target_->action_values().pool().ptr) {
217 out_ << " pool = ";
218 out_ << target_->action_values().pool().ptr->GetNinjaName(
219 settings_->default_toolchain_label());
220 out_ << std::endl;
221 }
222 }
223 }
224
WriteOutputFilesForBuildLine(const SourceFile & source,std::vector<OutputFile> * output_files)225 void NinjaActionTargetWriter::WriteOutputFilesForBuildLine(
226 const SourceFile& source,
227 std::vector<OutputFile>* output_files) {
228 size_t first_output_index = output_files->size();
229
230 SubstitutionWriter::ApplyListToSourceAsOutputFile(
231 target_, settings_, target_->action_values().outputs(), source,
232 output_files);
233
234 for (size_t i = first_output_index; i < output_files->size(); i++) {
235 out_ << " ";
236 path_output_.WriteFile(out_, (*output_files)[i]);
237 }
238 }
239
WriteDepfile(const SourceFile & source)240 void NinjaActionTargetWriter::WriteDepfile(const SourceFile& source) {
241 out_ << " depfile = ";
242 path_output_.WriteFile(
243 out_,
244 SubstitutionWriter::ApplyPatternToSourceAsOutputFile(
245 target_, settings_, target_->action_values().depfile(), source));
246 out_ << std::endl;
247 // Using "deps = gcc" allows Ninja to read and store the depfile content in
248 // its internal database which improves performance, especially for large
249 // depfiles. The use of this feature with depfiles that contain multiple
250 // outputs require Ninja version 1.9.0 or newer.
251 if (settings_->build_settings()->ninja_required_version() >=
252 Version{1, 9, 0}) {
253 out_ << " deps = gcc" << std::endl;
254 }
255 }
256
WriteNinjaVariablesForAction()257 void NinjaActionTargetWriter::WriteNinjaVariablesForAction() {
258 SubstitutionBits subst;
259 target_->action_values().args().FillRequiredTypes(&subst);
260 WriteRustCompilerVars(subst, /*indent=*/true, /*always_write=*/false);
261 WriteCCompilerVars(subst, /*indent=*/true, /*respect_source_types=*/false);
262 }
263