diff options
Diffstat (limited to 'src/main/cpp/blaze_util.cc')
-rw-r--r-- | src/main/cpp/blaze_util.cc | 336 |
1 files changed, 336 insertions, 0 deletions
diff --git a/src/main/cpp/blaze_util.cc b/src/main/cpp/blaze_util.cc new file mode 100644 index 0000000000..287cea012a --- /dev/null +++ b/src/main/cpp/blaze_util.cc @@ -0,0 +1,336 @@ +// Copyright 2014 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 "blaze_util.h" + +#include <errno.h> +#include <fcntl.h> +#include <limits.h> +#include <pwd.h> +#include <stdarg.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <sys/ioctl.h> +#include <sys/socket.h> +#include <sys/stat.h> +#include <sys/xattr.h> +#include <unistd.h> +#include <sstream> + +#include "util/numbers.h" +#include "util/strings.h" + +using std::vector; + +namespace blaze { + +void die(const int exit_status, const char *format, ...) { + va_list ap; + va_start(ap, format); + vfprintf(stderr, format, ap); + va_end(ap); + fputc('\n', stderr); + exit(exit_status); +} + +void pdie(const int exit_status, const char *format, ...) { + fprintf(stderr, "Error: "); + va_list ap; + va_start(ap, format); + vfprintf(stderr, format, ap); + va_end(ap); + fprintf(stderr, ": %s\n", strerror(errno)); + exit(exit_status); +} + +string GetUserName() { + const char *user = getenv("USER"); + if (user && user[0] != '\0') 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; +} + +// Returns the given path in absolute form. Does not change paths that are +// already absolute. +// +// If called from working directory "/bar": +// MakeAbsolute("foo") --> "/bar/foo" +// MakeAbsolute("/foo") ---> "/foo" +string MakeAbsolute(string path) { + // Check if path is already absolute. + if (path.empty() || path[0] == '/') { + return path; + } + + char cwdbuf[PATH_MAX]; + if (getcwd(cwdbuf, sizeof cwdbuf) == NULL) { + pdie(blaze_exit_code::LOCAL_ENVIRONMENTAL_ERROR, "getcwd() failed"); + } + + // Determine whether the cwd ends with "/" or not. + string separator = (cwdbuf[strlen(cwdbuf) - 1] == '/') ? "" : "/"; + return cwdbuf + separator + path; +} + +// mkdir -p path. Returns -1 on failure, sets errno. +int MakeDirectories(string path, int mode) { + path.push_back('\0'); + char *buf = &path[0]; + for (char *slash = strchr(buf + 1, '/'); slash != NULL; + slash = strchr(slash + 1, '/')) { + *slash = '\0'; + if (mkdir(buf, mode) == -1 && errno != EEXIST) { + return -1; + } + *slash = '/'; + } + // TODO(bazel-team): EEXIST does not prove that it's a directory! + if (mkdir(buf, mode) == -1 && errno != EEXIST) { + return -1; + } + return 0; +} + +// Replaces 'content' with contents of file 'filename'. +// Returns false on error. +bool ReadFile(const string &filename, string *content) { + content->clear(); + int fd = open(filename.c_str(), O_RDONLY); + if (fd == -1) return false; + char buf[4096]; + // OPT: This loop generates one spurious read on regular files. + while (int r = read(fd, buf, sizeof buf)) { + if (r == -1) { + if (errno == EINTR) continue; + return false; + } + content->append(buf, r); + } + close(fd); + return true; +} + +// Writes 'content' into file 'filename', and makes it executable. +// Returns false on failure, sets errno. +bool WriteFile(const string &content, const string &filename) { + unlink(filename.c_str()); + int fd = open(filename.c_str(), O_CREAT|O_WRONLY|O_TRUNC, 0755); // chmod +x + if (fd == -1) return false; + int r = write(fd, content.data(), content.size()); + int saved_errno = errno; + if (close(fd)) return false; // Can fail on NFS. + errno = saved_errno; // Caller should see errno from write(). + return r == content.size(); +} + +// 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") == nullptr ? "" : getenv("TERM"); + string emacs = getenv("EMACS") == nullptr ? "" : getenv("EMACS"); + if (term == "" || term == "dumb" || term == "emacs" || term == "xterm-mono" || + term == "symbolics" || term == "9term" || emacs == "t") { + 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; + } + const char* columns_env = getenv("COLUMNS"); + if (columns_env != NULL && columns_env[0] != '\0') { + char* endptr; + int columns = blaze_util::strto32(columns_env, &endptr, 10); + if (*endptr == '\0') { // $COLUMNS is a valid number + return columns; + } + } + return 80; // default if not a terminal. +} + +// Replace the current process with the given program in the given working +// directory, using the given argument vector. +// This function does not return on success. +void ExecuteProgram(string exe, const vector<string>& args_vector) { + if (VerboseLogging()) { + string dbg; + for (const auto& s : args_vector) { + dbg.append(s); + dbg.append(" "); + } + + char cwd[PATH_MAX] = {}; + getcwd(cwd, sizeof(cwd)); + + fprintf(stderr, "Invoking binary %s in %s:\n %s\n", + exe.c_str(), cwd, dbg.c_str()); + } + + // Copy to a char* array for execv: + int n = args_vector.size(); + const char **argv = new const char *[n + 1]; + for (int i = 0; i < n; ++i) { + argv[i] = args_vector[i].c_str(); + } + argv[n] = NULL; + + execv(exe.c_str(), const_cast<char**>(argv)); +} + +// Re-execute the blaze command line with a different binary as argv[0]. +// This function does not return on success. +void ReExecute(const string &executable, int argc, const char *argv[]) { + vector<string> args; + args.push_back(executable); + for (int i = 1; i < argc; i++) { + args.push_back(argv[i]); + } + ExecuteProgram(args[0], args); +} + +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; +} + +blaze_exit_code::ExitCode CheckValidPort( + const string &str, const string &option, string *error) { + int number; + if (blaze_util::safe_strto32(str, &number) && number > 0 && number < 65536) { + return blaze_exit_code::SUCCESS; + } + + blaze_util::StringPrintf(error, + "Invalid argument to %s: '%s' (must be a valid port number).", + option.c_str(), str.c_str()); + return blaze_exit_code::BAD_ARGV; +} + +bool VerboseLogging() { + return getenv("VERBOSE_BLAZE_CLIENT") != NULL; +} + +// Read the Jvm version from a file descriptor. The read fd +// should contains a similar output as the java -version output. +string ReadJvmVersion(int fd) { + static const int bytes_to_read = 255; + char buf[bytes_to_read + 1]; // leave one extra space for null + ssize_t size = read(fd, buf, bytes_to_read); + close(fd); + if (size > 0) { + buf[size] = 0; + // try to look out for 'version "' + static const char version_pattern[] = "version \""; + char *ptr = strstr(buf, version_pattern); + if (ptr != NULL) { + ptr += sizeof(version_pattern)-1; + // If we found "version \"", process until next '"' + char *endptr = strchr(ptr, '"'); + if (endptr != NULL) { + *endptr = 0; + } + return string(ptr); + } + } + return ""; +} + +string GetJvmVersion(string java_exe) { + vector<string> args; + args.push_back("java"); + args.push_back("-version"); + + int fds[2]; + if (pipe(fds)) { + pdie(blaze_exit_code::INTERNAL_ERROR, "pipe creation failed"); + } + + int child = fork(); + if (child == -1) { + pdie(blaze_exit_code::INTERNAL_ERROR, "fork() failed"); + } else if (child > 0) { // we're the parent + close(fds[1]); // parent keeps only the reading side + return ReadJvmVersion(fds[0]); + } else { + close(fds[0]); // child keeps only the writing side + // Redirect output to the writing side of the dup. + dup2(fds[1], STDOUT_FILENO); + dup2(fds[1], STDERR_FILENO); + // Execute java -version + ExecuteProgram(java_exe, args); + pdie(blaze_exit_code::INTERNAL_ERROR, "Failed to run java -version"); + } +} + +bool CheckJavaVersionIsAtLeast(string jvm_version, string version_spec) { + vector<string> jvm_version_vect = blaze_util::Split(jvm_version, '.'); + vector<string> version_spec_vect = blaze_util::Split(version_spec, '.'); + int i; + for (i = 0; i < jvm_version_vect.size() && i < version_spec_vect.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_vect.size()) { + for (; i < version_spec_vect.size(); i++) { + if (version_spec_vect[i] != "0") { + return false; + } + } + } + return true; +} + +} // namespace blaze |