aboutsummaryrefslogtreecommitdiff
path: root/src/libutil/util.cc
diff options
context:
space:
mode:
authorEelco Dolstra <edolstra@gmail.com>2018-09-25 12:36:11 +0200
committerEelco Dolstra <edolstra@gmail.com>2019-10-29 13:30:51 +0100
commit63b99af85a1e280876da3d807e4bc96a5c5dde39 (patch)
treee369a7b60b86dba503893c1a74a0c30fad85c5b4 /src/libutil/util.cc
parent2d37e88319441958b7eed876042a106e79c7c85d (diff)
Move Unix domain socket creation to libutil
Also drop multithread-unfriendly hacks like doing a temporary chmod/umask.
Diffstat (limited to 'src/libutil/util.cc')
-rw-r--r--src/libutil/util.cc31
1 files changed, 31 insertions, 0 deletions
diff --git a/src/libutil/util.cc b/src/libutil/util.cc
index 2e416edef..2a1272885 100644
--- a/src/libutil/util.cc
+++ b/src/libutil/util.cc
@@ -21,7 +21,9 @@
#include <pwd.h>
#include <sys/ioctl.h>
#include <sys/types.h>
+#include <sys/socket.h>
#include <sys/wait.h>
+#include <sys/un.h>
#include <unistd.h>
#ifdef __APPLE__
@@ -1562,4 +1564,33 @@ std::unique_ptr<InterruptCallback> createInterruptCallback(std::function<void()>
return std::unique_ptr<InterruptCallback>(res.release());
}
+
+AutoCloseFD createUnixDomainSocket(const Path & path, mode_t mode)
+{
+ AutoCloseFD fdSocket = socket(PF_UNIX, SOCK_STREAM | SOCK_CLOEXEC, 0);
+ if (!fdSocket)
+ throw SysError("cannot create Unix domain socket");
+
+ closeOnExec(fdSocket.get());
+
+ struct sockaddr_un addr;
+ addr.sun_family = AF_UNIX;
+ if (path.size() >= sizeof(addr.sun_path))
+ throw Error("socket path '%1%' is too long", path);
+ strcpy(addr.sun_path, path.c_str());
+
+ unlink(path.c_str());
+
+ if (bind(fdSocket.get(), (struct sockaddr *) &addr, sizeof(addr)) == -1)
+ throw SysError("cannot bind to socket '%1%'", path);
+
+ if (chmod(path.c_str(), mode) == -1)
+ throw SysError("changing permissions on '%1%'", path);
+
+ if (listen(fdSocket.get(), 5) == -1)
+ throw SysError("cannot listen on socket '%1%'", path);
+
+ return fdSocket;
+}
+
}