summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorYu Watanabe <watanabe.yu+github@gmail.com>2021-01-21 07:20:04 +0900
committerGitHub <noreply@github.com>2021-01-21 07:20:04 +0900
commit4723205968d480039ee797a7661ae8cba39b21b2 (patch)
treef766dd705e71c59f5bc87d15e548950df875b8f5
parent7504f599e1331c6e0ba1780a307178fd0a7f4d3a (diff)
parent301265ea1075b0fbaa6fb80dfaa4cf6c15d69623 (diff)
downloadsystemd-4723205968d480039ee797a7661ae8cba39b21b2.tar.gz
Merge pull request #18311 from poettering/sysext-fixups
sysext: post-merge fixups
-rw-r--r--man/systemd-sysext.xml42
-rw-r--r--src/sysext/sysext.c464
-rw-r--r--units/systemd-sysext.service4
3 files changed, 276 insertions, 234 deletions
diff --git a/man/systemd-sysext.xml b/man/systemd-sysext.xml
index 6bda5f4fc6..e5f2e36899 100644
--- a/man/systemd-sysext.xml
+++ b/man/systemd-sysext.xml
@@ -139,7 +139,7 @@
with newer ones, for example to install a locally compiled development version of some low-level
component over the immutable OS image without doing a full OS rebuild or modifying the nominally
immutable image. (e.g. "install" a locally built package with <command>DESTDIR=/var/lib/extensions/mytest
- make install &amp;&amp; systemd-sysext --refresh</command>, making it available in
+ make install &amp;&amp; systemd-sysext refresh</command>, making it available in
<filename>/usr/</filename> as if it was installed in the OS image itself.) This case works regardless if
the underlying host <filename>/usr/</filename> is managed as immutable disk image or is a traditional
package manager controlled (i.e. writable) tree.</para>
@@ -148,12 +148,19 @@
<refsect1>
<title>Commands</title>
- <para>The following command switches are understood:</para>
+ <para>The following commands are understood:</para>
<variablelist>
<varlistentry>
- <term><option>--merge</option></term>
- <term><option>-m</option></term>
+ <term><option>status</option></term>
+
+ <listitem><para>When invoked without any command verb, or when <option>status</option> is specified
+ the current merge status is shown, separately for both <filename>/usr/</filename> and
+ <filename>/opt/</filename>.</para></listitem>
+ </varlistentry>
+
+ <varlistentry>
+ <term><option>merge</option></term>
<listitem><para>Merges all currently installed system extension images into
<filename>/usr/</filename> and <filename>/opt/</filename>, by overmounting these hierarchies with an
<literal>overlayfs</literal> file system combining the underlying hierarchies with those included in
@@ -161,22 +168,20 @@
</varlistentry>
<varlistentry>
- <term><option>--unmerge</option></term>
- <term><option>-u</option></term>
+ <term><option>unmerge</option></term>
<listitem><para>Unmerges all currently installed system extension images from
<filename>/usr/</filename> and <filename>/opt/</filename>, by unmounting the
- <literal>overlayfs</literal> file systems created by <option>--merge</option>
+ <literal>overlayfs</literal> file systems created by <option>merge</option>
prior.</para></listitem>
</varlistentry>
<varlistentry>
- <term><option>--refresh</option></term>
- <term><option>-R</option></term>
- <listitem><para>A combination of <option>--unmerge</option> and <option>--merge</option>: if already
+ <term><option>refresh</option></term>
+ <listitem><para>A combination of <option>unmerge</option> and <option>merge</option>: if already
mounted the existing <literal>overlayfs</literal> instance is unmounted temporarily, and then
replaced by a new version. This command is useful after installing/removing system extension images,
in order to update the <literal>overlayfs</literal> file system accordingly. If no system extensions
- are installed when this command is executed, the equivalent of <option>--unmerge</option> is
+ are installed when this command is executed, the equivalent of <option>unmerge</option> is
executed, without establishing any new <literal>overlayfs</literal> instance. Note that currently
there's a brief moment where neither the old nor the new <literal>overlayfs</literal> file system is
mounted. This implies that all resources supplied by a system extension will briefly disappear — even
@@ -184,8 +189,7 @@
</varlistentry>
<varlistentry>
- <term><option>--list</option></term>
- <term><option>-l</option></term>
+ <term><option>list</option></term>
<listitem><para>A brief list of installed extension images is shown.</para></listitem>
</varlistentry>
@@ -193,9 +197,6 @@
<xi:include href="standard-options.xml" xpointer="help" />
<xi:include href="standard-options.xml" xpointer="version" />
</variablelist>
-
- <para>When invoked without any command switches, the current merge status is shown, separately for both
- <filename>/usr/</filename> and <filename>/opt/</filename>.</para>
</refsect1>
<refsect1>
@@ -218,6 +219,15 @@
output style, or explicitly disabling JSON output.</para></listitem>
</varlistentry>
+ <varlistentry>
+ <term><option>--force</option></term>
+
+ <listitem><para>When merging system extensions into <filename>/usr/</filename> and
+ <filename>/opt/</filename>, ignore version incompatibilities, i.e. force merging regardless of
+ whether the version information included in the extension images matches the host or
+ not.</para></listitem>
+ </varlistentry>
+
<xi:include href="standard-options.xml" xpointer="no-pager" />
</variablelist>
</refsect1>
diff --git a/src/sysext/sysext.c b/src/sysext/sysext.c
index 8a8cd7535e..590dfc9c20 100644
--- a/src/sysext/sysext.c
+++ b/src/sysext/sysext.c
@@ -29,18 +29,13 @@
#include "stat-util.h"
#include "terminal-util.h"
#include "user-util.h"
+#include "verbs.h"
-static enum {
- ACTION_STATUS,
- ACTION_MERGE,
- ACTION_UNMERGE,
- ACTION_REFRESH,
- ACTION_LIST,
-} arg_action = ACTION_STATUS;
static char **arg_hierarchies = NULL; /* "/usr" + "/opt" by default */
static char *arg_root = NULL;
static JsonFormatFlags arg_json_format_flags = JSON_FORMAT_OFF;
static PagerFlags arg_pager_flags = 0;
+static bool arg_force = false;
STATIC_DESTRUCTOR_REGISTER(arg_hierarchies, strv_freep);
STATIC_DESTRUCTOR_REGISTER(arg_root, freep);
@@ -149,7 +144,15 @@ static int unmerge(void) {
return ret;
}
-static int status(void) {
+static int verb_unmerge(int argc, char **argv, void *userdata) {
+
+ if (!have_effective_cap(CAP_SYS_ADMIN))
+ return log_error_errno(SYNTHETIC_ERRNO(EPERM), "Need to be privileged.");
+
+ return unmerge();
+}
+
+static int verb_status(int argc, char **argv, void *userdata) {
_cleanup_(table_unrefp) Table *t = NULL;
int r, ret = 0;
char **p;
@@ -399,6 +402,76 @@ static int strverscmpp(char *const* a, char *const* b) {
return strverscmp(*a, *b);
}
+static int validate_version(
+ const char *root,
+ const char *name,
+ const char *host_os_release_id,
+ const char *host_os_release_version_id,
+ const char *host_os_release_sysext_level) {
+
+ _cleanup_free_ char *extension_release_id = NULL, *extension_release_version_id = NULL, *extension_release_sysext_level = NULL;
+ int r;
+
+ assert(root);
+ assert(name);
+
+ if (arg_force) {
+ log_debug("Force mode enabled, skipping version validation.");
+ return 1;
+ }
+
+ /* Insist that extension images do not overwrite the underlying OS release file (it's fine if
+ * they place one in /etc/os-release, i.e. where things don't matter, as they aren't
+ * merged.) */
+ r = chase_symlinks("/usr/lib/os-release", root, CHASE_PREFIX_ROOT, NULL, NULL);
+ if (r < 0) {
+ if (r != -ENOENT)
+ return log_error_errno(r, "Failed to determine whether /usr/lib/os-release exists in the extension image: %m");
+ } else
+ return log_error_errno(SYNTHETIC_ERRNO(EINVAL),
+ "Extension image contains /usr/lib/os-release file, which is not allowed (it may carry /etc/os-release), refusing.");
+
+ /* Now that we can look into the extension image, let's see if the OS version is compatible */
+ r = parse_extension_release(
+ root,
+ name,
+ "ID", &extension_release_id,
+ "VERSION_ID", &extension_release_version_id,
+ "SYSEXT_LEVEL", &extension_release_sysext_level,
+ NULL);
+ if (r == -ENOENT) {
+ log_notice_errno(r, "Extension '%s' carries no extension-release data, ignoring extension.", name);
+ return 0;
+ }
+ if (r < 0)
+ return log_error_errno(r, "Failed to acquire 'os-release' data of extension '%s': %m", name);
+
+ if (!streq_ptr(host_os_release_id, extension_release_id)) {
+ log_notice("Extension '%s' is for OS '%s', but running on '%s', ignoring extension.",
+ name, strna(extension_release_id), strna(host_os_release_id));
+ return 0;
+ }
+
+ /* If the extension has a sysext API level declared, then it must match the host API
+ * level. Otherwise, compare OS version as a whole */
+ if (extension_release_sysext_level) {
+ if (!streq_ptr(host_os_release_sysext_level, extension_release_sysext_level)) {
+ log_notice("Extension '%s' is for sysext API level '%s', but running on sysext API level '%s', ignoring extension.",
+ name, extension_release_sysext_level, strna(host_os_release_sysext_level));
+ return 0;
+ }
+ } else {
+ if (!streq_ptr(host_os_release_version_id, extension_release_version_id)) {
+ log_notice("Extension '%s' is for OS version '%s', but running on OS version '%s', ignoring extension.",
+ name, extension_release_version_id, strna(host_os_release_version_id));
+ return 0;
+ }
+ }
+
+ log_debug("Version info of extension '%s' matches host.", name);
+ return 1;
+}
+
static int merge_subprocess(Hashmap *images, const char *workspace) {
_cleanup_free_ char *host_os_release_id = NULL, *host_os_release_version_id = NULL, *host_os_release_sysext_level = NULL,
*buf = NULL;
@@ -440,8 +513,7 @@ static int merge_subprocess(Hashmap *images, const char *workspace) {
/* Let's now mount all images */
HASHMAP_FOREACH(img, images) {
- _cleanup_free_ char *p = NULL,
- *extension_release_id = NULL, *extension_release_version_id = NULL, *extension_release_sysext_level = NULL;
+ _cleanup_free_ char *p = NULL;
p = path_join(workspace, "extensions", img->name);
if (!p)
@@ -523,57 +595,17 @@ static int merge_subprocess(Hashmap *images, const char *workspace) {
assert_not_reached("Unsupported image type");
}
- /* Insist that extension images do not overwrite the underlying OS release file (it's fine if
- * they place one in /etc/os-release, i.e. where things don't matter, as they aren't
- * merged.) */
- r = chase_symlinks("/usr/lib/os-release", p, CHASE_PREFIX_ROOT, NULL, NULL);
- if (r < 0) {
- if (r != -ENOENT)
- return log_error_errno(r, "Failed to determine whether /usr/lib/os-release exists in the extension image: %m");
- } else
- return log_error_errno(SYNTHETIC_ERRNO(EINVAL),
- "Extension image contains /usr/lib/os-release file, which is not allowed (it may carry /etc/os-release), refusing.");
-
- /* Now that we can look into the extension image, let's see if the OS version is compatible */
- r = parse_extension_release(
+ r = validate_version(
p,
img->name,
- "ID", &extension_release_id,
- "VERSION_ID", &extension_release_version_id,
- "SYSEXT_LEVEL", &extension_release_sysext_level,
- NULL);
- if (r == -ENOENT) {
- log_notice_errno(r, "Extension '%s' carries no extension-release data, ignoring extension.", img->name);
+ host_os_release_id,
+ host_os_release_version_id,
+ host_os_release_sysext_level);
+ if (r < 0)
+ return r;
+ if (r == 0) {
n_ignored++;
continue;
- } else if (r < 0)
- return log_error_errno(r, "Failed to acquire 'os-release' data of extension '%s': %m", img->name);
- else {
- if (!streq_ptr(host_os_release_id, extension_release_id)) {
- log_notice("Extension '%s' is for OS '%s', but running on '%s', ignoring extension.",
- img->name, strna(extension_release_id), strna(host_os_release_id));
- n_ignored++;
- continue;
- }
-
- /* If the extension has a sysext API level declared, then it must match the host API level. Otherwise, compare OS version as a whole */
- if (extension_release_sysext_level) {
- if (!streq_ptr(host_os_release_sysext_level, extension_release_sysext_level)) {
- log_notice("Extension '%s' is for sysext API level '%s', but running on sysext API level '%s', ignoring extension.",
- img->name, extension_release_sysext_level, strna(host_os_release_sysext_level));
- n_ignored++;
- continue;
- }
- } else {
- if (!streq_ptr(host_os_release_version_id, extension_release_version_id)) {
- log_notice("Extension '%s' is for OS version '%s', but running on OS version '%s', ignoring extension.",
- img->name, extension_release_version_id, strna(host_os_release_version_id));
- n_ignored++;
- continue;
- }
- }
-
- log_debug("Version info of extension '%s' matches host.", img->name);
}
/* Noice! This one is an extension we want. */
@@ -711,7 +743,134 @@ static int merge(Hashmap *images) {
return r != 123; /* exit code 123 means: didn't do anything */
}
-static int help(void) {
+static int verb_merge(int argc, char **argv, void *userdata) {
+ _cleanup_(hashmap_freep) Hashmap *images = NULL;
+ char **p;
+ int r;
+
+ if (!have_effective_cap(CAP_SYS_ADMIN))
+ return log_error_errno(SYNTHETIC_ERRNO(EPERM), "Need to be privileged.");
+
+ images = hashmap_new(&image_hash_ops);
+ if (!images)
+ return log_oom();
+
+ r = image_discover(IMAGE_EXTENSION, arg_root, images);
+ if (r < 0)
+ return log_error_errno(r, "Failed to discover extension images: %m");
+
+ /* In merge mode fail if things are already merged. (In --refresh mode below we'll unmerge if we find
+ * things are already merged...) */
+ STRV_FOREACH(p, arg_hierarchies) {
+ _cleanup_free_ char *resolved = NULL;
+
+ r = chase_symlinks(*p, arg_root, CHASE_PREFIX_ROOT, &resolved, NULL);
+ if (r == -ENOENT) {
+ log_debug_errno(r, "Hierarchy '%s%s' does not exist, ignoring.", strempty(arg_root), *p);
+ continue;
+ }
+ if (r < 0)
+ return log_error_errno(r, "Failed to resolve path to hierarchy '%s%s': %m", strempty(arg_root), *p);
+
+ r = is_our_mount_point(resolved);
+ if (r < 0)
+ return r;
+ if (r > 0)
+ return log_error_errno(SYNTHETIC_ERRNO(EBUSY),
+ "Hierarchy '%s' is already merged.", *p);
+ }
+
+ return merge(images);
+}
+
+static int verb_refresh(int argc, char **argv, void *userdata) {
+ _cleanup_(hashmap_freep) Hashmap *images = NULL;
+ int r;
+
+ if (!have_effective_cap(CAP_SYS_ADMIN))
+ return log_error_errno(SYNTHETIC_ERRNO(EPERM), "Need to be privileged.");
+
+ images = hashmap_new(&image_hash_ops);
+ if (!images)
+ return log_oom();
+
+ r = image_discover(IMAGE_EXTENSION, arg_root, images);
+ if (r < 0)
+ return log_error_errno(r, "Failed to discover extension images: %m");
+
+ r = merge(images); /* Returns > 0 if it did something, i.e. a new overlayfs is mounted now. When it
+ * does so it implicitly unmounts any overlayfs placed there before. Returns == 0
+ * if it did nothing, i.e. no extension images found. In this case the old
+ * overlayfs remains in place if there was one. */
+ if (r < 0)
+ return r;
+ if (r == 0) /* No images found? Then unmerge. The goal of --refresh is after all that after having
+ * called there's a guarantee that the merge status matches the installed extensions. */
+ r = unmerge();
+
+ /* Net result here is that:
+ *
+ * 1. If an overlayfs was mounted before and no extensions exist anymore, we'll have unmerged things.
+ *
+ * 2. If an overlayfs was mounted before, and there are still extensions installed' we'll have
+ * unmerged and then merged things again.
+ *
+ * 3. If an overlayfs so far wasn't mounted, and there are extensions installed, we'll have it
+ * mounted now.
+ *
+ * 4. If there was no overlayfs mount so far, and no extensions installed, we implement a NOP.
+ */
+
+ return 0;
+}
+
+static int verb_list(int argc, char **argv, void *userdata) {
+ _cleanup_(hashmap_freep) Hashmap *images = NULL;
+ _cleanup_(table_unrefp) Table *t = NULL;
+ Image *img;
+ int r;
+
+ images = hashmap_new(&image_hash_ops);
+ if (!images)
+ return log_oom();
+
+ r = image_discover(IMAGE_EXTENSION, arg_root, images);
+ if (r < 0)
+ return log_error_errno(r, "Failed to discover extension images: %m");
+
+ if ((arg_json_format_flags & JSON_FORMAT_OFF) && hashmap_isempty(images)) {
+ log_info("No OS extensions found.");
+ return 0;
+ }
+
+ t = table_new("name", "type", "path", "time");
+ if (!t)
+ return log_oom();
+
+ HASHMAP_FOREACH(img, images) {
+ r = table_add_many(
+ t,
+ TABLE_STRING, img->name,
+ TABLE_STRING, image_type_to_string(img->type),
+ TABLE_PATH, img->path,
+ TABLE_TIMESTAMP, img->mtime != 0 ? img->mtime : img->crtime);
+ if (r < 0)
+ return table_log_add_error(r);
+ }
+
+ (void) table_set_sort(t, (size_t) 0, (size_t) -1);
+
+ if (arg_json_format_flags & (JSON_FORMAT_OFF|JSON_FORMAT_PRETTY|JSON_FORMAT_PRETTY_AUTO))
+ (void) pager_open(arg_pager_flags);
+
+ r = table_print_json(t, stdout, arg_json_format_flags);
+ if (r < 0)
+ return table_log_print_error(r);
+
+ return 0;
+}
+
+static int verb_help(int argc, char **argv, void *userdata) {
_cleanup_free_ char *link = NULL;
int r;
@@ -722,17 +881,19 @@ static int help(void) {
printf("%1$s [OPTIONS...] [DEVICE]\n"
"\n%5$sMerge extension images into /usr/ and /opt/ hierarchies.%6$s\n"
"\n%3$sCommands:%4$s\n"
+ " status Show current merge status (default)\n"
+ " merge Merge extensions into /usr/ and /opt/\n"
+ " unmerge Unmerge extensions from /usr/ and /opt/\n"
+ " refresh Unmerge/merge extensions again\n"
+ " list List installed extensions\n"
" -h --help Show this help\n"
" --version Show package version\n"
- " -m --merge Merge extensions into /usr/ and /opt/\n"
- " -u --unmerge Unmerge extensions from /usr/ and /opt/\n"
- " -R --refresh Unmerge/merge extensions again\n"
- " -l --list List all OS images\n"
"\n%3$sOptions:%4$s\n"
" --no-pager Do not pipe output into a pager\n"
" --root=PATH Operate relative to root path\n"
" --json=pretty|short|off\n"
" Generate JSON output\n"
+ " --force Ignore version incompatibilities\n"
"\nSee the %2$s for details.\n"
, program_invocation_short_name
, link
@@ -748,12 +909,9 @@ static int parse_argv(int argc, char *argv[]) {
enum {
ARG_VERSION = 0x100,
ARG_NO_PAGER,
- ARG_MERGE,
- ARG_UNMERGE,
- ARG_REFRESH,
- ARG_LIST,
ARG_ROOT,
ARG_JSON,
+ ARG_FORCE,
};
static const struct option options[] = {
@@ -761,11 +919,8 @@ static int parse_argv(int argc, char *argv[]) {
{ "version", no_argument, NULL, ARG_VERSION },
{ "no-pager", no_argument, NULL, ARG_NO_PAGER },
{ "root", required_argument, NULL, ARG_ROOT },
- { "merge", no_argument, NULL, 'm' },
- { "unmerge", no_argument, NULL, 'u' },
- { "refresh", no_argument, NULL, 'R' },
- { "list", no_argument, NULL, 'l' },
{ "json", required_argument, NULL, ARG_JSON },
+ { "force", no_argument, NULL, ARG_FORCE },
{}
};
@@ -774,12 +929,12 @@ static int parse_argv(int argc, char *argv[]) {
assert(argc >= 0);
assert(argv);
- while ((c = getopt_long(argc, argv, "hmuRl", options, NULL)) >= 0)
+ while ((c = getopt_long(argc, argv, "h", options, NULL)) >= 0)
switch (c) {
case 'h':
- return help();
+ return verb_help(argc, argv, NULL);
case ARG_VERSION:
return version();
@@ -788,22 +943,6 @@ static int parse_argv(int argc, char *argv[]) {
arg_pager_flags |= PAGER_DISABLE;
break;
- case 'm':
- arg_action = ACTION_MERGE;
- break;
-
- case 'u':
- arg_action = ACTION_UNMERGE;
- break;
-
- case 'R':
- arg_action = ACTION_REFRESH;
- break;
-
- case 'l':
- arg_action = ACTION_LIST;
- break;
-
case ARG_ROOT:
r = parse_path_argument_and_warn(optarg, false, &arg_root);
if (r < 0)
@@ -817,6 +956,10 @@ static int parse_argv(int argc, char *argv[]) {
break;
+ case ARG_FORCE:
+ arg_force = true;
+ break;
+
case '?':
return -EINVAL;
@@ -824,10 +967,6 @@ static int parse_argv(int argc, char *argv[]) {
assert_not_reached("Unhandled option");
}
- if (argc - optind > 0)
- return log_error_errno(SYNTHETIC_ERRNO(EINVAL),
- "Unexpected argument.");
-
return 1;
}
@@ -871,13 +1010,25 @@ static int parse_env(void) {
return 0;
}
+static int sysext_main(int argc, char *argv[]) {
+
+ static const Verb verbs[] = {
+ { "status", VERB_ANY, 1, VERB_DEFAULT, verb_status },
+ { "merge", VERB_ANY, 1, 0, verb_merge },
+ { "unmerge", VERB_ANY, 1, 0, verb_unmerge },
+ { "refresh", VERB_ANY, 1, 0, verb_refresh },
+ { "list", VERB_ANY, 1, 0, verb_list },
+ { "help", VERB_ANY, 1, 0, verb_help },
+ {}
+ };
+
+ return dispatch_verb(argc, argv, verbs, NULL);
+}
+
static int run(int argc, char *argv[]) {
- _cleanup_(hashmap_freep) Hashmap *images = NULL;
int r;
- log_show_color(true);
- log_parse_environment();
- log_open();
+ log_setup_cli();
r = parse_argv(argc, argv);
if (r <= 0)
@@ -893,126 +1044,7 @@ static int run(int argc, char *argv[]) {
return log_oom();
}
- /* Given that things deep down in the child process will fail, let's catch the no-privilege issue
- * early on */
- if (!IN_SET(arg_action, ACTION_STATUS, ACTION_LIST) && !have_effective_cap(CAP_SYS_ADMIN))
- return log_error_errno(SYNTHETIC_ERRNO(EPERM), "Need to be privileged.");
-
- if (arg_action == ACTION_STATUS)
- return status();
-
- if (arg_action == ACTION_UNMERGE)
- return unmerge();
-
- images = hashmap_new(&image_hash_ops);
- if (!images)
- return log_oom();
-
- r = image_discover(IMAGE_EXTENSION, arg_root, images);
- if (r < 0)
- return log_error_errno(r, "Failed to discover extension images: %m");
-
- switch (arg_action) {
-
- case ACTION_LIST: {
- _cleanup_(table_unrefp) Table *t = NULL;
- Image *img;
-
- if ((arg_json_format_flags & JSON_FORMAT_OFF) && hashmap_isempty(images)) {
- log_info("No OS extensions found.");
- return 0;
- }
-
- t = table_new("name", "type", "path", "time");
- if (!t)
- return log_oom();
-
- HASHMAP_FOREACH(img, images) {
- r = table_add_many(
- t,
- TABLE_STRING, img->name,
- TABLE_STRING, image_type_to_string(img->type),
- TABLE_PATH, img->path,
- TABLE_TIMESTAMP, img->mtime != 0 ? img->mtime : img->crtime);
- if (r < 0)
- return table_log_add_error(r);
- }
-
- (void) table_set_sort(t, (size_t) 0, (size_t) -1);
-
- if (arg_json_format_flags & (JSON_FORMAT_OFF|JSON_FORMAT_PRETTY|JSON_FORMAT_PRETTY_AUTO))
- (void) pager_open(arg_pager_flags);
-
- r = table_print_json(t, stdout, arg_json_format_flags);
- if (r < 0)
- return table_log_print_error(r);
-
- r = 0;
- break;
- }
-
- case ACTION_MERGE: {
- char **p;
-
- /* In merge mode fail if things are already merged. (In --refresh mode below we'll unmerge if
- * we find things are already merged...) */
- STRV_FOREACH(p, arg_hierarchies) {
- _cleanup_free_ char *resolved = NULL;
-
- r = chase_symlinks(*p, arg_root, CHASE_PREFIX_ROOT, &resolved, NULL);
- if (r == -ENOENT) {
- log_debug_errno(r, "Hierarchy '%s%s' does not exist, ignoring.", strempty(arg_root), *p);
- continue;
- }
- if (r < 0)
- return log_error_errno(r, "Failed to resolve path to hierarchy '%s%s': %m", strempty(arg_root), *p);
-
- r = is_our_mount_point(resolved);
- if (r < 0)
- return r;
- if (r > 0)
- return log_error_errno(SYNTHETIC_ERRNO(EBUSY),
- "Hierarchy '%s' is already merged.", *p);
- }
-
- r = merge(images);
- break;
- }
-
- case ACTION_REFRESH:
- r = merge(images); /* Returns > 0 if it did something, i.e. a new overlayfs is mounted
- * now. When it does so it implicitly unmounts any overlayfs placed there
- * before. Returns == 0 if it did nothing, i.e. no extension images
- * found. In this case the old overlayfs remains in place if there was
- * one. */
- if (r < 0)
- return r;
- if (r == 0) /* No images found? Then unmerge. The goal of --refresh is after all that after
- * having called there's a guarantee that the merge status matches the installed
- * extensions. */
- r = unmerge();
-
- /* Net result here is that:
- *
- * 1. If an overlayfs was mounted before and no extensions exist anymore, we'll have unmerged
- * things.
- *
- * 2. If an overlayfs was mounted before, and there are still extensions installed' we'll
- * have unmerged and then merged things again.
- *
- * 3. If an overlayfs so far wasn't mounted, and there are extensions installed, we'll have
- * it mounted now.
- *
- * 4. If there was no overlayfs mount so far, and no extensions installed, we implement a
- * NOP.
- */
- break;
-
- default:
- assert_not_reached("Uneexpected action");
- }
-
- return r;
+ return sysext_main(argc, argv);
}
DEFINE_MAIN_FUNCTION(run);
diff --git a/units/systemd-sysext.service b/units/systemd-sysext.service
index cc31f9ca9b..35b5edca1d 100644
--- a/units/systemd-sysext.service
+++ b/units/systemd-sysext.service
@@ -24,8 +24,8 @@ ConditionDirectoryNotEmpty=|/usr/lib/extensions
[Service]
Type=oneshot
RemainAfterExit=yes
-ExecStart=systemd-sysext --merge
-ExecStop=systemd-sysext --unmerge
+ExecStart=systemd-sysext merge
+ExecStop=systemd-sysext unmerge
[Install]
WantedBy=sysinit.target