diff options
author | David Gibson <david@gibson.dropbear.id.au> | 2022-10-14 15:25:31 +1100 |
---|---|---|
committer | Stefano Brivio <sbrivio@redhat.com> | 2022-10-15 02:10:36 +0200 |
commit | 096e48669b56273d44a3242d7af4840b38335961 (patch) | |
tree | f88f70e1843ebc5edc7c323a817bf86d1b3cd014 /isolation.c | |
parent | 6909a8e3390913d656f55803fa6cf5d0b1d46812 (diff) | |
download | passt-096e48669b56273d44a3242d7af4840b38335961.tar passt-096e48669b56273d44a3242d7af4840b38335961.tar.gz passt-096e48669b56273d44a3242d7af4840b38335961.tar.bz2 passt-096e48669b56273d44a3242d7af4840b38335961.tar.lz passt-096e48669b56273d44a3242d7af4840b38335961.tar.xz passt-096e48669b56273d44a3242d7af4840b38335961.tar.zst passt-096e48669b56273d44a3242d7af4840b38335961.zip |
isolation: Clarify various self-isolation steps
We have a number of steps of self-isolation scattered across our code.
Improve function names and add comments to make it clearer what the self
isolation model is, what the steps do, and why they happen at the points
they happen.
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
Signed-off-by: Stefano Brivio <sbrivio@redhat.com>
Diffstat (limited to 'isolation.c')
-rw-r--r-- | isolation.c | 85 |
1 files changed, 80 insertions, 5 deletions
diff --git a/isolation.c b/isolation.c index b54c325..a3ea842 100644 --- a/isolation.c +++ b/isolation.c @@ -12,6 +12,48 @@ * Author: Stefano Brivio <sbrivio@redhat.com> * Author: David Gibson <david@gibson.dropbear.id.au> */ +/** + * DOC: Theory of Operation + * + * For security the passt/pasta process performs a number of + * self-isolations steps, dropping capabilities, setting namespaces + * and otherwise minimising the impact we can have on the system at + * large if we were compromised. + * + * Obviously we can't isolate ourselves from resources before we've + * done anything we need to do with those resources, so we have + * multiple stages of self-isolation. In order these are: + * + * 1. isolate_initial() + * ==================== + * + * Executed immediately after startup, drops capabilities we don't + * need at any point during execution (or which we gain back when we + * need by joining other namespaces). + * + * 2. isolate_user() + * ================= + * + * Executed once we know what user and user namespace we want to + * operate in. Sets our final UID & GID, and enters the correct user + * namespace. + * + * 3. isolate_prefork() + * ==================== + * + * Executed after all setup, but before daemonising (fork()ing into + * the background). Uses mount namespace and pivot_root() to remove + * our access to the filesystem. + * + * 4. isolate_postfork() + * ===================== + * + * Executed immediately after daemonizing, but before entering the + * actual packet forwarding phase of operation. Or, if not + * daemonizing, immediately after isolate_prefork(). Uses seccomp() + * to restrict ourselves to the handful of syscalls we need during + * runtime operation. + */ #include <errno.h> #include <fcntl.h> @@ -48,7 +90,7 @@ /** * drop_caps() - Drop capabilities we might have except for CAP_NET_BIND_SERVICE */ -void drop_caps(void) +static void drop_caps(void) { int i; @@ -61,11 +103,30 @@ void drop_caps(void) } /** + * isolate_initial() - Early, config independent self isolation + * + * Should: + * - drop unneeded capabilities + * Musn't: + * - remove filesytem access (we need to access files during setup) + */ +void isolate_initial(void) +{ + drop_caps(); +} + +/** * isolate_user() - Switch to final UID/GID and move into userns * @uid: User ID to run as (in original userns) * @gid: Group ID to run as (in original userns) * @use_userns: Whether to join or create a userns * @userns: userns path to enter, may be empty + * + * Should: + * - set our final UID and GID + * - enter our final user namespace + * Mustn't: + * - remove filesystem access (we need that for further setup) */ void isolate_user(uid_t uid, gid_t gid, bool use_userns, const char *userns) { @@ -135,11 +196,19 @@ void isolate_user(uid_t uid, gid_t gid, bool use_userns, const char *userns) } /** - * sandbox() - Unshare IPC, mount, PID, UTS, and user namespaces, "unmount" root + * isolate_prefork() - Self isolation before daemonizing + * @c: Execution context * * Return: negative error code on failure, zero on success + * + * Should: + * - Move us to our own IPC and UTS namespaces + * - Move us to a mount namespace with only an empty directory + * - Drop unneeded capabilities (in the new user namespace) + * Mustn't: + * - Remove syscalls we need to daemonise */ -int sandbox(struct ctx *c) +int isolate_prefork(struct ctx *c) { int flags = CLONE_NEWIPC | CLONE_NEWNS | CLONE_NEWUTS; @@ -188,13 +257,19 @@ int sandbox(struct ctx *c) } /** - * seccomp() - Set up seccomp filters depending on mode, won't return on failure + * isolate_postfork() - Self isolation after daemonizing * @c: Execution context + * + * Should: + * - disable core dumps + * - limit to a minimal set of syscalls */ -void seccomp(const struct ctx *c) +void isolate_postfork(const struct ctx *c) { struct sock_fprog prog; + prctl(PR_SET_DUMPABLE, 0); + if (c->mode == MODE_PASST) { prog.len = (unsigned short)ARRAY_SIZE(filter_passt); prog.filter = filter_passt; |