blob: 803dec78a6c2d2a9cec60971028ef78ab27f7ca4 [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/util/file_platform.h"
#include <errno.h>
#include <dirent.h> // DIR, dirent, opendir, closedir
#include <fcntl.h> // O_RDONLY
#include <limits.h> // PATH_MAX
#include <stdlib.h> // getenv
#include <sys/stat.h>
#include <unistd.h> // access, open, close, fsync
#include <utime.h> // utime
#include <vector>
#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/strings.h"
namespace blaze_util {
using std::pair;
using std::string;
class PosixPipe : public IPipe {
public:
PosixPipe(int recv_socket, int send_socket)
: _recv_socket(recv_socket), _send_socket(send_socket) {}
PosixPipe() = delete;
virtual ~PosixPipe() {
close(_recv_socket);
close(_send_socket);
}
bool Send(void* buffer, int size) override {
return size >= 0 && write(_send_socket, buffer, size) == size;
}
int Receive(void* buffer, int size) override {
return size < 0 ? -1 : read(_recv_socket, buffer, size);
}
private:
int _recv_socket;
int _send_socket;
};
IPipe* CreatePipe() {
int fd[2];
if (pipe(fd) < 0) {
pdie(blaze_exit_code::LOCAL_ENVIRONMENTAL_ERROR, "pipe()");
}
if (fcntl(fd[0], F_SETFD, FD_CLOEXEC) == -1) {
pdie(blaze_exit_code::LOCAL_ENVIRONMENTAL_ERROR,
"fcntl(F_SETFD, FD_CLOEXEC) failed");
}
if (fcntl(fd[1], F_SETFD, FD_CLOEXEC) == -1) {
pdie(blaze_exit_code::LOCAL_ENVIRONMENTAL_ERROR,
"fcntl(F_SETFD, FD_CLOEXEC) failed");
}
return new PosixPipe(fd[0], fd[1]);
}
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 UnlinkPath(const string &file_path) {
return unlink(file_path.c_str()) == 0;
}
string Which(const string &executable) {
char *path_cstr = getenv("PATH");
if (path_cstr == NULL || path_cstr[0] == '\0') {
die(blaze_exit_code::LOCAL_ENVIRONMENTAL_ERROR,
"Could not get PATH to find %s", executable.c_str());
}
string path(path_cstr);
std::vector<std::string> pieces = blaze_util::Split(path, ':');
for (auto piece : pieces) {
if (piece.empty()) {
piece = ".";
}
struct stat file_stat;
string candidate = blaze_util::JoinPath(piece, executable);
if (access(candidate.c_str(), X_OK) == 0 &&
stat(candidate.c_str(), &file_stat) == 0 &&
S_ISREG(file_stat.st_mode)) {
return candidate;
}
}
return "";
}
bool PathExists(const string& path) {
return access(path.c_str(), F_OK) == 0;
}
string MakeCanonical(const char *path) {
char *resolved_path = realpath(path, NULL);
if (resolved_path == NULL) {
return "";
} else {
string ret = resolved_path;
free(resolved_path);
return ret;
}
}
bool CanAccess(const string& path, bool read, bool write, bool exec) {
int mode = 0;
if (read) {
mode |= R_OK;
}
if (write) {
mode |= W_OK;
}
if (exec) {
mode |= X_OK;
}
return access(path.c_str(), mode) == 0;
}
bool IsDirectory(const string& path) {
struct stat buf;
return stat(path.c_str(), &buf) == 0 && S_ISDIR(buf.st_mode);
}
void SyncFile(const string& path) {
// fsync always fails on Cygwin with "Permission denied" for some reason.
#ifndef __CYGWIN__
const char* file_path = path.c_str();
int fd = open(file_path, O_RDONLY);
if (fd < 0) {
pdie(blaze_exit_code::LOCAL_ENVIRONMENTAL_ERROR,
"failed to open '%s' for syncing", file_path);
}
if (fsync(fd) < 0) {
pdie(blaze_exit_code::LOCAL_ENVIRONMENTAL_ERROR, "failed to sync '%s'",
file_path);
}
close(fd);
#endif
}
time_t GetMtimeMillisec(const string& path) {
struct stat buf;
if (stat(path.c_str(), &buf)) {
return -1;
} else {
return buf.st_mtime;
}
}
bool SetMtimeMillisec(const string& path, time_t mtime) {
struct utimbuf times = {mtime, mtime};
return utime(path.c_str(), &times) == 0;
}
string GetCwd() {
char cwdbuf[PATH_MAX];
if (getcwd(cwdbuf, sizeof cwdbuf) == NULL) {
pdie(blaze_exit_code::INTERNAL_ERROR, "getcwd() failed");
}
return string(cwdbuf);
}
bool ChangeDirectory(const string& path) {
return chdir(path.c_str()) == 0;
}
void ForEachDirectoryEntry(const string &path,
DirectoryEntryConsumer *consume) {
DIR *dir;
struct dirent *ent;
if ((dir = opendir(path.c_str())) == NULL) {
// This is not a directory or it cannot be opened.
return;
}
while ((ent = readdir(dir)) != NULL) {
if (!strcmp(ent->d_name, ".") || !strcmp(ent->d_name, "..")) {
continue;
}
string filename(blaze_util::JoinPath(path, ent->d_name));
bool is_directory;
if (ent->d_type == DT_UNKNOWN) {
struct stat buf;
if (lstat(filename.c_str(), &buf) == -1) {
die(blaze_exit_code::INTERNAL_ERROR, "stat failed");
}
is_directory = S_ISDIR(buf.st_mode);
} else {
is_directory = (ent->d_type == DT_DIR);
}
consume->Consume(filename, is_directory);
}
closedir(dir);
}
} // namespace blaze_util