mirror of
https://github.com/klzgrad/naiveproxy.git
synced 2024-12-01 01:36:09 +03:00
210 lines
6.4 KiB
C++
210 lines
6.4 KiB
C++
|
// 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() {
|
||
|
}
|
||
|
|
||
|
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,
|
||
|
SUBSTITUTION_RSP_FILE_NAME);
|
||
|
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;
|
||
|
}
|