284 lines
13 KiB
Diff
284 lines
13 KiB
Diff
From 4fb8f1e88322b94b0fa051d3c6fd19cac0227aaa Mon Sep 17 00:00:00 2001
|
|
From: Lennart Poettering <lennart@poettering.net>
|
|
Date: Wed, 29 Mar 2023 22:10:01 +0200
|
|
Subject: [PATCH] service: allow freeing the fdstore via cleaning
|
|
|
|
Now that we have a potentially pinned fdstore let's add a concept for
|
|
cleaning it explicitly on user requested. Let's expose this via
|
|
"systemctl clean", i.e. the same way as user directories are cleaned.
|
|
|
|
Conflict:Adaptation context.
|
|
Reference:https://github.com/systemd/systemd/commit/4fb8f1e88322b94b0fa051d3c6fd19cac0227aaa
|
|
|
|
---
|
|
man/systemctl.xml | 28 +++++++++++-------
|
|
src/core/dbus-unit.c | 21 ++++++-------
|
|
src/core/execute.c | 17 +++++++++++
|
|
src/core/execute.h | 5 +++-
|
|
src/core/service.c | 36 ++++++++++++++++++++---
|
|
src/core/timer.c | 1 +
|
|
src/systemctl/systemctl-clean-or-freeze.c | 2 +-
|
|
src/systemctl/systemctl.c | 3 +-
|
|
8 files changed, 86 insertions(+), 27 deletions(-)
|
|
|
|
diff --git a/man/systemctl.xml b/man/systemctl.xml
|
|
index 8402b95..a2cc54c 100644
|
|
--- a/man/systemctl.xml
|
|
+++ b/man/systemctl.xml
|
|
@@ -484,12 +484,16 @@ Jan 12 10:46:45 example.com bluetoothd[8900]: gatt-time-server: Input/output err
|
|
<varname>StateDirectory=</varname>, <varname>CacheDirectory=</varname>,
|
|
<varname>LogsDirectory=</varname> and <varname>RuntimeDirectory=</varname>, see
|
|
<citerefentry><refentrytitle>systemd.exec</refentrytitle><manvolnum>5</manvolnum></citerefentry>
|
|
- for details. For timer units this may be used to clear out the persistent timestamp data if
|
|
+ for details. It may also be used to clear the file decriptor store as enabled via
|
|
+ <varname>FileDescriptorStoreMax=</varname>, see
|
|
+ <citerefentry><refentrytitle>systemd.service</refentrytitle><manvolnum>5</manvolnum></citerefentry>
|
|
+ for details. For timer units this may be used to clear out the persistent timestamp data if
|
|
<varname>Persistent=</varname> is used and <option>--what=state</option> is selected, see
|
|
<citerefentry><refentrytitle>systemd.timer</refentrytitle><manvolnum>5</manvolnum></citerefentry>. This
|
|
command only applies to units that use either of these settings. If <option>--what=</option> is
|
|
- not specified, both the cache and runtime data are removed (as these two types of data are
|
|
- generally redundant and reproducible on the next invocation of the unit).</para>
|
|
+ not specified, the cache and runtime data as well as the file descriptor store are removed (as
|
|
+ these three types of resources are generally redundant and reproducible on the next invocation of
|
|
+ the unit). Note that the specified units must be stopped to invoke this operation.</para>
|
|
</listitem>
|
|
</varlistentry>
|
|
<varlistentry>
|
|
@@ -2076,13 +2080,17 @@ Jan 12 10:46:45 example.com bluetoothd[8900]: gatt-time-server: Input/output err
|
|
|
|
<listitem>
|
|
<para>Select what type of per-unit resources to remove when the <command>clean</command> command is
|
|
- invoked, see below. Takes one of <constant>configuration</constant>, <constant>state</constant>,
|
|
- <constant>cache</constant>, <constant>logs</constant>, <constant>runtime</constant> to select the
|
|
- type of resource. This option may be specified more than once, in which case all specified resource
|
|
- types are removed. Also accepts the special value <constant>all</constant> as a shortcut for
|
|
- specifying all five resource types. If this option is not specified defaults to the combination of
|
|
- <constant>cache</constant> and <constant>runtime</constant>, i.e. the two kinds of resources that
|
|
- are generally considered to be redundant and can be reconstructed on next invocation.</para>
|
|
+ invoked, see above. Takes one of <constant>configuration</constant>, <constant>state</constant>,
|
|
+ <constant>cache</constant>, <constant>logs</constant>, <constant>runtime</constant>,
|
|
+ <constant>fdstore</constant> to select the type of resource. This option may be specified more than
|
|
+ once, in which case all specified resource types are removed. Also accepts the special value
|
|
+ <constant>all</constant> as a shortcut for specifying all six resource types. If this option is not
|
|
+ specified defaults to the combination of <constant>cache</constant>, <constant>runtime</constant>
|
|
+ and <constant>fdstore</constant>, i.e. the three kinds of resources that are generally considered
|
|
+ to be redundant and can be reconstructed on next invocation. Note that the explicit removal of the
|
|
+ <constant>fdstore</constant> resource type is only useful if the
|
|
+ <varname>FileDescriptorStorePreserve=</varname> option is enabled, since the file descriptor store
|
|
+ is otherwise cleaned automatically when the unit is stopped.</para>
|
|
</listitem>
|
|
</varlistentry>
|
|
|
|
diff --git a/src/core/dbus-unit.c b/src/core/dbus-unit.c
|
|
index 24e4d25..eefb08e 100644
|
|
--- a/src/core/dbus-unit.c
|
|
+++ b/src/core/dbus-unit.c
|
|
@@ -91,6 +91,12 @@ static int property_get_can_clean(
|
|
return r;
|
|
}
|
|
|
|
+ if (FLAGS_SET(mask, EXEC_CLEAN_FDSTORE)) {
|
|
+ r = sd_bus_message_append(reply, "s", "fdstore");
|
|
+ if (r < 0)
|
|
+ return r;
|
|
+ }
|
|
+
|
|
return sd_bus_message_close_container(reply);
|
|
}
|
|
|
|
@@ -684,6 +690,7 @@ int bus_unit_method_clean(sd_bus_message *message, void *userdata, sd_bus_error
|
|
return r;
|
|
|
|
for (;;) {
|
|
+ ExecCleanMask m;
|
|
const char *i;
|
|
|
|
r = sd_bus_message_read(message, "s", &i);
|
|
@@ -692,17 +699,11 @@ int bus_unit_method_clean(sd_bus_message *message, void *userdata, sd_bus_error
|
|
if (r == 0)
|
|
break;
|
|
|
|
- if (streq(i, "all"))
|
|
- mask |= EXEC_CLEAN_ALL;
|
|
- else {
|
|
- ExecDirectoryType t;
|
|
-
|
|
- t = exec_resource_type_from_string(i);
|
|
- if (t < 0)
|
|
- return sd_bus_error_setf(error, SD_BUS_ERROR_INVALID_ARGS, "Invalid resource type: %s", i);
|
|
+ m = exec_clean_mask_from_string(i);
|
|
+ if (m < 0)
|
|
+ return sd_bus_error_setf(error, SD_BUS_ERROR_INVALID_ARGS, "Invalid resource type: %s", i);
|
|
|
|
- mask |= 1U << t;
|
|
- }
|
|
+ mask |= m;
|
|
}
|
|
|
|
r = sd_bus_message_exit_container(message);
|
|
diff --git a/src/core/execute.c b/src/core/execute.c
|
|
index 9185a6f..e4c9e94 100644
|
|
--- a/src/core/execute.c
|
|
+++ b/src/core/execute.c
|
|
@@ -6584,6 +6584,23 @@ ExecSetCredential *exec_set_credential_free(ExecSetCredential *sc) {
|
|
return mfree(sc);
|
|
}
|
|
|
|
+ExecCleanMask exec_clean_mask_from_string(const char *s) {
|
|
+ ExecDirectoryType t;
|
|
+
|
|
+ assert(s);
|
|
+
|
|
+ if (streq(s, "all"))
|
|
+ return EXEC_CLEAN_ALL;
|
|
+ if (streq(s, "fdstore"))
|
|
+ return EXEC_CLEAN_FDSTORE;
|
|
+
|
|
+ t = exec_resource_type_from_string(s);
|
|
+ if (t < 0)
|
|
+ return (ExecCleanMask) t;
|
|
+
|
|
+ return 1U << t;
|
|
+}
|
|
+
|
|
DEFINE_HASH_OPS_WITH_VALUE_DESTRUCTOR(exec_set_credential_hash_ops, char, string_hash_func, string_compare_func, ExecSetCredential, exec_set_credential_free);
|
|
|
|
static const char* const exec_input_table[_EXEC_INPUT_MAX] = {
|
|
diff --git a/src/core/execute.h b/src/core/execute.h
|
|
index f1f0ee4..fd83a8c 100644
|
|
--- a/src/core/execute.h
|
|
+++ b/src/core/execute.h
|
|
@@ -145,8 +145,9 @@ typedef enum ExecCleanMask {
|
|
EXEC_CLEAN_CACHE = 1U << EXEC_DIRECTORY_CACHE,
|
|
EXEC_CLEAN_LOGS = 1U << EXEC_DIRECTORY_LOGS,
|
|
EXEC_CLEAN_CONFIGURATION = 1U << EXEC_DIRECTORY_CONFIGURATION,
|
|
+ EXEC_CLEAN_FDSTORE = 1U << _EXEC_DIRECTORY_TYPE_MAX,
|
|
EXEC_CLEAN_NONE = 0,
|
|
- EXEC_CLEAN_ALL = (1U << _EXEC_DIRECTORY_TYPE_MAX) - 1,
|
|
+ EXEC_CLEAN_ALL = (1U << (_EXEC_DIRECTORY_TYPE_MAX+1)) - 1,
|
|
_EXEC_CLEAN_MASK_INVALID = -EINVAL,
|
|
} ExecCleanMask;
|
|
|
|
@@ -459,6 +460,8 @@ bool exec_context_get_cpu_affinity_from_numa(const ExecContext *c);
|
|
ExecSetCredential *exec_set_credential_free(ExecSetCredential *sc);
|
|
DEFINE_TRIVIAL_CLEANUP_FUNC(ExecSetCredential*, exec_set_credential_free);
|
|
|
|
+ExecCleanMask exec_clean_mask_from_string(const char *s);
|
|
+
|
|
extern const struct hash_ops exec_set_credential_hash_ops;
|
|
|
|
const char* exec_output_to_string(ExecOutput i) _const_;
|
|
diff --git a/src/core/service.c b/src/core/service.c
|
|
index 25aec40..d52ffe7 100644
|
|
--- a/src/core/service.c
|
|
+++ b/src/core/service.c
|
|
@@ -4500,22 +4500,39 @@ static int service_exit_status(Unit *u) {
|
|
|
|
static int service_clean(Unit *u, ExecCleanMask mask) {
|
|
_cleanup_strv_free_ char **l = NULL;
|
|
+ bool may_clean_fdstore = false;
|
|
Service *s = SERVICE(u);
|
|
int r;
|
|
|
|
assert(s);
|
|
assert(mask != 0);
|
|
|
|
- if (s->state != SERVICE_DEAD)
|
|
+ if (!IN_SET(s->state, SERVICE_DEAD, SERVICE_DEAD_RESOURCES_PINNED))
|
|
return -EBUSY;
|
|
|
|
+ /* Determine if there's anything we could potentially clean */
|
|
r = exec_context_get_clean_directories(&s->exec_context, u->manager->prefix, mask, &l);
|
|
if (r < 0)
|
|
return r;
|
|
|
|
- if (strv_isempty(l))
|
|
- return -EUNATCH;
|
|
+ if (mask & EXEC_CLEAN_FDSTORE)
|
|
+ may_clean_fdstore = s->n_fd_store > 0 || s->n_fd_store_max > 0;
|
|
+
|
|
+ if (strv_isempty(l) && !may_clean_fdstore)
|
|
+ return -EUNATCH; /* Nothing to potentially clean */
|
|
+
|
|
+ /* Let's clean the stuff we can clean quickly */
|
|
+ if (may_clean_fdstore)
|
|
+ service_release_fd_store(s);
|
|
+
|
|
+ /* If we are done, leave quickly */
|
|
+ if (strv_isempty(l)) {
|
|
+ if (s->state == SERVICE_DEAD_RESOURCES_PINNED && !s->fd_store)
|
|
+ service_set_state(s, SERVICE_DEAD);
|
|
+ return 0;
|
|
+ }
|
|
|
|
+ /* We need to clean disk stuff. This is slow, hence do it out of process, and change state */
|
|
service_unwatch_control_pid(s);
|
|
s->clean_result = SERVICE_SUCCESS;
|
|
s->control_command = NULL;
|
|
@@ -4542,10 +4559,21 @@ fail:
|
|
|
|
static int service_can_clean(Unit *u, ExecCleanMask *ret) {
|
|
Service *s = SERVICE(u);
|
|
+ ExecCleanMask mask = 0;
|
|
+ int r;
|
|
|
|
assert(s);
|
|
+ assert(ret);
|
|
+
|
|
+ r = exec_context_get_clean_mask(&s->exec_context, &mask);
|
|
+ if (r < 0)
|
|
+ return r;
|
|
|
|
- return exec_context_get_clean_mask(&s->exec_context, ret);
|
|
+ if (s->n_fd_store_max > 0)
|
|
+ mask |= EXEC_CLEAN_FDSTORE;
|
|
+
|
|
+ *ret = mask;
|
|
+ return 0;
|
|
}
|
|
|
|
static const char *service_finished_job(Unit *u, JobType t, JobResult result) {
|
|
diff --git a/src/core/timer.c b/src/core/timer.c
|
|
index 77c8ce4..b038004 100644
|
|
--- a/src/core/timer.c
|
|
+++ b/src/core/timer.c
|
|
@@ -891,6 +891,7 @@ static int timer_can_clean(Unit *u, ExecCleanMask *ret) {
|
|
Timer *t = TIMER(u);
|
|
|
|
assert(t);
|
|
+ assert(ret);
|
|
|
|
*ret = t->persistent ? EXEC_CLEAN_STATE : 0;
|
|
return 0;
|
|
diff --git a/src/systemctl/systemctl-clean-or-freeze.c b/src/systemctl/systemctl-clean-or-freeze.c
|
|
index eca3a6d..0fe9441 100644
|
|
--- a/src/systemctl/systemctl-clean-or-freeze.c
|
|
+++ b/src/systemctl/systemctl-clean-or-freeze.c
|
|
@@ -22,7 +22,7 @@ int clean_or_freeze_unit(int argc, char *argv[], void *userdata) {
|
|
polkit_agent_open_maybe();
|
|
|
|
if (!arg_clean_what) {
|
|
- arg_clean_what = strv_new("cache", "runtime");
|
|
+ arg_clean_what = strv_new("cache", "runtime", "fdstore");
|
|
if (!arg_clean_what)
|
|
return log_oom();
|
|
}
|
|
diff --git a/src/systemctl/systemctl.c b/src/systemctl/systemctl.c
|
|
index 2f6f581..887bb47 100644
|
|
--- a/src/systemctl/systemctl.c
|
|
+++ b/src/systemctl/systemctl.c
|
|
@@ -870,7 +870,8 @@ static int systemctl_parse_argv(int argc, char *argv[]) {
|
|
"state\n"
|
|
"cache\n"
|
|
"logs\n"
|
|
- "configuration");
|
|
+ "configuration\n"
|
|
+ "fdstore");
|
|
return 0;
|
|
}
|
|
|
|
--
|
|
2.33.0
|
|
|