diff --git a/CMakeLists.txt b/CMakeLists.txt index b8fdee7d3a..f05d2cfcc5 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -169,7 +169,10 @@ if(WIN32) # errors by telling windows.h to not define those two. add_compile_definitions(NOMINMAX) else() - target_sources(libninja PRIVATE src/subprocess-posix.cc) + target_sources(libninja PRIVATE + src/subprocess-posix.cc + src/jobserver-posix.cc + ) if(CMAKE_SYSTEM_NAME STREQUAL "OS400" OR CMAKE_SYSTEM_NAME STREQUAL "AIX") target_sources(libninja PRIVATE src/getopt.c) # Build getopt.c, which can be compiled as either C or C++, as C++ diff --git a/configure.py b/configure.py index c88daad508..9d6175e35f 100755 --- a/configure.py +++ b/configure.py @@ -565,6 +565,7 @@ def has_re2c() -> bool: objs += cc('getopt') else: objs += cxx('subprocess-posix') + objs += cxx('jobserver-posix') if platform.is_aix(): objs += cc('getopt') if platform.is_msvc(): diff --git a/src/build.cc b/src/build.cc index deb8f04c8b..f79fe87df7 100644 --- a/src/build.cc +++ b/src/build.cc @@ -164,8 +164,23 @@ Edge* Plan::FindWork() { if (ready_.empty()) return NULL; +// TODO: jobserver client support for Windows +#ifndef _WIN32 + // Only initiate work if the jobserver can acquire a token. + if (builder_->jobserver_.Enabled() && !builder_->jobserver_.Acquire()) { + return NULL; + } +#endif + Edge* work = ready_.top(); ready_.pop(); + +// TODO: jobserver client support for Windows +#ifndef _WIN32 + // Mark this edge as using a job token to be released when work is finished. + work->has_job_token_ = builder_->jobserver_.Enabled(); +#endif + return work; } @@ -201,6 +216,15 @@ bool Plan::EdgeFinished(Edge* edge, EdgeResult result, string* err) { edge->pool()->EdgeFinished(*edge); edge->pool()->RetrieveReadyEdges(&ready_); +// TODO: jobserver client support for Windows +#ifndef _WIN32 + // If jobserver is used, return the token for this job. + if (edge->has_job_token_) { + builder_->jobserver_.Release(); + edge->has_job_token_ = false; + } +#endif + // The rest of this function only applies to successful commands. if (result != kEdgeSucceeded) return true; @@ -594,7 +618,9 @@ void Plan::Dump() const { } struct RealCommandRunner : public CommandRunner { - explicit RealCommandRunner(const BuildConfig& config) : config_(config) {} + explicit RealCommandRunner(const BuildConfig& config, Jobserver& jobserver) : + config_(config), jobserver_(jobserver) {} + virtual ~RealCommandRunner() {} virtual size_t CanRunMore() const; virtual bool StartCommand(Edge* edge); @@ -603,6 +629,7 @@ struct RealCommandRunner : public CommandRunner { virtual void Abort(); const BuildConfig& config_; + Jobserver& jobserver_; SubprocessSet subprocs_; map subproc_to_edge_; }; @@ -617,6 +644,10 @@ vector RealCommandRunner::GetActiveEdges() { void RealCommandRunner::Abort() { subprocs_.Clear(); +// TODO: jobserver client support for Windows +#ifndef _WIN32 + jobserver_.Clear(); +#endif } size_t RealCommandRunner::CanRunMore() const { @@ -631,6 +662,18 @@ size_t RealCommandRunner::CanRunMore() const { capacity = load_capacity; } +// TODO: jobserver client support for Windows +#ifndef _WIN32 + int job_tokens = jobserver_.Tokens(); + + // When initialized, behave as if the implicit token is acquired already. + // Otherwise, this occurs after a token is released but before it is replaced, + // so the base capacity is represented by job_tokens + 1 when positive. + // Add an extra loop on capacity for each job in order to get an extra token. + if (job_tokens) + capacity = abs(job_tokens) - subproc_number + 2; +#endif + if (capacity < 0) capacity = 0; @@ -670,10 +713,10 @@ bool RealCommandRunner::WaitForCommand(Result* result) { return true; } -Builder::Builder(State* state, const BuildConfig& config, BuildLog* build_log, - DepsLog* deps_log, DiskInterface* disk_interface, - Status* status, int64_t start_time_millis) - : state_(state), config_(config), plan_(this), status_(status), +Builder::Builder(State* state, const BuildConfig& config, Jobserver& jobserver, + BuildLog* build_log, DepsLog* deps_log, DiskInterface* disk_interface, + Status* status, int64_t start_time_millis) : plan_(this), + state_(state), config_(config), jobserver_(jobserver), status_(status), start_time_millis_(start_time_millis), disk_interface_(disk_interface), explanations_(g_explaining ? new Explanations() : nullptr), scan_(state, build_log, deps_log, disk_interface, @@ -778,7 +821,7 @@ bool Builder::Build(string* err) { if (config_.dry_run) command_runner_.reset(new DryRunCommandRunner); else - command_runner_.reset(new RealCommandRunner(config_)); + command_runner_.reset(new RealCommandRunner(config_, jobserver_)); } // We are about to start the build process. diff --git a/src/build.h b/src/build.h index 9bb0c70b5c..117f0a5efe 100644 --- a/src/build.h +++ b/src/build.h @@ -24,6 +24,7 @@ #include "depfile_parser.h" #include "exit_status.h" #include "graph.h" +#include "jobserver.h" #include "util.h" // int64_t struct BuildLog; @@ -187,9 +188,9 @@ struct BuildConfig { /// Builder wraps the build process: starting commands, updating status. struct Builder { - Builder(State* state, const BuildConfig& config, BuildLog* build_log, - DepsLog* deps_log, DiskInterface* disk_interface, Status* status, - int64_t start_time_millis); + Builder(State* state, const BuildConfig& config, Jobserver& jobserver, + BuildLog* build_log, DepsLog* deps_log, DiskInterface* disk_interface, + Status* status, int64_t start_time_millis); ~Builder(); /// Clean up after interrupted commands by deleting output files. @@ -224,6 +225,7 @@ struct Builder { State* state_; const BuildConfig& config_; + Jobserver& jobserver_; Plan plan_; std::unique_ptr command_runner_; Status* status_; diff --git a/src/graph.h b/src/graph.h index 314c44296a..7f1496130c 100644 --- a/src/graph.h +++ b/src/graph.h @@ -227,6 +227,7 @@ struct Edge { bool deps_loaded_ = false; bool deps_missing_ = false; bool generated_by_dep_loader_ = false; + bool has_job_token_ = false; TimeStamp command_start_time_ = 0; const Rule& rule() const { return *rule_; } diff --git a/src/jobserver-posix.cc b/src/jobserver-posix.cc new file mode 100644 index 0000000000..7dd8952a0a --- /dev/null +++ b/src/jobserver-posix.cc @@ -0,0 +1,160 @@ +// Copyright 2024 Google Inc. All Rights Reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +#include "jobserver.h" + +#include +#include + +#include +#include + +#include "util.h" + +Jobserver::Jobserver() { + assert(rfd_ < 0); + assert(wfd_ < 0); + + // Return early if no makeflags are passed in the environment + char* makeflags = std::getenv("MAKEFLAGS"); + if (makeflags == nullptr) { + return; + } + + // Tokenize string to characters in flag_, then words in flags_ + while (flag_char_ < strlen(makeflags)) { + while (flag_char_ < strlen(makeflags) && + !isblank(static_cast(makeflags[flag_char_]))) { + flag_.push_back(static_cast(makeflags[flag_char_])); + flag_char_++; + } + + if (flag_.size() > 0) + flags_.push_back(flag_); + + flag_.clear(); + flag_char_++; + } + + // Search for --jobserver-auth + for (size_t n = 0; n < flags_.size(); n++) + if (flags_[n].find(AUTH_KEY) == 0) + flag_ = flags_[n].substr(strlen(AUTH_KEY)); + + // Fallback to --jobserver-fds + if (flag_.empty()) + for (size_t n = 0; n < flags_.size(); n++) + if (flags_[n].find(FDS_KEY) == 0) + flag_ = flags_[n].substr(strlen(FDS_KEY)); + + jobserver_name_.assign(flag_); + + const char* jobserver = jobserver_name_.c_str(); + + // Return early if the flag's value is empty + if (jobserver_name_.empty()) { + Warning("invalid jobserver value: '%s'", jobserver); + return; + } + + jobserver_fifo_ = jobserver_name_.find(FIFO_KEY) == 0 ? true : false; + + // Return early if jobserver type is unknown (neither fifo nor pipe) + if (!jobserver_fifo_ && sscanf(jobserver, "%d,%d", &rfd_, &wfd_) != 2) { + Warning("invalid jobserver value: '%s'", jobserver); + return; + } + + if (jobserver_fifo_) { + rfd_ = open(jobserver + strlen(FIFO_KEY), O_RDONLY | O_NONBLOCK); + wfd_ = open(jobserver + strlen(FIFO_KEY), O_WRONLY); + } + + if (rfd_ == -1 || wfd_ == -1) + Fatal("failed to open jobserver: %s: %s", + jobserver, errno ? strerror(errno) : "Bad file descriptor"); + else if (rfd_ > 0 && wfd_ > 0) + Info("using jobserver: %s", jobserver); + else + Fatal("Make provided an invalid pipe: %s (mark the command as recursive)", + jobserver); + + // Signal that we have initialized but do not have a token yet + token_count_ = -1; +} + +Jobserver::~Jobserver() { + Clear(); + + if (rfd_ >= 0) + close(rfd_); + if (wfd_ >= 0) + close(wfd_); + + rfd_ = -1; + wfd_ = -1; +} + +bool Jobserver::Enabled() const { + return rfd_ >= 0 && wfd_ >= 0; +} + +int Jobserver::Tokens() const { + return token_count_; +} + +bool Jobserver::Acquire() { + // The first token is implicitly handed to a process + if (token_count_ <= 0) { + token_count_ = 1; + return true; + } + + char token; + int ret = read(rfd_, &token, 1); + if (ret < 0 && errno != EAGAIN && errno != EWOULDBLOCK) { + if (!jobserver_fifo_) + Warning("Make closed the pipe: %d (mark the command as recursive)", rfd_); + Fatal("failed to read token from jobserver: %d: %s", rfd_, strerror(errno)); + } + + if (ret > 0) + token_count_++; + + return ret > 0; +} + +void Jobserver::Release() { + if (token_count_ < 0) + token_count_ = 0; + if (token_count_ > 0) + token_count_--; + + // The first token is implicitly handed to a process + if (token_count_ == 0) + return; + + char token = '+'; + int ret = write(wfd_, &token, 1); + if (ret != 1) { + if (!jobserver_fifo_) + Warning("Make closed the pipe: %d (mark the command as recursive)", wfd_); + Fatal("failed to return token to jobserver: %d: %s", rfd_, strerror(errno)); + } +} + +void Jobserver::Clear() { + while (token_count_) + Release(); +} diff --git a/src/jobserver.h b/src/jobserver.h new file mode 100644 index 0000000000..f94a1ef475 --- /dev/null +++ b/src/jobserver.h @@ -0,0 +1,107 @@ +// Copyright 2024 Google Inc. All Rights Reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +#pragma once + +#include +#include + +#define AUTH_KEY "--jobserver-auth=" +#define FDS_KEY "--jobserver-fds=" +#define FIFO_KEY "fifo:" + +/// The GNU jobserver limits parallelism by assigning a token from an external +/// pool for each command. On posix systems, the pool is a fifo or simple pipe +/// with N characters. On windows systems, the pool is a semaphore initialized +/// to N. When a command is finished, the acquired token is released by writing +/// it back to the fifo or pipe or by increasing the semaphore count. +/// +/// The jobserver functionality is enabled by passing --jobserver-auth= +/// (previously --jobserver-fds= in older versions of Make) in the MAKEFLAGS +/// environment variable and creating the respective file descriptors or objects. +/// On posix systems, is 'fifo:' or ',' for pipes. +/// On windows systems, is the name of the semaphore. +/// +/// The class parses the MAKEFLAGS variable and opens the object handle if needed. +/// Once enabled, Acquire() must be called to acquire a token from the pool. +/// If a token is acquired, a new command can be started. +/// Once the command is completed, Release() must be called to return a token. +/// Make does not care which order a token is recieved or returned. +struct Jobserver { +// TODO: jobserver client support for Windows +#ifndef _WIN32 + /// Parse the MAKEFLAGS environment variable to receive the path / FDs / name + /// of the token pool, and open the handle to the pool if it is an object. + /// If a jobserver argument is found in the MAKEFLAGS environment variable, + /// and the handle is successfully opened, later calls to Enable() return true. + /// If a jobserver argument is found, but the handle fails to be opened, + /// the ninja process is aborted with an error. + Jobserver(); + + /// Before exiting Ninja, ensure that tokens are returned and handles closed. + ~Jobserver(); +#endif + + /// Return true if jobserver functionality is enabled and initialized. + bool Enabled() const; + + /// Return current token count or initialization signal if negative. + int Tokens() const; + + /// Implementation-specific method to acquire a token from the external pool + /// which is called for all tokens but returns early for the first token. + /// This method is called every time Ninja needs to start a command process. + /// Return true on success (token acquired), and false on failure (no tokens + /// available). First call always succeeds. Ninja is aborted on read errors. + bool Acquire(); + + /// Implementation-specific method to release a token to the external pool + /// which is called for all tokens but returns early for the last token. + /// Return a previously acquired token to the external token pool. + /// It must be called for each successful call to Acquire() after the command + /// even if subprocesses fail or in the case of errors causing Ninja to exit. + /// Ninja is aborted on write errors, and otherwise calls always succeed. + void Release(); + + /// Loop through Release() to return all tokens. Called before Ninja exits. + void Clear(); + +private: + /// The number of currently acquired tokens, or the jobserver status if negative. + /// Used to verify that all acquired tokens have been released before exiting, + /// and when the implicit (first) token has been acquired (initialization). + /// -1: initialized without a token + /// 0: uninitialized or disabled + /// +n: number of tokens in use + int token_count_ = 0; + + /// String of the parsed value of the jobserver flag passed to environment. + std::string jobserver_name_; + +// TODO: jobserver client support for Windows +#ifdef _WIN32 +#else + /// Whether the type of jobserver pipe supplied to ninja is named + bool jobserver_fifo_; + + /// File descriptors to communicate with upstream jobserver token pool. + int rfd_ = -1; + int wfd_ = -1; +#endif + + /// Helpers for initialization + std::vector flags_; + std::string flag_; + std::string::size_type flag_char_ = 0; +}; diff --git a/src/ninja.cc b/src/ninja.cc index 2902359f15..4351bca593 100644 --- a/src/ninja.cc +++ b/src/ninja.cc @@ -84,8 +84,8 @@ struct Options { /// The Ninja main() loads up a series of data structures; various tools need /// to poke into these, so store them as fields on an object. struct NinjaMain : public BuildLogUser { - NinjaMain(const char* ninja_command, const BuildConfig& config) : - ninja_command_(ninja_command), config_(config), + NinjaMain(const char* ninja_command, const BuildConfig& config, Jobserver& jobserver) : + ninja_command_(ninja_command), config_(config), jobserver_(jobserver), start_time_millis_(GetTimeMillis()) {} /// Command line used to run Ninja. @@ -94,6 +94,9 @@ struct NinjaMain : public BuildLogUser { /// Build configuration set from flags (e.g. parallelism). const BuildConfig& config_; + /// Reference to jobserver client. + Jobserver& jobserver_; + /// Loaded state (rules, nodes). State state_; @@ -267,7 +270,8 @@ bool NinjaMain::RebuildManifest(const char* input_file, string* err, if (!node) return false; - Builder builder(&state_, config_, &build_log_, &deps_log_, &disk_interface_, + Builder builder(&state_, config_, jobserver_, + &build_log_, &deps_log_, &disk_interface_, status, start_time_millis_); if (!builder.AddTarget(node, err)) return false; @@ -1355,7 +1359,8 @@ int NinjaMain::RunBuild(int argc, char** argv, Status* status) { disk_interface_.AllowStatCache(g_experimental_statcache); - Builder builder(&state_, config_, &build_log_, &deps_log_, &disk_interface_, + Builder builder(&state_, config_, jobserver_, + &build_log_, &deps_log_, &disk_interface_, status, start_time_millis_); for (size_t i = 0; i < targets.size(); ++i) { if (!builder.AddTarget(targets[i], &err)) { @@ -1542,6 +1547,8 @@ NORETURN void real_main(int argc, char** argv) { Status* status = Status::factory(config); + Jobserver jobserver; + if (options.working_dir) { // The formatting of this string, complete with funny quotes, is // so Emacs can properly identify that the cwd has changed for @@ -1558,14 +1565,14 @@ NORETURN void real_main(int argc, char** argv) { if (options.tool && options.tool->when == Tool::RUN_AFTER_FLAGS) { // None of the RUN_AFTER_FLAGS actually use a NinjaMain, but it's needed // by other tools. - NinjaMain ninja(ninja_command, config); + NinjaMain ninja(ninja_command, config, jobserver); exit((ninja.*options.tool->func)(&options, argc, argv)); } // Limit number of rebuilds, to prevent infinite loops. const int kCycleLimit = 100; for (int cycle = 1; cycle <= kCycleLimit; ++cycle) { - NinjaMain ninja(ninja_command, config); + NinjaMain ninja(ninja_command, config, jobserver); ManifestParserOptions parser_opts; if (options.phony_cycle_should_err) {