12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238 |
- #include "config.h"
- #include "util.hh"
- #include "affinity.hh"
- #include <iostream>
- #include <cerrno>
- #include <cstdio>
- #include <cstdlib>
- #include <sstream>
- #include <cstring>
- #include <sys/wait.h>
- #include <unistd.h>
- #include <fcntl.h>
- #include <limits.h>
- #ifdef __APPLE__
- #include <sys/syscall.h>
- #endif
- #ifdef __linux__
- #include <sys/prctl.h>
- #endif
- extern char * * environ;
- namespace nix {
- BaseError::BaseError(const FormatOrString & fs, unsigned int status)
- : status(status)
- {
- err = fs.s;
- }
- BaseError & BaseError::addPrefix(const FormatOrString & fs)
- {
- prefix_ = fs.s + prefix_;
- return *this;
- }
- SysError::SysError(const FormatOrString & fs)
- : Error(format("%1%: %2%") % fs.s % strerror(errno))
- , errNo(errno)
- {
- }
- string getEnv(const string & key, const string & def)
- {
- char * value = getenv(key.c_str());
- return value ? string(value) : def;
- }
- Path absPath(Path path, Path dir)
- {
- if (path[0] != '/') {
- if (dir == "") {
- #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
- char buf[PATH_MAX];
- if (!getcwd(buf, sizeof(buf)))
- #endif
- throw SysError("cannot get cwd");
- dir = buf;
- #ifdef __GNU__
- free(buf);
- #endif
- }
- path = dir + "/" + path;
- }
- return canonPath(path);
- }
- Path canonPath(const Path & path, bool resolveSymlinks)
- {
- string s;
- if (path[0] != '/')
- throw Error(format("not an absolute path: `%1%'") % path);
- string::const_iterator i = path.begin(), end = path.end();
- 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;
- 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. */
- else if (*i == '.' && i + 1 < end && i[1] == '.' &&
- (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++;
- /* If s points to a symlink, resolve it and restart (since
- the symlink target might contain new symlinks). */
- if (resolveSymlinks && isLink(s)) {
- if (++followCount >= maxFollow)
- throw Error(format("infinite symlink recursion in path `%1%'") % path);
- temp = absPath(readLink(s), dirOf(s))
- + string(i, end);
- i = temp.begin(); /* restart */
- end = temp.end();
- s = "";
- }
- }
- }
- return s.empty() ? "/" : s;
- }
- Path dirOf(const Path & path)
- {
- Path::size_type pos = path.rfind('/');
- if (pos == string::npos)
- throw Error(format("invalid file name `%1%'") % path);
- return pos == 0 ? "/" : Path(path, 0, pos);
- }
- string baseNameOf(const Path & path)
- {
- Path::size_type pos = path.rfind('/');
- if (pos == string::npos)
- throw Error(format("invalid file name `%1%'") % path);
- return string(path, pos + 1);
- }
- bool isInDir(const Path & path, const Path & dir)
- {
- return path[0] == '/'
- && string(path, 0, dir.size()) == dir
- && path.size() >= dir.size() + 2
- && path[dir.size()] == '/';
- }
- struct stat lstat(const Path & path)
- {
- struct stat st;
- if (lstat(path.c_str(), &st))
- throw SysError(format("getting status of `%1%'") % path);
- return st;
- }
- bool pathExists(const Path & path)
- {
- int res;
- #ifdef HAVE_STATX
- struct statx st;
- res = statx(AT_FDCWD, path.c_str(), AT_SYMLINK_NOFOLLOW, 0, &st);
- #else
- struct stat st;
- res = lstat(path.c_str(), &st);
- #endif
- if (!res) return true;
- if (errno != ENOENT && errno != ENOTDIR)
- throw SysError(format("getting status of %1%") % path);
- return false;
- }
- Path readLink(const Path & path)
- {
- checkInterrupt();
- struct stat st = lstat(path);
- if (!S_ISLNK(st.st_mode))
- throw Error(format("`%1%' is not a symlink") % path);
- char buf[st.st_size];
- ssize_t rlsize = readlink(path.c_str(), buf, st.st_size);
- if (rlsize == -1)
- throw SysError(format("reading symbolic link '%1%'") % path);
- else if (rlsize > st.st_size)
- throw Error(format("symbolic link ‘%1%’ size overflow %2% > %3%")
- % path % rlsize % st.st_size);
- return string(buf, st.st_size);
- }
- bool isLink(const Path & path)
- {
- struct stat st = lstat(path);
- return S_ISLNK(st.st_mode);
- }
- DirEntries readDirectory(const Path & path)
- {
- DirEntries entries;
- entries.reserve(64);
- AutoCloseDir dir = opendir(path.c_str());
- if (!dir) throw SysError(format("opening directory `%1%'") % path);
- struct dirent * dirent;
- while (errno = 0, dirent = readdir(dir)) { /* sic */
- checkInterrupt();
- string name = dirent->d_name;
- if (name == "." || name == "..") continue;
- entries.emplace_back(name, dirent->d_ino, dirent->d_type);
- }
- if (errno) throw SysError(format("reading directory `%1%'") % path);
- return entries;
- }
- 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;
- }
- string readFile(int fd)
- {
- struct stat st;
- if (fstat(fd, &st) == -1)
- throw SysError("statting file");
- unsigned char * buf = new unsigned char[st.st_size];
- AutoDeleteArray<unsigned char> d(buf);
- readFull(fd, buf, st.st_size);
- return string((char *) buf, st.st_size);
- }
- string readFile(const Path & path, bool drain)
- {
- AutoCloseFD fd = open(path.c_str(), O_RDONLY);
- if (fd == -1)
- throw SysError(format("reading file `%1%'") % path);
- return drain ? drainFD(fd) : readFile(fd);
- }
- void writeFile(const Path & path, const string & s)
- {
- AutoCloseFD fd = open(path.c_str(), O_WRONLY | O_TRUNC | O_CREAT, 0666);
- if (fd == -1)
- throw SysError(format("writing file '%1%'") % path);
- writeFull(fd, s);
- }
- string readLine(int fd)
- {
- string s;
- while (1) {
- checkInterrupt();
- char ch;
- ssize_t rd = read(fd, &ch, 1);
- if (rd == -1) {
- if (errno != EINTR)
- throw SysError("reading a line");
- } else if (rd == 0)
- throw EndOfFile("unexpected EOF reading a line");
- else {
- if (ch == '\n') return s;
- s += ch;
- }
- }
- }
- void writeLine(int fd, string s)
- {
- s += '\n';
- writeFull(fd, s);
- }
- static void _deletePath(const Path & path, unsigned long long & bytesFreed, size_t linkThreshold)
- {
- checkInterrupt();
- printMsg(lvlVomit, format("%1%") % path);
- #ifdef HAVE_STATX
- # define st_mode stx_mode
- # define st_size stx_size
- # define st_nlink stx_nlink
- struct statx st;
- if (statx(AT_FDCWD, path.c_str(),
- AT_SYMLINK_NOFOLLOW,
- STATX_SIZE | STATX_NLINK | STATX_MODE, &st) == -1)
- throw SysError(format("getting status of `%1%'") % path);
- #else
- struct stat st = lstat(path);
- #endif
- if (!S_ISDIR(st.st_mode) && st.st_nlink <= linkThreshold)
- bytesFreed += st.st_size;
- if (S_ISDIR(st.st_mode)) {
- /* Make the directory writable. */
- if (!(st.st_mode & S_IWUSR)) {
- if (chmod(path.c_str(), st.st_mode | S_IWUSR) == -1)
- throw SysError(format("making `%1%' writable") % path);
- }
- for (auto & i : readDirectory(path))
- _deletePath(path + "/" + i.name, bytesFreed, linkThreshold);
- }
- int ret;
- ret = S_ISDIR(st.st_mode) ? rmdir(path.c_str()) : unlink(path.c_str());
- if (ret == -1)
- throw SysError(format("cannot unlink `%1%'") % path);
- #undef st_mode
- #undef st_size
- #undef st_nlink
- }
- void deletePath(const Path & path)
- {
- unsigned long long dummy;
- deletePath(path, dummy);
- }
- void deletePath(const Path & path, unsigned long long & bytesFreed, size_t linkThreshold)
- {
- startNest(nest, lvlDebug,
- format("recursively deleting path `%1%'") % path);
- bytesFreed = 0;
- _deletePath(path, bytesFreed, linkThreshold);
- }
- static Path tempName(Path tmpRoot, const Path & prefix, bool includePid,
- int & counter)
- {
- 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();
- }
- Path createTempDir(const Path & tmpRoot, const Path & prefix,
- bool includePid, bool useGlobalCounter, mode_t mode)
- {
- static int globalCounter = 0;
- int localCounter = 0;
- int & counter(useGlobalCounter ? globalCounter : localCounter);
- while (1) {
- checkInterrupt();
- Path tmpDir = tempName(tmpRoot, prefix, includePid, counter);
- if (mkdir(tmpDir.c_str(), mode) == 0) {
- /* 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)
- throw SysError(format("setting group of directory `%1%'") % tmpDir);
- return tmpDir;
- }
- if (errno != EEXIST)
- throw SysError(format("creating directory `%1%'") % tmpDir);
- }
- }
- Paths createDirs(const Path & path)
- {
- Paths created;
- if (path == "/") return created;
- struct stat st;
- if (lstat(path.c_str(), &st) == -1) {
- created = createDirs(dirOf(path));
- if (mkdir(path.c_str(), 0777) == -1 && errno != EEXIST)
- throw SysError(format("creating directory `%1%'") % path);
- st = lstat(path);
- created.push_back(path);
- }
- if (S_ISLNK(st.st_mode) && stat(path.c_str(), &st) == -1)
- throw SysError(format("statting symlink `%1%'") % path);
- if (!S_ISDIR(st.st_mode)) throw Error(format("`%1%' is not a directory") % path);
- return created;
- }
- void createSymlink(const Path & target, const Path & link)
- {
- if (symlink(target.c_str(), link.c_str()))
- throw SysError(format("creating symlink from `%1%' to `%2%'") % link % target);
- }
- LogType logType = ltPretty;
- Verbosity verbosity = lvlInfo;
- static int nestingLevel = 0;
- Nest::Nest()
- {
- nest = false;
- }
- Nest::~Nest()
- {
- close();
- }
- static string escVerbosity(Verbosity level)
- {
- return std::to_string((int) level);
- }
- void Nest::open(Verbosity level, const FormatOrString & fs)
- {
- if (level <= verbosity) {
- if (logType == ltEscapes)
- std::cerr << "\033[" << escVerbosity(level) << "p"
- << fs.s << "\n";
- else
- printMsg_(level, fs);
- nest = true;
- nestingLevel++;
- }
- }
- void Nest::close()
- {
- if (nest) {
- nestingLevel--;
- if (logType == ltEscapes)
- std::cerr << "\033[q";
- nest = false;
- }
- }
- void printMsg_(Verbosity level, const FormatOrString & fs)
- {
- checkInterrupt();
- if (level > verbosity) return;
- string prefix;
- if (logType == ltPretty)
- for (int i = 0; i < nestingLevel; i++)
- prefix += "| ";
- else if (logType == ltEscapes && level != lvlInfo)
- prefix = "\033[" + escVerbosity(level) + "s";
- string s = (format("%1%%2%\n") % prefix % fs.s).str();
- writeToStderr(s);
- }
- void warnOnce(bool & haveWarned, const FormatOrString & fs)
- {
- if (!haveWarned) {
- printMsg(lvlError, format("warning: %1%") % fs.s);
- haveWarned = true;
- }
- }
- void writeToStderr(const string & s)
- {
- try {
- if (_writeToStderr)
- _writeToStderr((const unsigned char *) s.data(), s.size());
- else
- writeFull(STDERR_FILENO, s);
- } catch (SysError & e) {
- /* Ignore failing writes to stderr if we're in an exception
- handler, otherwise throw an exception. We need to ignore
- write errors in exception handlers to ensure that cleanup
- code runs to completion if the other side of stderr has
- been closed unexpectedly. */
- if (!std::uncaught_exception()) throw;
- }
- }
- void (*_writeToStderr) (const unsigned char * buf, size_t count) = 0;
- void readFull(int fd, unsigned char * buf, size_t count)
- {
- while (count) {
- checkInterrupt();
- ssize_t res = read(fd, (char *) buf, count);
- if (res == -1) {
- if (errno == EINTR) continue;
- throw SysError("reading from file");
- }
- if (res == 0) throw EndOfFile("unexpected end-of-file");
- count -= res;
- buf += res;
- }
- }
- void writeFull(int fd, const unsigned char * buf, size_t count)
- {
- while (count) {
- checkInterrupt();
- ssize_t res = write(fd, (char *) buf, count);
- if (res == -1) {
- if (errno == EINTR) continue;
- throw SysError("writing to file");
- }
- count -= res;
- buf += res;
- }
- }
- void writeFull(int fd, const string & s)
- {
- writeFull(fd, (const unsigned char *) s.data(), s.size());
- }
- string drainFD(int fd)
- {
- string result;
- unsigned char buffer[4096];
- while (1) {
- checkInterrupt();
- 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;
- }
- //////////////////////////////////////////////////////////////////////
- AutoDelete::AutoDelete(const string & p, bool recursive) : path(p)
- {
- del = true;
- this->recursive = recursive;
- }
- AutoDelete::~AutoDelete()
- {
- try {
- if (del) {
- if (recursive)
- deletePath(path);
- else {
- if (remove(path.c_str()) == -1)
- throw SysError(format("cannot unlink `%1%'") % path);
- }
- }
- } catch (...) {
- ignoreException();
- }
- }
- void AutoDelete::cancel()
- {
- del = false;
- }
- //////////////////////////////////////////////////////////////////////
- AutoCloseFD::AutoCloseFD()
- {
- fd = -1;
- }
- AutoCloseFD::AutoCloseFD(int fd)
- {
- this->fd = fd;
- }
- AutoCloseFD::AutoCloseFD(const AutoCloseFD & fd)
- {
- /* Copying an AutoCloseFD isn't allowed (who should get to close
- it?). But as an edge case, allow copying of closed
- AutoCloseFDs. This is necessary due to tiresome reasons
- involving copy constructor use on default object values in STL
- containers (like when you do `map[value]' where value isn't in
- the map yet). */
- this->fd = fd.fd;
- if (this->fd != -1) abort();
- }
- AutoCloseFD::~AutoCloseFD()
- {
- try {
- close();
- } catch (...) {
- ignoreException();
- }
- }
- void AutoCloseFD::operator =(int fd)
- {
- if (this->fd != fd) close();
- this->fd = fd;
- }
- AutoCloseFD::operator int() const
- {
- return fd;
- }
- void AutoCloseFD::close()
- {
- if (fd != -1) {
- if (::close(fd) == -1)
- /* This should never happen. */
- throw SysError(format("closing file descriptor %1%") % fd);
- fd = -1;
- }
- }
- bool AutoCloseFD::isOpen()
- {
- return fd != -1;
- }
- /* Pass responsibility for closing this fd to the caller. */
- int AutoCloseFD::borrow()
- {
- int oldFD = fd;
- fd = -1;
- return oldFD;
- }
- void Pipe::create()
- {
- int fds[2];
- if (pipe(fds) != 0) throw SysError("creating pipe");
- readSide = fds[0];
- writeSide = fds[1];
- closeOnExec(readSide);
- closeOnExec(writeSide);
- }
- //////////////////////////////////////////////////////////////////////
- AutoCloseDir::AutoCloseDir()
- {
- dir = 0;
- }
- AutoCloseDir::AutoCloseDir(DIR * dir)
- {
- this->dir = dir;
- }
- AutoCloseDir::~AutoCloseDir()
- {
- close();
- }
- void AutoCloseDir::operator =(DIR * dir)
- {
- this->dir = dir;
- }
- AutoCloseDir::operator DIR *()
- {
- return dir;
- }
- void AutoCloseDir::close()
- {
- if (dir) {
- closedir(dir);
- dir = 0;
- }
- }
- //////////////////////////////////////////////////////////////////////
- Pid::Pid()
- : pid(-1), separatePG(false), killSignal(SIGKILL)
- {
- }
- Pid::Pid(pid_t pid)
- : pid(pid), separatePG(false), killSignal(SIGKILL)
- {
- }
- Pid::~Pid()
- {
- kill();
- }
- void Pid::operator =(pid_t pid)
- {
- if (this->pid != pid) kill();
- this->pid = pid;
- killSignal = SIGKILL; // reset signal to default
- }
- Pid::operator pid_t()
- {
- return pid;
- }
- void Pid::kill(bool quiet)
- {
- if (pid == -1 || pid == 0) return;
- if (!quiet)
- printMsg(lvlError, format("killing process %1%") % pid);
- /* 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). */
- if (::kill(separatePG ? -pid : pid, killSignal) != 0)
- printMsg(lvlError, (SysError(format("killing process %1%") % pid).msg()));
- /* Wait until the child dies, disregarding the exit status. */
- int status;
- while (waitpid(pid, &status, 0) == -1) {
- checkInterrupt();
- if (errno != EINTR) {
- printMsg(lvlError,
- (SysError(format("waiting for process %1%") % pid).msg()));
- break;
- }
- }
- pid = -1;
- }
- int Pid::wait(bool block)
- {
- assert(pid != -1);
- while (1) {
- int status;
- int res = waitpid(pid, &status, block ? 0 : WNOHANG);
- if (res == pid) {
- pid = -1;
- return status;
- }
- if (res == 0 && !block) return -1;
- if (errno != EINTR)
- throw SysError("cannot get child exit status");
- checkInterrupt();
- }
- }
- void Pid::setSeparatePG(bool separatePG)
- {
- this->separatePG = separatePG;
- }
- void Pid::setKillSignal(int signal)
- {
- this->killSignal = signal;
- }
- void killUser(uid_t uid)
- {
- debug(format("killing all processes running under uid `%1%'") % uid);
- 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. */
- Pid pid = startProcess([&]() {
- if (setuid(uid) == -1)
- throw SysError("setting uid");
- while (true) {
- #ifdef __APPLE__
- /* 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
- */
- if (syscall(SYS_kill, -1, SIGKILL, false) == 0) break;
- #elif __GNU__
- /* Killing all a user's processes using PID=-1 does currently
- not work on the Hurd. */
- if (kill(getpid(), SIGKILL) == 0) break;
- #else
- if (kill(-1, SIGKILL) == 0) break;
- #endif
- if (errno == ESRCH) break; /* no more processes */
- if (errno != EINTR)
- throw SysError(format("cannot kill processes for uid `%1%'") % uid);
- }
- _exit(0);
- });
- int status = pid.wait(true);
- #if __GNU__
- /* When the child killed itself, status = SIGKILL. */
- if (status == SIGKILL) return;
- #endif
- if (status != 0)
- throw Error(format("cannot kill processes for uid `%1%': %2%") % uid % statusToString(status));
- /* !!! 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'. */
- }
- //////////////////////////////////////////////////////////////////////
- pid_t startProcess(std::function<void()> fun,
- bool dieWithParent, const string & errorPrefix, bool runExitHandlers)
- {
- pid_t pid = fork();
- if (pid == -1) throw SysError("unable to fork");
- if (pid == 0) {
- _writeToStderr = 0;
- try {
- #if __linux__
- if (dieWithParent && prctl(PR_SET_PDEATHSIG, SIGKILL) == -1)
- throw SysError("setting death signal");
- #endif
- restoreAffinity();
- fun();
- } catch (std::exception & e) {
- try {
- std::cerr << errorPrefix << e.what() << "\n";
- } catch (...) { }
- } catch (...) { }
- if (runExitHandlers)
- exit(1);
- else
- _exit(1);
- }
- return pid;
- }
- std::vector<char *> stringsToCharPtrs(const Strings & ss)
- {
- std::vector<char *> res;
- for (auto & s : ss) res.push_back((char *) s.c_str());
- res.push_back(0);
- return res;
- }
- string runProgram(Path program, bool searchPath, const Strings & args)
- {
- checkInterrupt();
- /* Create a pipe. */
- Pipe pipe;
- pipe.create();
- /* Fork. */
- Pid pid = startProcess([&]() {
- if (dup2(pipe.writeSide, STDOUT_FILENO) == -1)
- throw SysError("dupping stdout");
- Strings args_(args);
- args_.push_front(program);
- if (searchPath)
- execvp(program.c_str(), stringsToCharPtrs(args_).data());
- else
- execv(program.c_str(), stringsToCharPtrs(args_).data());
- throw SysError(format("executing `%1%'") % program);
- });
- pipe.writeSide.close();
- string result = drainFD(pipe.readSide);
- /* Wait for the child to finish. */
- int status = pid.wait(true);
- if (!statusOk(status))
- throw ExecError(format("program `%1%' %2%")
- % program % statusToString(status));
- return result;
- }
- void closeMostFDs(const set<int> & exceptions)
- {
- int maxFD = 0;
- maxFD = sysconf(_SC_OPEN_MAX);
- for (int fd = 0; fd < maxFD; ++fd)
- if (fd != STDIN_FILENO && fd != STDOUT_FILENO && fd != STDERR_FILENO
- && exceptions.find(fd) == exceptions.end())
- close(fd); /* ignore result */
- }
- 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");
- }
- //////////////////////////////////////////////////////////////////////
- volatile sig_atomic_t _isInterrupted = 0;
- void _interrupted()
- {
- /* Block user interrupts while an exception is being handled.
- Throwing an exception while another exception is being handled
- kills the program! */
- if (!std::uncaught_exception()) {
- _isInterrupted = 0;
- throw Interrupted("interrupted by the user");
- }
- }
- //////////////////////////////////////////////////////////////////////
- template<class C> C tokenizeString(const string & s, const string & separators)
- {
- C result;
- 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);
- result.insert(result.end(), token);
- pos = s.find_first_not_of(separators, end);
- }
- return result;
- }
- template Strings tokenizeString(const string & s, const string & separators);
- template StringSet tokenizeString(const string & s, const string & separators);
- template vector<string> tokenizeString(const string & s, const string & separators);
- string concatStringsSep(const string & sep, const Strings & ss)
- {
- string s;
- foreach (Strings::const_iterator, i, ss) {
- if (s.size() != 0) s += sep;
- s += *i;
- }
- return s;
- }
- string concatStringsSep(const string & sep, const StringSet & ss)
- {
- string s;
- foreach (StringSet::const_iterator, i, ss) {
- if (s.size() != 0) s += sep;
- s += *i;
- }
- return s;
- }
- string chomp(const string & s)
- {
- size_t i = s.find_last_not_of(" \n\r\t");
- return i == string::npos ? "" : string(s, 0, i + 1);
- }
- string statusToString(int status)
- {
- if (!WIFEXITED(status) || WEXITSTATUS(status) != 0) {
- if (WIFEXITED(status))
- return (format("failed with exit code %1%") % WEXITSTATUS(status)).str();
- else if (WIFSIGNALED(status)) {
- int sig = WTERMSIG(status);
- #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
- }
- else
- return "died abnormally";
- } else return "succeeded";
- }
- bool statusOk(int status)
- {
- return WIFEXITED(status) && WEXITSTATUS(status) == 0;
- }
- bool hasSuffix(const string & s, const string & suffix)
- {
- return s.size() >= suffix.size() && string(s, s.size() - suffix.size()) == suffix;
- }
- void expect(std::istream & str, const string & s)
- {
- char s2[s.size()];
- str.read(s2, s.size());
- if (string(s2, s.size()) != s)
- throw FormatError(format("expected string `%1%'") % s);
- }
- string parseString(std::istream & str)
- {
- string res;
- expect(str, "\"");
- int c;
- while ((c = str.get()) != '"')
- if (c == '\\') {
- c = str.get();
- if (c == 'n') res += '\n';
- else if (c == 'r') res += '\r';
- else if (c == 't') res += '\t';
- else res += c;
- }
- else res += c;
- return res;
- }
- bool endOfList(std::istream & str)
- {
- if (str.peek() == ',') {
- str.get();
- return false;
- }
- if (str.peek() == ']') {
- str.get();
- return true;
- }
- return false;
- }
- void ignoreException()
- {
- try {
- throw;
- } catch (std::exception & e) {
- printMsg(lvlError, format("error (ignored): %1%") % e.what());
- }
- }
- static const string pathNullDevice = "/dev/null";
- /* Common initialisation performed in child processes. */
- void commonChildInit(Pipe & logPipe)
- {
- /* Put the child in a separate session (and thus a separate
- process group) so that it has no controlling terminal (meaning
- that e.g. ssh cannot open /dev/tty) and it doesn't receive
- terminal signals. */
- if (setsid() == -1)
- throw SysError(format("creating a new session"));
- /* Dup the write side of the logger pipe into stderr. */
- if (dup2(logPipe.writeSide, STDERR_FILENO) == -1)
- throw SysError("cannot pipe standard error into log file");
- /* Dup stderr to stdout. */
- if (dup2(STDERR_FILENO, STDOUT_FILENO) == -1)
- throw SysError("cannot dup stderr into stdout");
- /* Reroute stdin to /dev/null. */
- int fdDevNull = open(pathNullDevice.c_str(), O_RDWR);
- if (fdDevNull == -1)
- throw SysError(format("cannot open `%1%'") % pathNullDevice);
- if (dup2(fdDevNull, STDIN_FILENO) == -1)
- throw SysError("cannot dup null device into stdin");
- close(fdDevNull);
- }
- //////////////////////////////////////////////////////////////////////
- Agent::Agent(const string &command, const Strings &args, const std::map<string, string> &env)
- {
- debug(format("starting agent '%1%'") % command);
- /* Create a pipe to get the output of the child. */
- fromAgent.create();
- /* Create the communication pipes. */
- toAgent.create();
- /* Create a pipe to get the output of the builder. */
- builderOut.create();
- /* Fork the hook. */
- pid = startProcess([&]() {
- commonChildInit(fromAgent);
- for (auto pair: env) {
- setenv(pair.first.c_str(), pair.second.c_str(), 1);
- }
- if (chdir("/") == -1) throw SysError("changing into `/");
- /* Dup the communication pipes. */
- if (dup2(toAgent.readSide, STDIN_FILENO) == -1)
- throw SysError("dupping to-hook read side");
- /* Use fd 4 for the builder's stdout/stderr. */
- if (dup2(builderOut.writeSide, 4) == -1)
- throw SysError("dupping builder's stdout/stderr");
- Strings allArgs;
- allArgs.push_back(command);
- allArgs.insert(allArgs.end(), args.begin(), args.end()); // append
- execv(command.c_str(), stringsToCharPtrs(allArgs).data());
- throw SysError(format("executing `%1%'") % command);
- });
- pid.setSeparatePG(true);
- fromAgent.writeSide.close();
- toAgent.readSide.close();
- }
- Agent::~Agent()
- {
- try {
- toAgent.writeSide.close();
- pid.kill(true);
- } catch (...) {
- ignoreException();
- }
- }
- }
|