Skip to content

Commit

Permalink
Add GNU make jobserver client support
Browse files Browse the repository at this point in the history
- add new TokenPool interface
- GNU make implementation for TokenPool parses and verifies the magic
  information from the MAKEFLAGS environment variable
- RealCommandRunner tries to acquire TokenPool
  * if no token pool is available then there is no change in behaviour
- When a token pool is available then RealCommandRunner behaviour
  changes as follows
  * CanRunMore() only returns true if TokenPool::Acquire() returns true
  * StartCommand() calls TokenPool::Reserve()
  * WaitForCommand() calls TokenPool::Release()

Documentation for GNU make jobserver

  http://make.mad-scientist.net/papers/jobserver-implementation/

Fixes ninja-build#1139
  • Loading branch information
Stefan Becker committed May 27, 2016
1 parent 63a8584 commit 1b05452
Show file tree
Hide file tree
Showing 5 changed files with 289 additions and 4 deletions.
2 changes: 2 additions & 0 deletions configure.py
Original file line number Diff line number Diff line change
Expand Up @@ -494,6 +494,7 @@ def has_re2c():
objs += cxx(name)
if platform.is_windows():
for name in ['subprocess-win32',
'tokenpool-none',
'includes_normalize-win32',
'msvc_helper-win32',
'msvc_helper_main-win32']:
Expand All @@ -503,6 +504,7 @@ def has_re2c():
objs += cc('getopt')
else:
objs += cxx('subprocess-posix')
objs += cxx('tokenpool-gnu-make')
if platform.is_aix():
objs += cc('getopt')
if platform.is_msvc():
Expand Down
27 changes: 23 additions & 4 deletions src/build.cc
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@
#include "graph.h"
#include "state.h"
#include "subprocess.h"
#include "tokenpool.h"
#include "util.h"

