blob: 0ab9b763133139404ceaad861b149d757dd438cd [file] [log] [blame]
// Copyright 2014 The Bazel Authors. 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 "src/main/cpp/blaze_util.h"
#include <errno.h>
#include <fcntl.h>
#include <pwd.h>
#include <stdarg.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/file.h>
#include <sys/ioctl.h>
#include <sys/socket.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include <sstream>
#include "src/main/cpp/blaze_util_platform.h"
#include "src/main/cpp/util/errors.h"
#include "src/main/cpp/util/exit_code.h"
#include "src/main/cpp/util/file.h"
#include "src/main/cpp/util/file_platform.h"
#include "src/main/cpp/util/numbers.h"
#include "src/main/cpp/util/strings.h"
#include "src/main/cpp/util/port.h"
using blaze_util::die;
using blaze_util::pdie;
namespace blaze {
using std::string;
using std::vector;
const char kServerPidFile[] = "server.pid.txt";
const char kServerPidSymlink[] = "server.pid";
string GetUserName() {
string user = GetEnv("USER");
if (!user.empty()) {
return user;
}
errno = 0;
passwd *pwent = getpwuid(getuid()); // NOLINT (single-threaded)
if (pwent == NULL || pwent->pw_name == NULL) {
pdie(blaze_exit_code::LOCAL_ENVIRONMENTAL_ERROR,
"$USER is not set, and unable to look up name of current user");
}
return pwent->pw_name;
}
string MakeAbsolute(const string &path) {
// Check if path is already absolute.
if (path.empty() || path[0] == '/' || (isalpha(path[0]) && path[1] == ':')) {
return path;
}
string cwd = blaze_util::GetCwd();
// Determine whether the cwd ends with "/" or not.
string separator = cwd.back() == '/' ? "" : "/";
return cwd + separator + path;
}
bool ReadFrom(const std::function<int(void *, int)> &read_func, string *content,
int max_size) {
content->clear();
char buf[4096];
// OPT: This loop generates one spurious read on regular files.
while (int r = read_func(
buf, max_size > 0
? std::min(max_size, static_cast<int>(sizeof buf))
: sizeof buf)) {
if (r == -1) {
if (errno == EINTR || errno == EAGAIN) continue;
return false;
}
content->append(buf, r);
if (max_size > 0) {
if (max_size > r) {
max_size -= r;
} else {
break;
}
}
}
return true;
}
bool ReadFile(const string &filename, string *content, int max_size) {
int fd = open(filename.c_str(), O_RDONLY);
if (fd == -1) return false;
bool result =
ReadFrom([fd](void *buf, int len) { return read(fd, buf, len); }, content,
max_size);
close(fd);
return result;
}
bool WriteFile(const void* data, size_t size, const string &filename) {
UnlinkPath(filename); // We don't care about the success of this.
int fd = open(filename.c_str(), O_CREAT|O_WRONLY|O_TRUNC, 0755); // chmod +x
if (fd == -1) {
return false;
}
bool result = WriteTo(
[fd](const void *buf, size_t bufsize) { return write(fd, buf, bufsize); },
data, size);
int saved_errno = errno;
if (close(fd)) {
return false; // Can fail on NFS.
}
errno = saved_errno; // Caller should see errno from write().
return result;
}
bool WriteTo(const std::function<int(const void *, size_t)> &write_func,
const void *data, size_t size) {
int r = write_func(data, size);
if (r == -1) {
return false;
}
return static_cast<uint>(r) == size;
}
bool WriteFile(const std::string &content, const std::string &filename) {
return WriteFile(content.c_str(), content.size(), filename);
}
bool UnlinkPath(const string &file_path) {
return unlink(file_path.c_str()) == 0;
}
bool IsEmacsTerminal() {
string emacs = GetEnv("EMACS");
string inside_emacs = GetEnv("INSIDE_EMACS");
// GNU Emacs <25.1 (and ~all non-GNU emacsen) set EMACS=t, but >=25.1 doesn't
// do that and instead sets INSIDE_EMACS=<stuff> (where <stuff> can look like
// e.g. "25.1.1,comint"). So we check both variables for maximum
// compatibility.
return emacs == "t" || !inside_emacs.empty();
}
// Returns true iff both stdout and stderr are connected to a
// terminal, and it can support color and cursor movement
// (this is computed heuristically based on the values of
// environment variables).
bool IsStandardTerminal() {
string term = GetEnv("TERM");
if (term.empty() || term == "dumb" || term == "emacs" ||
term == "xterm-mono" || term == "symbolics" || term == "9term" ||
IsEmacsTerminal()) {
return false;
}
return isatty(STDOUT_FILENO) && isatty(STDERR_FILENO);
}
// Returns the number of columns of the terminal to which stdout is
// connected, or $COLUMNS (default 80) if there is no such terminal.
int GetTerminalColumns() {
struct winsize ws;
if (ioctl(STDOUT_FILENO, TIOCGWINSZ, &ws) != -1) {
return ws.ws_col;
}
string columns_env = GetEnv("COLUMNS");
if (!columns_env.empty()) {
char* endptr;
int columns = blaze_util::strto32(columns_env.c_str(), &endptr, 10);
if (*endptr == '\0') { // $COLUMNS is a valid number
return columns;
}
}
return 80; // default if not a terminal.
}
const char* GetUnaryOption(const char *arg,
const char *next_arg,
const char *key) {
const char *value = blaze_util::var_strprefix(arg, key);
if (value == NULL) {
return NULL;
} else if (value[0] == '=') {
return value + 1;
} else if (value[0]) {
return NULL; // trailing garbage in key name
}
return next_arg;
}
bool GetNullaryOption(const char *arg, const char *key) {
const char *value = blaze_util::var_strprefix(arg, key);
if (value == NULL) {
return false;
} else if (value[0] == '=') {
die(blaze_exit_code::BAD_ARGV,
"In argument '%s': option '%s' does not take a value.", arg, key);
} else if (value[0]) {
return false; // trailing garbage in key name
}
return true;
}
bool VerboseLogging() { return !GetEnv("VERBOSE_BLAZE_CLIENT").empty(); }
// Read the Jvm version from a file descriptor. The read fd
// should contains a similar output as the java -version output.
string ReadJvmVersion(const string& version_string) {
// try to look out for 'version "'
static const string version_pattern = "version \"";
size_t found = version_string.find(version_pattern);
if (found != string::npos) {
found += version_pattern.size();
// If we found "version \"", process until next '"'
size_t end = version_string.find("\"", found);
if (end == string::npos) { // consider end of string as a '"'
end = version_string.size();
}
return version_string.substr(found, end - found);
}
return "";
}
string GetJvmVersion(const string &java_exe) {
vector<string> args;
args.push_back("java");
args.push_back("-version");
string version_string = RunProgram(java_exe, args);
return ReadJvmVersion(version_string);
}
bool CheckJavaVersionIsAtLeast(const string &jvm_version,
const string &version_spec) {
vector<string> jvm_version_vect = blaze_util::Split(jvm_version, '.');
int jvm_version_size = static_cast<int>(jvm_version_vect.size());
vector<string> version_spec_vect = blaze_util::Split(version_spec, '.');
int version_spec_size = static_cast<int>(version_spec_vect.size());
int i;
for (i = 0; i < jvm_version_size && i < version_spec_size; i++) {
int jvm = blaze_util::strto32(jvm_version_vect[i].c_str(), NULL, 10);
int spec = blaze_util::strto32(version_spec_vect[i].c_str(), NULL, 10);
if (jvm > spec) {
return true;
} else if (jvm < spec) {
return false;
}
}
if (i < version_spec_size) {
for (; i < version_spec_size; i++) {
if (version_spec_vect[i] != "0") {
return false;
}
}
}
return true;
}
uint64_t AcquireLock(const string& output_base, bool batch_mode, bool block,
BlazeLock* blaze_lock) {
string lockfile = output_base + "/lock";
int lockfd = open(lockfile.c_str(), O_CREAT|O_RDWR, 0644);
if (lockfd < 0) {
pdie(blaze_exit_code::LOCAL_ENVIRONMENTAL_ERROR,
"cannot open lockfile '%s' for writing", lockfile.c_str());
}
// Keep server from inheriting a useless fd if we are not in batch mode
if (!batch_mode) {
if (fcntl(lockfd, F_SETFD, FD_CLOEXEC) == -1) {
pdie(blaze_exit_code::LOCAL_ENVIRONMENTAL_ERROR,
"fcntl(F_SETFD) failed for lockfile");
}
}
struct flock lock;
lock.l_type = F_WRLCK;
lock.l_whence = SEEK_SET;
lock.l_start = 0;
// This doesn't really matter now, but allows us to subdivide the lock
// later if that becomes meaningful. (Ranges beyond EOF can be locked.)
lock.l_len = 4096;
uint64_t wait_time = 0;
// Try to take the lock, without blocking.
if (fcntl(lockfd, F_SETLK, &lock) == -1) {
if (errno != EACCES && errno != EAGAIN) {
pdie(blaze_exit_code::LOCAL_ENVIRONMENTAL_ERROR,
"unexpected result from F_SETLK");
}
// We didn't get the lock. Find out who has it.
struct flock probe = lock;
probe.l_pid = 0;
if (fcntl(lockfd, F_GETLK, &probe) == -1) {
pdie(blaze_exit_code::LOCAL_ENVIRONMENTAL_ERROR,
"unexpected result from F_GETLK");
}
if (!block) {
die(blaze_exit_code::BAD_ARGV,
"Another command is running (pid=%d). Exiting immediately.",
probe.l_pid);
}
fprintf(stderr, "Another command is running (pid = %d). "
"Waiting for it to complete...", probe.l_pid);
fflush(stderr);
// Take a clock sample for that start of the waiting time
uint64_t st = GetMillisecondsMonotonic();
// Try to take the lock again (blocking).
int r;
do {
r = fcntl(lockfd, F_SETLKW, &lock);
} while (r == -1 && errno == EINTR);
fprintf(stderr, "\n");
if (r == -1) {
pdie(blaze_exit_code::LOCAL_ENVIRONMENTAL_ERROR,
"couldn't acquire file lock");
}
// Take another clock sample, calculate elapsed
uint64_t et = GetMillisecondsMonotonic();
wait_time = et - st;
}
// Identify ourselves in the lockfile.
(void) ftruncate(lockfd, 0);
const char *tty = ttyname(STDIN_FILENO); // NOLINT (single-threaded)
string msg = "owner=launcher\npid="
+ ToString(getpid()) + "\ntty=" + (tty ? tty : "") + "\n";
// The contents are currently meant only for debugging.
(void) write(lockfd, msg.data(), msg.size());
blaze_lock->lockfd = lockfd;
return wait_time;
}
void ReleaseLock(BlazeLock* blaze_lock) {
close(blaze_lock->lockfd);
}
bool IsArg(const string& arg) {
return blaze_util::starts_with(arg, "-") && (arg != "--help")
&& (arg != "-help") && (arg != "-h");
}
} // namespace blaze