| // 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/c_tool.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/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()) { |
| const char* tool_name; |
| if (!target->GetOutputFilesForSource(source, &tool_name, &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) {} |
| |
| 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; |
| case GENERATED_FILE: |
| return functions::kGeneratedFile; |
| 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() || |
| dep.ptr->toolchain()->propagates_configs()) |
| 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; |
| |
| if (!FillOutputFiles(err)) |
| return false; |
| |
| 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); |
| |
| if (output_type_ == GENERATED_FILE) { |
| DCHECK(!computed_outputs_.empty()); |
| g_scheduler->AddGeneratedFile( |
| computed_outputs_[0].AsSourceFile(settings()->build_settings())); |
| } |
| |
| 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 || output_type_ == RUST_LIBRARY; |
| } |
| |
| bool Target::IsLinkable() const { |
| return output_type_ == STATIC_LIBRARY || output_type_ == SHARED_LIBRARY || |
| output_type_ == RUST_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(), |
| Tool::GetToolTypeForTargetFinalOutput(this))); |
| } |
| return false; |
| } |
| |
| bool Target::GetOutputFilesForSource(const SourceFile& source, |
| const char** computed_tool_type, |
| std::vector<OutputFile>* outputs) const { |
| outputs->clear(); |
| *computed_tool_type = Tool::kToolNone; |
| |
| SourceFile::Type file_type = source.type(); |
| if (file_type == SourceFile::SOURCE_UNKNOWN) |
| return false; |
| if (file_type == SourceFile::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 = Tool::GetToolTypeForSourceType(file_type); |
| if (*computed_tool_type == Tool::kToolNone) |
| 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() || |
| pair.ptr->toolchain()->propagates_configs()) |
| MergeAllDependentConfigsFrom(pair.ptr, &configs_, |
| &all_dependent_configs_); |
| } |
| for (const auto& pair : GetDeps(DEPS_LINKED)) { |
| if (pair.ptr->toolchain() == toolchain() || |
| pair.ptr->toolchain()->propagates_configs()) |
| 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 || dep->output_type() == RUST_LIBRARY) |
| 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); |
| } |
| |
| bool Target::FillOutputFiles(Err* err) { |
| 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: |
| case GENERATED_FILE: { |
| // 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 RUST_LIBRARY: |
| 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 (const CTool* ctool = tool->AsC()) { |
| if (ctool->link_output().empty() && ctool->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 (!ctool->link_output().empty()) { |
| link_output_file_ = |
| SubstitutionWriter::ApplyPatternToLinkerAsOutputFile( |
| this, tool, ctool->link_output()); |
| } |
| if (!ctool->depend_output().empty()) { |
| dependency_output_file_ = |
| SubstitutionWriter::ApplyPatternToLinkerAsOutputFile( |
| this, tool, ctool->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_); |
| } |
| } else if (const RustTool* rstool = tool->AsRust()) { |
| // Default behavior, use the first output file for both. |
| link_output_file_ = dependency_output_file_ = |
| SubstitutionWriter::ApplyPatternToLinkerAsOutputFile( |
| this, tool, tool->outputs().list()[0]); |
| } |
| break; |
| case UNKNOWN: |
| default: |
| NOTREACHED(); |
| } |
| |
| // Count anything generated from bundle_data dependencies. |
| if (output_type_ == CREATE_BUNDLE) { |
| if (!bundle_data_.GetOutputFiles(settings(), this, &computed_outputs_, err)) |
| return false; |
| } |
| |
| // 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)); |
| |
| return true; |
| } |
| |
| 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) || |
| g_scheduler->IsFileGeneratedByTarget(source); |
| // 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); |
| } |
| } |
| |
| bool Target::GetMetadata(const std::vector<std::string>& keys_to_extract, |
| const std::vector<std::string>& keys_to_walk, |
| const SourceDir& rebase_dir, |
| bool deps_only, |
| std::vector<Value>* result, |
| std::set<const Target*>* targets_walked, |
| Err* err) const { |
| std::vector<Value> next_walk_keys; |
| std::vector<Value> current_result; |
| // If deps_only, this is the top-level target and thus we don't want to |
| // collect its metadata, only that of its deps and data_deps. |
| if (deps_only) { |
| // Empty string will be converted below to mean all deps and data_deps. |
| // Origin is null because this isn't declared anywhere, and should never |
| // trigger any errors. |
| next_walk_keys.push_back(Value(nullptr, "")); |
| } else { |
| // Otherwise, we walk this target and collect the appropriate data. |
| if (!metadata_.WalkStep(settings()->build_settings(), keys_to_extract, |
| keys_to_walk, rebase_dir, &next_walk_keys, |
| ¤t_result, err)) |
| return false; |
| } |
| |
| // Gather walk keys and find the appropriate target. Targets identified in |
| // the walk key set must be deps or data_deps of the declaring target. |
| const DepsIteratorRange& all_deps = GetDeps(Target::DEPS_ALL); |
| const SourceDir current_dir("//"); |
| for (const auto& next : next_walk_keys) { |
| DCHECK(next.type() == Value::STRING); |
| |
| // If we hit an empty string in this list, add all deps and data_deps. The |
| // ordering in the resulting list of values as a result will be the data |
| // from each explicitly listed dep prior to this, followed by all data in |
| // walk order of the remaining deps. |
| if (next.string_value().empty()) { |
| for (const auto& dep : all_deps) { |
| // If we haven't walked this dep yet, go down into it. |
| auto pair = targets_walked->insert(dep.ptr); |
| if (pair.second) { |
| if (!dep.ptr->GetMetadata(keys_to_extract, keys_to_walk, rebase_dir, |
| false, result, targets_walked, err)) |
| return false; |
| } |
| } |
| |
| // Any other walk keys are superfluous, as they can only be a subset of |
| // all deps. |
| break; |
| } |
| |
| // Otherwise, look through the target's deps for the specified one. |
| // Canonicalize the label if possible. |
| Label next_label = |
| Label::Resolve(current_dir, settings()->toolchain_label(), next, err); |
| if (next_label.is_null()) { |
| *err = Err(next.origin(), std::string("Failed to canonicalize ") + |
| next.string_value() + std::string(".")); |
| } |
| std::string canonicalize_next_label = next_label.GetUserVisibleName(true); |
| |
| bool found_next = false; |
| for (const auto& dep : all_deps) { |
| // Match against the label with the toolchain. |
| if (dep.label.GetUserVisibleName(true) == canonicalize_next_label) { |
| // If we haven't walked this dep yet, go down into it. |
| auto pair = targets_walked->insert(dep.ptr); |
| if (pair.second) { |
| if (!dep.ptr->GetMetadata(keys_to_extract, keys_to_walk, rebase_dir, |
| false, result, targets_walked, err)) |
| return false; |
| } |
| // We found it, so we can exit this search now. |
| found_next = true; |
| break; |
| } |
| } |
| // If we didn't find the specified dep in the target, that's an error. |
| // Propagate it back to the user. |
| if (!found_next) { |
| *err = Err(next.origin(), |
| std::string("I was expecting ") + canonicalize_next_label + |
| std::string(" to be a dependency of ") + |
| label().GetUserVisibleName(true) + |
| ". Make sure it's included in the deps or data_deps, and " |
| "that you've specified the appropriate toolchain."); |
| return false; |
| } |
| } |
| result->insert(result->end(), std::make_move_iterator(current_result.begin()), |
| std::make_move_iterator(current_result.end())); |
| return true; |
| } |