|  | // 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); | 
|  | } | 
|  | } |