2017-05-05 17:40:12 +03:00
|
|
|
#include "lazy.hh"
|
2014-07-10 17:50:51 +03:00
|
|
|
#include "util.hh"
|
|
|
|
#include "affinity.hh"
|
2017-01-17 19:21:02 +02:00
|
|
|
#include "sync.hh"
|
2017-03-15 15:40:47 +02:00
|
|
|
#include "finally.hh"
|
2014-07-10 17:50:51 +03:00
|
|
|
|
2017-01-17 19:21:02 +02:00
|
|
|
#include <cctype>
|
2003-09-11 11:31:29 +03:00
|
|
|
#include <cerrno>
|
|
|
|
#include <cstdio>
|
2008-05-21 14:17:31 +03:00
|
|
|
#include <cstdlib>
|
2007-12-14 16:49:35 +02:00
|
|
|
#include <cstring>
|
2017-01-17 19:21:02 +02:00
|
|
|
#include <iostream>
|
|
|
|
#include <sstream>
|
|
|
|
#include <thread>
|
2017-03-15 15:40:47 +02:00
|
|
|
#include <future>
|
2003-06-16 16:33:38 +03:00
|
|
|
|
2006-12-02 17:45:51 +02:00
|
|
|
#include <fcntl.h>
|
2010-06-24 20:51:13 +03:00
|
|
|
#include <limits.h>
|
2017-05-05 17:40:12 +03:00
|
|
|
#include <pwd.h>
|
2017-08-25 16:57:49 +03:00
|
|
|
#include <sys/ioctl.h>
|
2017-05-05 17:40:12 +03:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/wait.h>
|
|
|
|
#include <unistd.h>
|
2006-09-28 00:04:07 +03:00
|
|
|
|
2013-03-18 17:13:53 +02:00
|
|
|
#ifdef __APPLE__
|
|
|
|
#include <sys/syscall.h>
|
|
|
|
#endif
|
|
|
|
|
2014-08-21 16:31:43 +03:00
|
|
|
#ifdef __linux__
|
|
|
|
#include <sys/prctl.h>
|
|
|
|
#endif
|
|
|
|
|
2003-05-26 16:45:00 +03:00
|
|
|
|
2006-12-07 18:40:41 +02:00
|
|
|
extern char * * environ;
|
|
|
|
|
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
namespace nix {
|
|
|
|
|
|
|
|
|
2014-03-28 17:59:26 +02:00
|
|
|
BaseError & BaseError::addPrefix(const FormatOrString & fs)
|
2006-03-08 16:11:19 +02:00
|
|
|
{
|
2014-03-28 17:59:26 +02:00
|
|
|
prefix_ = fs.s + prefix_;
|
2006-03-08 16:11:19 +02:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-09-21 17:00:03 +03:00
|
|
|
std::string SysError::addErrno(const std::string & s)
|
2003-06-16 16:33:38 +03:00
|
|
|
{
|
2016-09-21 17:00:03 +03:00
|
|
|
errNo = errno;
|
|
|
|
return s + ": " + strerror(errNo);
|
2003-06-16 16:33:38 +03:00
|
|
|
}
|
|
|
|
|
2003-05-26 16:45:00 +03:00
|
|
|
|
2004-05-12 12:35:51 +03:00
|
|
|
string getEnv(const string & key, const string & def)
|
|
|
|
{
|
|
|
|
char * value = getenv(key.c_str());
|
|
|
|
return value ? string(value) : def;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-09-20 16:39:08 +03:00
|
|
|
std::map<std::string, std::string> getEnv()
|
|
|
|
{
|
|
|
|
std::map<std::string, std::string> env;
|
|
|
|
for (size_t i = 0; environ[i]; ++i) {
|
|
|
|
auto s = environ[i];
|
|
|
|
auto eq = strchr(s, '=');
|
|
|
|
if (!eq)
|
|
|
|
// invalid env, just keep going
|
|
|
|
continue;
|
|
|
|
env.emplace(std::string(s, eq), std::string(eq + 1));
|
|
|
|
}
|
|
|
|
return env;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-07 17:37:41 +03:00
|
|
|
Path absPath(Path path, Path dir)
|
2003-05-26 16:45:00 +03:00
|
|
|
{
|
2003-06-16 16:33:38 +03:00
|
|
|
if (path[0] != '/') {
|
2003-05-26 16:45:00 +03:00
|
|
|
if (dir == "") {
|
2010-02-10 17:55:50 +02:00
|
|
|
#ifdef __GNU__
|
|
|
|
/* GNU (aka. GNU/Hurd) doesn't have any limitation on path
|
|
|
|
lengths and doesn't define `PATH_MAX'. */
|
|
|
|
char *buf = getcwd(NULL, 0);
|
|
|
|
if (buf == NULL)
|
|
|
|
#else
|
2003-05-26 16:45:00 +03:00
|
|
|
char buf[PATH_MAX];
|
|
|
|
if (!getcwd(buf, sizeof(buf)))
|
2010-02-10 17:55:50 +02:00
|
|
|
#endif
|
2003-06-16 16:33:38 +03:00
|
|
|
throw SysError("cannot get cwd");
|
2003-05-26 16:45:00 +03:00
|
|
|
dir = buf;
|
2010-02-10 17:55:50 +02:00
|
|
|
#ifdef __GNU__
|
|
|
|
free(buf);
|
|
|
|
#endif
|
2003-05-26 16:45:00 +03:00
|
|
|
}
|
2003-06-16 16:33:38 +03:00
|
|
|
path = dir + "/" + path;
|
2003-05-26 16:45:00 +03:00
|
|
|
}
|
2003-07-07 12:25:26 +03:00
|
|
|
return canonPath(path);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-01-08 19:16:03 +02:00
|
|
|
Path canonPath(const Path & path, bool resolveSymlinks)
|
2003-07-07 12:25:26 +03:00
|
|
|
{
|
2017-04-13 16:32:43 +03:00
|
|
|
assert(path != "");
|
|
|
|
|
2003-07-08 22:58:41 +03:00
|
|
|
string s;
|
|
|
|
|
|
|
|
if (path[0] != '/')
|
2017-07-30 14:27:57 +03:00
|
|
|
throw Error(format("not an absolute path: '%1%'") % path);
|
2003-07-08 22:58:41 +03:00
|
|
|
|
|
|
|
string::const_iterator i = path.begin(), end = path.end();
|
2006-01-08 19:16:03 +02:00
|
|
|
string temp;
|
|
|
|
|
|
|
|
/* Count the number of times we follow a symlink and stop at some
|
|
|
|
arbitrary (but high) limit to prevent infinite loops. */
|
|
|
|
unsigned int followCount = 0, maxFollow = 1024;
|
2003-07-08 22:58:41 +03:00
|
|
|
|
|
|
|
while (1) {
|
|
|
|
|
|
|
|
/* Skip slashes. */
|
|
|
|
while (i != end && *i == '/') i++;
|
|
|
|
if (i == end) break;
|
|
|
|
|
|
|
|
/* Ignore `.'. */
|
|
|
|
if (*i == '.' && (i + 1 == end || i[1] == '/'))
|
|
|
|
i++;
|
|
|
|
|
|
|
|
/* If `..', delete the last component. */
|
2013-01-03 14:00:46 +02:00
|
|
|
else if (*i == '.' && i + 1 < end && i[1] == '.' &&
|
2003-07-08 22:58:41 +03:00
|
|
|
(i + 2 == end || i[2] == '/'))
|
|
|
|
{
|
|
|
|
if (!s.empty()) s.erase(s.rfind('/'));
|
|
|
|
i += 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Normal component; copy it. */
|
|
|
|
else {
|
|
|
|
s += '/';
|
|
|
|
while (i != end && *i != '/') s += *i++;
|
2006-01-08 19:16:03 +02:00
|
|
|
|
|
|
|
/* If s points to a symlink, resolve it and restart (since
|
|
|
|
the symlink target might contain new symlinks). */
|
|
|
|
if (resolveSymlinks && isLink(s)) {
|
2007-11-29 18:18:24 +02:00
|
|
|
if (++followCount >= maxFollow)
|
2017-07-30 14:27:57 +03:00
|
|
|
throw Error(format("infinite symlink recursion in path '%1%'") % path);
|
2006-01-08 19:16:03 +02:00
|
|
|
temp = absPath(readLink(s), dirOf(s))
|
|
|
|
+ string(i, end);
|
|
|
|
i = temp.begin(); /* restart */
|
|
|
|
end = temp.end();
|
|
|
|
s = "";
|
|
|
|
}
|
2003-07-08 22:58:41 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return s.empty() ? "/" : s;
|
2003-06-16 16:33:38 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-07 17:37:41 +03:00
|
|
|
Path dirOf(const Path & path)
|
2003-06-16 16:33:38 +03:00
|
|
|
{
|
2006-05-11 05:19:43 +03:00
|
|
|
Path::size_type pos = path.rfind('/');
|
2003-07-04 15:18:06 +03:00
|
|
|
if (pos == string::npos)
|
2017-07-30 14:27:57 +03:00
|
|
|
throw Error(format("invalid file name '%1%'") % path);
|
2006-01-09 16:52:46 +02:00
|
|
|
return pos == 0 ? "/" : Path(path, 0, pos);
|
2003-05-26 16:45:00 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-07 17:37:41 +03:00
|
|
|
string baseNameOf(const Path & path)
|
2003-05-26 16:45:00 +03:00
|
|
|
{
|
2015-07-13 15:25:13 +03:00
|
|
|
if (path.empty())
|
2016-01-27 18:18:31 +02:00
|
|
|
return "";
|
2015-07-13 15:25:13 +03:00
|
|
|
|
|
|
|
Path::size_type last = path.length() - 1;
|
|
|
|
if (path[last] == '/' && last > 0)
|
|
|
|
last -= 1;
|
|
|
|
|
|
|
|
Path::size_type pos = path.rfind('/', last);
|
2003-07-04 15:18:06 +03:00
|
|
|
if (pos == string::npos)
|
2015-07-13 15:25:13 +03:00
|
|
|
pos = 0;
|
|
|
|
else
|
|
|
|
pos += 1;
|
2016-01-27 18:18:31 +02:00
|
|
|
|
2015-07-13 15:25:13 +03:00
|
|
|
return string(path, pos, last - pos + 1);
|
2003-05-26 16:45:00 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-12 15:01:25 +03:00
|
|
|
bool isInDir(const Path & path, const Path & dir)
|
|
|
|
{
|
2015-10-22 00:40:35 +03:00
|
|
|
return path[0] == '/'
|
|
|
|
&& string(path, 0, dir.size()) == dir
|
|
|
|
&& path.size() >= dir.size() + 2
|
|
|
|
&& path[dir.size()] == '/';
|
2013-07-12 15:01:25 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-13 15:32:58 +02:00
|
|
|
struct stat lstat(const Path & path)
|
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
if (lstat(path.c_str(), &st))
|
2017-07-30 14:27:57 +03:00
|
|
|
throw SysError(format("getting status of '%1%'") % path);
|
2010-12-13 15:32:58 +02:00
|
|
|
return st;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-07 17:37:41 +03:00
|
|
|
bool pathExists(const Path & path)
|
2003-07-08 16:22:08 +03:00
|
|
|
{
|
|
|
|
int res;
|
|
|
|
struct stat st;
|
2004-02-06 12:59:06 +02:00
|
|
|
res = lstat(path.c_str(), &st);
|
2003-07-08 16:22:08 +03:00
|
|
|
if (!res) return true;
|
2004-08-04 12:25:21 +03:00
|
|
|
if (errno != ENOENT && errno != ENOTDIR)
|
2003-07-08 16:22:08 +03:00
|
|
|
throw SysError(format("getting status of %1%") % path);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-01-05 18:26:43 +02:00
|
|
|
Path readLink(const Path & path)
|
|
|
|
{
|
2007-09-17 19:08:24 +03:00
|
|
|
checkInterrupt();
|
2017-10-27 19:15:31 +03:00
|
|
|
for (ssize_t bufSize = PATH_MAX/4; true; bufSize += bufSize/2) {
|
|
|
|
char buf[bufSize];
|
|
|
|
ssize_t rlSize = readlink(path.c_str(), buf, bufSize);
|
|
|
|
if (rlSize == -1)
|
|
|
|
if (errno == EINVAL)
|
|
|
|
throw Error(format("'%1%' is not a symlink") % path);
|
|
|
|
else
|
|
|
|
throw SysError(format("reading symbolic link '%1%'") % path);
|
|
|
|
else if (rlSize < bufSize)
|
|
|
|
return string(buf, rlSize);
|
|
|
|
}
|
2004-01-05 18:26:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-02-01 15:48:46 +02:00
|
|
|
bool isLink(const Path & path)
|
|
|
|
{
|
2010-12-13 15:32:58 +02:00
|
|
|
struct stat st = lstat(path);
|
2005-02-01 15:48:46 +02:00
|
|
|
return S_ISLNK(st.st_mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-08-01 17:37:47 +03:00
|
|
|
DirEntries readDirectory(const Path & path)
|
2003-11-19 19:27:16 +02:00
|
|
|
{
|
2014-08-01 17:37:47 +03:00
|
|
|
DirEntries entries;
|
|
|
|
entries.reserve(64);
|
2003-11-19 19:27:16 +02:00
|
|
|
|
2017-01-16 23:39:27 +02:00
|
|
|
AutoCloseDir dir(opendir(path.c_str()));
|
2017-07-30 14:27:57 +03:00
|
|
|
if (!dir) throw SysError(format("opening directory '%1%'") % path);
|
2003-11-19 19:27:16 +02:00
|
|
|
|
|
|
|
struct dirent * dirent;
|
2017-01-16 23:39:27 +02:00
|
|
|
while (errno = 0, dirent = readdir(dir.get())) { /* sic */
|
2004-01-15 22:23:55 +02:00
|
|
|
checkInterrupt();
|
2003-11-19 19:27:16 +02:00
|
|
|
string name = dirent->d_name;
|
|
|
|
if (name == "." || name == "..") continue;
|
2016-01-05 15:05:11 +02:00
|
|
|
entries.emplace_back(name, dirent->d_ino,
|
|
|
|
#ifdef HAVE_STRUCT_DIRENT_D_TYPE
|
|
|
|
dirent->d_type
|
|
|
|
#else
|
|
|
|
DT_UNKNOWN
|
|
|
|
#endif
|
|
|
|
);
|
2003-11-19 19:27:16 +02:00
|
|
|
}
|
2017-07-30 14:27:57 +03:00
|
|
|
if (errno) throw SysError(format("reading directory '%1%'") % path);
|
2003-11-19 19:27:16 +02:00
|
|
|
|
2014-08-01 17:37:47 +03:00
|
|
|
return entries;
|
2003-11-19 19:27:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-10-03 23:37:51 +03:00
|
|
|
unsigned char getFileType(const Path & path)
|
|
|
|
{
|
|
|
|
struct stat st = lstat(path);
|
|
|
|
if (S_ISDIR(st.st_mode)) return DT_DIR;
|
|
|
|
if (S_ISLNK(st.st_mode)) return DT_LNK;
|
|
|
|
if (S_ISREG(st.st_mode)) return DT_REG;
|
|
|
|
return DT_UNKNOWN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-02-02 00:07:48 +02:00
|
|
|
string readFile(int fd)
|
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
if (fstat(fd, &st) == -1)
|
|
|
|
throw SysError("statting file");
|
2013-01-03 14:00:46 +02:00
|
|
|
|
2017-01-16 23:24:29 +02:00
|
|
|
auto buf = std::make_unique<unsigned char[]>(st.st_size);
|
|
|
|
readFull(fd, buf.get(), st.st_size);
|
2005-02-02 00:07:48 +02:00
|
|
|
|
2017-01-16 23:24:29 +02:00
|
|
|
return string((char *) buf.get(), st.st_size);
|
2005-02-02 00:07:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-20 22:27:30 +03:00
|
|
|
string readFile(const Path & path, bool drain)
|
2005-02-02 00:07:48 +02:00
|
|
|
{
|
2016-06-09 17:15:58 +03:00
|
|
|
AutoCloseFD fd = open(path.c_str(), O_RDONLY | O_CLOEXEC);
|
2016-07-11 22:44:44 +03:00
|
|
|
if (!fd)
|
2017-07-30 14:27:57 +03:00
|
|
|
throw SysError(format("opening file '%1%'") % path);
|
2016-07-11 22:44:44 +03:00
|
|
|
return drain ? drainFD(fd.get()) : readFile(fd.get());
|
2005-02-02 00:07:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-02-16 16:42:49 +02:00
|
|
|
void writeFile(const Path & path, const string & s, mode_t mode)
|
2005-02-09 11:50:29 +02:00
|
|
|
{
|
2017-02-16 16:42:49 +02:00
|
|
|
AutoCloseFD fd = open(path.c_str(), O_WRONLY | O_TRUNC | O_CREAT | O_CLOEXEC, mode);
|
2016-07-11 22:44:44 +03:00
|
|
|
if (!fd)
|
2017-07-30 14:27:57 +03:00
|
|
|
throw SysError(format("opening file '%1%'") % path);
|
2016-07-11 22:44:44 +03:00
|
|
|
writeFull(fd.get(), s);
|
2005-02-09 11:50:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-03-28 21:29:55 +02:00
|
|
|
string readLine(int fd)
|
|
|
|
{
|
|
|
|
string s;
|
|
|
|
while (1) {
|
|
|
|
checkInterrupt();
|
|
|
|
char ch;
|
2017-08-09 17:22:05 +03:00
|
|
|
// FIXME: inefficient
|
2009-03-28 21:29:55 +02:00
|
|
|
ssize_t rd = read(fd, &ch, 1);
|
|
|
|
if (rd == -1) {
|
|
|
|
if (errno != EINTR)
|
|
|
|
throw SysError("reading a line");
|
|
|
|
} else if (rd == 0)
|
2012-08-01 18:19:24 +03:00
|
|
|
throw EndOfFile("unexpected EOF reading a line");
|
2009-03-28 21:29:55 +02:00
|
|
|
else {
|
|
|
|
if (ch == '\n') return s;
|
|
|
|
s += ch;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void writeLine(int fd, string s)
|
|
|
|
{
|
|
|
|
s += '\n';
|
2014-12-12 15:35:44 +02:00
|
|
|
writeFull(fd, s);
|
2009-03-28 21:29:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-02 05:34:46 +03:00
|
|
|
static void _deletePath(const Path & path, unsigned long long & bytesFreed)
|
2003-06-23 17:40:49 +03:00
|
|
|
{
|
2004-01-15 22:23:55 +02:00
|
|
|
checkInterrupt();
|
|
|
|
|
2016-02-24 18:44:12 +02:00
|
|
|
struct stat st;
|
|
|
|
if (lstat(path.c_str(), &st) == -1) {
|
|
|
|
if (errno == ENOENT) return;
|
2017-07-30 14:27:57 +03:00
|
|
|
throw SysError(format("getting status of '%1%'") % path);
|
2016-02-24 18:44:12 +02:00
|
|
|
}
|
2003-06-23 17:40:49 +03:00
|
|
|
|
2012-08-02 05:34:46 +03:00
|
|
|
if (!S_ISDIR(st.st_mode) && st.st_nlink == 1)
|
|
|
|
bytesFreed += st.st_blocks * 512;
|
2005-12-15 23:11:39 +02:00
|
|
|
|
2003-06-23 17:40:49 +03:00
|
|
|
if (S_ISDIR(st.st_mode)) {
|
2016-07-26 01:00:08 +03:00
|
|
|
/* Make the directory accessible. */
|
|
|
|
const auto PERM_MASK = S_IRUSR | S_IWUSR | S_IXUSR;
|
|
|
|
if ((st.st_mode & PERM_MASK) != PERM_MASK) {
|
|
|
|
if (chmod(path.c_str(), st.st_mode | PERM_MASK) == -1)
|
2017-07-30 14:27:57 +03:00
|
|
|
throw SysError(format("chmod '%1%'") % path);
|
2013-01-03 14:00:46 +02:00
|
|
|
}
|
2003-08-22 23:12:44 +03:00
|
|
|
|
2014-08-01 17:37:47 +03:00
|
|
|
for (auto & i : readDirectory(path))
|
|
|
|
_deletePath(path + "/" + i.name, bytesFreed);
|
2003-06-23 17:40:49 +03:00
|
|
|
}
|
|
|
|
|
2016-02-24 18:44:12 +02:00
|
|
|
if (remove(path.c_str()) == -1) {
|
|
|
|
if (errno == ENOENT) return;
|
2017-07-30 14:27:57 +03:00
|
|
|
throw SysError(format("cannot unlink '%1%'") % path);
|
2016-02-24 18:44:12 +02:00
|
|
|
}
|
2003-08-22 23:12:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-22 23:42:28 +02:00
|
|
|
void deletePath(const Path & path)
|
2005-12-15 23:11:39 +02:00
|
|
|
{
|
2012-08-02 05:34:46 +03:00
|
|
|
unsigned long long dummy;
|
|
|
|
deletePath(path, dummy);
|
2005-12-15 23:11:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-02 05:34:46 +03:00
|
|
|
void deletePath(const Path & path, unsigned long long & bytesFreed)
|
2004-03-22 23:42:28 +02:00
|
|
|
{
|
2017-07-30 14:27:57 +03:00
|
|
|
//Activity act(*logger, lvlDebug, format("recursively deleting path '%1%'") % path);
|
2005-12-15 23:11:39 +02:00
|
|
|
bytesFreed = 0;
|
2012-08-02 05:34:46 +03:00
|
|
|
_deletePath(path, bytesFreed);
|
2004-03-22 23:42:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-03-27 15:45:17 +02:00
|
|
|
static Path tempName(Path tmpRoot, const Path & prefix, bool includePid,
|
|
|
|
int & counter)
|
2003-10-02 14:55:38 +03:00
|
|
|
{
|
2008-03-27 15:45:17 +02:00
|
|
|
tmpRoot = canonPath(tmpRoot.empty() ? getEnv("TMPDIR", "/tmp") : tmpRoot, true);
|
|
|
|
if (includePid)
|
|
|
|
return (format("%1%/%2%-%3%-%4%") % tmpRoot % prefix % getpid() % counter++).str();
|
|
|
|
else
|
|
|
|
return (format("%1%/%2%-%3%") % tmpRoot % prefix % counter++).str();
|
2003-10-02 14:55:38 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-03-27 15:45:17 +02:00
|
|
|
Path createTempDir(const Path & tmpRoot, const Path & prefix,
|
2012-07-26 22:04:40 +03:00
|
|
|
bool includePid, bool useGlobalCounter, mode_t mode)
|
2003-10-02 14:55:38 +03:00
|
|
|
{
|
2008-03-27 15:45:17 +02:00
|
|
|
static int globalCounter = 0;
|
|
|
|
int localCounter = 0;
|
|
|
|
int & counter(useGlobalCounter ? globalCounter : localCounter);
|
2013-01-03 14:00:46 +02:00
|
|
|
|
2003-10-02 14:55:38 +03:00
|
|
|
while (1) {
|
2004-01-15 22:23:55 +02:00
|
|
|
checkInterrupt();
|
2013-01-03 14:00:46 +02:00
|
|
|
Path tmpDir = tempName(tmpRoot, prefix, includePid, counter);
|
|
|
|
if (mkdir(tmpDir.c_str(), mode) == 0) {
|
2016-06-02 19:17:30 +03:00
|
|
|
#if __FreeBSD__
|
2013-01-03 14:00:46 +02:00
|
|
|
/* Explicitly set the group of the directory. This is to
|
|
|
|
work around around problems caused by BSD's group
|
|
|
|
ownership semantics (directories inherit the group of
|
|
|
|
the parent). For instance, the group of /tmp on
|
|
|
|
FreeBSD is "wheel", so all directories created in /tmp
|
|
|
|
will be owned by "wheel"; but if the user is not in
|
|
|
|
"wheel", then "tar" will fail to unpack archives that
|
|
|
|
have the setgid bit set on directories. */
|
|
|
|
if (chown(tmpDir.c_str(), (uid_t) -1, getegid()) != 0)
|
2017-07-30 14:27:57 +03:00
|
|
|
throw SysError(format("setting group of directory '%1%'") % tmpDir);
|
2016-06-02 19:17:30 +03:00
|
|
|
#endif
|
2013-01-03 14:00:46 +02:00
|
|
|
return tmpDir;
|
|
|
|
}
|
|
|
|
if (errno != EEXIST)
|
2017-07-30 14:27:57 +03:00
|
|
|
throw SysError(format("creating directory '%1%'") % tmpDir);
|
2003-10-02 14:55:38 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-05-05 17:40:12 +03:00
|
|
|
static Lazy<Path> getHome2([]() {
|
|
|
|
Path homeDir = getEnv("HOME");
|
|
|
|
if (homeDir.empty()) {
|
|
|
|
char buf[16384];
|
|
|
|
struct passwd pwbuf;
|
|
|
|
struct passwd * pw;
|
|
|
|
if (getpwuid_r(getuid(), &pwbuf, buf, sizeof(buf), &pw) != 0
|
|
|
|
|| !pw || !pw->pw_dir || !pw->pw_dir[0])
|
|
|
|
throw Error("cannot determine user's home directory");
|
|
|
|
homeDir = pw->pw_dir;
|
|
|
|
}
|
|
|
|
return homeDir;
|
|
|
|
});
|
|
|
|
|
|
|
|
Path getHome() { return getHome2(); }
|
|
|
|
|
|
|
|
|
2016-04-20 15:12:38 +03:00
|
|
|
Path getCacheDir()
|
|
|
|
{
|
|
|
|
Path cacheDir = getEnv("XDG_CACHE_HOME");
|
2017-05-05 17:40:12 +03:00
|
|
|
if (cacheDir.empty())
|
|
|
|
cacheDir = getHome() + "/.cache";
|
2016-04-20 15:12:38 +03:00
|
|
|
return cacheDir;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-04-20 15:58:16 +03:00
|
|
|
Path getConfigDir()
|
|
|
|
{
|
|
|
|
Path configDir = getEnv("XDG_CONFIG_HOME");
|
2017-05-05 17:40:12 +03:00
|
|
|
if (configDir.empty())
|
|
|
|
configDir = getHome() + "/.config";
|
2017-04-20 15:58:16 +03:00
|
|
|
return configDir;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-04-25 19:56:29 +03:00
|
|
|
Path getDataDir()
|
|
|
|
{
|
|
|
|
Path dataDir = getEnv("XDG_DATA_HOME");
|
2017-05-05 17:40:12 +03:00
|
|
|
if (dataDir.empty())
|
|
|
|
dataDir = getHome() + "/.local/share";
|
2017-04-25 19:56:29 +03:00
|
|
|
return dataDir;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-10-27 03:46:59 +03:00
|
|
|
Paths createDirs(const Path & path)
|
2005-03-24 19:46:38 +02:00
|
|
|
{
|
2008-06-09 16:52:45 +03:00
|
|
|
Paths created;
|
|
|
|
if (path == "/") return created;
|
2010-12-13 15:32:58 +02:00
|
|
|
|
|
|
|
struct stat st;
|
|
|
|
if (lstat(path.c_str(), &st) == -1) {
|
2008-06-09 16:52:45 +03:00
|
|
|
created = createDirs(dirOf(path));
|
2010-12-13 15:32:58 +02:00
|
|
|
if (mkdir(path.c_str(), 0777) == -1 && errno != EEXIST)
|
2017-07-30 14:27:57 +03:00
|
|
|
throw SysError(format("creating directory '%1%'") % path);
|
2010-12-13 15:32:58 +02:00
|
|
|
st = lstat(path);
|
2007-10-27 03:46:59 +03:00
|
|
|
created.push_back(path);
|
|
|
|
}
|
2010-12-13 15:32:58 +02:00
|
|
|
|
2014-10-03 17:53:28 +03:00
|
|
|
if (S_ISLNK(st.st_mode) && stat(path.c_str(), &st) == -1)
|
2017-07-30 14:27:57 +03:00
|
|
|
throw SysError(format("statting symlink '%1%'") % path);
|
2014-10-03 17:53:28 +03:00
|
|
|
|
2017-07-30 14:27:57 +03:00
|
|
|
if (!S_ISDIR(st.st_mode)) throw Error(format("'%1%' is not a directory") % path);
|
2013-01-03 14:00:46 +02:00
|
|
|
|
2007-10-27 03:46:59 +03:00
|
|
|
return created;
|
2005-03-24 19:46:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-02-28 00:17:53 +02:00
|
|
|
void createSymlink(const Path & target, const Path & link)
|
|
|
|
{
|
|
|
|
if (symlink(target.c_str(), link.c_str()))
|
2017-07-30 14:27:57 +03:00
|
|
|
throw SysError(format("creating symlink from '%1%' to '%2%'") % link % target);
|
2014-02-28 00:17:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-04-09 12:42:04 +03:00
|
|
|
void replaceSymlink(const Path & target, const Path & link)
|
|
|
|
{
|
2017-07-12 00:20:01 +03:00
|
|
|
for (unsigned int n = 0; true; n++) {
|
|
|
|
Path tmp = canonPath(fmt("%s/.%d_%s", dirOf(link), n, baseNameOf(link)));
|
|
|
|
|
|
|
|
try {
|
|
|
|
createSymlink(target, tmp);
|
|
|
|
} catch (SysError & e) {
|
|
|
|
if (e.errNo == EEXIST) continue;
|
|
|
|
throw;
|
|
|
|
}
|
2015-04-09 12:42:04 +03:00
|
|
|
|
2017-07-12 00:20:01 +03:00
|
|
|
if (rename(tmp.c_str(), link.c_str()) != 0)
|
2017-07-30 14:27:57 +03:00
|
|
|
throw SysError(format("renaming '%1%' to '%2%'") % tmp % link);
|
2015-04-09 12:42:04 +03:00
|
|
|
|
2017-07-12 00:20:01 +03:00
|
|
|
break;
|
|
|
|
}
|
2015-04-09 12:42:04 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-21 00:11:43 +03:00
|
|
|
void readFull(int fd, unsigned char * buf, size_t count)
|
|
|
|
{
|
|
|
|
while (count) {
|
2004-01-15 22:23:55 +02:00
|
|
|
checkInterrupt();
|
2003-07-21 00:11:43 +03:00
|
|
|
ssize_t res = read(fd, (char *) buf, count);
|
2004-05-11 16:48:25 +03:00
|
|
|
if (res == -1) {
|
|
|
|
if (errno == EINTR) continue;
|
|
|
|
throw SysError("reading from file");
|
|
|
|
}
|
2006-12-04 19:17:13 +02:00
|
|
|
if (res == 0) throw EndOfFile("unexpected end-of-file");
|
2003-07-21 00:11:43 +03:00
|
|
|
count -= res;
|
|
|
|
buf += res;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-09-16 19:52:42 +03:00
|
|
|
void writeFull(int fd, const unsigned char * buf, size_t count, bool allowInterrupts)
|
2003-07-21 00:11:43 +03:00
|
|
|
{
|
|
|
|
while (count) {
|
2017-04-06 18:18:56 +03:00
|
|
|
if (allowInterrupts) checkInterrupt();
|
2003-07-21 00:11:43 +03:00
|
|
|
ssize_t res = write(fd, (char *) buf, count);
|
2016-09-16 19:52:42 +03:00
|
|
|
if (res == -1 && errno != EINTR)
|
2004-05-11 16:48:25 +03:00
|
|
|
throw SysError("writing to file");
|
2016-09-16 19:52:42 +03:00
|
|
|
if (res > 0) {
|
|
|
|
count -= res;
|
|
|
|
buf += res;
|
2004-05-11 16:48:25 +03:00
|
|
|
}
|
2003-07-21 00:11:43 +03:00
|
|
|
}
|
|
|
|
}
|
2003-10-22 13:48:22 +03:00
|
|
|
|
|
|
|
|
2016-09-16 19:52:42 +03:00
|
|
|
void writeFull(int fd, const string & s, bool allowInterrupts)
|
2014-12-12 15:35:44 +02:00
|
|
|
{
|
2016-09-16 19:52:42 +03:00
|
|
|
writeFull(fd, (const unsigned char *) s.data(), s.size(), allowInterrupts);
|
2014-12-12 15:35:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-07-20 15:17:25 +03:00
|
|
|
string drainFD(int fd)
|
|
|
|
{
|
|
|
|
string result;
|
|
|
|
unsigned char buffer[4096];
|
|
|
|
while (1) {
|
2007-08-12 03:29:28 +03:00
|
|
|
checkInterrupt();
|
2006-07-20 15:17:25 +03:00
|
|
|
ssize_t rd = read(fd, buffer, sizeof buffer);
|
|
|
|
if (rd == -1) {
|
|
|
|
if (errno != EINTR)
|
|
|
|
throw SysError("reading from file");
|
|
|
|
}
|
|
|
|
else if (rd == 0) break;
|
|
|
|
else result.append((char *) buffer, rd);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-06-22 12:51:44 +03:00
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
2015-11-16 12:53:10 +02:00
|
|
|
AutoDelete::AutoDelete() : del{false} {}
|
|
|
|
|
2007-10-27 03:46:59 +03:00
|
|
|
AutoDelete::AutoDelete(const string & p, bool recursive) : path(p)
|
2003-10-22 13:48:22 +03:00
|
|
|
{
|
|
|
|
del = true;
|
2007-10-27 03:46:59 +03:00
|
|
|
this->recursive = recursive;
|
2003-10-22 13:48:22 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
AutoDelete::~AutoDelete()
|
|
|
|
{
|
2007-10-27 03:46:59 +03:00
|
|
|
try {
|
2008-05-21 14:17:31 +03:00
|
|
|
if (del) {
|
2007-10-27 03:46:59 +03:00
|
|
|
if (recursive)
|
|
|
|
deletePath(path);
|
|
|
|
else {
|
|
|
|
if (remove(path.c_str()) == -1)
|
2017-07-30 14:27:57 +03:00
|
|
|
throw SysError(format("cannot unlink '%1%'") % path);
|
2007-10-27 03:46:59 +03:00
|
|
|
}
|
2008-05-21 14:17:31 +03:00
|
|
|
}
|
2007-10-27 03:46:59 +03:00
|
|
|
} catch (...) {
|
|
|
|
ignoreException();
|
|
|
|
}
|
2003-10-22 13:48:22 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void AutoDelete::cancel()
|
|
|
|
{
|
|
|
|
del = false;
|
|
|
|
}
|
|
|
|
|
2015-11-16 12:55:55 +02:00
|
|
|
void AutoDelete::reset(const Path & p, bool recursive) {
|
2015-11-16 12:54:34 +02:00
|
|
|
path = p;
|
2015-11-16 12:53:10 +02:00
|
|
|
this->recursive = recursive;
|
|
|
|
del = true;
|
|
|
|
}
|
|
|
|
|
2003-10-22 13:48:22 +03:00
|
|
|
|
2004-06-22 12:51:44 +03:00
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
2016-07-11 22:44:44 +03:00
|
|
|
AutoCloseFD::AutoCloseFD() : fd{-1} {}
|
|
|
|
|
|
|
|
|
|
|
|
AutoCloseFD::AutoCloseFD(int fd) : fd{fd} {}
|
2003-10-22 13:48:22 +03:00
|
|
|
|
2004-06-22 12:51:44 +03:00
|
|
|
|
2016-07-11 22:44:44 +03:00
|
|
|
AutoCloseFD::AutoCloseFD(AutoCloseFD&& that) : fd{that.fd}
|
2003-10-22 13:48:22 +03:00
|
|
|
{
|
2016-07-11 22:44:44 +03:00
|
|
|
that.fd = -1;
|
2003-10-22 13:48:22 +03:00
|
|
|
}
|
|
|
|
|
2004-06-22 12:51:44 +03:00
|
|
|
|
2016-07-11 22:44:44 +03:00
|
|
|
AutoCloseFD& AutoCloseFD::operator =(AutoCloseFD&& that)
|
2005-01-31 12:27:25 +02:00
|
|
|
{
|
2016-07-11 22:44:44 +03:00
|
|
|
close();
|
|
|
|
fd = that.fd;
|
|
|
|
that.fd = -1;
|
|
|
|
return *this;
|
2005-01-31 12:27:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-22 13:48:22 +03:00
|
|
|
AutoCloseFD::~AutoCloseFD()
|
|
|
|
{
|
2004-06-15 16:49:42 +03:00
|
|
|
try {
|
|
|
|
close();
|
2007-05-01 18:16:17 +03:00
|
|
|
} catch (...) {
|
|
|
|
ignoreException();
|
2004-06-15 16:49:42 +03:00
|
|
|
}
|
2003-10-22 13:48:22 +03:00
|
|
|
}
|
|
|
|
|
2004-06-22 12:51:44 +03:00
|
|
|
|
2016-07-11 22:44:44 +03:00
|
|
|
int AutoCloseFD::get() const
|
2003-10-22 13:48:22 +03:00
|
|
|
{
|
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
|
2004-06-22 12:51:44 +03:00
|
|
|
|
2004-06-15 16:49:42 +03:00
|
|
|
void AutoCloseFD::close()
|
|
|
|
{
|
|
|
|
if (fd != -1) {
|
|
|
|
if (::close(fd) == -1)
|
|
|
|
/* This should never happen. */
|
2012-11-15 14:55:02 +02:00
|
|
|
throw SysError(format("closing file descriptor %1%") % fd);
|
2004-06-15 16:49:42 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-06-22 12:51:44 +03:00
|
|
|
|
2016-07-11 22:44:44 +03:00
|
|
|
AutoCloseFD::operator bool() const
|
2004-06-15 16:49:42 +03:00
|
|
|
{
|
|
|
|
return fd != -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-07-11 22:44:44 +03:00
|
|
|
int AutoCloseFD::release()
|
2005-01-27 14:19:25 +02:00
|
|
|
{
|
|
|
|
int oldFD = fd;
|
|
|
|
fd = -1;
|
|
|
|
return oldFD;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-06-15 16:49:42 +03:00
|
|
|
void Pipe::create()
|
|
|
|
{
|
|
|
|
int fds[2];
|
2016-06-09 17:15:58 +03:00
|
|
|
#if HAVE_PIPE2
|
|
|
|
if (pipe2(fds, O_CLOEXEC) != 0) throw SysError("creating pipe");
|
|
|
|
#else
|
2004-06-15 16:49:42 +03:00
|
|
|
if (pipe(fds) != 0) throw SysError("creating pipe");
|
2016-06-09 17:15:58 +03:00
|
|
|
closeOnExec(fds[0]);
|
|
|
|
closeOnExec(fds[1]);
|
|
|
|
#endif
|
2004-06-15 16:49:42 +03:00
|
|
|
readSide = fds[0];
|
|
|
|
writeSide = fds[1];
|
|
|
|
}
|
|
|
|
|
2003-10-22 13:48:22 +03:00
|
|
|
|
2004-06-22 12:51:44 +03:00
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
|
|
Pid::Pid()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-07-10 17:50:51 +03:00
|
|
|
Pid::Pid(pid_t pid)
|
2017-01-19 17:58:39 +02:00
|
|
|
: pid(pid)
|
2014-07-10 17:50:51 +03:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-06-22 12:51:44 +03:00
|
|
|
Pid::~Pid()
|
|
|
|
{
|
2017-01-19 17:58:39 +02:00
|
|
|
if (pid != -1) kill();
|
2004-06-22 12:51:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Pid::operator =(pid_t pid)
|
|
|
|
{
|
2017-01-19 17:58:39 +02:00
|
|
|
if (this->pid != -1 && this->pid != pid) kill();
|
2004-06-22 12:51:44 +03:00
|
|
|
this->pid = pid;
|
2007-03-19 14:48:45 +02:00
|
|
|
killSignal = SIGKILL; // reset signal to default
|
2004-06-22 12:51:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Pid::operator pid_t()
|
|
|
|
{
|
|
|
|
return pid;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-16 11:52:28 +02:00
|
|
|
int Pid::kill()
|
2004-06-22 12:51:44 +03:00
|
|
|
{
|
2017-01-19 17:58:39 +02:00
|
|
|
assert(pid != -1);
|
2012-11-09 19:00:33 +02:00
|
|
|
|
2017-03-16 11:52:28 +02:00
|
|
|
debug(format("killing process %1%") % pid);
|
2004-06-22 12:51:44 +03:00
|
|
|
|
2007-03-19 14:48:45 +02:00
|
|
|
/* Send the requested signal to the child. If it has its own
|
|
|
|
process group, send the signal to every process in the child
|
|
|
|
process group (which hopefully includes *all* its children). */
|
2017-06-12 19:34:48 +03:00
|
|
|
if (::kill(separatePG ? -pid : pid, killSignal) != 0) {
|
|
|
|
/* On BSDs, killing a process group will return EPERM if all
|
|
|
|
processes in the group are zombies (or something like
|
|
|
|
that). So try to detect and ignore that situation. */
|
|
|
|
#if __FreeBSD__ || __APPLE__
|
|
|
|
if (errno != EPERM || ::kill(pid, 0) != 0)
|
|
|
|
#endif
|
|
|
|
printError((SysError("killing process %d", pid).msg()));
|
|
|
|
}
|
2004-06-25 18:36:09 +03:00
|
|
|
|
2017-01-19 17:58:39 +02:00
|
|
|
return wait();
|
2004-06-22 12:51:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-01-19 17:58:39 +02:00
|
|
|
int Pid::wait()
|
2004-06-22 12:51:44 +03:00
|
|
|
{
|
2013-06-20 12:55:15 +03:00
|
|
|
assert(pid != -1);
|
2004-06-22 12:51:44 +03:00
|
|
|
while (1) {
|
|
|
|
int status;
|
2017-01-19 17:58:39 +02:00
|
|
|
int res = waitpid(pid, &status, 0);
|
2004-06-22 12:51:44 +03:00
|
|
|
if (res == pid) {
|
|
|
|
pid = -1;
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
if (errno != EINTR)
|
|
|
|
throw SysError("cannot get child exit status");
|
2006-12-04 19:17:13 +02:00
|
|
|
checkInterrupt();
|
2004-06-22 12:51:44 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Pid::setSeparatePG(bool separatePG)
|
|
|
|
{
|
|
|
|
this->separatePG = separatePG;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-03-19 14:48:45 +02:00
|
|
|
void Pid::setKillSignal(int signal)
|
|
|
|
{
|
|
|
|
this->killSignal = signal;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-10-12 16:49:37 +03:00
|
|
|
pid_t Pid::release()
|
|
|
|
{
|
|
|
|
pid_t p = pid;
|
|
|
|
pid = -1;
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-12-07 02:16:07 +02:00
|
|
|
void killUser(uid_t uid)
|
|
|
|
{
|
2017-07-30 14:27:57 +03:00
|
|
|
debug(format("killing all processes running under uid '%1%'") % uid);
|
2006-12-07 02:16:07 +02:00
|
|
|
|
|
|
|
assert(uid != 0); /* just to be safe... */
|
|
|
|
|
|
|
|
/* The system call kill(-1, sig) sends the signal `sig' to all
|
|
|
|
users to which the current process can send signals. So we
|
|
|
|
fork a process, switch to uid, and send a mass kill. */
|
|
|
|
|
2014-12-10 19:01:01 +02:00
|
|
|
ProcessOptions options;
|
|
|
|
options.allowVfork = false;
|
|
|
|
|
2014-07-10 17:50:51 +03:00
|
|
|
Pid pid = startProcess([&]() {
|
2006-12-07 02:16:07 +02:00
|
|
|
|
2014-07-10 17:50:51 +03:00
|
|
|
if (setuid(uid) == -1)
|
|
|
|
throw SysError("setting uid");
|
2006-12-07 02:16:07 +02:00
|
|
|
|
2014-07-10 17:50:51 +03:00
|
|
|
while (true) {
|
2013-03-18 17:13:53 +02:00
|
|
|
#ifdef __APPLE__
|
2014-07-10 17:50:51 +03:00
|
|
|
/* OSX's kill syscall takes a third parameter that, among
|
|
|
|
other things, determines if kill(-1, signo) affects the
|
|
|
|
calling process. In the OSX libc, it's set to true,
|
|
|
|
which means "follow POSIX", which we don't want here
|
2013-03-18 17:13:53 +02:00
|
|
|
*/
|
2014-07-10 17:50:51 +03:00
|
|
|
if (syscall(SYS_kill, -1, SIGKILL, false) == 0) break;
|
2013-03-18 17:13:53 +02:00
|
|
|
#else
|
2014-07-10 17:50:51 +03:00
|
|
|
if (kill(-1, SIGKILL) == 0) break;
|
2013-03-18 17:13:53 +02:00
|
|
|
#endif
|
2014-07-10 17:50:51 +03:00
|
|
|
if (errno == ESRCH) break; /* no more processes */
|
|
|
|
if (errno != EINTR)
|
2017-07-30 14:27:57 +03:00
|
|
|
throw SysError(format("cannot kill processes for uid '%1%'") % uid);
|
2006-12-07 02:16:07 +02:00
|
|
|
}
|
2014-07-10 17:50:51 +03:00
|
|
|
|
2012-11-09 17:42:10 +02:00
|
|
|
_exit(0);
|
2014-12-12 14:41:00 +02:00
|
|
|
}, options);
|
2013-01-03 14:00:46 +02:00
|
|
|
|
2017-01-19 17:58:39 +02:00
|
|
|
int status = pid.wait();
|
2010-03-19 13:36:34 +02:00
|
|
|
if (status != 0)
|
2017-07-30 14:27:57 +03:00
|
|
|
throw Error(format("cannot kill processes for uid '%1%': %2%") % uid % statusToString(status));
|
2006-12-07 02:16:07 +02:00
|
|
|
|
|
|
|
/* !!! We should really do some check to make sure that there are
|
|
|
|
no processes left running under `uid', but there is no portable
|
|
|
|
way to do so (I think). The most reliable way may be `ps -eo
|
|
|
|
uid | grep -q $uid'. */
|
|
|
|
}
|
|
|
|
|
2004-06-22 12:51:44 +03:00
|
|
|
|
2006-07-20 15:17:25 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
2014-12-10 17:35:42 +02:00
|
|
|
/* Wrapper around vfork to prevent the child process from clobbering
|
|
|
|
the caller's stack frame in the parent. */
|
|
|
|
static pid_t doFork(bool allowVfork, std::function<void()> fun) __attribute__((noinline));
|
|
|
|
static pid_t doFork(bool allowVfork, std::function<void()> fun)
|
2014-07-10 17:50:51 +03:00
|
|
|
{
|
2014-12-10 17:35:42 +02:00
|
|
|
#ifdef __linux__
|
|
|
|
pid_t pid = allowVfork ? vfork() : fork();
|
|
|
|
#else
|
2014-07-10 17:50:51 +03:00
|
|
|
pid_t pid = fork();
|
2014-12-10 17:35:42 +02:00
|
|
|
#endif
|
|
|
|
if (pid != 0) return pid;
|
|
|
|
fun();
|
|
|
|
abort();
|
|
|
|
}
|
2014-07-10 17:50:51 +03:00
|
|
|
|
2014-12-10 17:35:42 +02:00
|
|
|
|
|
|
|
pid_t startProcess(std::function<void()> fun, const ProcessOptions & options)
|
|
|
|
{
|
|
|
|
auto wrapper = [&]() {
|
2016-04-25 16:26:07 +03:00
|
|
|
if (!options.allowVfork)
|
|
|
|
logger = makeDefaultLogger();
|
2014-07-10 17:50:51 +03:00
|
|
|
try {
|
2014-08-21 16:31:43 +03:00
|
|
|
#if __linux__
|
2014-12-10 17:35:42 +02:00
|
|
|
if (options.dieWithParent && prctl(PR_SET_PDEATHSIG, SIGKILL) == -1)
|
2014-08-21 16:31:43 +03:00
|
|
|
throw SysError("setting death signal");
|
|
|
|
#endif
|
2014-07-10 17:50:51 +03:00
|
|
|
restoreAffinity();
|
|
|
|
fun();
|
|
|
|
} catch (std::exception & e) {
|
2014-07-23 20:11:26 +03:00
|
|
|
try {
|
2014-12-10 17:35:42 +02:00
|
|
|
std::cerr << options.errorPrefix << e.what() << "\n";
|
2014-07-23 20:11:26 +03:00
|
|
|
} catch (...) { }
|
|
|
|
} catch (...) { }
|
2014-12-10 17:35:42 +02:00
|
|
|
if (options.runExitHandlers)
|
2014-11-19 18:09:27 +02:00
|
|
|
exit(1);
|
|
|
|
else
|
|
|
|
_exit(1);
|
2014-12-10 17:35:42 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
pid_t pid = doFork(options.allowVfork, wrapper);
|
|
|
|
if (pid == -1) throw SysError("unable to fork");
|
2014-07-10 17:50:51 +03:00
|
|
|
|
|
|
|
return pid;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-06-09 11:50:55 +03:00
|
|
|
std::vector<char *> stringsToCharPtrs(const Strings & ss)
|
2014-12-12 16:01:16 +02:00
|
|
|
{
|
2015-06-09 11:50:55 +03:00
|
|
|
std::vector<char *> res;
|
|
|
|
for (auto & s : ss) res.push_back((char *) s.c_str());
|
2014-12-12 16:01:16 +02:00
|
|
|
res.push_back(0);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-02-04 17:43:32 +02:00
|
|
|
string runProgram(Path program, bool searchPath, const Strings & args,
|
2017-03-15 15:40:47 +02:00
|
|
|
const std::experimental::optional<std::string> & input)
|
2017-11-01 19:43:11 +02:00
|
|
|
{
|
|
|
|
RunOptions opts(program, args);
|
|
|
|
opts.searchPath = searchPath;
|
|
|
|
opts.input = input;
|
|
|
|
|
|
|
|
auto res = runProgram(opts);
|
|
|
|
|
|
|
|
if (!statusOk(res.first))
|
|
|
|
throw ExecError(res.first, fmt("program '%1%' %2%", program, statusToString(res.first)));
|
|
|
|
|
|
|
|
return res.second;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::pair<int, std::string> runProgram(const RunOptions & options)
|
2006-07-20 15:17:25 +03:00
|
|
|
{
|
2007-08-12 03:29:28 +03:00
|
|
|
checkInterrupt();
|
2012-11-09 17:58:51 +02:00
|
|
|
|
2006-07-20 15:17:25 +03:00
|
|
|
/* Create a pipe. */
|
2015-05-13 10:37:56 +03:00
|
|
|
Pipe out, in;
|
|
|
|
out.create();
|
2017-11-01 19:43:11 +02:00
|
|
|
if (options.input) in.create();
|
2006-07-20 15:17:25 +03:00
|
|
|
|
|
|
|
/* Fork. */
|
2014-07-10 17:50:51 +03:00
|
|
|
Pid pid = startProcess([&]() {
|
2016-07-11 22:44:44 +03:00
|
|
|
if (dup2(out.writeSide.get(), STDOUT_FILENO) == -1)
|
2014-07-10 17:50:51 +03:00
|
|
|
throw SysError("dupping stdout");
|
2017-11-01 19:43:11 +02:00
|
|
|
if (options.input && dup2(in.readSide.get(), STDIN_FILENO) == -1)
|
2017-03-15 15:40:47 +02:00
|
|
|
throw SysError("dupping stdin");
|
2006-07-20 15:17:25 +03:00
|
|
|
|
2017-11-01 19:43:11 +02:00
|
|
|
Strings args_(options.args);
|
|
|
|
args_.push_front(options.program);
|
2014-12-12 16:01:16 +02:00
|
|
|
|
2017-02-01 14:00:21 +02:00
|
|
|
restoreSignals();
|
|
|
|
|
2017-11-01 19:43:11 +02:00
|
|
|
if (options.searchPath)
|
|
|
|
execvp(options.program.c_str(), stringsToCharPtrs(args_).data());
|
2014-07-10 17:50:51 +03:00
|
|
|
else
|
2017-11-01 19:43:11 +02:00
|
|
|
execv(options.program.c_str(), stringsToCharPtrs(args_).data());
|
2006-07-20 15:17:25 +03:00
|
|
|
|
2017-11-01 19:43:11 +02:00
|
|
|
throw SysError("executing '%1%'", options.program);
|
2014-07-10 17:50:51 +03:00
|
|
|
});
|
2006-07-20 15:17:25 +03:00
|
|
|
|
2016-07-11 22:44:44 +03:00
|
|
|
out.writeSide = -1;
|
2015-02-04 17:43:32 +02:00
|
|
|
|
2017-03-13 15:56:33 +02:00
|
|
|
std::thread writerThread;
|
|
|
|
|
2017-03-15 15:40:47 +02:00
|
|
|
std::promise<void> promise;
|
|
|
|
|
|
|
|
Finally doJoin([&]() {
|
|
|
|
if (writerThread.joinable())
|
|
|
|
writerThread.join();
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2017-11-01 19:43:11 +02:00
|
|
|
if (options.input) {
|
2016-07-11 22:44:44 +03:00
|
|
|
in.readSide = -1;
|
2017-03-13 15:56:33 +02:00
|
|
|
writerThread = std::thread([&]() {
|
2017-03-15 15:40:47 +02:00
|
|
|
try {
|
2017-11-01 19:43:11 +02:00
|
|
|
writeFull(in.writeSide.get(), *options.input);
|
2017-03-15 15:40:47 +02:00
|
|
|
promise.set_value();
|
|
|
|
} catch (...) {
|
|
|
|
promise.set_exception(std::current_exception());
|
|
|
|
}
|
2017-03-13 15:56:33 +02:00
|
|
|
in.writeSide = -1;
|
|
|
|
});
|
2015-02-04 17:43:32 +02:00
|
|
|
}
|
2006-07-20 15:17:25 +03:00
|
|
|
|
2016-07-11 22:44:44 +03:00
|
|
|
string result = drainFD(out.readSide.get());
|
2006-07-20 15:17:25 +03:00
|
|
|
|
|
|
|
/* Wait for the child to finish. */
|
2017-01-19 17:58:39 +02:00
|
|
|
int status = pid.wait();
|
2006-07-20 15:17:25 +03:00
|
|
|
|
2017-03-15 15:40:47 +02:00
|
|
|
/* Wait for the writer thread to finish. */
|
2017-11-01 19:43:11 +02:00
|
|
|
if (options.input) promise.get_future().get();
|
2017-03-13 15:56:33 +02:00
|
|
|
|
2017-11-01 19:43:11 +02:00
|
|
|
return {status, result};
|
2006-07-20 15:17:25 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-08-02 15:54:35 +03:00
|
|
|
void closeMostFDs(const set<int> & exceptions)
|
|
|
|
{
|
2017-08-09 17:22:05 +03:00
|
|
|
#if __linux__
|
|
|
|
try {
|
|
|
|
for (auto & s : readDirectory("/proc/self/fd")) {
|
|
|
|
auto fd = std::stoi(s.name);
|
|
|
|
if (!exceptions.count(fd)) {
|
|
|
|
debug("closing leaked FD %d", fd);
|
|
|
|
close(fd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
} catch (SysError &) {
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2008-08-02 15:54:35 +03:00
|
|
|
int maxFD = 0;
|
|
|
|
maxFD = sysconf(_SC_OPEN_MAX);
|
|
|
|
for (int fd = 0; fd < maxFD; ++fd)
|
2017-08-09 17:22:05 +03:00
|
|
|
if (!exceptions.count(fd))
|
2008-08-02 15:54:35 +03:00
|
|
|
close(fd); /* ignore result */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-05 21:29:00 +02:00
|
|
|
void closeOnExec(int fd)
|
|
|
|
{
|
|
|
|
int prev;
|
|
|
|
if ((prev = fcntl(fd, F_GETFD, 0)) == -1 ||
|
|
|
|
fcntl(fd, F_SETFD, prev | FD_CLOEXEC) == -1)
|
|
|
|
throw SysError("setting close-on-exec flag");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-06-22 12:51:44 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
2017-01-17 19:21:02 +02:00
|
|
|
bool _isInterrupted = false;
|
2004-01-15 22:23:55 +02:00
|
|
|
|
2017-04-21 17:28:10 +03:00
|
|
|
static thread_local bool interruptThrown = false;
|
2017-09-08 16:31:24 +03:00
|
|
|
thread_local std::function<bool()> interruptCheck;
|
2017-04-21 17:28:10 +03:00
|
|
|
|
|
|
|
void setInterruptThrown()
|
|
|
|
{
|
|
|
|
interruptThrown = true;
|
|
|
|
}
|
2016-03-29 16:08:24 +03:00
|
|
|
|
2004-01-15 22:23:55 +02:00
|
|
|
void _interrupted()
|
|
|
|
{
|
2004-05-11 16:48:25 +03:00
|
|
|
/* Block user interrupts while an exception is being handled.
|
|
|
|
Throwing an exception while another exception is being handled
|
|
|
|
kills the program! */
|
2016-03-29 16:08:24 +03:00
|
|
|
if (!interruptThrown && !std::uncaught_exception()) {
|
|
|
|
interruptThrown = true;
|
2006-12-04 19:17:13 +02:00
|
|
|
throw Interrupted("interrupted by the user");
|
2004-05-11 16:48:25 +03:00
|
|
|
}
|
2004-01-15 22:23:55 +02:00
|
|
|
}
|
2004-06-20 16:37:51 +03:00
|
|
|
|
|
|
|
|
2004-06-22 12:51:44 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
2012-09-19 22:43:23 +03:00
|
|
|
template<class C> C tokenizeString(const string & s, const string & separators)
|
2005-09-22 18:43:22 +03:00
|
|
|
{
|
2012-09-19 22:43:23 +03:00
|
|
|
C result;
|
2005-09-22 18:43:22 +03:00
|
|
|
string::size_type pos = s.find_first_not_of(separators, 0);
|
|
|
|
while (pos != string::npos) {
|
|
|
|
string::size_type end = s.find_first_of(separators, pos + 1);
|
|
|
|
if (end == string::npos) end = s.size();
|
|
|
|
string token(s, pos, end - pos);
|
2012-11-26 16:39:10 +02:00
|
|
|
result.insert(result.end(), token);
|
2005-09-22 18:43:22 +03:00
|
|
|
pos = s.find_first_not_of(separators, end);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2012-09-19 22:43:23 +03:00
|
|
|
template Strings tokenizeString(const string & s, const string & separators);
|
2012-11-26 16:39:10 +02:00
|
|
|
template StringSet tokenizeString(const string & s, const string & separators);
|
2012-09-19 22:43:23 +03:00
|
|
|
template vector<string> tokenizeString(const string & s, const string & separators);
|
|
|
|
|
2005-09-22 18:43:22 +03:00
|
|
|
|
2010-08-27 16:18:13 +03:00
|
|
|
string concatStringsSep(const string & sep, const Strings & ss)
|
|
|
|
{
|
|
|
|
string s;
|
2015-07-17 20:24:28 +03:00
|
|
|
for (auto & i : ss) {
|
2010-08-27 16:18:13 +03:00
|
|
|
if (s.size() != 0) s += sep;
|
2015-07-17 20:24:28 +03:00
|
|
|
s += i;
|
2010-08-27 16:18:13 +03:00
|
|
|
}
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-26 16:39:10 +02:00
|
|
|
string concatStringsSep(const string & sep, const StringSet & ss)
|
|
|
|
{
|
|
|
|
string s;
|
2015-07-17 20:24:28 +03:00
|
|
|
for (auto & i : ss) {
|
2012-11-26 16:39:10 +02:00
|
|
|
if (s.size() != 0) s += sep;
|
2015-07-17 20:24:28 +03:00
|
|
|
s += i;
|
2012-11-26 16:39:10 +02:00
|
|
|
}
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-01 18:19:24 +03:00
|
|
|
string chomp(const string & s)
|
|
|
|
{
|
|
|
|
size_t i = s.find_last_not_of(" \n\r\t");
|
2012-08-02 00:21:47 +03:00
|
|
|
return i == string::npos ? "" : string(s, 0, i + 1);
|
2012-08-01 18:19:24 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-04-09 12:42:04 +03:00
|
|
|
string trim(const string & s, const string & whitespace)
|
|
|
|
{
|
|
|
|
auto i = s.find_first_not_of(whitespace);
|
|
|
|
if (i == string::npos) return "";
|
|
|
|
auto j = s.find_last_not_of(whitespace);
|
|
|
|
return string(s, i, j == string::npos ? j : j - i + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-06-17 17:20:11 +03:00
|
|
|
string replaceStrings(const std::string & s,
|
|
|
|
const std::string & from, const std::string & to)
|
|
|
|
{
|
|
|
|
if (from.empty()) return s;
|
|
|
|
string res = s;
|
|
|
|
size_t pos = 0;
|
|
|
|
while ((pos = res.find(from, pos)) != std::string::npos) {
|
|
|
|
res.replace(pos, from.size(), to);
|
|
|
|
pos += to.size();
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-06-22 11:50:25 +03:00
|
|
|
string statusToString(int status)
|
|
|
|
{
|
|
|
|
if (!WIFEXITED(status) || WEXITSTATUS(status) != 0) {
|
|
|
|
if (WIFEXITED(status))
|
2004-06-22 20:04:10 +03:00
|
|
|
return (format("failed with exit code %1%") % WEXITSTATUS(status)).str();
|
2007-12-14 16:49:35 +02:00
|
|
|
else if (WIFSIGNALED(status)) {
|
2013-01-03 14:00:46 +02:00
|
|
|
int sig = WTERMSIG(status);
|
2007-12-14 16:49:35 +02:00
|
|
|
#if HAVE_STRSIGNAL
|
|
|
|
const char * description = strsignal(sig);
|
|
|
|
return (format("failed due to signal %1% (%2%)") % sig % description).str();
|
|
|
|
#else
|
|
|
|
return (format("failed due to signal %1%") % sig).str();
|
|
|
|
#endif
|
2013-01-03 14:00:46 +02:00
|
|
|
}
|
2004-06-22 11:50:25 +03:00
|
|
|
else
|
|
|
|
return "died abnormally";
|
|
|
|
} else return "succeeded";
|
|
|
|
}
|
2004-06-22 14:03:41 +03:00
|
|
|
|
|
|
|
|
|
|
|
bool statusOk(int status)
|
|
|
|
{
|
|
|
|
return WIFEXITED(status) && WEXITSTATUS(status) == 0;
|
|
|
|
}
|
2004-09-10 16:32:08 +03:00
|
|
|
|
|
|
|
|
2017-05-01 18:28:19 +03:00
|
|
|
bool hasPrefix(const string & s, const string & prefix)
|
2016-04-29 22:04:40 +03:00
|
|
|
{
|
2017-05-01 18:28:19 +03:00
|
|
|
return s.compare(0, prefix.size(), prefix) == 0;
|
2016-04-29 22:04:40 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-08-25 16:31:57 +03:00
|
|
|
bool hasSuffix(const string & s, const string & suffix)
|
|
|
|
{
|
|
|
|
return s.size() >= suffix.size() && string(s, s.size() - suffix.size()) == suffix;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-09-14 15:42:15 +03:00
|
|
|
std::string toLower(const std::string & s)
|
|
|
|
{
|
|
|
|
std::string r(s);
|
|
|
|
for (auto & c : r)
|
|
|
|
c = std::tolower(c);
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-10-25 14:01:50 +03:00
|
|
|
std::string shellEscape(const std::string & s)
|
|
|
|
{
|
|
|
|
std::string r = "'";
|
|
|
|
for (auto & i : s)
|
|
|
|
if (i == '\'') r += "'\\''"; else r += i;
|
|
|
|
r += '\'';
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-05-01 18:16:17 +03:00
|
|
|
void ignoreException()
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
throw;
|
|
|
|
} catch (std::exception & e) {
|
2016-09-21 17:11:01 +03:00
|
|
|
printError(format("error (ignored): %1%") % e.what());
|
2007-05-01 18:16:17 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-15 16:01:02 +02:00
|
|
|
|
2014-08-20 17:01:16 +03:00
|
|
|
string filterANSIEscapes(const string & s, bool nixOnly)
|
|
|
|
{
|
|
|
|
string t, r;
|
|
|
|
enum { stTop, stEscape, stCSI } state = stTop;
|
|
|
|
for (auto c : s) {
|
|
|
|
if (state == stTop) {
|
|
|
|
if (c == '\e') {
|
|
|
|
state = stEscape;
|
|
|
|
r = c;
|
|
|
|
} else
|
|
|
|
t += c;
|
|
|
|
} else if (state == stEscape) {
|
|
|
|
r += c;
|
|
|
|
if (c == '[')
|
|
|
|
state = stCSI;
|
|
|
|
else {
|
|
|
|
t += r;
|
|
|
|
state = stTop;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
r += c;
|
2017-08-25 16:44:15 +03:00
|
|
|
if (c >= 0x40 && c <= 0x7e) {
|
2014-08-20 17:01:16 +03:00
|
|
|
if (nixOnly && (c != 'p' && c != 'q' && c != 's' && c != 'a' && c != 'b'))
|
|
|
|
t += r;
|
|
|
|
state = stTop;
|
|
|
|
r.clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
t += r;
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-02-09 16:09:39 +02:00
|
|
|
static char base64Chars[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
|
|
|
|
|
|
|
|
|
|
|
|
string base64Encode(const string & s)
|
|
|
|
{
|
|
|
|
string res;
|
|
|
|
int data = 0, nbits = 0;
|
|
|
|
|
|
|
|
for (char c : s) {
|
|
|
|
data = data << 8 | (unsigned char) c;
|
|
|
|
nbits += 8;
|
|
|
|
while (nbits >= 6) {
|
|
|
|
nbits -= 6;
|
|
|
|
res.push_back(base64Chars[data >> nbits & 0x3f]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nbits) res.push_back(base64Chars[data << (6 - nbits) & 0x3f]);
|
|
|
|
while (res.size() % 4) res.push_back('=');
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
string base64Decode(const string & s)
|
|
|
|
{
|
|
|
|
bool init = false;
|
|
|
|
char decode[256];
|
|
|
|
if (!init) {
|
|
|
|
// FIXME: not thread-safe.
|
|
|
|
memset(decode, -1, sizeof(decode));
|
|
|
|
for (int i = 0; i < 64; i++)
|
|
|
|
decode[(int) base64Chars[i]] = i;
|
|
|
|
init = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
string res;
|
|
|
|
unsigned int d = 0, bits = 0;
|
|
|
|
|
|
|
|
for (char c : s) {
|
|
|
|
if (c == '=') break;
|
|
|
|
if (c == '\n') continue;
|
|
|
|
|
|
|
|
char digit = decode[(unsigned char) c];
|
|
|
|
if (digit == -1)
|
|
|
|
throw Error("invalid character in Base64 string");
|
|
|
|
|
|
|
|
bits += 6;
|
|
|
|
d = d << 6 | digit;
|
|
|
|
if (bits >= 8) {
|
|
|
|
res.push_back(d >> (bits - 8) & 0xff);
|
|
|
|
bits -= 8;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-09-20 18:49:31 +03:00
|
|
|
void callFailure(const std::function<void(std::exception_ptr exc)> & failure, std::exception_ptr exc)
|
2016-09-16 19:54:14 +03:00
|
|
|
{
|
|
|
|
try {
|
2016-09-20 18:49:31 +03:00
|
|
|
failure(exc);
|
2016-09-16 19:54:14 +03:00
|
|
|
} catch (std::exception & e) {
|
2016-09-21 17:11:01 +03:00
|
|
|
printError(format("uncaught exception: %s") % e.what());
|
2016-09-16 19:54:14 +03:00
|
|
|
abort();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-08-25 16:57:49 +03:00
|
|
|
static Sync<std::pair<unsigned short, unsigned short>> windowSize{{0, 0}};
|
|
|
|
|
|
|
|
|
|
|
|
static void updateWindowSize()
|
|
|
|
{
|
|
|
|
struct winsize ws;
|
|
|
|
if (ioctl(1, TIOCGWINSZ, &ws) == 0) {
|
|
|
|
auto windowSize_(windowSize.lock());
|
|
|
|
windowSize_->first = ws.ws_row;
|
|
|
|
windowSize_->second = ws.ws_col;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
std::pair<unsigned short, unsigned short> getWindowSize()
|
|
|
|
{
|
|
|
|
return *windowSize.lock();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-01-17 19:21:02 +02:00
|
|
|
static Sync<std::list<std::function<void()>>> _interruptCallbacks;
|
|
|
|
|
|
|
|
static void signalHandlerThread(sigset_t set)
|
|
|
|
{
|
|
|
|
while (true) {
|
|
|
|
int signal = 0;
|
|
|
|
sigwait(&set, &signal);
|
|
|
|
|
2017-01-25 14:37:02 +02:00
|
|
|
if (signal == SIGINT || signal == SIGTERM || signal == SIGHUP)
|
|
|
|
triggerInterrupt();
|
2017-08-25 16:57:49 +03:00
|
|
|
|
|
|
|
else if (signal == SIGWINCH) {
|
|
|
|
updateWindowSize();
|
|
|
|
}
|
2017-01-25 14:37:02 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void triggerInterrupt()
|
|
|
|
{
|
2017-04-06 18:18:56 +03:00
|
|
|
_isInterrupted = true;
|
2017-01-25 14:37:02 +02:00
|
|
|
|
|
|
|
{
|
|
|
|
auto interruptCallbacks(_interruptCallbacks.lock());
|
|
|
|
for (auto & callback : *interruptCallbacks) {
|
|
|
|
try {
|
|
|
|
callback();
|
|
|
|
} catch (...) {
|
|
|
|
ignoreException();
|
2017-01-17 19:21:02 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-01 14:00:21 +02:00
|
|
|
static sigset_t savedSignalMask;
|
|
|
|
|
2017-01-17 19:21:02 +02:00
|
|
|
void startSignalHandlerThread()
|
|
|
|
{
|
2017-08-25 16:57:49 +03:00
|
|
|
updateWindowSize();
|
|
|
|
|
2017-02-01 14:00:21 +02:00
|
|
|
if (sigprocmask(SIG_BLOCK, nullptr, &savedSignalMask))
|
|
|
|
throw SysError("quering signal mask");
|
|
|
|
|
2017-01-17 19:21:02 +02:00
|
|
|
sigset_t set;
|
|
|
|
sigemptyset(&set);
|
|
|
|
sigaddset(&set, SIGINT);
|
|
|
|
sigaddset(&set, SIGTERM);
|
|
|
|
sigaddset(&set, SIGHUP);
|
2017-02-01 14:00:21 +02:00
|
|
|
sigaddset(&set, SIGPIPE);
|
2017-08-25 16:57:49 +03:00
|
|
|
sigaddset(&set, SIGWINCH);
|
2017-01-17 19:21:02 +02:00
|
|
|
if (pthread_sigmask(SIG_BLOCK, &set, nullptr))
|
|
|
|
throw SysError("blocking signals");
|
|
|
|
|
|
|
|
std::thread(signalHandlerThread, set).detach();
|
|
|
|
}
|
|
|
|
|
2017-02-01 14:00:21 +02:00
|
|
|
void restoreSignals()
|
|
|
|
{
|
|
|
|
if (sigprocmask(SIG_SETMASK, &savedSignalMask, nullptr))
|
|
|
|
throw SysError("restoring signals");
|
|
|
|
}
|
|
|
|
|
2017-01-17 19:21:02 +02:00
|
|
|
/* RAII helper to automatically deregister a callback. */
|
|
|
|
struct InterruptCallbackImpl : InterruptCallback
|
|
|
|
{
|
|
|
|
std::list<std::function<void()>>::iterator it;
|
|
|
|
~InterruptCallbackImpl() override
|
|
|
|
{
|
|
|
|
_interruptCallbacks.lock()->erase(it);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
std::unique_ptr<InterruptCallback> createInterruptCallback(std::function<void()> callback)
|
|
|
|
{
|
|
|
|
auto interruptCallbacks(_interruptCallbacks.lock());
|
|
|
|
interruptCallbacks->push_back(callback);
|
|
|
|
|
|
|
|
auto res = std::make_unique<InterruptCallbackImpl>();
|
|
|
|
res->it = interruptCallbacks->end();
|
|
|
|
res->it--;
|
|
|
|
|
2017-01-24 11:55:28 +02:00
|
|
|
return std::unique_ptr<InterruptCallback>(res.release());
|
2017-01-17 19:21:02 +02:00
|
|
|
}
|
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
}
|