mirror of
https://github.com/systemd/systemd.git
synced 2025-01-18 10:04:04 +03:00
Merge pull request #33092 from YHNdnzj/freezer-cleanup
UnitFreezer: several cleanups
This commit is contained in:
commit
bbddfe5249
@ -140,7 +140,9 @@ static const FreezerState freezer_state_finish_table[_FREEZER_STATE_MAX] = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
FreezerState freezer_state_finish(FreezerState state) {
|
FreezerState freezer_state_finish(FreezerState state) {
|
||||||
assert(state >= 0 && state < _FREEZER_STATE_MAX);
|
assert(state >= 0);
|
||||||
|
assert(state < _FREEZER_STATE_MAX);
|
||||||
|
|
||||||
return freezer_state_finish_table[state];
|
return freezer_state_finish_table[state];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4044,12 +4044,8 @@ static int unit_check_cgroup_events(Unit *u) {
|
|||||||
/* Disregard freezer state changes due to operations not initiated by us.
|
/* Disregard freezer state changes due to operations not initiated by us.
|
||||||
* See: https://github.com/systemd/systemd/pull/13512/files#r416469963 and
|
* See: https://github.com/systemd/systemd/pull/13512/files#r416469963 and
|
||||||
* https://github.com/systemd/systemd/pull/13512#issuecomment-573007207 */
|
* https://github.com/systemd/systemd/pull/13512#issuecomment-573007207 */
|
||||||
if (values[1] && IN_SET(u->freezer_state, FREEZER_FREEZING, FREEZER_FREEZING_BY_PARENT, FREEZER_THAWING)) {
|
if (values[1] && IN_SET(u->freezer_state, FREEZER_FREEZING, FREEZER_FREEZING_BY_PARENT, FREEZER_THAWING))
|
||||||
if (streq(values[1], "0"))
|
unit_freezer_complete(u, streq(values[1], "0") ? FREEZER_RUNNING : FREEZER_FROZEN);
|
||||||
unit_thawed(u);
|
|
||||||
else
|
|
||||||
unit_frozen(u);
|
|
||||||
}
|
|
||||||
|
|
||||||
free(values[0]);
|
free(values[0]);
|
||||||
free(values[1]);
|
free(values[1]);
|
||||||
@ -5106,35 +5102,35 @@ static int unit_cgroup_freezer_kernel_state(Unit *u, FreezerState *ret) {
|
|||||||
|
|
||||||
int unit_cgroup_freezer_action(Unit *u, FreezerAction action) {
|
int unit_cgroup_freezer_action(Unit *u, FreezerAction action) {
|
||||||
_cleanup_free_ char *path = NULL;
|
_cleanup_free_ char *path = NULL;
|
||||||
FreezerState target, current, next;
|
FreezerState current, next, objective;
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
assert(u);
|
assert(u);
|
||||||
assert(IN_SET(action, FREEZER_FREEZE, FREEZER_PARENT_FREEZE,
|
assert(action >= 0);
|
||||||
FREEZER_THAW, FREEZER_PARENT_THAW));
|
assert(action < _FREEZER_ACTION_MAX);
|
||||||
|
|
||||||
if (!cg_freezer_supported())
|
if (!cg_freezer_supported())
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
unit_next_freezer_state(u, action, &next, &target);
|
unit_next_freezer_state(u, action, &next, &objective);
|
||||||
|
|
||||||
CGroupRuntime *crt = unit_get_cgroup_runtime(u);
|
CGroupRuntime *crt = unit_get_cgroup_runtime(u);
|
||||||
if (!crt || !crt->cgroup_path) {
|
if (!crt || !crt->cgroup_path)
|
||||||
/* No realized cgroup = nothing to freeze */
|
/* No realized cgroup = nothing to freeze */
|
||||||
u->freezer_state = freezer_state_finish(next);
|
goto skip;
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
r = unit_cgroup_freezer_kernel_state(u, ¤t);
|
r = unit_cgroup_freezer_kernel_state(u, ¤t);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
if (current == target)
|
if (current == objective)
|
||||||
next = freezer_state_finish(next);
|
goto skip;
|
||||||
else if (IN_SET(next, FREEZER_FROZEN, FREEZER_FROZEN_BY_PARENT, FREEZER_RUNNING)) {
|
|
||||||
/* We're transitioning into a finished state, which implies that the cgroup's
|
if (next == freezer_state_finish(next)) {
|
||||||
* current state already matches the target and thus we'd return 0. But, reality
|
/* We're directly transitioning into a finished state, which in theory means that
|
||||||
* shows otherwise. This indicates that our freezer_state tracking has diverged
|
* the cgroup's current state already matches the objective and thus we'd return 0.
|
||||||
|
* But, reality shows otherwise (such case would have been handled by current == objective
|
||||||
|
* branch above). This indicates that our freezer_state tracking has diverged
|
||||||
* from the real state of the cgroup, which can happen if someone meddles with the
|
* from the real state of the cgroup, which can happen if someone meddles with the
|
||||||
* cgroup from underneath us. This really shouldn't happen during normal operation,
|
* cgroup from underneath us. This really shouldn't happen during normal operation,
|
||||||
* though. So, let's warn about it and fix up the state to be valid */
|
* though. So, let's warn about it and fix up the state to be valid */
|
||||||
@ -5148,22 +5144,24 @@ int unit_cgroup_freezer_action(Unit *u, FreezerAction action) {
|
|||||||
next = FREEZER_FREEZING_BY_PARENT;
|
next = FREEZER_FREEZING_BY_PARENT;
|
||||||
else if (next == FREEZER_RUNNING)
|
else if (next == FREEZER_RUNNING)
|
||||||
next = FREEZER_THAWING;
|
next = FREEZER_THAWING;
|
||||||
|
else
|
||||||
|
assert_not_reached();
|
||||||
}
|
}
|
||||||
|
|
||||||
r = cg_get_path(SYSTEMD_CGROUP_CONTROLLER, crt->cgroup_path, "cgroup.freeze", &path);
|
r = cg_get_path(SYSTEMD_CGROUP_CONTROLLER, crt->cgroup_path, "cgroup.freeze", &path);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
log_unit_debug(u, "Unit freezer state was %s, now %s.",
|
r = write_string_file(path, one_zero(objective == FREEZER_FROZEN), WRITE_STRING_FILE_DISABLE_BUFFER);
|
||||||
freezer_state_to_string(u->freezer_state),
|
|
||||||
freezer_state_to_string(next));
|
|
||||||
|
|
||||||
r = write_string_file(path, one_zero(target == FREEZER_FROZEN), WRITE_STRING_FILE_DISABLE_BUFFER);
|
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
u->freezer_state = next;
|
unit_set_freezer_state(u, next);
|
||||||
return target != current;
|
return 1; /* Wait for cgroup event before replying */
|
||||||
|
|
||||||
|
skip:
|
||||||
|
unit_set_freezer_state(u, freezer_state_finish(next));
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int unit_get_cpuset(Unit *u, CPUSet *cpus, const char *name) {
|
int unit_get_cpuset(Unit *u, CPUSet *cpus, const char *name) {
|
||||||
|
@ -60,7 +60,6 @@ typedef enum FreezerAction {
|
|||||||
FREEZER_PARENT_FREEZE,
|
FREEZER_PARENT_FREEZE,
|
||||||
FREEZER_THAW,
|
FREEZER_THAW,
|
||||||
FREEZER_PARENT_THAW,
|
FREEZER_PARENT_THAW,
|
||||||
|
|
||||||
_FREEZER_ACTION_MAX,
|
_FREEZER_ACTION_MAX,
|
||||||
_FREEZER_ACTION_INVALID = -EINVAL,
|
_FREEZER_ACTION_INVALID = -EINVAL,
|
||||||
} FreezerAction;
|
} FreezerAction;
|
||||||
@ -514,6 +513,7 @@ void unit_cgroup_catchup(Unit *u);
|
|||||||
bool unit_cgroup_delegate(Unit *u);
|
bool unit_cgroup_delegate(Unit *u);
|
||||||
|
|
||||||
int unit_get_cpuset(Unit *u, CPUSet *cpus, const char *name);
|
int unit_get_cpuset(Unit *u, CPUSet *cpus, const char *name);
|
||||||
|
|
||||||
int unit_cgroup_freezer_action(Unit *u, FreezerAction action);
|
int unit_cgroup_freezer_action(Unit *u, FreezerAction action);
|
||||||
|
|
||||||
const char* freezer_action_to_string(FreezerAction a) _const_;
|
const char* freezer_action_to_string(FreezerAction a) _const_;
|
||||||
|
@ -34,9 +34,9 @@
|
|||||||
static BUS_DEFINE_PROPERTY_GET_ENUM(property_get_collect_mode, collect_mode, CollectMode);
|
static BUS_DEFINE_PROPERTY_GET_ENUM(property_get_collect_mode, collect_mode, CollectMode);
|
||||||
static BUS_DEFINE_PROPERTY_GET_ENUM(property_get_load_state, unit_load_state, UnitLoadState);
|
static BUS_DEFINE_PROPERTY_GET_ENUM(property_get_load_state, unit_load_state, UnitLoadState);
|
||||||
static BUS_DEFINE_PROPERTY_GET_ENUM(property_get_job_mode, job_mode, JobMode);
|
static BUS_DEFINE_PROPERTY_GET_ENUM(property_get_job_mode, job_mode, JobMode);
|
||||||
|
static BUS_DEFINE_PROPERTY_GET_ENUM(property_get_freezer_state, freezer_state, FreezerState);
|
||||||
static BUS_DEFINE_PROPERTY_GET(property_get_description, "s", Unit, unit_description);
|
static BUS_DEFINE_PROPERTY_GET(property_get_description, "s", Unit, unit_description);
|
||||||
static BUS_DEFINE_PROPERTY_GET2(property_get_active_state, "s", Unit, unit_active_state, unit_active_state_to_string);
|
static BUS_DEFINE_PROPERTY_GET2(property_get_active_state, "s", Unit, unit_active_state, unit_active_state_to_string);
|
||||||
static BUS_DEFINE_PROPERTY_GET2(property_get_freezer_state, "s", Unit, unit_freezer_state, freezer_state_to_string);
|
|
||||||
static BUS_DEFINE_PROPERTY_GET(property_get_sub_state, "s", Unit, unit_sub_state_to_string);
|
static BUS_DEFINE_PROPERTY_GET(property_get_sub_state, "s", Unit, unit_sub_state_to_string);
|
||||||
static BUS_DEFINE_PROPERTY_GET2(property_get_unit_file_state, "s", Unit, unit_get_unit_file_state, unit_file_state_to_string);
|
static BUS_DEFINE_PROPERTY_GET2(property_get_unit_file_state, "s", Unit, unit_get_unit_file_state, unit_file_state_to_string);
|
||||||
static BUS_DEFINE_PROPERTY_GET(property_get_can_reload, "b", Unit, unit_can_reload);
|
static BUS_DEFINE_PROPERTY_GET(property_get_can_reload, "b", Unit, unit_can_reload);
|
||||||
@ -755,8 +755,8 @@ static int bus_unit_method_freezer_generic(sd_bus_message *message, void *userda
|
|||||||
return sd_bus_error_set(error, BUS_ERROR_UNIT_INACTIVE, "Unit is not active");
|
return sd_bus_error_set(error, BUS_ERROR_UNIT_INACTIVE, "Unit is not active");
|
||||||
if (r == -EALREADY)
|
if (r == -EALREADY)
|
||||||
return sd_bus_error_set(error, BUS_ERROR_UNIT_BUSY, "Previously requested freezer operation for unit is still in progress");
|
return sd_bus_error_set(error, BUS_ERROR_UNIT_BUSY, "Previously requested freezer operation for unit is still in progress");
|
||||||
if (r == -ECHILD)
|
if (r == -EDEADLK)
|
||||||
return sd_bus_error_set(error, SD_BUS_ERROR_FAILED, "Unit is frozen by a parent slice");
|
return sd_bus_error_set(error, BUS_ERROR_FROZEN_BY_PARENT, "Unit is frozen by a parent slice");
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
@ -864,7 +864,7 @@ const sd_bus_vtable bus_unit_vtable[] = {
|
|||||||
SD_BUS_PROPERTY("AccessSELinuxContext", "s", NULL, offsetof(Unit, access_selinux_context), SD_BUS_VTABLE_PROPERTY_CONST),
|
SD_BUS_PROPERTY("AccessSELinuxContext", "s", NULL, offsetof(Unit, access_selinux_context), SD_BUS_VTABLE_PROPERTY_CONST),
|
||||||
SD_BUS_PROPERTY("LoadState", "s", property_get_load_state, offsetof(Unit, load_state), SD_BUS_VTABLE_PROPERTY_CONST),
|
SD_BUS_PROPERTY("LoadState", "s", property_get_load_state, offsetof(Unit, load_state), SD_BUS_VTABLE_PROPERTY_CONST),
|
||||||
SD_BUS_PROPERTY("ActiveState", "s", property_get_active_state, 0, SD_BUS_VTABLE_PROPERTY_EMITS_CHANGE),
|
SD_BUS_PROPERTY("ActiveState", "s", property_get_active_state, 0, SD_BUS_VTABLE_PROPERTY_EMITS_CHANGE),
|
||||||
SD_BUS_PROPERTY("FreezerState", "s", property_get_freezer_state, 0, SD_BUS_VTABLE_PROPERTY_EMITS_CHANGE),
|
SD_BUS_PROPERTY("FreezerState", "s", property_get_freezer_state, offsetof(Unit, freezer_state), SD_BUS_VTABLE_PROPERTY_EMITS_CHANGE),
|
||||||
SD_BUS_PROPERTY("SubState", "s", property_get_sub_state, 0, SD_BUS_VTABLE_PROPERTY_EMITS_CHANGE),
|
SD_BUS_PROPERTY("SubState", "s", property_get_sub_state, 0, SD_BUS_VTABLE_PROPERTY_EMITS_CHANGE),
|
||||||
SD_BUS_PROPERTY("FragmentPath", "s", NULL, offsetof(Unit, fragment_path), SD_BUS_VTABLE_PROPERTY_CONST),
|
SD_BUS_PROPERTY("FragmentPath", "s", NULL, offsetof(Unit, fragment_path), SD_BUS_VTABLE_PROPERTY_CONST),
|
||||||
SD_BUS_PROPERTY("SourcePath", "s", NULL, offsetof(Unit, source_path), SD_BUS_VTABLE_PROPERTY_CONST),
|
SD_BUS_PROPERTY("SourcePath", "s", NULL, offsetof(Unit, source_path), SD_BUS_VTABLE_PROPERTY_CONST),
|
||||||
@ -1807,9 +1807,7 @@ int bus_unit_queue_job_one(
|
|||||||
type = JOB_TRY_RELOAD;
|
type = JOB_TRY_RELOAD;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (type == JOB_STOP &&
|
if (type == JOB_STOP && UNIT_IS_LOAD_ERROR(u->load_state) && unit_active_state(u) == UNIT_INACTIVE)
|
||||||
IN_SET(u->load_state, UNIT_NOT_FOUND, UNIT_ERROR, UNIT_BAD_SETTING) &&
|
|
||||||
unit_active_state(u) == UNIT_INACTIVE)
|
|
||||||
return sd_bus_error_setf(error, BUS_ERROR_NO_SUCH_UNIT, "Unit %s not loaded.", u->id);
|
return sd_bus_error_setf(error, BUS_ERROR_NO_SUCH_UNIT, "Unit %s not loaded.", u->id);
|
||||||
|
|
||||||
if ((type == JOB_START && u->refuse_manual_start) ||
|
if ((type == JOB_START && u->refuse_manual_start) ||
|
||||||
|
@ -339,32 +339,31 @@ static void slice_enumerate_perpetual(Manager *m) {
|
|||||||
(void) slice_make_perpetual(m, SPECIAL_SYSTEM_SLICE, NULL);
|
(void) slice_make_perpetual(m, SPECIAL_SYSTEM_SLICE, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool slice_can_freeze(Unit *s) {
|
static bool slice_can_freeze(const Unit *u) {
|
||||||
|
assert(u);
|
||||||
|
|
||||||
Unit *member;
|
Unit *member;
|
||||||
|
UNIT_FOREACH_DEPENDENCY(member, u, UNIT_ATOM_SLICE_OF)
|
||||||
assert(s);
|
|
||||||
|
|
||||||
UNIT_FOREACH_DEPENDENCY(member, s, UNIT_ATOM_SLICE_OF)
|
|
||||||
if (!unit_can_freeze(member))
|
if (!unit_can_freeze(member))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int slice_freezer_action(Unit *s, FreezerAction action) {
|
static int slice_freezer_action(Unit *s, FreezerAction action) {
|
||||||
FreezerAction child_action;
|
FreezerAction child_action;
|
||||||
Unit *member;
|
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
assert(s);
|
assert(s);
|
||||||
assert(IN_SET(action, FREEZER_FREEZE, FREEZER_PARENT_FREEZE,
|
assert(action >= 0);
|
||||||
FREEZER_THAW, FREEZER_PARENT_THAW));
|
assert(action < _FREEZER_ACTION_MAX);
|
||||||
|
|
||||||
if (action == FREEZER_FREEZE && !slice_can_freeze(s)) {
|
if (action == FREEZER_FREEZE && !slice_can_freeze(s)) {
|
||||||
/* We're intentionally only checking for FREEZER_FREEZE here and ignoring the
|
/* We're intentionally only checking for FREEZER_FREEZE here and ignoring the
|
||||||
* _BY_PARENT variant. If we're being frozen by parent, that means someone has
|
* _BY_PARENT variant. If we're being frozen by parent, that means someone has
|
||||||
* already checked if we can be frozen further up the call stack. No point to
|
* already checked if we can be frozen further up the call stack. No point to
|
||||||
* redo that work */
|
* redo that work */
|
||||||
log_unit_warning(s, "Requested freezer operation is not supported by all children of the slice");
|
log_unit_warning(s, "Requested freezer operation is not supported by all children of the slice.");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -375,12 +374,10 @@ static int slice_freezer_action(Unit *s, FreezerAction action) {
|
|||||||
else
|
else
|
||||||
child_action = action;
|
child_action = action;
|
||||||
|
|
||||||
UNIT_FOREACH_DEPENDENCY(member, s, UNIT_ATOM_SLICE_OF) {
|
Unit *member;
|
||||||
if (UNIT_VTABLE(member)->freezer_action)
|
UNIT_FOREACH_DEPENDENCY(member, s, UNIT_ATOM_SLICE_OF)
|
||||||
|
if (UNIT_VTABLE(member)->freezer_action) {
|
||||||
r = UNIT_VTABLE(member)->freezer_action(member, child_action);
|
r = UNIT_VTABLE(member)->freezer_action(member, child_action);
|
||||||
else
|
|
||||||
/* Only thawing will reach here, since freezing checks for a method in can_freeze */
|
|
||||||
r = 0;
|
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
@ -117,7 +117,8 @@ int unit_serialize_state(Unit *u, FILE *f, FDSet *fds, bool switching_root) {
|
|||||||
if (!sd_id128_is_null(u->invocation_id))
|
if (!sd_id128_is_null(u->invocation_id))
|
||||||
(void) serialize_item_format(f, "invocation-id", SD_ID128_FORMAT_STR, SD_ID128_FORMAT_VAL(u->invocation_id));
|
(void) serialize_item_format(f, "invocation-id", SD_ID128_FORMAT_STR, SD_ID128_FORMAT_VAL(u->invocation_id));
|
||||||
|
|
||||||
(void) serialize_item_format(f, "freezer-state", "%s", freezer_state_to_string(unit_freezer_state(u)));
|
(void) serialize_item(f, "freezer-state", freezer_state_to_string(u->freezer_state));
|
||||||
|
|
||||||
(void) serialize_markers(f, u->markers);
|
(void) serialize_markers(f, u->markers);
|
||||||
|
|
||||||
bus_track_serialize(u->bus_track, f, "ref");
|
bus_track_serialize(u->bus_track, f, "ref");
|
||||||
|
115
src/core/unit.c
115
src/core/unit.c
@ -864,12 +864,6 @@ Unit* unit_free(Unit *u) {
|
|||||||
return mfree(u);
|
return mfree(u);
|
||||||
}
|
}
|
||||||
|
|
||||||
FreezerState unit_freezer_state(Unit *u) {
|
|
||||||
assert(u);
|
|
||||||
|
|
||||||
return u->freezer_state;
|
|
||||||
}
|
|
||||||
|
|
||||||
UnitActiveState unit_active_state(Unit *u) {
|
UnitActiveState unit_active_state(Unit *u) {
|
||||||
assert(u);
|
assert(u);
|
||||||
|
|
||||||
@ -6162,79 +6156,90 @@ bool unit_can_isolate_refuse_manual(Unit *u) {
|
|||||||
return unit_can_isolate(u) && !u->refuse_manual_start;
|
return unit_can_isolate(u) && !u->refuse_manual_start;
|
||||||
}
|
}
|
||||||
|
|
||||||
void unit_next_freezer_state(Unit *u, FreezerAction action, FreezerState *ret, FreezerState *ret_target) {
|
void unit_next_freezer_state(Unit *u, FreezerAction action, FreezerState *ret_next, FreezerState *ret_objective) {
|
||||||
Unit *slice;
|
FreezerState current, parent, next, objective;
|
||||||
FreezerState curr, parent, next, tgt;
|
|
||||||
|
|
||||||
assert(u);
|
assert(u);
|
||||||
assert(IN_SET(action, FREEZER_FREEZE, FREEZER_PARENT_FREEZE,
|
assert(action >= 0);
|
||||||
FREEZER_THAW, FREEZER_PARENT_THAW));
|
assert(action < _FREEZER_ACTION_MAX);
|
||||||
assert(ret);
|
assert(ret_next);
|
||||||
assert(ret_target);
|
assert(ret_objective);
|
||||||
|
|
||||||
/* This function determines the correct freezer state transitions for a unit
|
/* This function determines the correct freezer state transitions for a unit
|
||||||
* given the action being requested. It returns the next state, and also the "target",
|
* given the action being requested. It returns the next state, and also the "objective",
|
||||||
* which is either FREEZER_FROZEN or FREEZER_RUNNING, depending on what actual state we
|
* which is either FREEZER_FROZEN or FREEZER_RUNNING, depending on what actual state we
|
||||||
* ultimately want to achieve. */
|
* ultimately want to achieve. */
|
||||||
|
|
||||||
curr = u->freezer_state;
|
current = u->freezer_state;
|
||||||
slice = UNIT_GET_SLICE(u);
|
|
||||||
|
Unit *slice = UNIT_GET_SLICE(u);
|
||||||
if (slice)
|
if (slice)
|
||||||
parent = slice->freezer_state;
|
parent = slice->freezer_state;
|
||||||
else
|
else
|
||||||
parent = FREEZER_RUNNING;
|
parent = FREEZER_RUNNING;
|
||||||
|
|
||||||
if (action == FREEZER_FREEZE) {
|
switch (action) {
|
||||||
|
|
||||||
|
case FREEZER_FREEZE:
|
||||||
/* We always "promote" a freeze initiated by parent into a normal freeze */
|
/* We always "promote" a freeze initiated by parent into a normal freeze */
|
||||||
if (IN_SET(curr, FREEZER_FROZEN, FREEZER_FROZEN_BY_PARENT))
|
if (IN_SET(current, FREEZER_FROZEN, FREEZER_FROZEN_BY_PARENT))
|
||||||
next = FREEZER_FROZEN;
|
next = FREEZER_FROZEN;
|
||||||
else
|
else
|
||||||
next = FREEZER_FREEZING;
|
next = FREEZER_FREEZING;
|
||||||
} else if (action == FREEZER_THAW) {
|
break;
|
||||||
|
|
||||||
|
case FREEZER_THAW:
|
||||||
/* Thawing is the most complicated operation here, because we can't thaw a unit
|
/* Thawing is the most complicated operation here, because we can't thaw a unit
|
||||||
* if its parent is frozen. So we instead "demote" a normal freeze into a freeze
|
* if its parent is frozen. So we instead "demote" a normal freeze into a freeze
|
||||||
* initiated by parent if the parent is frozen */
|
* initiated by parent if the parent is frozen */
|
||||||
if (IN_SET(curr, FREEZER_RUNNING, FREEZER_THAWING, FREEZER_FREEZING_BY_PARENT, FREEZER_FROZEN_BY_PARENT))
|
if (IN_SET(current, FREEZER_RUNNING, FREEZER_THAWING,
|
||||||
next = curr;
|
FREEZER_FREEZING_BY_PARENT, FREEZER_FROZEN_BY_PARENT)) /* Should usually be refused by unit_freezer_action */
|
||||||
else if (curr == FREEZER_FREEZING) {
|
next = current;
|
||||||
|
else if (current == FREEZER_FREEZING) {
|
||||||
if (IN_SET(parent, FREEZER_RUNNING, FREEZER_THAWING))
|
if (IN_SET(parent, FREEZER_RUNNING, FREEZER_THAWING))
|
||||||
next = FREEZER_THAWING;
|
next = FREEZER_THAWING;
|
||||||
else
|
else
|
||||||
next = FREEZER_FREEZING_BY_PARENT;
|
next = FREEZER_FREEZING_BY_PARENT;
|
||||||
} else {
|
} else if (current == FREEZER_FROZEN) {
|
||||||
assert(curr == FREEZER_FROZEN);
|
|
||||||
if (IN_SET(parent, FREEZER_RUNNING, FREEZER_THAWING))
|
if (IN_SET(parent, FREEZER_RUNNING, FREEZER_THAWING))
|
||||||
next = FREEZER_THAWING;
|
next = FREEZER_THAWING;
|
||||||
else
|
else
|
||||||
next = FREEZER_FROZEN_BY_PARENT;
|
next = FREEZER_FROZEN_BY_PARENT;
|
||||||
}
|
} else
|
||||||
} else if (action == FREEZER_PARENT_FREEZE) {
|
assert_not_reached();
|
||||||
|
break;
|
||||||
|
|
||||||
|
case FREEZER_PARENT_FREEZE:
|
||||||
/* We need to avoid accidentally demoting units frozen manually */
|
/* We need to avoid accidentally demoting units frozen manually */
|
||||||
if (IN_SET(curr, FREEZER_FREEZING, FREEZER_FROZEN, FREEZER_FROZEN_BY_PARENT))
|
if (IN_SET(current, FREEZER_FREEZING, FREEZER_FROZEN, FREEZER_FROZEN_BY_PARENT))
|
||||||
next = curr;
|
next = current;
|
||||||
else
|
else
|
||||||
next = FREEZER_FREEZING_BY_PARENT;
|
next = FREEZER_FREEZING_BY_PARENT;
|
||||||
} else {
|
break;
|
||||||
assert(action == FREEZER_PARENT_THAW);
|
|
||||||
|
|
||||||
|
case FREEZER_PARENT_THAW:
|
||||||
/* We don't want to thaw units from a parent if they were frozen
|
/* We don't want to thaw units from a parent if they were frozen
|
||||||
* manually, so for such units this action is a no-op */
|
* manually, so for such units this action is a no-op */
|
||||||
if (IN_SET(curr, FREEZER_RUNNING, FREEZER_FREEZING, FREEZER_FROZEN))
|
if (IN_SET(current, FREEZER_RUNNING, FREEZER_FREEZING, FREEZER_FROZEN))
|
||||||
next = curr;
|
next = current;
|
||||||
else
|
else
|
||||||
next = FREEZER_THAWING;
|
next = FREEZER_THAWING;
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
assert_not_reached();
|
||||||
}
|
}
|
||||||
|
|
||||||
tgt = freezer_state_finish(next);
|
objective = freezer_state_finish(next);
|
||||||
if (tgt == FREEZER_FROZEN_BY_PARENT)
|
if (objective == FREEZER_FROZEN_BY_PARENT)
|
||||||
tgt = FREEZER_FROZEN;
|
objective = FREEZER_FROZEN;
|
||||||
assert(IN_SET(tgt, FREEZER_RUNNING, FREEZER_FROZEN));
|
assert(IN_SET(objective, FREEZER_RUNNING, FREEZER_FROZEN));
|
||||||
|
|
||||||
*ret = next;
|
*ret_next = next;
|
||||||
*ret_target = tgt;
|
*ret_objective = objective;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool unit_can_freeze(Unit *u) {
|
bool unit_can_freeze(const Unit *u) {
|
||||||
assert(u);
|
assert(u);
|
||||||
|
|
||||||
if (unit_has_name(u, SPECIAL_ROOT_SLICE) || unit_has_name(u, SPECIAL_INIT_SCOPE))
|
if (unit_has_name(u, SPECIAL_ROOT_SLICE) || unit_has_name(u, SPECIAL_INIT_SCOPE))
|
||||||
@ -6246,26 +6251,36 @@ bool unit_can_freeze(Unit *u) {
|
|||||||
return UNIT_VTABLE(u)->freezer_action;
|
return UNIT_VTABLE(u)->freezer_action;
|
||||||
}
|
}
|
||||||
|
|
||||||
void unit_frozen(Unit *u) {
|
void unit_set_freezer_state(Unit *u, FreezerState state) {
|
||||||
assert(u);
|
assert(u);
|
||||||
|
assert(state >= 0);
|
||||||
|
assert(state < _FREEZER_STATE_MAX);
|
||||||
|
|
||||||
u->freezer_state = u->freezer_state == FREEZER_FREEZING_BY_PARENT
|
if (u->freezer_state == state)
|
||||||
? FREEZER_FROZEN_BY_PARENT
|
return;
|
||||||
: FREEZER_FROZEN;
|
|
||||||
|
|
||||||
log_unit_debug(u, "Unit now %s.", freezer_state_to_string(u->freezer_state));
|
log_unit_debug(u, "Freezer state changed %s -> %s",
|
||||||
|
freezer_state_to_string(u->freezer_state), freezer_state_to_string(state));
|
||||||
|
|
||||||
bus_unit_send_pending_freezer_message(u, false);
|
u->freezer_state = state;
|
||||||
|
|
||||||
|
unit_add_to_dbus_queue(u);
|
||||||
}
|
}
|
||||||
|
|
||||||
void unit_thawed(Unit *u) {
|
void unit_freezer_complete(Unit *u, FreezerState kernel_state) {
|
||||||
|
bool expected;
|
||||||
|
|
||||||
assert(u);
|
assert(u);
|
||||||
|
assert(IN_SET(kernel_state, FREEZER_RUNNING, FREEZER_FROZEN));
|
||||||
|
|
||||||
u->freezer_state = FREEZER_RUNNING;
|
expected = IN_SET(u->freezer_state, FREEZER_RUNNING, FREEZER_THAWING) == (kernel_state == FREEZER_RUNNING);
|
||||||
|
|
||||||
log_unit_debug(u, "Unit thawed.");
|
unit_set_freezer_state(u, expected ? freezer_state_finish(u->freezer_state) : kernel_state);
|
||||||
|
log_unit_info(u, "Unit now %s.", u->freezer_state == FREEZER_RUNNING ? "thawed" :
|
||||||
|
freezer_state_to_string(u->freezer_state));
|
||||||
|
|
||||||
bus_unit_send_pending_freezer_message(u, false);
|
/* If the cgroup's final state is against what's requested by us, report as canceled. */
|
||||||
|
bus_unit_send_pending_freezer_message(u, /* canceled = */ !expected);
|
||||||
}
|
}
|
||||||
|
|
||||||
int unit_freezer_action(Unit *u, FreezerAction action) {
|
int unit_freezer_action(Unit *u, FreezerAction action) {
|
||||||
@ -6293,7 +6308,7 @@ int unit_freezer_action(Unit *u, FreezerAction action) {
|
|||||||
if (action == FREEZER_THAW && u->freezer_state == FREEZER_THAWING)
|
if (action == FREEZER_THAW && u->freezer_state == FREEZER_THAWING)
|
||||||
return -EALREADY;
|
return -EALREADY;
|
||||||
if (action == FREEZER_THAW && IN_SET(u->freezer_state, FREEZER_FREEZING_BY_PARENT, FREEZER_FROZEN_BY_PARENT))
|
if (action == FREEZER_THAW && IN_SET(u->freezer_state, FREEZER_FREEZING_BY_PARENT, FREEZER_FROZEN_BY_PARENT))
|
||||||
return -ECHILD;
|
return -EDEADLK;
|
||||||
|
|
||||||
r = UNIT_VTABLE(u)->freezer_action(u, action);
|
r = UNIT_VTABLE(u)->freezer_action(u, action);
|
||||||
if (r <= 0)
|
if (r <= 0)
|
||||||
|
@ -577,7 +577,7 @@ typedef struct UnitVTable {
|
|||||||
/* Freeze or thaw the unit. Returns > 0 to indicate that the request will be handled asynchronously; unit_frozen
|
/* Freeze or thaw the unit. Returns > 0 to indicate that the request will be handled asynchronously; unit_frozen
|
||||||
* or unit_thawed should be called once the operation is done. Returns 0 if done successfully, or < 0 on error. */
|
* or unit_thawed should be called once the operation is done. Returns 0 if done successfully, or < 0 on error. */
|
||||||
int (*freezer_action)(Unit *u, FreezerAction a);
|
int (*freezer_action)(Unit *u, FreezerAction a);
|
||||||
bool (*can_freeze)(Unit *u);
|
bool (*can_freeze)(const Unit *u);
|
||||||
|
|
||||||
/* Return which kind of data can be cleaned */
|
/* Return which kind of data can be cleaned */
|
||||||
int (*can_clean)(Unit *u, ExecCleanMask *ret);
|
int (*can_clean)(Unit *u, ExecCleanMask *ret);
|
||||||
@ -849,7 +849,6 @@ const char* unit_status_string(Unit *u, char **combined);
|
|||||||
bool unit_has_name(const Unit *u, const char *name);
|
bool unit_has_name(const Unit *u, const char *name);
|
||||||
|
|
||||||
UnitActiveState unit_active_state(Unit *u);
|
UnitActiveState unit_active_state(Unit *u);
|
||||||
FreezerState unit_freezer_state(Unit *u);
|
|
||||||
|
|
||||||
const char* unit_sub_state_to_string(Unit *u);
|
const char* unit_sub_state_to_string(Unit *u);
|
||||||
|
|
||||||
@ -1037,11 +1036,11 @@ bool unit_can_start_refuse_manual(Unit *u);
|
|||||||
bool unit_can_stop_refuse_manual(Unit *u);
|
bool unit_can_stop_refuse_manual(Unit *u);
|
||||||
bool unit_can_isolate_refuse_manual(Unit *u);
|
bool unit_can_isolate_refuse_manual(Unit *u);
|
||||||
|
|
||||||
bool unit_can_freeze(Unit *u);
|
bool unit_can_freeze(const Unit *u);
|
||||||
int unit_freezer_action(Unit *u, FreezerAction action);
|
int unit_freezer_action(Unit *u, FreezerAction action);
|
||||||
void unit_next_freezer_state(Unit *u, FreezerAction a, FreezerState *ret, FreezerState *ret_tgt);
|
void unit_next_freezer_state(Unit *u, FreezerAction action, FreezerState *ret_next, FreezerState *ret_objective);
|
||||||
void unit_frozen(Unit *u);
|
void unit_set_freezer_state(Unit *u, FreezerState state);
|
||||||
void unit_thawed(Unit *u);
|
void unit_freezer_complete(Unit *u, FreezerState kernel_state);
|
||||||
|
|
||||||
Condition *unit_find_failed_condition(Unit *u);
|
Condition *unit_find_failed_condition(Unit *u);
|
||||||
|
|
||||||
|
@ -1870,7 +1870,7 @@ static int home_inspect(UserRecord *h, UserRecord **ret_home) {
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int user_session_freezer(uid_t uid, bool freeze_now, UnitFreezer **ret) {
|
static int user_session_freezer_new(uid_t uid, UnitFreezer **ret) {
|
||||||
_cleanup_free_ char *unit = NULL;
|
_cleanup_free_ char *unit = NULL;
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
@ -1881,10 +1881,6 @@ static int user_session_freezer(uid_t uid, bool freeze_now, UnitFreezer **ret) {
|
|||||||
if (r < 0 && r != -ENXIO)
|
if (r < 0 && r != -ENXIO)
|
||||||
log_warning_errno(r, "Cannot parse value of $SYSTEMD_HOME_LOCK_FREEZE_SESSION, ignoring: %m");
|
log_warning_errno(r, "Cannot parse value of $SYSTEMD_HOME_LOCK_FREEZE_SESSION, ignoring: %m");
|
||||||
else if (r == 0) {
|
else if (r == 0) {
|
||||||
if (freeze_now)
|
|
||||||
log_notice("Session remains unfrozen on explicit request ($SYSTEMD_HOME_LOCK_FREEZE_SESSION=0).\n"
|
|
||||||
"This is not recommended, and might result in unexpected behavior including data loss!");
|
|
||||||
|
|
||||||
*ret = NULL;
|
*ret = NULL;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -1892,9 +1888,6 @@ static int user_session_freezer(uid_t uid, bool freeze_now, UnitFreezer **ret) {
|
|||||||
if (asprintf(&unit, "user-" UID_FMT ".slice", uid) < 0)
|
if (asprintf(&unit, "user-" UID_FMT ".slice", uid) < 0)
|
||||||
return log_oom();
|
return log_oom();
|
||||||
|
|
||||||
if (freeze_now)
|
|
||||||
r = unit_freezer_new_freeze(unit, ret);
|
|
||||||
else
|
|
||||||
r = unit_freezer_new(unit, ret);
|
r = unit_freezer_new(unit, ret);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
@ -1921,9 +1914,16 @@ static int home_lock(UserRecord *h) {
|
|||||||
|
|
||||||
_cleanup_(unit_freezer_freep) UnitFreezer *f = NULL;
|
_cleanup_(unit_freezer_freep) UnitFreezer *f = NULL;
|
||||||
|
|
||||||
r = user_session_freezer(h->uid, /* freeze_now= */ true, &f);
|
r = user_session_freezer_new(h->uid, &f);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
if (r > 0) {
|
||||||
|
r = unit_freezer_freeze(f);
|
||||||
|
if (r < 0)
|
||||||
|
return r;
|
||||||
|
} else
|
||||||
|
log_notice("Session remains unfrozen on explicit request ($SYSTEMD_HOME_LOCK_FREEZE_SESSION=0).\n"
|
||||||
|
"This is not recommended, and might result in unexpected behavior including data loss!");
|
||||||
|
|
||||||
r = home_lock_luks(h, &setup);
|
r = home_lock_luks(h, &setup);
|
||||||
if (r < 0) {
|
if (r < 0) {
|
||||||
@ -1966,7 +1966,7 @@ static int home_unlock(UserRecord *h) {
|
|||||||
_cleanup_(unit_freezer_freep) UnitFreezer *f = NULL;
|
_cleanup_(unit_freezer_freep) UnitFreezer *f = NULL;
|
||||||
|
|
||||||
/* We want to thaw the session only after it's safe to access $HOME */
|
/* We want to thaw the session only after it's safe to access $HOME */
|
||||||
r = user_session_freezer(h->uid, /* freeze_now= */ false, &f);
|
r = user_session_freezer_new(h->uid, &f);
|
||||||
if (r > 0)
|
if (r > 0)
|
||||||
r = unit_freezer_thaw(f);
|
r = unit_freezer_thaw(f);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
|
@ -34,6 +34,7 @@ BUS_ERROR_MAP_ELF_REGISTER const sd_bus_error_map bus_common_errors[] = {
|
|||||||
SD_BUS_ERROR_MAP(BUS_ERROR_DISK_FULL, ENOSPC),
|
SD_BUS_ERROR_MAP(BUS_ERROR_DISK_FULL, ENOSPC),
|
||||||
SD_BUS_ERROR_MAP(BUS_ERROR_FILE_DESCRIPTOR_STORE_DISABLED,
|
SD_BUS_ERROR_MAP(BUS_ERROR_FILE_DESCRIPTOR_STORE_DISABLED,
|
||||||
EHOSTDOWN),
|
EHOSTDOWN),
|
||||||
|
SD_BUS_ERROR_MAP(BUS_ERROR_FROZEN_BY_PARENT, EDEADLK),
|
||||||
|
|
||||||
SD_BUS_ERROR_MAP(BUS_ERROR_NO_SUCH_MACHINE, ENXIO),
|
SD_BUS_ERROR_MAP(BUS_ERROR_NO_SUCH_MACHINE, ENXIO),
|
||||||
SD_BUS_ERROR_MAP(BUS_ERROR_NO_SUCH_IMAGE, ENOENT),
|
SD_BUS_ERROR_MAP(BUS_ERROR_NO_SUCH_IMAGE, ENOENT),
|
||||||
|
@ -34,6 +34,7 @@
|
|||||||
#define BUS_ERROR_FREEZE_CANCELLED "org.freedesktop.systemd1.FreezeCancelled"
|
#define BUS_ERROR_FREEZE_CANCELLED "org.freedesktop.systemd1.FreezeCancelled"
|
||||||
#define BUS_ERROR_FILE_DESCRIPTOR_STORE_DISABLED \
|
#define BUS_ERROR_FILE_DESCRIPTOR_STORE_DISABLED \
|
||||||
"org.freedesktop.systemd1.FileDescriptorStoreDisabled"
|
"org.freedesktop.systemd1.FileDescriptorStoreDisabled"
|
||||||
|
#define BUS_ERROR_FROZEN_BY_PARENT "org.freedesktop.systemd1.FrozenByParent"
|
||||||
|
|
||||||
#define BUS_ERROR_NO_SUCH_MACHINE "org.freedesktop.machine1.NoSuchMachine"
|
#define BUS_ERROR_NO_SUCH_MACHINE "org.freedesktop.machine1.NoSuchMachine"
|
||||||
#define BUS_ERROR_NO_SUCH_IMAGE "org.freedesktop.machine1.NoSuchImage"
|
#define BUS_ERROR_NO_SUCH_IMAGE "org.freedesktop.machine1.NoSuchImage"
|
||||||
|
@ -3025,22 +3025,3 @@ int unit_freezer_freeze(UnitFreezer *f) {
|
|||||||
int unit_freezer_thaw(UnitFreezer *f) {
|
int unit_freezer_thaw(UnitFreezer *f) {
|
||||||
return unit_freezer_action(f, false);
|
return unit_freezer_action(f, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
int unit_freezer_new_freeze(const char *name, UnitFreezer **ret) {
|
|
||||||
_cleanup_(unit_freezer_freep) UnitFreezer *f = NULL;
|
|
||||||
int r;
|
|
||||||
|
|
||||||
assert(name);
|
|
||||||
assert(ret);
|
|
||||||
|
|
||||||
r = unit_freezer_new(name, &f);
|
|
||||||
if (r < 0)
|
|
||||||
return r;
|
|
||||||
|
|
||||||
r = unit_freezer_freeze(f);
|
|
||||||
if (r < 0)
|
|
||||||
return r;
|
|
||||||
|
|
||||||
*ret = TAKE_PTR(f);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
@ -45,5 +45,3 @@ int unit_freezer_new(const char *name, UnitFreezer **ret);
|
|||||||
|
|
||||||
int unit_freezer_freeze(UnitFreezer *f);
|
int unit_freezer_freeze(UnitFreezer *f);
|
||||||
int unit_freezer_thaw(UnitFreezer *f);
|
int unit_freezer_thaw(UnitFreezer *f);
|
||||||
|
|
||||||
int unit_freezer_new_freeze(const char *name, UnitFreezer **ret);
|
|
||||||
|
@ -603,10 +603,14 @@ static int run(int argc, char *argv[]) {
|
|||||||
/* Freeze the user sessions */
|
/* Freeze the user sessions */
|
||||||
r = getenv_bool("SYSTEMD_SLEEP_FREEZE_USER_SESSIONS");
|
r = getenv_bool("SYSTEMD_SLEEP_FREEZE_USER_SESSIONS");
|
||||||
if (r < 0 && r != -ENXIO)
|
if (r < 0 && r != -ENXIO)
|
||||||
log_warning_errno(r, "Cannot parse value of $SYSTEMD_SLEEP_FREEZE_USER_SESSIONS, ignoring.");
|
log_warning_errno(r, "Cannot parse value of $SYSTEMD_SLEEP_FREEZE_USER_SESSIONS, ignoring: %m");
|
||||||
if (r != 0)
|
if (r != 0) {
|
||||||
(void) unit_freezer_new_freeze(SPECIAL_USER_SLICE, &user_slice_freezer);
|
r = unit_freezer_new(SPECIAL_USER_SLICE, &user_slice_freezer);
|
||||||
else
|
if (r < 0)
|
||||||
|
return r;
|
||||||
|
|
||||||
|
(void) unit_freezer_freeze(user_slice_freezer);
|
||||||
|
} else
|
||||||
log_notice("User sessions remain unfrozen on explicit request ($SYSTEMD_SLEEP_FREEZE_USER_SESSIONS=0).\n"
|
log_notice("User sessions remain unfrozen on explicit request ($SYSTEMD_SLEEP_FREEZE_USER_SESSIONS=0).\n"
|
||||||
"This is not recommended, and might result in unexpected behavior, particularly\n"
|
"This is not recommended, and might result in unexpected behavior, particularly\n"
|
||||||
"in suspend-then-hibernate operations or setups with encrypted home directories.");
|
"in suspend-then-hibernate operations or setups with encrypted home directories.");
|
||||||
@ -631,10 +635,13 @@ static int run(int argc, char *argv[]) {
|
|||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
case SLEEP_SUSPEND:
|
||||||
|
case SLEEP_HIBERNATE:
|
||||||
r = execute(sleep_config, arg_operation, NULL);
|
r = execute(sleep_config, arg_operation, NULL);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
assert_not_reached();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (user_slice_freezer)
|
if (user_slice_freezer)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user