| // 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 "gn/loader.h" |
| |
| #include <memory> |
| |
| #include "gn/build_settings.h" |
| #include "gn/err.h" |
| #include "gn/filesystem_utils.h" |
| #include "gn/input_file_manager.h" |
| #include "gn/parse_tree.h" |
| #include "gn/scheduler.h" |
| #include "gn/scope_per_file_provider.h" |
| #include "gn/settings.h" |
| #include "gn/source_dir.h" |
| #include "gn/source_file.h" |
| #include "gn/trace.h" |
| |
| namespace { |
| |
| struct SourceFileAndOrigin { |
| SourceFileAndOrigin(const SourceFile& f, const LocationRange& o) |
| : file(f), origin(o) {} |
| |
| SourceFile file; |
| LocationRange origin; |
| }; |
| |
| } // namespace |
| |
| // Identifies one time a file is loaded in a given toolchain so we don't load |
| // it more than once. |
| struct LoaderImpl::LoadID { |
| LoadID() = default; |
| LoadID(const SourceFile& f, const Label& tc_name) |
| : file(f), toolchain_name(tc_name) {} |
| |
| bool operator<(const LoadID& other) const { |
| if (file == other.file) |
| return toolchain_name < other.toolchain_name; |
| return file < other.file; |
| } |
| |
| SourceFile file; |
| Label toolchain_name; |
| }; |
| |
| // Our tracking information for a toolchain. |
| struct LoaderImpl::ToolchainRecord { |
| // The default toolchain label can be empty for the first time the default |
| // toolchain is loaded, since we don't know it yet. This will be fixed up |
| // later. It should be valid in all other cases. |
| ToolchainRecord(const BuildSettings* build_settings, |
| const Label& toolchain_label, |
| const Label& default_toolchain_label) |
| : settings( |
| build_settings, |
| GetOutputSubdirName(toolchain_label, |
| toolchain_label == default_toolchain_label)), |
| is_toolchain_loaded(false), |
| is_config_loaded(false) { |
| settings.set_default_toolchain_label(default_toolchain_label); |
| settings.set_toolchain_label(toolchain_label); |
| } |
| |
| Settings settings; |
| |
| bool is_toolchain_loaded; |
| bool is_config_loaded; |
| |
| std::vector<SourceFileAndOrigin> waiting_on_me; |
| }; |
| |
| // ----------------------------------------------------------------------------- |
| |
| const void* const Loader::kDefaultToolchainKey = &kDefaultToolchainKey; |
| |
| Loader::Loader() = default; |
| |
| Loader::~Loader() = default; |
| |
| void Loader::Load(const Label& label, const LocationRange& origin) { |
| Load(BuildFileForLabel(label), origin, label.GetToolchainLabel()); |
| } |
| |
| // ----------------------------------------------------------------------------- |
| |
| LoaderImpl::LoaderImpl(const BuildSettings* build_settings) |
| : pending_loads_(0), build_settings_(build_settings) { |
| // There may not be an active TaskRunner at this point. When that's the case, |
| // the calling code is expected to call set_task_runner(). |
| task_runner_ = MsgLoop::Current(); |
| } |
| |
| LoaderImpl::~LoaderImpl() = default; |
| |
| void LoaderImpl::Load(const SourceFile& file, |
| const LocationRange& origin, |
| const Label& in_toolchain_name) { |
| const Label& toolchain_name = in_toolchain_name.is_null() |
| ? default_toolchain_label_ |
| : in_toolchain_name; |
| LoadID load_id(file, toolchain_name); |
| if (!invocations_.insert(load_id).second) |
| return; // Already in set, so this file was already loaded or schedulerd. |
| |
| if (toolchain_records_.empty()) { |
| // Nothing loaded, need to load the default build config. The initial load |
| // should not specify a toolchain. |
| DCHECK(toolchain_name.is_null()); |
| |
| std::unique_ptr<ToolchainRecord> new_record = |
| std::make_unique<ToolchainRecord>(build_settings_, Label(), Label()); |
| ToolchainRecord* record = new_record.get(); |
| Label empty_label; // VS issues spurious warning using ...[Label()]. |
| toolchain_records_[empty_label] = std::move(new_record); |
| |
| // The default build config is no dependent on the toolchain definition, |
| // since we need to load the build config before we know what the default |
| // toolchain name is. |
| record->is_toolchain_loaded = true; |
| |
| record->waiting_on_me.push_back(SourceFileAndOrigin(file, origin)); |
| ScheduleLoadBuildConfig(&record->settings, Scope::KeyValueMap()); |
| |
| return; |
| } |
| |
| ToolchainRecord* record; |
| if (toolchain_name.is_null()) |
| record = toolchain_records_[default_toolchain_label_].get(); |
| else |
| record = toolchain_records_[toolchain_name].get(); |
| |
| if (!record) { |
| DCHECK(!default_toolchain_label_.is_null()); |
| |
| // No reference to this toolchain found yet, make one. |
| std::unique_ptr<ToolchainRecord> new_record = |
| std::make_unique<ToolchainRecord>(build_settings_, toolchain_name, |
| default_toolchain_label_); |
| record = new_record.get(); |
| toolchain_records_[toolchain_name] = std::move(new_record); |
| |
| // Schedule a load of the toolchain using the default one. |
| Load(BuildFileForLabel(toolchain_name), origin, default_toolchain_label_); |
| } |
| |
| if (record->is_config_loaded) |
| ScheduleLoadFile(&record->settings, origin, file); |
| else |
| record->waiting_on_me.push_back(SourceFileAndOrigin(file, origin)); |
| } |
| |
| void LoaderImpl::ToolchainLoaded(const Toolchain* toolchain) { |
| ToolchainRecord* record = toolchain_records_[toolchain->label()].get(); |
| if (!record) { |
| DCHECK(!default_toolchain_label_.is_null()); |
| std::unique_ptr<ToolchainRecord> new_record = |
| std::make_unique<ToolchainRecord>(build_settings_, toolchain->label(), |
| default_toolchain_label_); |
| record = new_record.get(); |
| toolchain_records_[toolchain->label()] = std::move(new_record); |
| } |
| record->is_toolchain_loaded = true; |
| |
| // The default build config is loaded first, then its toolchain. Secondary |
| // ones are loaded in the opposite order so we can pass toolchain parameters |
| // to the build config. So we may or may not have a config at this point. |
| if (!record->is_config_loaded) { |
| ScheduleLoadBuildConfig(&record->settings, toolchain->args()); |
| } else { |
| // There should be nobody waiting on this if the build config is already |
| // loaded. |
| DCHECK(record->waiting_on_me.empty()); |
| } |
| } |
| |
| Label LoaderImpl::GetDefaultToolchain() const { |
| return default_toolchain_label_; |
| } |
| |
| const Settings* LoaderImpl::GetToolchainSettings(const Label& label) const { |
| ToolchainRecordMap::const_iterator found_toolchain; |
| if (label.is_null()) { |
| if (default_toolchain_label_.is_null()) |
| return nullptr; |
| found_toolchain = toolchain_records_.find(default_toolchain_label_); |
| } else { |
| found_toolchain = toolchain_records_.find(label); |
| } |
| |
| if (found_toolchain == toolchain_records_.end()) |
| return nullptr; |
| return &found_toolchain->second->settings; |
| } |
| |
| SourceFile LoaderImpl::BuildFileForLabel(const Label& label) const { |
| return SourceFile( |
| label.dir().value() + "BUILD" + build_file_extension_ + ".gn"); |
| } |
| |
| void LoaderImpl::ScheduleLoadFile(const Settings* settings, |
| const LocationRange& origin, |
| const SourceFile& file) { |
| Err err; |
| pending_loads_++; |
| if (!AsyncLoadFile( |
| origin, settings->build_settings(), file, |
| [this, settings, file, origin](const ParseNode* parse_node) { |
| BackgroundLoadFile(settings, file, origin, parse_node); |
| }, |
| &err)) { |
| g_scheduler->FailWithError(err); |
| DecrementPendingLoads(); |
| } |
| } |
| |
| void LoaderImpl::ScheduleLoadBuildConfig( |
| Settings* settings, |
| const Scope::KeyValueMap& toolchain_overrides) { |
| Err err; |
| pending_loads_++; |
| if (!AsyncLoadFile( |
| LocationRange(), settings->build_settings(), |
| settings->build_settings()->build_config_file(), |
| [this, settings, toolchain_overrides](const ParseNode* root) { |
| BackgroundLoadBuildConfig(settings, toolchain_overrides, root); |
| }, |
| &err)) { |
| g_scheduler->FailWithError(err); |
| DecrementPendingLoads(); |
| } |
| } |
| |
| void LoaderImpl::BackgroundLoadFile(const Settings* settings, |
| const SourceFile& file_name, |
| const LocationRange& origin, |
| const ParseNode* root) { |
| if (!root) { |
| task_runner_->PostTask([this]() { DecrementPendingLoads(); }); |
| return; |
| } |
| |
| if (g_scheduler->verbose_logging()) { |
| g_scheduler->Log("Running", |
| file_name.value() + " with toolchain " + |
| settings->toolchain_label().GetUserVisibleName(false)); |
| } |
| |
| Scope our_scope(settings->base_config()); |
| ScopePerFileProvider per_file_provider(&our_scope, true); |
| our_scope.set_source_dir(file_name.GetDir()); |
| our_scope.AddBuildDependencyFile(file_name); |
| |
| // Targets, etc. generated as part of running this file will end up here. |
| Scope::ItemVector collected_items; |
| our_scope.set_item_collector(&collected_items); |
| |
| ScopedTrace trace(TraceItem::TRACE_FILE_EXECUTE, file_name.value()); |
| trace.SetToolchain(settings->toolchain_label()); |
| |
| Err err; |
| root->Execute(&our_scope, &err); |
| if (!err.has_error()) |
| our_scope.CheckForUnusedVars(&err); |
| |
| if (err.has_error()) { |
| if (!origin.is_null()) |
| err.AppendSubErr(Err(origin, "which caused the file to be included.")); |
| |
| if (!settings->is_default()) |
| err.set_toolchain_label(settings->toolchain_label()); |
| |
| g_scheduler->FailWithError(err); |
| } |
| |
| // Pass all of the items that were defined off to the builder. |
| for (auto& item : collected_items) |
| settings->build_settings()->ItemDefined(std::move(item)); |
| |
| trace.Done(); |
| |
| task_runner_->PostTask([this]() { DidLoadFile(); }); |
| } |
| |
| void LoaderImpl::BackgroundLoadBuildConfig( |
| Settings* settings, |
| const Scope::KeyValueMap& toolchain_overrides, |
| const ParseNode* root) { |
| if (!root) { |
| task_runner_->PostTask([this]() { DecrementPendingLoads(); }); |
| return; |
| } |
| |
| Scope* base_config = settings->base_config(); |
| base_config->set_source_dir(SourceDir("//")); |
| base_config->AddBuildDependencyFile( |
| settings->build_settings()->build_config_file()); |
| |
| settings->build_settings()->build_args().SetupRootScope(base_config, |
| toolchain_overrides); |
| |
| base_config->SetProcessingBuildConfig(); |
| |
| // See kDefaultToolchainKey in the header. |
| Label default_toolchain_label; |
| if (settings->is_default()) |
| base_config->SetProperty(kDefaultToolchainKey, &default_toolchain_label); |
| |
| ScopedTrace trace(TraceItem::TRACE_FILE_EXECUTE, |
| settings->build_settings()->build_config_file().value()); |
| trace.SetToolchain(settings->toolchain_label()); |
| |
| // Run the BUILDCONFIG with its directory as the current one. We want |
| // BUILDCONFIG to modify the base_config so can't make a copy or a nested one. |
| base_config->set_source_dir( |
| settings->build_settings()->build_config_file().GetDir()); |
| |
| Err err; |
| root->Execute(base_config, &err); |
| |
| // Put back the root as the default source dir. This probably isn't necessary |
| // as other scopes will set their directories to their own path, but it's a |
| // better default than the build config's directory. |
| base_config->set_source_dir(SourceDir("//")); |
| |
| // Clear all private variables left in the scope. We want the root build |
| // config to be like a .gni file in that variables beginning with an |
| // underscore aren't exported. |
| base_config->RemovePrivateIdentifiers(); |
| |
| trace.Done(); |
| |
| if (err.has_error()) { |
| if (!settings->is_default()) |
| err.set_toolchain_label(settings->toolchain_label()); |
| |
| g_scheduler->FailWithError(err); |
| } |
| |
| base_config->ClearProcessingBuildConfig(); |
| if (settings->is_default()) { |
| // The default toolchain must have been set in the default build config |
| // file. |
| if (default_toolchain_label.is_null()) { |
| g_scheduler->FailWithError(Err( |
| Location(), |
| "The default build config file did not call set_default_toolchain()", |
| "If you don't call this, I can't figure out what toolchain to use\n" |
| "for all of this code.")); |
| } else { |
| DCHECK(settings->toolchain_label().is_null()); |
| settings->set_toolchain_label(default_toolchain_label); |
| } |
| } |
| |
| task_runner_->PostTask( |
| [this, toolchain_label = settings->toolchain_label()]() { |
| DidLoadBuildConfig(toolchain_label); |
| }); |
| } |
| |
| void LoaderImpl::DidLoadFile() { |
| DecrementPendingLoads(); |
| } |
| |
| void LoaderImpl::DidLoadBuildConfig(const Label& label) { |
| // Do not return early, we must call DecrementPendingLoads() at the bottom. |
| |
| ToolchainRecordMap::iterator found_toolchain = toolchain_records_.find(label); |
| ToolchainRecord* record = nullptr; |
| if (found_toolchain == toolchain_records_.end()) { |
| // When loading the default build config, we'll insert it into the record |
| // map with an empty label since we don't yet know what to call it. |
| // |
| // In this case, we should have exactly one entry in the map with an empty |
| // label. We now need to fix up the naming so it refers to the "real" one. |
| CHECK_EQ(1U, toolchain_records_.size()); |
| ToolchainRecordMap::iterator empty_label = toolchain_records_.find(Label()); |
| CHECK(empty_label != toolchain_records_.end()); |
| |
| // Fix up the toolchain record. |
| std::unique_ptr<ToolchainRecord> moved_record = |
| std::move(empty_label->second); |
| record = moved_record.get(); |
| toolchain_records_[label] = std::move(moved_record); |
| toolchain_records_.erase(empty_label); |
| |
| // Save the default toolchain label. |
| default_toolchain_label_ = label; |
| DCHECK(record->settings.default_toolchain_label().is_null()); |
| record->settings.set_default_toolchain_label(label); |
| |
| // The settings object should have the toolchain label already set. |
| DCHECK(!record->settings.toolchain_label().is_null()); |
| |
| // Update any stored invocations that refer to the empty toolchain label. |
| // This will normally only be one, for the root build file, so brute-force |
| // is OK. |
| LoadIDSet old_loads; |
| invocations_.swap(old_loads); |
| for (const auto& load : old_loads) { |
| if (load.toolchain_name.is_null()) { |
| // Fix up toolchain label |
| invocations_.emplace(load.file, label); |
| } else { |
| // Can keep the old one. |
| invocations_.insert(load); |
| } |
| } |
| } else { |
| record = found_toolchain->second.get(); |
| } |
| |
| DCHECK(!record->is_config_loaded); |
| DCHECK(record->is_toolchain_loaded); |
| record->is_config_loaded = true; |
| |
| // Schedule all waiting file loads. |
| for (const auto& waiting : record->waiting_on_me) |
| ScheduleLoadFile(&record->settings, waiting.origin, waiting.file); |
| record->waiting_on_me.clear(); |
| |
| DecrementPendingLoads(); |
| } |
| |
| void LoaderImpl::DecrementPendingLoads() { |
| DCHECK_GT(pending_loads_, 0); |
| pending_loads_--; |
| if (pending_loads_ == 0 && complete_callback_) |
| complete_callback_(); |
| } |
| |
| bool LoaderImpl::AsyncLoadFile(const LocationRange& origin, |
| const BuildSettings* build_settings, |
| const SourceFile& file_name, |
| std::function<void(const ParseNode*)> callback, |
| Err* err) { |
| if (async_load_file_) { |
| return async_load_file_(origin, build_settings, file_name, |
| std::move(callback), err); |
| } |
| return g_scheduler->input_file_manager()->AsyncLoadFile( |
| origin, build_settings, file_name, std::move(callback), err); |
| } |