namespace {
Expand Down Expand Up @@ -502,8 +503,8 @@ void Plan::Dump() {
}

struct RealCommandRunner : public CommandRunner {
explicit RealCommandRunner(const BuildConfig& config) : config_(config) {}
virtual ~RealCommandRunner() {}
explicit RealCommandRunner(const BuildConfig& config);
virtual ~RealCommandRunner();
virtual bool CanRunMore();
virtual bool StartCommand(Edge* edge);
virtual bool WaitForCommand(Result* result);
Expand All @@ -512,9 +513,18 @@ struct RealCommandRunner : public CommandRunner {

const BuildConfig& config_;
SubprocessSet subprocs_;
TokenPool *tokens_;
map<Subprocess*, Edge*> subproc_to_edge_;
};

RealCommandRunner::RealCommandRunner(const BuildConfig& config) : config_(config) {
tokens_ = TokenPool::Get();
}

RealCommandRunner::~RealCommandRunner() {
delete tokens_;
}

vector<Edge*> RealCommandRunner::GetActiveEdges() {
vector<Edge*> edges;
for (map<Subprocess*, Edge*>::iterator e = subproc_to_edge_.begin();
Expand All @@ -525,21 +535,27 @@ vector<Edge*> RealCommandRunner::GetActiveEdges() {

void RealCommandRunner::Abort() {
subprocs_.Clear();
if (tokens_)
tokens_->Clear();
}

bool RealCommandRunner::CanRunMore() {
size_t subproc_number =
subprocs_.running_.size() + subprocs_.finished_.size();
return (int)subproc_number < config_.parallelism
&& ((subprocs_.running_.empty() || config_.max_load_average <= 0.0f)
|| GetLoadAverage() < config_.max_load_average);
&& (subprocs_.running_.empty()
|| ((!tokens_ || tokens_->Acquire())
&& (config_.max_load_average <= 0.0f
|| GetLoadAverage() < config_.max_load_average)));
}

bool RealCommandRunner::StartCommand(Edge* edge) {
string command = edge->EvaluateCommand();
Subprocess* subproc = subprocs_.Add(command, edge->use_console());
if (!subproc)
return false;
if (tokens_)
tokens_->Reserve();
subproc_to_edge_.insert(make_pair(subproc, edge));

return true;
Expand All @@ -553,6 +569,9 @@ bool RealCommandRunner::WaitForCommand(Result* result) {
return false;
}

if (tokens_)
tokens_->Release();

result->status = subproc->Finish();
result->output = subproc->GetOutput();

Expand Down
211 changes: 211 additions & 0 deletions src/tokenpool-gnu-make.cc
Original file line number Diff line number Diff line change
@@ -0,0 +1,211 @@
// Copyright 2016 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 "tokenpool.h"

#include <errno.h>
#include <fcntl.h>
#include <poll.h>
#include <unistd.h>
#include <signal.h>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>

// TokenPool implementation for GNU make jobserver
// (http://make.mad-scientist.net/papers/jobserver-implementation/)
struct GNUmakeTokenPool : public TokenPool {
GNUmakeTokenPool();
virtual ~GNUmakeTokenPool();

virtual bool Acquire();
virtual void Reserve();
virtual void Release();
virtual void Clear();

bool Setup();

private:
int available_;
int used_;

#ifdef _WIN32
// @TODO
#else
int rfd_;
int wfd_;

struct sigaction old_act_;
bool restore_;

static int dup_rfd_;
static void CloseDupRfd(int signum);

bool CheckFd(int fd);
bool SetAlarmHandler();
#endif

void Return();
};

// every instance owns an implicit token -> available_ == 1
GNUmakeTokenPool::GNUmakeTokenPool() : available_(1), used_(0),
rfd_(-1), wfd_(-1), restore_(false) {
}

GNUmakeTokenPool::~GNUmakeTokenPool() {
Clear();
if (restore_)
sigaction(SIGALRM, &old_act_, NULL);
}

bool GNUmakeTokenPool::CheckFd(int fd) {
if (fd < 0)
return false;
int ret = fcntl(fd, F_GETFD);
if (ret < 0)
return false;
return true;
}

int GNUmakeTokenPool::dup_rfd_ = -1;

void GNUmakeTokenPool::CloseDupRfd(int signum) {
close(dup_rfd_);
dup_rfd_ = -1;
}

bool GNUmakeTokenPool::SetAlarmHandler() {
struct sigaction act;
memset(&act, 0, sizeof(act));
act.sa_handler = CloseDupRfd;
if (sigaction(SIGALRM, &act, &old_act_) < 0) {
perror("sigaction:");
return(false);
} else {
restore_ = true;
return(true);
}
}

bool GNUmakeTokenPool::Setup() {
const char *value = getenv("MAKEFLAGS");
if (value) {
// GNU make <= 4.1
const char *jobserver = strstr(value, "--jobserver-fds=");
// GNU make => 4.2
if (!jobserver)
jobserver = strstr(value, "--jobserver-auth=");
if (jobserver) {
int rfd = -1;
int wfd = -1;
if ((sscanf(jobserver, "%*[^=]=%d,%d", &rfd, &wfd) == 2) &&
CheckFd(rfd) &&
CheckFd(wfd) &&
SetAlarmHandler()) {
printf("ninja: using GNU make jobserver.\n");
rfd_ = rfd;
wfd_ = wfd;
return true;
}
}
}

return false;
}

bool GNUmakeTokenPool::Acquire() {
if (available_ > 0)
return true;

#ifdef USE_PPOLL
pollfd pollfds[] = {{rfd_, POLLIN, 0}};
int ret = poll(pollfds, 1, 0);
#else
fd_set set;
struct timeval timeout = { 0, 0 };
FD_ZERO(&set);
FD_SET(rfd_, &set);
int ret = select(rfd_ + 1, &set, NULL, NULL, &timeout);
#endif
if (ret > 0) {
dup_rfd_ = dup(rfd_);

if (dup_rfd_ != -1) {
struct sigaction act, old_act;
int ret = 0;

memset(&act, 0, sizeof(act));
act.sa_handler = CloseDupRfd;
if (sigaction(SIGCHLD, &act, &old_act) == 0) {
char buf;

// block until token read, child exits or timeout
alarm(1);
ret = read(dup_rfd_, &buf, 1);
alarm(0);

sigaction(SIGCHLD, &old_act, NULL);
}

CloseDupRfd(0);

if (ret > 0) {
available_++;
return true;
}
}
}
return false;
}

void GNUmakeTokenPool::Reserve() {
available_--;
used_++;
}

void GNUmakeTokenPool::Return() {
const char buf = '+';
while (1) {
int ret = write(wfd_, &buf, 1);
if (ret > 0)
available_--;
if ((ret != -1) || (errno != EINTR))
return;
// write got interrupted - retry
}
}

void GNUmakeTokenPool::Release() {
available_++;
used_--;
if (available_ > 1)
Return();
}

void GNUmakeTokenPool::Clear() {
while (used_ > 0)
Release();
while (available_ > 1)
Return();
}

struct TokenPool *TokenPool::Get(void) {
GNUmakeTokenPool *tokenpool = new GNUmakeTokenPool;
if (tokenpool->Setup())
return tokenpool;
else
delete tokenpool;
return NULL;
}
27 changes: 27 additions & 0 deletions src/tokenpool-none.cc
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
// Copyright 2016 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 "tokenpool.h"

#include <fcntl.h>
#include <poll.h>
#include <unistd.h>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>

// No-op TokenPool implementation
struct TokenPool *TokenPool::Get(void) {
return NULL;
}
26 changes: 26 additions & 0 deletions src/tokenpool.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
// Copyright 2016 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.

// interface to token pool
struct TokenPool {
virtual ~TokenPool() {}

virtual bool Acquire() = 0;
virtual void Reserve() = 0;
virtual void Release() = 0;
virtual void Clear() = 0;

// returns NULL if token pool is not available
static struct TokenPool *Get(void);
};

0 comments on commit 1b05452

Please sign in to comment.