| // 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/scheduler.h" | 
 |  | 
 | #include <algorithm> | 
 |  | 
 | #include "base/bind.h" | 
 | #include "base/single_thread_task_runner.h" | 
 | #include "base/task_scheduler/post_task.h" | 
 | #include "base/threading/thread_task_runner_handle.h" | 
 | #include "tools/gn/standard_out.h" | 
 | #include "tools/gn/target.h" | 
 |  | 
 | Scheduler* g_scheduler = nullptr; | 
 |  | 
 | Scheduler::Scheduler() | 
 |     : main_thread_task_runner_(base::ThreadTaskRunnerHandle::Get()), | 
 |       input_file_manager_(new InputFileManager), | 
 |       verbose_logging_(false), | 
 |       pool_work_count_cv_(&pool_work_count_lock_), | 
 |       is_failed_(false), | 
 |       suppress_output_for_testing_(false), | 
 |       has_been_shutdown_(false) { | 
 |   g_scheduler = this; | 
 | } | 
 |  | 
 | Scheduler::~Scheduler() { | 
 |   WaitForPoolTasks(); | 
 |   g_scheduler = nullptr; | 
 | } | 
 |  | 
 | bool Scheduler::Run() { | 
 |   runner_.Run(); | 
 |   bool local_is_failed; | 
 |   { | 
 |     base::AutoLock lock(lock_); | 
 |     local_is_failed = is_failed(); | 
 |     has_been_shutdown_ = true; | 
 |   } | 
 |   // Don't do this while holding |lock_|, since it will block on the workers, | 
 |   // which may be in turn waiting on the lock. | 
 |   WaitForPoolTasks(); | 
 |   return !local_is_failed; | 
 | } | 
 |  | 
 | void Scheduler::Log(const std::string& verb, const std::string& msg) { | 
 |   if (task_runner()->BelongsToCurrentThread()) { | 
 |     LogOnMainThread(verb, msg); | 
 |   } else { | 
 |     // The run loop always joins on the sub threads, so the lifetime of this | 
 |     // object outlives the invocations of this function, hence "unretained". | 
 |     task_runner()->PostTask(FROM_HERE, | 
 |                             base::Bind(&Scheduler::LogOnMainThread, | 
 |                                        base::Unretained(this), verb, msg)); | 
 |   } | 
 | } | 
 |  | 
 | void Scheduler::FailWithError(const Err& err) { | 
 |   DCHECK(err.has_error()); | 
 |   { | 
 |     base::AutoLock lock(lock_); | 
 |  | 
 |     if (is_failed_ || has_been_shutdown_) | 
 |       return;  // Ignore errors once we see one. | 
 |     is_failed_ = true; | 
 |   } | 
 |  | 
 |   if (task_runner()->BelongsToCurrentThread()) { | 
 |     FailWithErrorOnMainThread(err); | 
 |   } else { | 
 |     // The run loop always joins on the sub threads, so the lifetime of this | 
 |     // object outlives the invocations of this function, hence "unretained". | 
 |     task_runner()->PostTask(FROM_HERE, | 
 |                             base::Bind(&Scheduler::FailWithErrorOnMainThread, | 
 |                                        base::Unretained(this), err)); | 
 |   } | 
 | } | 
 |  | 
 | void Scheduler::ScheduleWork(base::OnceClosure work) { | 
 |   IncrementWorkCount(); | 
 |   pool_work_count_.Increment(); | 
 |   base::PostTaskWithTraits( | 
 |       FROM_HERE, {base::MayBlock()}, | 
 |       base::BindOnce(&Scheduler::DoWork, base::Unretained(this), | 
 |                      std::move(work))); | 
 | } | 
 |  | 
 | void Scheduler::AddGenDependency(const base::FilePath& file) { | 
 |   base::AutoLock lock(lock_); | 
 |   gen_dependencies_.push_back(file); | 
 | } | 
 |  | 
 | std::vector<base::FilePath> Scheduler::GetGenDependencies() const { | 
 |   base::AutoLock lock(lock_); | 
 |   return gen_dependencies_; | 
 | } | 
 |  | 
 | void Scheduler::AddWrittenFile(const SourceFile& file) { | 
 |   base::AutoLock lock(lock_); | 
 |   written_files_.push_back(file); | 
 | } | 
 |  | 
 | void Scheduler::AddUnknownGeneratedInput(const Target* target, | 
 |                                          const SourceFile& file) { | 
 |   base::AutoLock lock(lock_); | 
 |   unknown_generated_inputs_.insert(std::make_pair(file, target)); | 
 | } | 
 |  | 
 | void Scheduler::AddWriteRuntimeDepsTarget(const Target* target) { | 
 |   base::AutoLock lock(lock_); | 
 |   write_runtime_deps_targets_.push_back(target); | 
 | } | 
 |  | 
 | std::vector<const Target*> Scheduler::GetWriteRuntimeDepsTargets() const { | 
 |   base::AutoLock lock(lock_); | 
 |   return write_runtime_deps_targets_; | 
 | } | 
 |  | 
 | bool Scheduler::IsFileGeneratedByWriteRuntimeDeps( | 
 |     const OutputFile& file) const { | 
 |   base::AutoLock lock(lock_); | 
 |   // Number of targets should be quite small, so brute-force search is fine. | 
 |   for (const Target* target : write_runtime_deps_targets_) { | 
 |     if (file == target->write_runtime_deps_output()) { | 
 |       return true; | 
 |     } | 
 |   } | 
 |   return false; | 
 | } | 
 |  | 
 | std::multimap<SourceFile, const Target*> | 
 |     Scheduler::GetUnknownGeneratedInputs() const { | 
 |   base::AutoLock lock(lock_); | 
 |  | 
 |   // Remove all unknown inputs that were written files. These are OK as inputs | 
 |   // to build steps since they were written as a side-effect of running GN. | 
 |   // | 
 |   // It's assumed that this function is called once during cleanup to check for | 
 |   // errors, so performing this work in the lock doesn't matter. | 
 |   std::multimap<SourceFile, const Target*> filtered = unknown_generated_inputs_; | 
 |   for (const SourceFile& file : written_files_) | 
 |     filtered.erase(file); | 
 |  | 
 |   return filtered; | 
 | } | 
 |  | 
 | void Scheduler::ClearUnknownGeneratedInputsAndWrittenFiles() { | 
 |   base::AutoLock lock(lock_); | 
 |   unknown_generated_inputs_.clear(); | 
 |   written_files_.clear(); | 
 | } | 
 |  | 
 | void Scheduler::IncrementWorkCount() { | 
 |   work_count_.Increment(); | 
 | } | 
 |  | 
 | void Scheduler::DecrementWorkCount() { | 
 |   if (!work_count_.Decrement()) { | 
 |     if (task_runner()->BelongsToCurrentThread()) { | 
 |       OnComplete(); | 
 |     } else { | 
 |       task_runner()->PostTask(FROM_HERE, base::Bind(&Scheduler::OnComplete, | 
 |                                                     base::Unretained(this))); | 
 |     } | 
 |   } | 
 | } | 
 |  | 
 | void Scheduler::SuppressOutputForTesting(bool suppress) { | 
 |   base::AutoLock lock(lock_); | 
 |   suppress_output_for_testing_ = suppress; | 
 | } | 
 |  | 
 | void Scheduler::LogOnMainThread(const std::string& verb, | 
 |                                 const std::string& msg) { | 
 |   OutputString(verb, DECORATION_YELLOW); | 
 |   OutputString(" " + msg + "\n"); | 
 | } | 
 |  | 
 | void Scheduler::FailWithErrorOnMainThread(const Err& err) { | 
 |   if (!suppress_output_for_testing_) | 
 |     err.PrintToStdout(); | 
 |   runner_.Quit(); | 
 | } | 
 |  | 
 | void Scheduler::DoWork(base::OnceClosure closure) { | 
 |   std::move(closure).Run(); | 
 |   DecrementWorkCount(); | 
 |   if (!pool_work_count_.Decrement()) { | 
 |     base::AutoLock auto_lock(pool_work_count_lock_); | 
 |     pool_work_count_cv_.Signal(); | 
 |   } | 
 | } | 
 |  | 
 | void Scheduler::OnComplete() { | 
 |   // Should be called on the main thread. | 
 |   DCHECK(task_runner()->BelongsToCurrentThread()); | 
 |   runner_.Quit(); | 
 | } | 
 |  | 
 | void Scheduler::WaitForPoolTasks() { | 
 |   base::AutoLock lock(pool_work_count_lock_); | 
 |   while (!pool_work_count_.IsZero()) | 
 |     pool_work_count_cv_.Wait(); | 
 | } |