udevd.c revision cb6268f41935a217f3694bf54f482d8770d8c3bc
/*
* Copyright (C) 2004-2009 Kay Sievers <kay.sievers@vrfy.org>
* Copyright (C) 2004 Chris Friesen <chris_friesen@sympatico.ca>
* Copyright (C) 2009 Canonical Ltd.
* Copyright (C) 2009 Scott James Remnant <scott@netsplit.com>
*
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include <stddef.h>
#include <signal.h>
#include <unistd.h>
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
#include <string.h>
#include <ctype.h>
#include <fcntl.h>
#include <time.h>
#include <getopt.h>
#include <dirent.h>
#include <sys/signalfd.h>
#include "udev.h"
#define UDEVD_PRIORITY -4
#define UDEV_PRIORITY -2
static bool debug;
{
if (debug) {
char buf[1024];
} else {
}
}
static bool debug_trace;
static struct udev_rules *rules;
static struct udev_queue_export *udev_queue_export;
static struct udev_monitor *monitor;
static int worker_watch[2];
static pid_t settle_pid;
static bool stop_exec_queue;
static bool reload_config;
static int max_childs;
static int childs;
static struct udev_list_node event_list;
static struct udev_list_node worker_list;
static bool udev_exit;
static volatile sig_atomic_t worker_exit;
enum poll_fd {
};
};
enum event_state {
};
struct event {
struct udev_list_node node;
struct udev_device *dev;
enum event_state state;
int exitcode;
unsigned long long int delaying_seqnum;
unsigned long long int seqnum;
const char *devpath;
const char *devpath_old;
};
{
char *event;
}
enum worker_state {
};
struct worker {
struct udev_list_node node;
int refcount;
struct udev_monitor *monitor;
enum worker_state state;
};
/* passed from worker to main process */
struct worker_message {
int exitcode;
};
{
char *worker;
}
{
/* mark as failed, if "add" event returns non-zero */
else
}
static void event_sig_handler(int signum)
{
switch (signum) {
case SIGALRM:
_exit(1);
break;
case SIGTERM:
worker_exit = true;
break;
}
}
{
return worker;
}
{
return;
childs--;
}
{
struct udev_monitor *worker_monitor;
/* listen for new events */
if (worker_monitor == NULL)
return;
/* allow the main daemon netlink address to send devices to the worker */
return;
/* worker + event reference */
switch (pid) {
case 0: {
struct udev_device *dev;
};
udev_log_init("udevd-work");
/* set signal handlers */
/* unblock SIGALRM */
/* SIGTERM is unblocked in ppoll() */
/* request TERM signal if parent exits */
/* initial device */
do {
struct udev_event *udev_event;
struct worker_message msg = {};
int err;
int failed = 0;
if (udev_event == NULL)
_exit(3);
/* set timeout to prevent hanging processes */
/* apply rules, create node, symlinks */
if (udev_device_get_event_timeout(dev) >= 0)
/* execute RUN= */
/* reset alarm */
alarm(0);
}
/* send processed event back to libudev listeners */
/* send back the result of the event execution */
if (err != 0)
else if (failed != 0)
/* wait for more device messages or signal from udevd */
while (!worker_exit) {
int fdcount;
if (fdcount < 0)
continue;
break;
}
}
exit(0);
}
case -1:
break;
default:
/* close monitor, but keep address around */
childs++;
break;
}
}
{
struct udev_list_node *loop;
continue;
if (count < 0) {
continue;
}
return;
}
if (childs >= max_childs) {
return;
}
/* start new worker and pass initial device */
}
{
return;
/* run all events with a timeout set immediately */
if (udev_device_get_timeout(dev) > 0) {
return;
}
}
{
struct udev_list_node *loop;
int max;
return;
if (max-- <= 0)
break;
continue;
}
}
static int mem_size_mb(void)
{
FILE *f;
char buf[4096];
long int memsize = -1;
if (f == NULL)
return -1;
long int value;
break;
}
}
fclose(f);
return memsize;
}
/* lookup event for identical, parent, child device */
{
struct udev_list_node *loop;
/* check if queue contains events we depend on */
/* we already found a later event, earlier can not block us, no need to check again */
continue;
/* event we checked earlier still exists, no need to check again */
return 2;
/* found ourself, no later event can block us */
break;
/* check our old name */
return 3;
}
/* compare devpath */
/* one devpath is contained in the other? */
continue;
/* identical device event found */
return 4;
}
/* parent device event found */
return 5;
}
/* child device event found */
return 6;
}
/* no matching device */
continue;
}
return 0;
}
{
struct udev_list_node *loop;
continue;
/* do not start event if parent or child event is still running */
if (devpath_busy(event) != 0) {
continue;
}
}
}
static void worker_returned(void)
{
while (1) {
struct worker_message msg;
struct udev_list_node *loop;
if (size != sizeof(struct worker_message))
break;
/* lookup worker who sent the signal */
continue;
/* worker returned */
break;
}
}
}
/* receive the udevd message from userspace */
{
struct udev_ctrl_msg *ctrl_msg;
const char *str;
int i;
return;
if (i >= 0) {
worker_kill(udev, 0);
}
if (udev_ctrl_get_stop_exec_queue(ctrl_msg) > 0) {
stop_exec_queue = true;
}
if (udev_ctrl_get_start_exec_queue(ctrl_msg) > 0) {
stop_exec_queue = false;
}
if (udev_ctrl_get_reload_rules(ctrl_msg) > 0) {
reload_config = true;
}
char *key;
char *val;
val[0] = '\0';
if (val[0] == '\0') {
} else {
}
} else {
}
}
worker_kill(udev, 0);
}
if (i >= 0) {
max_childs = i;
}
if (settle_pid > 0) {
settle_pid = 0;
}
}
/* read inotify messages */
{
char *buf;
struct inotify_event *ev;
return 0;
return -1;
}
struct udev_device *dev;
reload_config = true;
continue;
}
char filename[UTIL_PATH_SIZE];
int fd;
}
}
}
return 0;
}
{
switch (signo) {
case SIGINT:
case SIGTERM:
udev_exit = true;
break;
case SIGCHLD:
while (1) {
int status;
if (pid <= 0)
break;
continue;
/* drop reference from running event */
}
}
break;
}
}
break;
case SIGHUP:
reload_config = true;
break;
}
}
{
FILE *f;
char path[UTIL_PATH_SIZE];
if (f != NULL)
const char *depr_str =
"udev: missing sysfs features; please update the kernel "
"or disable the kernel's CONFIG_SYSFS_DEPRECATED option; "
"udev may fail to work correctly";
if (f != NULL)
sleep(15);
}
if (f != NULL)
fclose(f);
}
{
int fd;
const char *value;
int daemonize = false;
int resolve_names = 1;
{}
};
int rc = 1;
goto exit;
udev_log_init("udevd");
while (1) {
int option;
if (option == -1)
break;
switch (option) {
case 'd':
daemonize = true;
break;
case 't':
debug_trace = true;
break;
case 'D':
debug = true;
break;
case 'N':
resolve_names = 1;
resolve_names = 0;
resolve_names = -1;
} else {
goto exit;
}
break;
case 'h':
printf("Usage: udevd [--help] [--daemon] [--debug-trace] [--debug] "
"[--resolve-names=early|late|never] [--version]\n");
goto exit;
case 'V':
goto exit;
default:
goto exit;
}
}
if (getuid() != 0) {
goto exit;
}
/* make sure std{in,out,err} fd's are in a sane state */
if (fd < 0) {
}
if (write(STDOUT_FILENO, 0, 0) < 0)
if (write(STDERR_FILENO, 0, 0) < 0)
/* init control socket, bind() ensures, that only one udevd instance is running */
rc = 1;
goto exit;
}
if (udev_ctrl_enable_receiving(udev_ctrl) < 0) {
rc = 1;
goto exit;
}
rc = 3;
goto exit;
}
rc = 4;
goto exit;
}
} else {
char filename[UTIL_PATH_SIZE];
/* watch dynamic rules directory */
}
/* block and listen to all signals on signalfd */
sigfillset(&mask);
rc = 5;
goto exit;
}
/* unnamed socket from workers to the main daemon */
rc = 6;
goto exit;
}
goto exit;
}
if (udev_queue_export == NULL) {
goto exit;
}
if (daemonize) {
switch (pid) {
case 0:
break;
case -1:
rc = 4;
goto exit;
default:
rc = 0;
goto exit;
}
}
/* redirect std{out,err} */
if (!debug && !debug_trace) {
}
if (fd > STDERR_FILENO)
/* set scheduling priority for the daemon */
chdir("/");
umask(022);
setsid();
/* OOM_DISABLE == -17 */
if (fd < 0) {
} else {
}
/* in trace mode run one event after the other */
if (debug_trace) {
max_childs = 1;
} else {
int memsize = mem_size_mb();
if (memsize > 0)
else
max_childs = 128;
}
/* possibly overwrite maximum limit of executed events */
if (value)
while (!udev_exit) {
int fdcount;
int timeout;
/* set timeout to kill idle workers */
else
timeout = -1;
/* wait for events */
if (fdcount < 0)
continue;
/* timeout - kill idle workers */
if (fdcount == 0)
/* event has finished */
/* get kernel uevent */
struct udev_device *dev;
else
}
/* start new events */
/* get signal */
struct signalfd_siginfo fdsi;
if (size == sizeof(struct signalfd_siginfo))
}
/* device node and rules directory inotify watch */
/*
* get control message
*
* This needs to be after the inotify handling, to make sure,
* that the settle signal is send back after the possibly generated
* "change" events by the inotify device node watch.
*/
/* rules changed, set by inotify or a HUP signal */
if (reload_config) {
struct udev_rules *rules_new;
worker_kill(udev, 0);
}
reload_config = 0;
}
}
rc = 0;
exit:
if (worker_watch[READ_END] >= 0)
if (worker_watch[WRITE_END] >= 0)
return rc;
}