• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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