| // Copyright (c) 2013 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 "tools/gn/action_target_generator.h" |
| |
| #include "base/stl_util.h" |
| #include "tools/gn/build_settings.h" |
| #include "tools/gn/err.h" |
| #include "tools/gn/filesystem_utils.h" |
| #include "tools/gn/functions.h" |
| #include "tools/gn/parse_tree.h" |
| #include "tools/gn/scope.h" |
| #include "tools/gn/value.h" |
| #include "tools/gn/value_extractors.h" |
| #include "tools/gn/variables.h" |
| |
| ActionTargetGenerator::ActionTargetGenerator( |
| Target* target, |
| Scope* scope, |
| const FunctionCallNode* function_call, |
| Target::OutputType type, |
| Err* err) |
| : TargetGenerator(target, scope, function_call, err), output_type_(type) {} |
| |
| ActionTargetGenerator::~ActionTargetGenerator() = default; |
| |
| void ActionTargetGenerator::DoRun() { |
| target_->set_output_type(output_type_); |
| |
| if (!FillSources()) |
| return; |
| if (output_type_ == Target::ACTION_FOREACH && target_->sources().empty()) { |
| // Foreach rules must always have some sources to have an effect. |
| *err_ = |
| Err(function_call_, "action_foreach target has no sources.", |
| "If you don't specify any sources, there is nothing to run your\n" |
| "script over."); |
| return; |
| } |
| |
| if (!FillInputs()) |
| return; |
| |
| if (!FillScript()) |
| return; |
| |
| if (!FillScriptArgs()) |
| return; |
| |
| if (!FillResponseFileContents()) |
| return; |
| |
| if (!FillOutputs(output_type_ == Target::ACTION_FOREACH)) |
| return; |
| |
| if (!FillDepfile()) |
| return; |
| |
| if (!FillPool()) |
| return; |
| |
| if (!FillCheckIncludes()) |
| return; |
| |
| if (!CheckOutputs()) |
| return; |
| |
| // Action outputs don't depend on the current toolchain so we can skip adding |
| // that dependency. |
| |
| // response_file_contents and {{response_file_name}} in the args must go |
| // together. |
| const auto& required_args_substitutions = |
| target_->action_values().args().required_types(); |
| bool has_rsp_file_name = base::ContainsValue(required_args_substitutions, |
| &SubstitutionRspFileName); |
| if (target_->action_values().uses_rsp_file() && !has_rsp_file_name) { |
| *err_ = Err( |
| function_call_, "Missing {{response_file_name}} in args.", |
| "This target defines response_file_contents but doesn't use\n" |
| "{{response_file_name}} in the args, which means the response file\n" |
| "will be unused."); |
| return; |
| } |
| if (!target_->action_values().uses_rsp_file() && has_rsp_file_name) { |
| *err_ = Err( |
| function_call_, "Missing response_file_contents definition.", |
| "This target uses {{response_file_name}} in the args, but does not\n" |
| "define response_file_contents which means the response file\n" |
| "will be empty."); |
| return; |
| } |
| } |
| |
| bool ActionTargetGenerator::FillScript() { |
| // If this gets called, the target type requires a script, so error out |
| // if it doesn't have one. |
| const Value* value = scope_->GetValue(variables::kScript, true); |
| if (!value) { |
| *err_ = Err(function_call_, "This target type requires a \"script\"."); |
| return false; |
| } |
| if (!value->VerifyTypeIs(Value::STRING, err_)) |
| return false; |
| |
| SourceFile script_file = scope_->GetSourceDir().ResolveRelativeFile( |
| *value, err_, scope_->settings()->build_settings()->root_path_utf8()); |
| if (err_->has_error()) |
| return false; |
| target_->action_values().set_script(script_file); |
| return true; |
| } |
| |
| bool ActionTargetGenerator::FillScriptArgs() { |
| const Value* value = scope_->GetValue(variables::kArgs, true); |
| if (!value) |
| return true; // Nothing to do. |
| |
| if (!target_->action_values().args().Parse(*value, err_)) |
| return false; |
| if (!EnsureValidSubstitutions( |
| target_->action_values().args().required_types(), |
| &IsValidScriptArgsSubstitution, value->origin(), err_)) |
| return false; |
| |
| return true; |
| } |
| |
| bool ActionTargetGenerator::FillResponseFileContents() { |
| const Value* value = scope_->GetValue(variables::kResponseFileContents, true); |
| if (!value) |
| return true; // Nothing to do. |
| |
| if (!target_->action_values().rsp_file_contents().Parse(*value, err_)) |
| return false; |
| if (!EnsureValidSubstitutions( |
| target_->action_values().rsp_file_contents().required_types(), |
| &IsValidSourceSubstitution, value->origin(), err_)) |
| return false; |
| |
| return true; |
| } |
| |
| bool ActionTargetGenerator::FillDepfile() { |
| const Value* value = scope_->GetValue(variables::kDepfile, true); |
| if (!value) |
| return true; |
| |
| SubstitutionPattern depfile; |
| if (!depfile.Parse(*value, err_)) |
| return false; |
| if (!EnsureSubstitutionIsInOutputDir(depfile, *value)) |
| return false; |
| |
| target_->action_values().set_depfile(depfile); |
| return true; |
| } |
| |
| bool ActionTargetGenerator::FillPool() { |
| const Value* value = scope_->GetValue(variables::kPool, true); |
| if (!value) |
| return true; |
| |
| Label label = Label::Resolve(scope_->GetSourceDir(), |
| ToolchainLabelForScope(scope_), *value, err_); |
| if (err_->has_error()) |
| return false; |
| |
| LabelPtrPair<Pool> pair(label); |
| pair.origin = target_->defined_from(); |
| |
| target_->action_values().set_pool(std::move(pair)); |
| return true; |
| } |
| |
| bool ActionTargetGenerator::CheckOutputs() { |
| const SubstitutionList& outputs = target_->action_values().outputs(); |
| if (outputs.list().empty()) { |
| *err_ = |
| Err(function_call_, "Action has no outputs.", |
| "If you have no outputs, the build system can not tell when your\n" |
| "script needs to be run."); |
| return false; |
| } |
| |
| if (output_type_ == Target::ACTION) { |
| if (!outputs.required_types().empty()) { |
| *err_ = Err( |
| function_call_, "Action has patterns in the output.", |
| "An action target should have the outputs completely specified. If\n" |
| "you want to provide a mapping from source to output, use an\n" |
| "\"action_foreach\" target."); |
| return false; |
| } |
| } else if (output_type_ == Target::ACTION_FOREACH) { |
| // A foreach target should always have a pattern in the outputs. |
| if (outputs.required_types().empty()) { |
| *err_ = Err( |
| function_call_, "action_foreach should have a pattern in the output.", |
| "An action_foreach target should have a source expansion pattern in\n" |
| "it to map source file to unique output file name. Otherwise, the\n" |
| "build system can't determine when your script needs to be run."); |
| return false; |
| } |
| } |
| return true; |
| } |
| |
| bool ActionTargetGenerator::FillInputs() { |
| const Value* value = scope_->GetValue(variables::kInputs, true); |
| if (!value) |
| return true; |
| |
| Target::FileList dest_inputs; |
| if (!ExtractListOfRelativeFiles(scope_->settings()->build_settings(), *value, |
| scope_->GetSourceDir(), &dest_inputs, err_)) |
| return false; |
| target_->config_values().inputs().swap(dest_inputs); |
| return true; |
| } |