1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
4 This file is part of systemd.
6 Copyright 2010 Lennart Poettering
8 systemd is free software; you can redistribute it and/or modify it
9 under the terms of the GNU Lesser General Public License as published by
10 the Free Software Foundation; either version 2.1 of the License, or
11 (at your option) any later version.
13 systemd is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 Lesser General Public License for more details.
18 You should have received a copy of the GNU Lesser General Public License
19 along with systemd; If not, see <http://www.gnu.org/licenses/>.
23 #include <sys/epoll.h>
24 #include <sys/inotify.h>
27 #include "bus-error.h"
29 #include "dbus-path.h"
33 #include "string-util.h"
34 #include "unit-name.h"
38 static const UnitActiveState state_translation_table
[_PATH_STATE_MAX
] = {
39 [PATH_DEAD
] = UNIT_INACTIVE
,
40 [PATH_WAITING
] = UNIT_ACTIVE
,
41 [PATH_RUNNING
] = UNIT_ACTIVE
,
42 [PATH_FAILED
] = UNIT_FAILED
45 static int path_dispatch_io(sd_event_source
*source
, int fd
, uint32_t revents
, void *userdata
);
47 int path_spec_watch(PathSpec
*s
, sd_event_io_handler_t handler
) {
49 static const int flags_table
[_PATH_TYPE_MAX
] = {
50 [PATH_EXISTS
] = IN_DELETE_SELF
|IN_MOVE_SELF
|IN_ATTRIB
,
51 [PATH_EXISTS_GLOB
] = IN_DELETE_SELF
|IN_MOVE_SELF
|IN_ATTRIB
,
52 [PATH_CHANGED
] = IN_DELETE_SELF
|IN_MOVE_SELF
|IN_ATTRIB
|IN_CLOSE_WRITE
|IN_CREATE
|IN_DELETE
|IN_MOVED_FROM
|IN_MOVED_TO
,
53 [PATH_MODIFIED
] = IN_DELETE_SELF
|IN_MOVE_SELF
|IN_ATTRIB
|IN_CLOSE_WRITE
|IN_CREATE
|IN_DELETE
|IN_MOVED_FROM
|IN_MOVED_TO
|IN_MODIFY
,
54 [PATH_DIRECTORY_NOT_EMPTY
] = IN_DELETE_SELF
|IN_MOVE_SELF
|IN_ATTRIB
|IN_CREATE
|IN_MOVED_TO
58 char *slash
, *oldslash
= NULL
;
67 s
->inotify_fd
= inotify_init1(IN_NONBLOCK
|IN_CLOEXEC
);
68 if (s
->inotify_fd
< 0) {
73 r
= sd_event_add_io(s
->unit
->manager
->event
, &s
->event_source
, s
->inotify_fd
, EPOLLIN
, handler
, s
);
77 (void) sd_event_source_set_description(s
->event_source
, "path");
79 /* This assumes the path was passed through path_kill_slashes()! */
81 for (slash
= strchr(s
->path
, '/'); ; slash
= strchr(slash
+1, '/')) {
87 cut
= slash
+ (slash
== s
->path
);
91 flags
= IN_MOVE_SELF
| IN_DELETE_SELF
| IN_ATTRIB
| IN_CREATE
| IN_MOVED_TO
;
93 flags
= flags_table
[s
->type
];
95 r
= inotify_add_watch(s
->inotify_fd
, s
->path
, flags
);
97 if (errno
== EACCES
|| errno
== ENOENT
) {
103 r
= log_warning_errno(errno
, "Failed to add watch on %s: %s", s
->path
, errno
== ENOSPC
? "too many watches" : strerror(-r
));
110 /* Path exists, we don't need to watch parent
113 char *cut2
= oldslash
+ (oldslash
== s
->path
);
117 inotify_add_watch(s
->inotify_fd
, s
->path
, IN_MOVE_SELF
);
118 /* Error is ignored, the worst can happen is
119 we get spurious events. */
131 /* whole path has been iterated over */
138 r
= log_error_errno(errno
, "Failed to add watch on any of the components of %s: %m", s
->path
);
139 /* either EACCESS or ENOENT */
146 path_spec_unwatch(s
);
150 void path_spec_unwatch(PathSpec
*s
) {
153 s
->event_source
= sd_event_source_unref(s
->event_source
);
154 s
->inotify_fd
= safe_close(s
->inotify_fd
);
157 int path_spec_fd_event(PathSpec
*s
, uint32_t revents
) {
158 union inotify_event_buffer buffer
;
159 struct inotify_event
*e
;
163 if (revents
!= EPOLLIN
) {
164 log_error("Got invalid poll event on inotify.");
168 l
= read(s
->inotify_fd
, &buffer
, sizeof(buffer
));
170 if (errno
== EAGAIN
|| errno
== EINTR
)
173 return log_error_errno(errno
, "Failed to read inotify event: %m");
176 FOREACH_INOTIFY_EVENT(e
, buffer
, l
) {
177 if ((s
->type
== PATH_CHANGED
|| s
->type
== PATH_MODIFIED
) &&
178 s
->primary_wd
== e
->wd
)
185 static bool path_spec_check_good(PathSpec
*s
, bool initial
) {
191 good
= access(s
->path
, F_OK
) >= 0;
194 case PATH_EXISTS_GLOB
:
195 good
= glob_exists(s
->path
) > 0;
198 case PATH_DIRECTORY_NOT_EMPTY
: {
201 k
= dir_is_empty(s
->path
);
202 good
= !(k
== -ENOENT
|| k
> 0);
207 case PATH_MODIFIED
: {
210 b
= access(s
->path
, F_OK
) >= 0;
211 good
= !initial
&& b
!= s
->previous_exists
;
212 s
->previous_exists
= b
;
223 static void path_spec_mkdir(PathSpec
*s
, mode_t mode
) {
226 if (s
->type
== PATH_EXISTS
|| s
->type
== PATH_EXISTS_GLOB
)
229 r
= mkdir_p_label(s
->path
, mode
);
231 log_warning_errno(r
, "mkdir(%s) failed: %m", s
->path
);
234 static void path_spec_dump(PathSpec
*s
, FILE *f
, const char *prefix
) {
238 path_type_to_string(s
->type
),
242 void path_spec_done(PathSpec
*s
) {
244 assert(s
->inotify_fd
== -1);
249 static void path_init(Unit
*u
) {
253 assert(u
->load_state
== UNIT_STUB
);
255 p
->directory_mode
= 0755;
258 void path_free_specs(Path
*p
) {
263 while ((s
= p
->specs
)) {
264 path_spec_unwatch(s
);
265 LIST_REMOVE(spec
, p
->specs
, s
);
271 static void path_done(Unit
*u
) {
279 static int path_add_mount_links(Path
*p
) {
285 LIST_FOREACH(spec
, s
, p
->specs
) {
286 r
= unit_require_mounts_for(UNIT(p
), s
->path
);
294 static int path_verify(Path
*p
) {
297 if (UNIT(p
)->load_state
!= UNIT_LOADED
)
301 log_unit_error(UNIT(p
), "Path unit lacks path setting. Refusing.");
308 static int path_add_default_dependencies(Path
*p
) {
313 r
= unit_add_dependency_by_name(UNIT(p
), UNIT_BEFORE
,
314 SPECIAL_PATHS_TARGET
, NULL
, true);
318 if (UNIT(p
)->manager
->running_as
== MANAGER_SYSTEM
) {
319 r
= unit_add_two_dependencies_by_name(UNIT(p
), UNIT_AFTER
, UNIT_REQUIRES
,
320 SPECIAL_SYSINIT_TARGET
, NULL
, true);
325 return unit_add_two_dependencies_by_name(UNIT(p
), UNIT_BEFORE
, UNIT_CONFLICTS
,
326 SPECIAL_SHUTDOWN_TARGET
, NULL
, true);
329 static int path_load(Unit
*u
) {
334 assert(u
->load_state
== UNIT_STUB
);
336 r
= unit_load_fragment_and_dropin(u
);
340 if (u
->load_state
== UNIT_LOADED
) {
342 if (set_isempty(u
->dependencies
[UNIT_TRIGGERS
])) {
345 r
= unit_load_related_unit(u
, ".service", &x
);
349 r
= unit_add_two_dependencies(u
, UNIT_BEFORE
, UNIT_TRIGGERS
, x
, true);
354 r
= path_add_mount_links(p
);
358 if (UNIT(p
)->default_dependencies
) {
359 r
= path_add_default_dependencies(p
);
365 return path_verify(p
);
368 static void path_dump(Unit
*u
, FILE *f
, const char *prefix
) {
376 trigger
= UNIT_TRIGGER(u
);
382 "%sMakeDirectory: %s\n"
383 "%sDirectoryMode: %04o\n",
384 prefix
, path_state_to_string(p
->state
),
385 prefix
, path_result_to_string(p
->result
),
386 prefix
, trigger
? trigger
->id
: "n/a",
387 prefix
, yes_no(p
->make_directory
),
388 prefix
, p
->directory_mode
);
390 LIST_FOREACH(spec
, s
, p
->specs
)
391 path_spec_dump(s
, f
, prefix
);
394 static void path_unwatch(Path
*p
) {
399 LIST_FOREACH(spec
, s
, p
->specs
)
400 path_spec_unwatch(s
);
403 static int path_watch(Path
*p
) {
409 LIST_FOREACH(spec
, s
, p
->specs
) {
410 r
= path_spec_watch(s
, path_dispatch_io
);
418 static void path_set_state(Path
*p
, PathState state
) {
422 old_state
= p
->state
;
425 if (state
!= PATH_WAITING
&&
426 (state
!= PATH_RUNNING
|| p
->inotify_triggered
))
429 if (state
!= old_state
)
430 log_unit_debug(UNIT(p
), "Changed %s -> %s", path_state_to_string(old_state
), path_state_to_string(state
));
432 unit_notify(UNIT(p
), state_translation_table
[old_state
], state_translation_table
[state
], true);
435 static void path_enter_waiting(Path
*p
, bool initial
, bool recheck
);
437 static int path_coldplug(Unit
*u
) {
441 assert(p
->state
== PATH_DEAD
);
443 if (p
->deserialized_state
!= p
->state
) {
445 if (p
->deserialized_state
== PATH_WAITING
||
446 p
->deserialized_state
== PATH_RUNNING
)
447 path_enter_waiting(p
, true, true);
449 path_set_state(p
, p
->deserialized_state
);
455 static void path_enter_dead(Path
*p
, PathResult f
) {
458 if (f
!= PATH_SUCCESS
)
461 path_set_state(p
, p
->result
!= PATH_SUCCESS
? PATH_FAILED
: PATH_DEAD
);
464 static void path_enter_running(Path
*p
) {
465 _cleanup_bus_error_free_ sd_bus_error error
= SD_BUS_ERROR_NULL
;
470 /* Don't start job if we are supposed to go down */
471 if (unit_stop_pending(UNIT(p
)))
474 r
= manager_add_job(UNIT(p
)->manager
, JOB_START
, UNIT_TRIGGER(UNIT(p
)),
475 JOB_REPLACE
, true, &error
, NULL
);
479 p
->inotify_triggered
= false;
485 path_set_state(p
, PATH_RUNNING
);
489 log_unit_warning(UNIT(p
), "Failed to queue unit startup job: %s", bus_error_message(&error
, r
));
490 path_enter_dead(p
, PATH_FAILURE_RESOURCES
);
493 static bool path_check_good(Path
*p
, bool initial
) {
499 LIST_FOREACH(spec
, s
, p
->specs
) {
500 good
= path_spec_check_good(s
, initial
);
509 static void path_enter_waiting(Path
*p
, bool initial
, bool recheck
) {
513 if (path_check_good(p
, initial
)) {
514 log_unit_debug(UNIT(p
), "Got triggered.");
515 path_enter_running(p
);
523 /* Hmm, so now we have created inotify watches, but the file
524 * might have appeared/been removed by now, so we must
528 if (path_check_good(p
, false)) {
529 log_unit_debug(UNIT(p
), "Got triggered.");
530 path_enter_running(p
);
534 path_set_state(p
, PATH_WAITING
);
538 log_unit_warning_errno(UNIT(p
), r
, "Failed to enter waiting state: %m");
539 path_enter_dead(p
, PATH_FAILURE_RESOURCES
);
542 static void path_mkdir(Path
*p
) {
547 if (!p
->make_directory
)
550 LIST_FOREACH(spec
, s
, p
->specs
)
551 path_spec_mkdir(s
, p
->directory_mode
);
554 static int path_start(Unit
*u
) {
558 assert(p
->state
== PATH_DEAD
|| p
->state
== PATH_FAILED
);
560 if (UNIT_TRIGGER(u
)->load_state
!= UNIT_LOADED
)
565 p
->result
= PATH_SUCCESS
;
566 path_enter_waiting(p
, true, true);
571 static int path_stop(Unit
*u
) {
575 assert(p
->state
== PATH_WAITING
|| p
->state
== PATH_RUNNING
);
577 path_enter_dead(p
, PATH_SUCCESS
);
581 static int path_serialize(Unit
*u
, FILE *f
, FDSet
*fds
) {
588 unit_serialize_item(u
, f
, "state", path_state_to_string(p
->state
));
589 unit_serialize_item(u
, f
, "result", path_result_to_string(p
->result
));
594 static int path_deserialize_item(Unit
*u
, const char *key
, const char *value
, FDSet
*fds
) {
602 if (streq(key
, "state")) {
605 state
= path_state_from_string(value
);
607 log_unit_debug(u
, "Failed to parse state value: %s", value
);
609 p
->deserialized_state
= state
;
611 } else if (streq(key
, "result")) {
614 f
= path_result_from_string(value
);
616 log_unit_debug(u
, "Failed to parse result value: %s", value
);
617 else if (f
!= PATH_SUCCESS
)
621 log_unit_debug(u
, "Unknown serialization key: %s", key
);
626 _pure_
static UnitActiveState
path_active_state(Unit
*u
) {
629 return state_translation_table
[PATH(u
)->state
];
632 _pure_
static const char *path_sub_state_to_string(Unit
*u
) {
635 return path_state_to_string(PATH(u
)->state
);
638 static int path_dispatch_io(sd_event_source
*source
, int fd
, uint32_t revents
, void *userdata
) {
639 PathSpec
*s
= userdata
;
649 if (p
->state
!= PATH_WAITING
&&
650 p
->state
!= PATH_RUNNING
)
653 /* log_debug("inotify wakeup on %s.", u->id); */
655 LIST_FOREACH(spec
, s
, p
->specs
)
656 if (path_spec_owns_inotify_fd(s
, fd
))
660 log_error("Got event on unknown fd.");
664 changed
= path_spec_fd_event(s
, revents
);
668 /* If we are already running, then remember that one event was
669 * dispatched so that we restart the service only if something
670 * actually changed on disk */
671 p
->inotify_triggered
= true;
674 path_enter_running(p
);
676 path_enter_waiting(p
, false, true);
681 path_enter_dead(p
, PATH_FAILURE_RESOURCES
);
685 static void path_trigger_notify(Unit
*u
, Unit
*other
) {
691 /* Invoked whenever the unit we trigger changes state or gains
694 if (other
->load_state
!= UNIT_LOADED
)
697 if (p
->state
== PATH_RUNNING
&&
698 UNIT_IS_INACTIVE_OR_FAILED(unit_active_state(other
))) {
699 log_unit_debug(UNIT(p
), "Got notified about unit deactivation.");
701 /* Hmm, so inotify was triggered since the
702 * last activation, so I guess we need to
703 * recheck what is going on. */
704 path_enter_waiting(p
, false, p
->inotify_triggered
);
708 static void path_reset_failed(Unit
*u
) {
713 if (p
->state
== PATH_FAILED
)
714 path_set_state(p
, PATH_DEAD
);
716 p
->result
= PATH_SUCCESS
;
719 static const char* const path_type_table
[_PATH_TYPE_MAX
] = {
720 [PATH_EXISTS
] = "PathExists",
721 [PATH_EXISTS_GLOB
] = "PathExistsGlob",
722 [PATH_DIRECTORY_NOT_EMPTY
] = "DirectoryNotEmpty",
723 [PATH_CHANGED
] = "PathChanged",
724 [PATH_MODIFIED
] = "PathModified",
727 DEFINE_STRING_TABLE_LOOKUP(path_type
, PathType
);
729 static const char* const path_result_table
[_PATH_RESULT_MAX
] = {
730 [PATH_SUCCESS
] = "success",
731 [PATH_FAILURE_RESOURCES
] = "resources",
734 DEFINE_STRING_TABLE_LOOKUP(path_result
, PathResult
);
736 const UnitVTable path_vtable
= {
737 .object_size
= sizeof(Path
),
748 .coldplug
= path_coldplug
,
755 .serialize
= path_serialize
,
756 .deserialize_item
= path_deserialize_item
,
758 .active_state
= path_active_state
,
759 .sub_state_to_string
= path_sub_state_to_string
,
761 .trigger_notify
= path_trigger_notify
,
763 .reset_failed
= path_reset_failed
,
765 .bus_vtable
= bus_path_vtable