mirror of
https://github.com/klzgrad/naiveproxy.git
synced 2024-11-24 14:26:09 +03:00
875 lines
33 KiB
C++
875 lines
33 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/target.h"
|
||
|
|
||
|
#include <stddef.h>
|
||
|
|
||
|
#include "base/bind.h"
|
||
|
#include "base/stl_util.h"
|
||
|
#include "base/strings/string_util.h"
|
||
|
#include "base/strings/stringprintf.h"
|
||
|
#include "tools/gn/config_values_extractors.h"
|
||
|
#include "tools/gn/deps_iterator.h"
|
||
|
#include "tools/gn/filesystem_utils.h"
|
||
|
#include "tools/gn/functions.h"
|
||
|
#include "tools/gn/scheduler.h"
|
||
|
#include "tools/gn/source_file_type.h"
|
||
|
#include "tools/gn/substitution_writer.h"
|
||
|
#include "tools/gn/tool.h"
|
||
|
#include "tools/gn/toolchain.h"
|
||
|
#include "tools/gn/trace.h"
|
||
|
|
||
|
namespace {
|
||
|
|
||
|
typedef std::set<const Config*> ConfigSet;
|
||
|
|
||
|
// Merges the public configs from the given target to the given config list.
|
||
|
void MergePublicConfigsFrom(const Target* from_target,
|
||
|
UniqueVector<LabelConfigPair>* dest) {
|
||
|
const UniqueVector<LabelConfigPair>& pub = from_target->public_configs();
|
||
|
dest->Append(pub.begin(), pub.end());
|
||
|
}
|
||
|
|
||
|
// Like MergePublicConfigsFrom above except does the "all dependent" ones. This
|
||
|
// additionally adds all configs to the all_dependent_configs_ of the dest
|
||
|
// target given in *all_dest.
|
||
|
void MergeAllDependentConfigsFrom(const Target* from_target,
|
||
|
UniqueVector<LabelConfigPair>* dest,
|
||
|
UniqueVector<LabelConfigPair>* all_dest) {
|
||
|
for (const auto& pair : from_target->all_dependent_configs()) {
|
||
|
all_dest->push_back(pair);
|
||
|
dest->push_back(pair);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Err MakeTestOnlyError(const Target* from, const Target* to) {
|
||
|
return Err(from->defined_from(), "Test-only dependency not allowed.",
|
||
|
from->label().GetUserVisibleName(false) + "\n"
|
||
|
"which is NOT marked testonly can't depend on\n" +
|
||
|
to->label().GetUserVisibleName(false) + "\n"
|
||
|
"which is marked testonly. Only targets with \"testonly = true\"\n"
|
||
|
"can depend on other test-only targets.\n"
|
||
|
"\n"
|
||
|
"Either mark it test-only or don't do this dependency.");
|
||
|
}
|
||
|
|
||
|
// Set check_private_deps to true for the first invocation since a target
|
||
|
// can see all of its dependencies. For recursive invocations this will be set
|
||
|
// to false to follow only public dependency paths.
|
||
|
//
|
||
|
// Pass a pointer to an empty set for the first invocation. This will be used
|
||
|
// to avoid duplicate checking.
|
||
|
//
|
||
|
// Checking of object files is optional because it is much slower. This allows
|
||
|
// us to check targets for normal outputs, and then as a second pass check
|
||
|
// object files (since we know it will be an error otherwise). This allows
|
||
|
// us to avoid computing all object file names in the common case.
|
||
|
bool EnsureFileIsGeneratedByDependency(const Target* target,
|
||
|
const OutputFile& file,
|
||
|
bool check_private_deps,
|
||
|
bool consider_object_files,
|
||
|
bool check_data_deps,
|
||
|
std::set<const Target*>* seen_targets) {
|
||
|
if (seen_targets->find(target) != seen_targets->end())
|
||
|
return false; // Already checked this one and it's not found.
|
||
|
seen_targets->insert(target);
|
||
|
|
||
|
// Assume that we have relatively few generated inputs so brute-force
|
||
|
// searching here is OK. If this becomes a bottleneck, consider storing
|
||
|
// computed_outputs as a hash set.
|
||
|
for (const OutputFile& cur : target->computed_outputs()) {
|
||
|
if (file == cur)
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
if (file == target->write_runtime_deps_output())
|
||
|
return true;
|
||
|
|
||
|
// Check binary target intermediate files if requested.
|
||
|
if (consider_object_files && target->IsBinary()) {
|
||
|
std::vector<OutputFile> source_outputs;
|
||
|
for (const SourceFile& source : target->sources()) {
|
||
|
Toolchain::ToolType tool_type;
|
||
|
if (!target->GetOutputFilesForSource(source, &tool_type, &source_outputs))
|
||
|
continue;
|
||
|
if (base::ContainsValue(source_outputs, file))
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (check_data_deps) {
|
||
|
check_data_deps = false; // Consider only direct data_deps.
|
||
|
for (const auto& pair : target->data_deps()) {
|
||
|
if (EnsureFileIsGeneratedByDependency(pair.ptr, file, false,
|
||
|
consider_object_files,
|
||
|
check_data_deps, seen_targets))
|
||
|
return true; // Found a path.
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Check all public dependencies (don't do data ones since those are
|
||
|
// runtime-only).
|
||
|
for (const auto& pair : target->public_deps()) {
|
||
|
if (EnsureFileIsGeneratedByDependency(pair.ptr, file, false,
|
||
|
consider_object_files,
|
||
|
check_data_deps, seen_targets))
|
||
|
return true; // Found a path.
|
||
|
}
|
||
|
|
||
|
// Only check private deps if requested.
|
||
|
if (check_private_deps) {
|
||
|
for (const auto& pair : target->private_deps()) {
|
||
|
if (EnsureFileIsGeneratedByDependency(pair.ptr, file, false,
|
||
|
consider_object_files,
|
||
|
check_data_deps, seen_targets))
|
||
|
return true; // Found a path.
|
||
|
}
|
||
|
if (target->output_type() == Target::CREATE_BUNDLE) {
|
||
|
for (auto* dep : target->bundle_data().bundle_deps()) {
|
||
|
if (EnsureFileIsGeneratedByDependency(dep, file, false,
|
||
|
consider_object_files,
|
||
|
check_data_deps, seen_targets))
|
||
|
return true; // Found a path.
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// check_this indicates if the given target should be matched against the
|
||
|
// patterns. It should be set to false for the first call since assert_no_deps
|
||
|
// shouldn't match the target itself.
|
||
|
//
|
||
|
// visited should point to an empty set, this will be used to prevent
|
||
|
// multiple visits.
|
||
|
//
|
||
|
// *failure_path_str will be filled with a string describing the path of the
|
||
|
// dependency failure, and failure_pattern will indicate the pattern in
|
||
|
// assert_no that matched the target.
|
||
|
//
|
||
|
// Returns true if everything is OK. failure_path_str and failure_pattern_index
|
||
|
// will be unchanged in this case.
|
||
|
bool RecursiveCheckAssertNoDeps(const Target* target,
|
||
|
bool check_this,
|
||
|
const std::vector<LabelPattern>& assert_no,
|
||
|
std::set<const Target*>* visited,
|
||
|
std::string* failure_path_str,
|
||
|
const LabelPattern** failure_pattern) {
|
||
|
static const char kIndentPath[] = " ";
|
||
|
|
||
|
if (visited->find(target) != visited->end())
|
||
|
return true; // Already checked this target.
|
||
|
visited->insert(target);
|
||
|
|
||
|
if (check_this) {
|
||
|
// Check this target against the given list of patterns.
|
||
|
for (const LabelPattern& pattern : assert_no) {
|
||
|
if (pattern.Matches(target->label())) {
|
||
|
// Found a match.
|
||
|
*failure_pattern = &pattern;
|
||
|
*failure_path_str =
|
||
|
kIndentPath + target->label().GetUserVisibleName(false);
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Recursively check dependencies.
|
||
|
for (const auto& pair : target->GetDeps(Target::DEPS_ALL)) {
|
||
|
if (pair.ptr->output_type() == Target::EXECUTABLE)
|
||
|
continue;
|
||
|
if (!RecursiveCheckAssertNoDeps(pair.ptr, true, assert_no, visited,
|
||
|
failure_path_str, failure_pattern)) {
|
||
|
// To reconstruct the path, prepend the current target to the error.
|
||
|
std::string prepend_path =
|
||
|
kIndentPath + target->label().GetUserVisibleName(false) + " ->\n";
|
||
|
failure_path_str->insert(0, prepend_path);
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
} // namespace
|
||
|
|
||
|
const char kExecution_Help[] =
|
||
|
R"(Build graph and execution overview
|
||
|
|
||
|
Overall build flow
|
||
|
|
||
|
1. Look for ".gn" file (see "gn help dotfile") in the current directory and
|
||
|
walk up the directory tree until one is found. Set this directory to be
|
||
|
the "source root" and interpret this file to find the name of the build
|
||
|
config file.
|
||
|
|
||
|
2. Execute the build config file identified by .gn to set up the global
|
||
|
variables and default toolchain name. Any arguments, variables, defaults,
|
||
|
etc. set up in this file will be visible to all files in the build.
|
||
|
|
||
|
3. Load the //BUILD.gn (in the source root directory).
|
||
|
|
||
|
4. Recursively evaluate rules and load BUILD.gn in other directories as
|
||
|
necessary to resolve dependencies. If a BUILD file isn't found in the
|
||
|
specified location, GN will look in the corresponding location inside
|
||
|
the secondary_source defined in the dotfile (see "gn help dotfile").
|
||
|
|
||
|
5. When a target's dependencies are resolved, write out the `.ninja`
|
||
|
file to disk.
|
||
|
|
||
|
6. When all targets are resolved, write out the root build.ninja file.
|
||
|
|
||
|
Executing target definitions and templates
|
||
|
|
||
|
Build files are loaded in parallel. This means it is impossible to
|
||
|
interrogate a target from GN code for any information not derivable from its
|
||
|
label (see "gn help label"). The exception is the get_target_outputs()
|
||
|
function which requires the target being interrogated to have been defined
|
||
|
previously in the same file.
|
||
|
|
||
|
Targets are declared by their type and given a name:
|
||
|
|
||
|
static_library("my_static_library") {
|
||
|
... target parameter definitions ...
|
||
|
}
|
||
|
|
||
|
There is also a generic "target" function for programmatically defined types
|
||
|
(see "gn help target"). You can define new types using templates (see "gn
|
||
|
help template"). A template defines some custom code that expands to one or
|
||
|
more other targets.
|
||
|
|
||
|
Before executing the code inside the target's { }, the target defaults are
|
||
|
applied (see "gn help set_defaults"). It will inject implicit variable
|
||
|
definitions that can be overridden by the target code as necessary. Typically
|
||
|
this mechanism is used to inject a default set of configs that define the
|
||
|
global compiler and linker flags.
|
||
|
|
||
|
Which targets are built
|
||
|
|
||
|
All targets encountered in the default toolchain (see "gn help toolchain")
|
||
|
will have build rules generated for them, even if no other targets reference
|
||
|
them. Their dependencies must resolve and they will be added to the implicit
|
||
|
"all" rule (see "gn help ninja_rules").
|
||
|
|
||
|
Targets in non-default toolchains will only be generated when they are
|
||
|
required (directly or transitively) to build a target in the default
|
||
|
toolchain.
|
||
|
|
||
|
See also "gn help ninja_rules".
|
||
|
|
||
|
Dependencies
|
||
|
|
||
|
The only difference between "public_deps" and "deps" except for pushing
|
||
|
configs around the build tree and allowing includes for the purposes of "gn
|
||
|
check".
|
||
|
|
||
|
A target's "data_deps" are guaranteed to be built whenever the target is
|
||
|
built, but the ordering is not defined. The meaning of this is dependencies
|
||
|
required at runtime. Currently data deps will be complete before the target
|
||
|
is linked, but this is not semantically guaranteed and this is undesirable
|
||
|
from a build performance perspective. Since we hope to change this in the
|
||
|
future, do not rely on this behavior.
|
||
|
)";
|
||
|
|
||
|
Target::Target(const Settings* settings,
|
||
|
const Label& label,
|
||
|
const std::set<SourceFile>& build_dependency_files)
|
||
|
: Item(settings, label, build_dependency_files),
|
||
|
output_type_(UNKNOWN),
|
||
|
output_prefix_override_(false),
|
||
|
output_extension_set_(false),
|
||
|
all_headers_public_(true),
|
||
|
check_includes_(true),
|
||
|
complete_static_lib_(false),
|
||
|
testonly_(false),
|
||
|
toolchain_(nullptr) {}
|
||
|
|
||
|
Target::~Target() = default;
|
||
|
|
||
|
// static
|
||
|
const char* Target::GetStringForOutputType(OutputType type) {
|
||
|
switch (type) {
|
||
|
case UNKNOWN:
|
||
|
return "unknown";
|
||
|
case GROUP:
|
||
|
return functions::kGroup;
|
||
|
case EXECUTABLE:
|
||
|
return functions::kExecutable;
|
||
|
case LOADABLE_MODULE:
|
||
|
return functions::kLoadableModule;
|
||
|
case SHARED_LIBRARY:
|
||
|
return functions::kSharedLibrary;
|
||
|
case STATIC_LIBRARY:
|
||
|
return functions::kStaticLibrary;
|
||
|
case SOURCE_SET:
|
||
|
return functions::kSourceSet;
|
||
|
case COPY_FILES:
|
||
|
return functions::kCopy;
|
||
|
case ACTION:
|
||
|
return functions::kAction;
|
||
|
case ACTION_FOREACH:
|
||
|
return functions::kActionForEach;
|
||
|
case BUNDLE_DATA:
|
||
|
return functions::kBundleData;
|
||
|
case CREATE_BUNDLE:
|
||
|
return functions::kCreateBundle;
|
||
|
default:
|
||
|
return "";
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Target* Target::AsTarget() {
|
||
|
return this;
|
||
|
}
|
||
|
|
||
|
const Target* Target::AsTarget() const {
|
||
|
return this;
|
||
|
}
|
||
|
|
||
|
bool Target::OnResolved(Err* err) {
|
||
|
DCHECK(output_type_ != UNKNOWN);
|
||
|
DCHECK(toolchain_) << "Toolchain should have been set before resolving.";
|
||
|
|
||
|
ScopedTrace trace(TraceItem::TRACE_ON_RESOLVED, label());
|
||
|
trace.SetToolchain(settings()->toolchain_label());
|
||
|
|
||
|
// Copy this target's own dependent and public configs to the list of configs
|
||
|
// applying to it.
|
||
|
configs_.Append(all_dependent_configs_.begin(), all_dependent_configs_.end());
|
||
|
MergePublicConfigsFrom(this, &configs_);
|
||
|
|
||
|
// Copy public configs from all dependencies into the list of configs
|
||
|
// applying to this target (configs_).
|
||
|
PullDependentTargetConfigs();
|
||
|
|
||
|
// Copies public dependencies' public configs to this target's public
|
||
|
// configs. These configs have already been applied to this target by
|
||
|
// PullDependentTargetConfigs above, along with the public configs from
|
||
|
// private deps. This step re-exports them as public configs for targets that
|
||
|
// depend on this one.
|
||
|
for (const auto& dep : public_deps_) {
|
||
|
if (dep.ptr->toolchain() == toolchain()) {
|
||
|
public_configs_.Append(dep.ptr->public_configs().begin(),
|
||
|
dep.ptr->public_configs().end());
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Copy our own libs and lib_dirs to the final set. This will be from our
|
||
|
// target and all of our configs. We do this specially since these must be
|
||
|
// inherited through the dependency tree (other flags don't work this way).
|
||
|
//
|
||
|
// This needs to happen after we pull dependent target configs for the
|
||
|
// public config's libs to be included here. And it needs to happen
|
||
|
// before pulling the dependent target libs so the libs are in the correct
|
||
|
// order (local ones first, then the dependency's).
|
||
|
for (ConfigValuesIterator iter(this); !iter.done(); iter.Next()) {
|
||
|
const ConfigValues& cur = iter.cur();
|
||
|
all_lib_dirs_.append(cur.lib_dirs().begin(), cur.lib_dirs().end());
|
||
|
all_libs_.append(cur.libs().begin(), cur.libs().end());
|
||
|
}
|
||
|
|
||
|
PullRecursiveBundleData();
|
||
|
PullDependentTargetLibs();
|
||
|
PullRecursiveHardDeps();
|
||
|
if (!ResolvePrecompiledHeaders(err))
|
||
|
return false;
|
||
|
|
||
|
FillOutputFiles();
|
||
|
|
||
|
if (!CheckVisibility(err))
|
||
|
return false;
|
||
|
if (!CheckTestonly(err))
|
||
|
return false;
|
||
|
if (!CheckAssertNoDeps(err))
|
||
|
return false;
|
||
|
CheckSourcesGenerated();
|
||
|
|
||
|
if (!write_runtime_deps_output_.value().empty())
|
||
|
g_scheduler->AddWriteRuntimeDepsTarget(this);
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool Target::IsBinary() const {
|
||
|
return output_type_ == EXECUTABLE ||
|
||
|
output_type_ == SHARED_LIBRARY ||
|
||
|
output_type_ == LOADABLE_MODULE ||
|
||
|
output_type_ == STATIC_LIBRARY ||
|
||
|
output_type_ == SOURCE_SET;
|
||
|
}
|
||
|
|
||
|
bool Target::IsLinkable() const {
|
||
|
return output_type_ == STATIC_LIBRARY || output_type_ == SHARED_LIBRARY;
|
||
|
}
|
||
|
|
||
|
bool Target::IsFinal() const {
|
||
|
return output_type_ == EXECUTABLE ||
|
||
|
output_type_ == SHARED_LIBRARY ||
|
||
|
output_type_ == LOADABLE_MODULE ||
|
||
|
output_type_ == ACTION ||
|
||
|
output_type_ == ACTION_FOREACH ||
|
||
|
output_type_ == COPY_FILES ||
|
||
|
output_type_ == CREATE_BUNDLE ||
|
||
|
(output_type_ == STATIC_LIBRARY && complete_static_lib_);
|
||
|
}
|
||
|
|
||
|
DepsIteratorRange Target::GetDeps(DepsIterationType type) const {
|
||
|
if (type == DEPS_LINKED) {
|
||
|
return DepsIteratorRange(DepsIterator(
|
||
|
&public_deps_, &private_deps_, nullptr));
|
||
|
}
|
||
|
// All deps.
|
||
|
return DepsIteratorRange(DepsIterator(
|
||
|
&public_deps_, &private_deps_, &data_deps_));
|
||
|
}
|
||
|
|
||
|
std::string Target::GetComputedOutputName() const {
|
||
|
DCHECK(toolchain_)
|
||
|
<< "Toolchain must be specified before getting the computed output name.";
|
||
|
|
||
|
const std::string& name = output_name_.empty() ? label().name()
|
||
|
: output_name_;
|
||
|
|
||
|
std::string result;
|
||
|
const Tool* tool = toolchain_->GetToolForTargetFinalOutput(this);
|
||
|
if (tool) {
|
||
|
// Only add the prefix if the name doesn't already have it and it's not
|
||
|
// being overridden.
|
||
|
if (!output_prefix_override_ &&
|
||
|
!base::StartsWith(name, tool->output_prefix(),
|
||
|
base::CompareCase::SENSITIVE))
|
||
|
result = tool->output_prefix();
|
||
|
}
|
||
|
result.append(name);
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
bool Target::SetToolchain(const Toolchain* toolchain, Err* err) {
|
||
|
DCHECK(!toolchain_);
|
||
|
DCHECK_NE(UNKNOWN, output_type_);
|
||
|
toolchain_ = toolchain;
|
||
|
|
||
|
const Tool* tool = toolchain->GetToolForTargetFinalOutput(this);
|
||
|
if (tool)
|
||
|
return true;
|
||
|
|
||
|
// Tool not specified for this target type.
|
||
|
if (err) {
|
||
|
*err = Err(defined_from(), "This target uses an undefined tool.",
|
||
|
base::StringPrintf(
|
||
|
"The target %s\n"
|
||
|
"of type \"%s\"\n"
|
||
|
"uses toolchain %s\n"
|
||
|
"which doesn't have the tool \"%s\" defined.\n\n"
|
||
|
"Alas, I can not continue.",
|
||
|
label().GetUserVisibleName(false).c_str(),
|
||
|
GetStringForOutputType(output_type_),
|
||
|
label().GetToolchainLabel().GetUserVisibleName(false).c_str(),
|
||
|
Toolchain::ToolTypeToName(
|
||
|
toolchain->GetToolTypeForTargetFinalOutput(this)).c_str()));
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
bool Target::GetOutputFilesForSource(const SourceFile& source,
|
||
|
Toolchain::ToolType* computed_tool_type,
|
||
|
std::vector<OutputFile>* outputs) const {
|
||
|
outputs->clear();
|
||
|
*computed_tool_type = Toolchain::TYPE_NONE;
|
||
|
|
||
|
SourceFileType file_type = GetSourceFileType(source);
|
||
|
if (file_type == SOURCE_UNKNOWN)
|
||
|
return false;
|
||
|
if (file_type == SOURCE_O) {
|
||
|
// Object files just get passed to the output and not compiled.
|
||
|
outputs->push_back(OutputFile(settings()->build_settings(), source));
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
*computed_tool_type = toolchain_->GetToolTypeForSourceType(file_type);
|
||
|
if (*computed_tool_type == Toolchain::TYPE_NONE)
|
||
|
return false; // No tool for this file (it's a header file or something).
|
||
|
const Tool* tool = toolchain_->GetTool(*computed_tool_type);
|
||
|
if (!tool)
|
||
|
return false; // Tool does not apply for this toolchain.file.
|
||
|
|
||
|
// Figure out what output(s) this compiler produces.
|
||
|
SubstitutionWriter::ApplyListToCompilerAsOutputFile(
|
||
|
this, source, tool->outputs(), outputs);
|
||
|
return !outputs->empty();
|
||
|
}
|
||
|
|
||
|
void Target::PullDependentTargetConfigs() {
|
||
|
for (const auto& pair : GetDeps(DEPS_LINKED)) {
|
||
|
if (pair.ptr->toolchain() == toolchain()) {
|
||
|
MergeAllDependentConfigsFrom(pair.ptr, &configs_,
|
||
|
&all_dependent_configs_);
|
||
|
}
|
||
|
}
|
||
|
for (const auto& pair : GetDeps(DEPS_LINKED)) {
|
||
|
if (pair.ptr->toolchain() == toolchain()) {
|
||
|
MergePublicConfigsFrom(pair.ptr, &configs_);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void Target::PullDependentTargetLibsFrom(const Target* dep, bool is_public) {
|
||
|
// Direct dependent libraries.
|
||
|
if (dep->output_type() == STATIC_LIBRARY ||
|
||
|
dep->output_type() == SHARED_LIBRARY ||
|
||
|
dep->output_type() == SOURCE_SET)
|
||
|
inherited_libraries_.Append(dep, is_public);
|
||
|
|
||
|
if (dep->output_type() == SHARED_LIBRARY) {
|
||
|
// Shared library dependendencies are inherited across public shared
|
||
|
// library boundaries.
|
||
|
//
|
||
|
// In this case:
|
||
|
// EXE -> INTERMEDIATE_SHLIB --[public]--> FINAL_SHLIB
|
||
|
// The EXE will also link to to FINAL_SHLIB. The public dependeny means
|
||
|
// that the EXE can use the headers in FINAL_SHLIB so the FINAL_SHLIB
|
||
|
// will need to appear on EXE's link line.
|
||
|
//
|
||
|
// However, if the dependency is private:
|
||
|
// EXE -> INTERMEDIATE_SHLIB --[private]--> FINAL_SHLIB
|
||
|
// the dependency will not be propagated because INTERMEDIATE_SHLIB is
|
||
|
// not granting permission to call functiosn from FINAL_SHLIB. If EXE
|
||
|
// wants to use functions (and link to) FINAL_SHLIB, it will need to do
|
||
|
// so explicitly.
|
||
|
//
|
||
|
// Static libraries and source sets aren't inherited across shared
|
||
|
// library boundaries because they will be linked into the shared
|
||
|
// library.
|
||
|
inherited_libraries_.AppendPublicSharedLibraries(
|
||
|
dep->inherited_libraries(), is_public);
|
||
|
} else if (!dep->IsFinal()) {
|
||
|
// The current target isn't linked, so propogate linked deps and
|
||
|
// libraries up the dependency tree.
|
||
|
inherited_libraries_.AppendInherited(dep->inherited_libraries(), is_public);
|
||
|
} else if (dep->complete_static_lib()) {
|
||
|
// Inherit only final targets through _complete_ static libraries.
|
||
|
//
|
||
|
// Inherited final libraries aren't linked into complete static libraries.
|
||
|
// They are forwarded here so that targets that depend on complete
|
||
|
// static libraries can link them in. Conversely, since complete static
|
||
|
// libraries link in non-final targets they shouldn't be inherited.
|
||
|
for (const auto& inherited :
|
||
|
dep->inherited_libraries().GetOrderedAndPublicFlag()) {
|
||
|
if (inherited.first->IsFinal()) {
|
||
|
inherited_libraries_.Append(inherited.first,
|
||
|
is_public && inherited.second);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Library settings are always inherited across static library boundaries.
|
||
|
if (!dep->IsFinal() || dep->output_type() == STATIC_LIBRARY) {
|
||
|
all_lib_dirs_.append(dep->all_lib_dirs());
|
||
|
all_libs_.append(dep->all_libs());
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void Target::PullDependentTargetLibs() {
|
||
|
for (const auto& dep : public_deps_)
|
||
|
PullDependentTargetLibsFrom(dep.ptr, true);
|
||
|
for (const auto& dep : private_deps_)
|
||
|
PullDependentTargetLibsFrom(dep.ptr, false);
|
||
|
}
|
||
|
|
||
|
void Target::PullRecursiveHardDeps() {
|
||
|
for (const auto& pair : GetDeps(DEPS_LINKED)) {
|
||
|
// Direct hard dependencies.
|
||
|
if (hard_dep() || pair.ptr->hard_dep()) {
|
||
|
recursive_hard_deps_.insert(pair.ptr);
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// If |pair.ptr| is binary target and |pair.ptr| has no public header,
|
||
|
// |this| target does not need to have |pair.ptr|'s hard_deps as its
|
||
|
// hard_deps to start compiles earlier.
|
||
|
if (pair.ptr->IsBinary() && !pair.ptr->all_headers_public() &&
|
||
|
pair.ptr->public_headers().empty()) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// Recursive hard dependencies of all dependencies.
|
||
|
recursive_hard_deps_.insert(pair.ptr->recursive_hard_deps().begin(),
|
||
|
pair.ptr->recursive_hard_deps().end());
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void Target::PullRecursiveBundleData() {
|
||
|
for (const auto& pair : GetDeps(DEPS_LINKED)) {
|
||
|
// Don't propagate bundle_data once they are added to a bundle.
|
||
|
if (pair.ptr->output_type() == CREATE_BUNDLE)
|
||
|
continue;
|
||
|
|
||
|
// Don't propagate across toolchain.
|
||
|
if (pair.ptr->toolchain() != toolchain())
|
||
|
continue;
|
||
|
|
||
|
// Direct dependency on a bundle_data target.
|
||
|
if (pair.ptr->output_type() == BUNDLE_DATA)
|
||
|
bundle_data_.AddBundleData(pair.ptr);
|
||
|
|
||
|
// Recursive bundle_data informations from all dependencies.
|
||
|
for (auto* target : pair.ptr->bundle_data().bundle_deps())
|
||
|
bundle_data_.AddBundleData(target);
|
||
|
}
|
||
|
|
||
|
bundle_data_.OnTargetResolved(this);
|
||
|
}
|
||
|
|
||
|
void Target::FillOutputFiles() {
|
||
|
const Tool* tool = toolchain_->GetToolForTargetFinalOutput(this);
|
||
|
bool check_tool_outputs = false;
|
||
|
switch (output_type_) {
|
||
|
case GROUP:
|
||
|
case BUNDLE_DATA:
|
||
|
case CREATE_BUNDLE:
|
||
|
case SOURCE_SET:
|
||
|
case COPY_FILES:
|
||
|
case ACTION:
|
||
|
case ACTION_FOREACH: {
|
||
|
// These don't get linked to and use stamps which should be the first
|
||
|
// entry in the outputs. These stamps are named
|
||
|
// "<target_out_dir>/<targetname>.stamp".
|
||
|
dependency_output_file_ =
|
||
|
GetBuildDirForTargetAsOutputFile(this, BuildDirType::OBJ);
|
||
|
dependency_output_file_.value().append(GetComputedOutputName());
|
||
|
dependency_output_file_.value().append(".stamp");
|
||
|
break;
|
||
|
}
|
||
|
case EXECUTABLE:
|
||
|
case LOADABLE_MODULE:
|
||
|
// Executables and loadable modules don't get linked to, but the first
|
||
|
// output is used for dependency management.
|
||
|
CHECK_GE(tool->outputs().list().size(), 1u);
|
||
|
check_tool_outputs = true;
|
||
|
dependency_output_file_ =
|
||
|
SubstitutionWriter::ApplyPatternToLinkerAsOutputFile(
|
||
|
this, tool, tool->outputs().list()[0]);
|
||
|
|
||
|
if (tool->runtime_outputs().list().empty()) {
|
||
|
// Default to the first output for the runtime output.
|
||
|
runtime_outputs_.push_back(dependency_output_file_);
|
||
|
} else {
|
||
|
SubstitutionWriter::ApplyListToLinkerAsOutputFile(
|
||
|
this, tool, tool->runtime_outputs(), &runtime_outputs_);
|
||
|
}
|
||
|
break;
|
||
|
case STATIC_LIBRARY:
|
||
|
// Static libraries both have dependencies and linking going off of the
|
||
|
// first output.
|
||
|
CHECK(tool->outputs().list().size() >= 1);
|
||
|
check_tool_outputs = true;
|
||
|
link_output_file_ = dependency_output_file_ =
|
||
|
SubstitutionWriter::ApplyPatternToLinkerAsOutputFile(
|
||
|
this, tool, tool->outputs().list()[0]);
|
||
|
break;
|
||
|
case SHARED_LIBRARY:
|
||
|
CHECK(tool->outputs().list().size() >= 1);
|
||
|
check_tool_outputs = true;
|
||
|
if (tool->link_output().empty() && tool->depend_output().empty()) {
|
||
|
// Default behavior, use the first output file for both.
|
||
|
link_output_file_ = dependency_output_file_ =
|
||
|
SubstitutionWriter::ApplyPatternToLinkerAsOutputFile(
|
||
|
this, tool, tool->outputs().list()[0]);
|
||
|
} else {
|
||
|
// Use the tool-specified ones.
|
||
|
if (!tool->link_output().empty()) {
|
||
|
link_output_file_ =
|
||
|
SubstitutionWriter::ApplyPatternToLinkerAsOutputFile(
|
||
|
this, tool, tool->link_output());
|
||
|
}
|
||
|
if (!tool->depend_output().empty()) {
|
||
|
dependency_output_file_ =
|
||
|
SubstitutionWriter::ApplyPatternToLinkerAsOutputFile(
|
||
|
this, tool, tool->depend_output());
|
||
|
}
|
||
|
}
|
||
|
if (tool->runtime_outputs().list().empty()) {
|
||
|
// Default to the link output for the runtime output.
|
||
|
runtime_outputs_.push_back(link_output_file_);
|
||
|
} else {
|
||
|
SubstitutionWriter::ApplyListToLinkerAsOutputFile(
|
||
|
this, tool, tool->runtime_outputs(), &runtime_outputs_);
|
||
|
}
|
||
|
break;
|
||
|
case UNKNOWN:
|
||
|
default:
|
||
|
NOTREACHED();
|
||
|
}
|
||
|
|
||
|
// Count anything generated from bundle_data dependencies.
|
||
|
if (output_type_ == CREATE_BUNDLE)
|
||
|
bundle_data_.GetOutputFiles(settings(), &computed_outputs_);
|
||
|
|
||
|
// Count all outputs from this tool as something generated by this target.
|
||
|
if (check_tool_outputs) {
|
||
|
SubstitutionWriter::ApplyListToLinkerAsOutputFile(
|
||
|
this, tool, tool->outputs(), &computed_outputs_);
|
||
|
|
||
|
// Output names aren't canonicalized in the same way that source files
|
||
|
// are. For example, the tool outputs often use
|
||
|
// {{some_var}}/{{output_name}} which expands to "./foo", but this won't
|
||
|
// match "foo" which is what we'll compute when converting a SourceFile to
|
||
|
// an OutputFile.
|
||
|
for (auto& out : computed_outputs_)
|
||
|
NormalizePath(&out.value());
|
||
|
}
|
||
|
|
||
|
// Also count anything the target has declared to be an output.
|
||
|
std::vector<SourceFile> outputs_as_sources;
|
||
|
action_values_.GetOutputsAsSourceFiles(this, &outputs_as_sources);
|
||
|
for (const SourceFile& out : outputs_as_sources)
|
||
|
computed_outputs_.push_back(OutputFile(settings()->build_settings(), out));
|
||
|
}
|
||
|
|
||
|
bool Target::ResolvePrecompiledHeaders(Err* err) {
|
||
|
// Precompiled headers are stored on a ConfigValues struct. This way, the
|
||
|
// build can set all the precompiled header settings in a config and apply
|
||
|
// it to many targets. Likewise, the precompiled header values may be
|
||
|
// specified directly on a target.
|
||
|
//
|
||
|
// Unlike other values on configs which are lists that just get concatenated,
|
||
|
// the precompiled header settings are unique values. We allow them to be
|
||
|
// specified anywhere, but if they are specified in more than one place all
|
||
|
// places must match.
|
||
|
|
||
|
// Track where the current settings came from for issuing errors.
|
||
|
const Label* pch_header_settings_from = NULL;
|
||
|
if (config_values_.has_precompiled_headers())
|
||
|
pch_header_settings_from = &label();
|
||
|
|
||
|
for (ConfigValuesIterator iter(this); !iter.done(); iter.Next()) {
|
||
|
if (!iter.GetCurrentConfig())
|
||
|
continue; // Skip the one on the target itself.
|
||
|
|
||
|
const Config* config = iter.GetCurrentConfig();
|
||
|
const ConfigValues& cur = config->resolved_values();
|
||
|
if (!cur.has_precompiled_headers())
|
||
|
continue; // This one has no precompiled header info, skip.
|
||
|
|
||
|
if (config_values_.has_precompiled_headers()) {
|
||
|
// Already have a precompiled header values, the settings must match.
|
||
|
if (config_values_.precompiled_header() != cur.precompiled_header() ||
|
||
|
config_values_.precompiled_source() != cur.precompiled_source()) {
|
||
|
*err = Err(defined_from(),
|
||
|
"Precompiled header setting conflict.",
|
||
|
"The target " + label().GetUserVisibleName(false) + "\n"
|
||
|
"has conflicting precompiled header settings.\n"
|
||
|
"\n"
|
||
|
"From " + pch_header_settings_from->GetUserVisibleName(false) +
|
||
|
"\n header: " + config_values_.precompiled_header() +
|
||
|
"\n source: " + config_values_.precompiled_source().value() +
|
||
|
"\n\n"
|
||
|
"From " + config->label().GetUserVisibleName(false) +
|
||
|
"\n header: " + cur.precompiled_header() +
|
||
|
"\n source: " + cur.precompiled_source().value());
|
||
|
return false;
|
||
|
}
|
||
|
} else {
|
||
|
// Have settings from a config, apply them to ourselves.
|
||
|
pch_header_settings_from = &config->label();
|
||
|
config_values_.set_precompiled_header(cur.precompiled_header());
|
||
|
config_values_.set_precompiled_source(cur.precompiled_source());
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool Target::CheckVisibility(Err* err) const {
|
||
|
for (const auto& pair : GetDeps(DEPS_ALL)) {
|
||
|
if (!Visibility::CheckItemVisibility(this, pair.ptr, err))
|
||
|
return false;
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool Target::CheckTestonly(Err* err) const {
|
||
|
// If the current target is marked testonly, it can include both testonly
|
||
|
// and non-testonly targets, so there's nothing to check.
|
||
|
if (testonly())
|
||
|
return true;
|
||
|
|
||
|
// Verify no deps have "testonly" set.
|
||
|
for (const auto& pair : GetDeps(DEPS_ALL)) {
|
||
|
if (pair.ptr->testonly()) {
|
||
|
*err = MakeTestOnlyError(this, pair.ptr);
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool Target::CheckAssertNoDeps(Err* err) const {
|
||
|
if (assert_no_deps_.empty())
|
||
|
return true;
|
||
|
|
||
|
std::set<const Target*> visited;
|
||
|
std::string failure_path_str;
|
||
|
const LabelPattern* failure_pattern = nullptr;
|
||
|
|
||
|
if (!RecursiveCheckAssertNoDeps(this, false, assert_no_deps_, &visited,
|
||
|
&failure_path_str, &failure_pattern)) {
|
||
|
*err = Err(defined_from(), "assert_no_deps failed.",
|
||
|
label().GetUserVisibleName(false) +
|
||
|
" has an assert_no_deps entry:\n " +
|
||
|
failure_pattern->Describe() +
|
||
|
"\nwhich fails for the dependency path:\n" +
|
||
|
failure_path_str);
|
||
|
return false;
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void Target::CheckSourcesGenerated() const {
|
||
|
// Checks that any inputs or sources to this target that are in the build
|
||
|
// directory are generated by a target that this one transitively depends on
|
||
|
// in some way. We already guarantee that all generated files are written
|
||
|
// to the build dir.
|
||
|
//
|
||
|
// See Scheduler::AddUnknownGeneratedInput's declaration for more.
|
||
|
for (const SourceFile& file : sources_)
|
||
|
CheckSourceGenerated(file);
|
||
|
for (ConfigValuesIterator iter(this); !iter.done(); iter.Next()) {
|
||
|
for (const SourceFile& file : iter.cur().inputs())
|
||
|
CheckSourceGenerated(file);
|
||
|
}
|
||
|
// TODO(agrieve): Check all_libs_ here as well (those that are source files).
|
||
|
// http://crbug.com/571731
|
||
|
}
|
||
|
|
||
|
void Target::CheckSourceGenerated(const SourceFile& source) const {
|
||
|
if (!IsStringInOutputDir(settings()->build_settings()->build_dir(),
|
||
|
source.value()))
|
||
|
return; // Not in output dir, this is OK.
|
||
|
|
||
|
// Tell the scheduler about unknown files. This will be noted for later so
|
||
|
// the list of files written by the GN build itself (often response files)
|
||
|
// can be filtered out of this list.
|
||
|
OutputFile out_file(settings()->build_settings(), source);
|
||
|
std::set<const Target*> seen_targets;
|
||
|
bool check_data_deps = false;
|
||
|
bool consider_object_files = false;
|
||
|
if (!EnsureFileIsGeneratedByDependency(this, out_file, true,
|
||
|
consider_object_files, check_data_deps,
|
||
|
&seen_targets)) {
|
||
|
seen_targets.clear();
|
||
|
// Allow dependency to be through data_deps for files generated by gn.
|
||
|
check_data_deps = g_scheduler->IsFileGeneratedByWriteRuntimeDeps(out_file);
|
||
|
// Check object files (much slower and very rare) only if the "normal"
|
||
|
// output check failed.
|
||
|
consider_object_files = !check_data_deps;
|
||
|
if (!EnsureFileIsGeneratedByDependency(this, out_file, true,
|
||
|
consider_object_files,
|
||
|
check_data_deps, &seen_targets))
|
||
|
g_scheduler->AddUnknownGeneratedInput(this, source);
|
||
|
}
|
||
|
}
|