1 /* SPDX-License-Identifier: LGPL-2.1+ */
3 #include "dbus-target.h"
8 #include "string-util.h"
10 #include "unit-name.h"
13 static const UnitActiveState state_translation_table
[_TARGET_STATE_MAX
] = {
14 [TARGET_DEAD
] = UNIT_INACTIVE
,
15 [TARGET_ACTIVE
] = UNIT_ACTIVE
18 static void target_set_state(Target
*t
, TargetState state
) {
19 TargetState old_state
;
22 if (t
->state
!= state
)
23 bus_unit_send_pending_change_signal(UNIT(t
), false);
28 if (state
!= old_state
)
29 log_debug("%s changed %s -> %s",
31 target_state_to_string(old_state
),
32 target_state_to_string(state
));
34 unit_notify(UNIT(t
), state_translation_table
[old_state
], state_translation_table
[state
], 0);
37 static int target_add_default_dependencies(Target
*t
) {
39 static const UnitDependency deps
[] = {
52 if (!UNIT(t
)->default_dependencies
)
55 /* Imply ordering for requirement dependencies on target units. Note that when the user created a contradicting
56 * ordering manually we won't add anything in here to make sure we don't create a loop. */
58 for (k
= 0; k
< ELEMENTSOF(deps
); k
++) {
63 HASHMAP_FOREACH_KEY(v
, other
, UNIT(t
)->dependencies
[deps
[k
]], i
) {
64 r
= unit_add_default_target_dependency(other
, UNIT(t
));
70 if (unit_has_name(UNIT(t
), SPECIAL_SHUTDOWN_TARGET
))
73 /* Make sure targets are unloaded on shutdown */
74 return unit_add_two_dependencies_by_name(UNIT(t
), UNIT_BEFORE
, UNIT_CONFLICTS
, SPECIAL_SHUTDOWN_TARGET
, true, UNIT_DEPENDENCY_DEFAULT
);
77 static int target_load(Unit
*u
) {
78 Target
*t
= TARGET(u
);
83 r
= unit_load_fragment_and_dropin(u
, true);
87 /* This is a new unit? Then let's add in some extras */
88 if (u
->load_state
== UNIT_LOADED
) {
89 r
= target_add_default_dependencies(t
);
97 static int target_coldplug(Unit
*u
) {
98 Target
*t
= TARGET(u
);
101 assert(t
->state
== TARGET_DEAD
);
103 if (t
->deserialized_state
!= t
->state
)
104 target_set_state(t
, t
->deserialized_state
);
109 static void target_dump(Unit
*u
, FILE *f
, const char *prefix
) {
110 Target
*t
= TARGET(u
);
116 "%sTarget State: %s\n",
117 prefix
, target_state_to_string(t
->state
));
120 static int target_start(Unit
*u
) {
121 Target
*t
= TARGET(u
);
125 assert(t
->state
== TARGET_DEAD
);
127 r
= unit_acquire_invocation_id(u
);
131 target_set_state(t
, TARGET_ACTIVE
);
135 static int target_stop(Unit
*u
) {
136 Target
*t
= TARGET(u
);
139 assert(t
->state
== TARGET_ACTIVE
);
141 target_set_state(t
, TARGET_DEAD
);
145 static int target_serialize(Unit
*u
, FILE *f
, FDSet
*fds
) {
146 Target
*s
= TARGET(u
);
152 (void) serialize_item(f
, "state", target_state_to_string(s
->state
));
156 static int target_deserialize_item(Unit
*u
, const char *key
, const char *value
, FDSet
*fds
) {
157 Target
*s
= TARGET(u
);
164 if (streq(key
, "state")) {
167 state
= target_state_from_string(value
);
169 log_debug("Failed to parse state value %s", value
);
171 s
->deserialized_state
= state
;
174 log_debug("Unknown serialization key '%s'", key
);
179 _pure_
static UnitActiveState
target_active_state(Unit
*u
) {
182 return state_translation_table
[TARGET(u
)->state
];
185 _pure_
static const char *target_sub_state_to_string(Unit
*u
) {
188 return target_state_to_string(TARGET(u
)->state
);
191 const UnitVTable target_vtable
= {
192 .object_size
= sizeof(Target
),
200 .coldplug
= target_coldplug
,
204 .start
= target_start
,
207 .serialize
= target_serialize
,
208 .deserialize_item
= target_deserialize_item
,
210 .active_state
= target_active_state
,
211 .sub_state_to_string
= target_sub_state_to_string
,
213 .bus_vtable
= bus_target_vtable
,
215 .status_message_formats
= {
216 .finished_start_job
= {
217 [JOB_DONE
] = "Reached target %s.",
219 .finished_stop_job
= {
220 [JOB_DONE
] = "Stopped target %s.",