From b0a94268f8761cfa0dfa532fb8b18620b381f440 Mon Sep 17 00:00:00 2001 From: Lennart Poettering Date: Tue, 7 Jan 2020 16:56:05 +0100 Subject: [PATCH] core: when we cannot open an image file for write, try read-only Closes: #14442 --- src/core/namespace.c | 2 +- src/shared/loop-util.c | 25 +++++++++++++++++++++---- 2 files changed, 22 insertions(+), 5 deletions(-) diff --git a/src/core/namespace.c b/src/core/namespace.c index fd38174ac02..a6f803671eb 100644 --- a/src/core/namespace.c +++ b/src/core/namespace.c @@ -1289,7 +1289,7 @@ int setup_namespace( dissect_image_flags |= DISSECT_IMAGE_READ_ONLY; r = loop_device_make_by_path(root_image, - dissect_image_flags & DISSECT_IMAGE_READ_ONLY ? O_RDONLY : O_RDWR, + FLAGS_SET(dissect_image_flags, DISSECT_IMAGE_READ_ONLY) ? O_RDONLY : -1 /* < 0 means writable if possible, read-only as fallback */, LO_FLAGS_PARTSCAN, &loop_device); if (r < 0) diff --git a/src/shared/loop-util.c b/src/shared/loop-util.c index d92ef961624..7aee239e33a 100644 --- a/src/shared/loop-util.c +++ b/src/shared/loop-util.c @@ -14,6 +14,7 @@ #include #include "alloc-util.h" +#include "errno-util.h" #include "fd-util.h" #include "fileio.h" #include "loop-util.h" @@ -157,14 +158,30 @@ int loop_device_make( int loop_device_make_by_path(const char *path, int open_flags, uint32_t loop_flags, LoopDevice **ret) { _cleanup_close_ int fd = -1; + int r; assert(path); assert(ret); - assert(IN_SET(open_flags, O_RDWR, O_RDONLY)); + assert(open_flags < 0 || IN_SET(open_flags, O_RDWR, O_RDONLY)); - fd = open(path, O_CLOEXEC|O_NONBLOCK|O_NOCTTY|open_flags); - if (fd < 0) - return -errno; + /* Passing < 0 as open_flags here means we'll try to open the device writable if we can, retrying + * read-only if we cannot. */ + + fd = open(path, O_CLOEXEC|O_NONBLOCK|O_NOCTTY|(open_flags >= 0 ? open_flags : O_RDWR)); + if (fd < 0) { + r = -errno; + + /* Retry read-only? */ + if (open_flags >= 0 || !(ERRNO_IS_PRIVILEGE(r) || r == -EROFS)) + return r; + + fd = open(path, O_CLOEXEC|O_NONBLOCK|O_NOCTTY|O_RDONLY); + if (fd < 0) + return r; /* Propagate original error */ + + open_flags = O_RDONLY; + } else if (open_flags < 0) + open_flags = O_RDWR; return loop_device_make(fd, open_flags, 0, 0, loop_flags, ret); }