2 This file is part of systemd.
4 Copyright 2003-2004 Greg Kroah-Hartman <greg@kroah.com>
5 Copyright 2004-2012 Kay Sievers <kay@vrfy.org>
7 systemd is free software; you can redistribute it and/or modify it
8 under the terms of the GNU Lesser General Public License as published by
9 the Free Software Foundation; either version 2.1 of the License, or
10 (at your option) any later version.
12 systemd is distributed in the hope that it will be useful, but
13 WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 Lesser General Public License for more details.
17 You should have received a copy of the GNU Lesser General Public License
18 along with systemd; If not, see <http://www.gnu.org/licenses/>.
26 #include <sys/mount.h>
27 #include <sys/signalfd.h>
30 #include "selinux-util.h"
31 #include "signal-util.h"
33 #include "udev-util.h"
35 static int fake_filesystems(void) {
36 static const struct fakefs
{
41 { "test/sys", "/sys", "failed to mount test /sys" },
42 { "test/dev", "/dev", "failed to mount test /dev" },
43 { "test/run", "/run", "failed to mount test /run" },
44 { "test/run", "/etc/udev/rules.d", "failed to mount empty /etc/udev/rules.d" },
45 { "test/run", "/usr/lib/udev/rules.d", "failed to mount empty /usr/lib/udev/rules.d" },
50 err
= unshare(CLONE_NEWNS
);
53 fprintf(stderr
, "failed to call unshare(): %m\n");
57 if (mount(NULL
, "/", NULL
, MS_PRIVATE
|MS_REC
, NULL
) < 0) {
59 fprintf(stderr
, "failed to mount / as private: %m\n");
63 for (i
= 0; i
< ELEMENTSOF(fakefss
); i
++) {
64 err
= mount(fakefss
[i
].src
, fakefss
[i
].target
, NULL
, MS_BIND
, NULL
);
67 fprintf(stderr
, "%s %m", fakefss
[i
].error
);
75 int main(int argc
, char *argv
[]) {
76 _cleanup_udev_unref_
struct udev
*udev
= NULL
;
77 _cleanup_udev_event_unref_
struct udev_event
*event
= NULL
;
78 _cleanup_udev_device_unref_
struct udev_device
*dev
= NULL
;
79 _cleanup_udev_rules_unref_
struct udev_rules
*rules
= NULL
;
80 char syspath
[UTIL_PATH_SIZE
];
85 err
= fake_filesystems();
93 log_debug("version %s", VERSION
);
94 mac_selinux_init("/dev");
98 log_error("action missing");
103 if (devpath
== NULL
) {
104 log_error("devpath missing");
108 rules
= udev_rules_new(udev
, 1);
110 strscpyl(syspath
, sizeof(syspath
), "/sys", devpath
, NULL
);
111 dev
= udev_device_new_from_synthetic_event(udev
, syspath
, action
);
113 log_debug("unknown device '%s'", devpath
);
117 event
= udev_event_new(dev
);
119 assert_se(sigprocmask_many(SIG_BLOCK
, NULL
, SIGTERM
, SIGINT
, SIGHUP
, SIGCHLD
, -1) >= 0);
121 /* do what devtmpfs usually provides us */
122 if (udev_device_get_devnode(dev
) != NULL
) {
125 if (streq(udev_device_get_subsystem(dev
), "block"))
130 if (!streq(action
, "remove")) {
131 mkdir_parents_label(udev_device_get_devnode(dev
), 0755);
132 mknod(udev_device_get_devnode(dev
), mode
, udev_device_get_devnum(dev
));
134 unlink(udev_device_get_devnode(dev
));
135 rmdir_parents(udev_device_get_devnode(dev
), "/");
139 udev_event_execute_rules(event
,
140 3 * USEC_PER_SEC
, USEC_PER_SEC
,
143 udev_event_execute_run(event
,
144 3 * USEC_PER_SEC
, USEC_PER_SEC
);
146 mac_selinux_finish();
148 return err
? EXIT_FAILURE
: EXIT_SUCCESS
;