udevd.c revision e598c5738c2dc85a3e93c3f68cd88e8eea51215b
/*
* Copyright (C) 2004-2008 Kay Sievers <kay.sievers@vrfy.org>
* Copyright (C) 2004 Chris Friesen <chris_friesen@sympatico.ca>
*
* 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 <string.h>
#include <ctype.h>
#include <dirent.h>
#include <fcntl.h>
#include <syslog.h>
#include <time.h>
#include <getopt.h>
#ifdef HAVE_INOTIFY
#endif
#include "udev.h"
#include "udev_rules.h"
#define UDEVD_PRIORITY -4
#define UDEV_PRIORITY -2
/* maximum limit of forked childs */
#define UDEVD_MAX_CHILDS 256
static int debug;
{
if (debug) {
} else {
}
}
struct udevd_uevent_msg {
int exitstatus;
char *action;
char *devpath;
char *subsystem;
char *driver;
unsigned long long seqnum;
char *devpath_old;
char *physdevpath;
unsigned int timeout;
char envbuf[];
};
static int debug_trace;
static struct udev_rules rules;
static int uevent_netlink_sock = -1;
static int inotify_fd = -1;
static volatile int sigchilds_waiting;
static volatile int udev_exit;
static volatile int reload_config;
static int run_exec_q;
static int stop_exec_q;
static int max_childs;
static char udev_log_env[32];
static LIST_HEAD(running_list);
{
exit(1);
}
{
int i;
int retval;
/* set signal handlers */
/* reset to default */
/* trigger timeout to prevent hanging processes */
/* reconstruct event environment from message */
return -1;
if (udevice->event_timeout >= 0)
/* run programs collected by RUN-key*/
return retval;
}
enum event_state {
};
{
char filename[UTIL_PATH_SIZE];
char filename_failed[UTIL_PATH_SIZE];
/* location of queue file */
snprintf(filename, sizeof(filename), "%s/.udev/queue/%llu", udev_get_dev_path(msg->udev), msg->seqnum);
/* location of failed file */
switch (state) {
case EVENT_QUEUED:
break;
case EVENT_FINISHED:
/* "move" event - rename failed file to current name, do not delete failed */
} else {
}
break;
case EVENT_FAILED:
/* move failed event to the failed directory */
/* clean up possibly empty queue directory */
break;
}
return;
}
{
/* mark as failed, if "add" event returns non-zero */
else
}
{
int retval;
switch (pid) {
case 0:
/* child */
if (inotify_fd >= 0)
logging_init("udevd-event");
if (retval)
exit(1);
exit(0);
case -1:
break;
default:
/* get SIGCHLD in main loop */
}
}
{
char filename[UTIL_PATH_SIZE];
int fd;
if (fd >= 0) {
char str[32];
int len;
}
/* run one event after the other in debug mode */
if (debug_trace) {
return;
}
/* run all events with a timeout set immediately */
return;
}
run_exec_q = 1;
}
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;
}
{
int i;
for (i = 0; i < UTIL_PATH_SIZE; i++) {
/* identical device event found */
return 1;
/* parent device event found */
return 2;
/* child device event found */
return 3;
/* no matching event */
break;
}
return 0;
}
/* lookup event for identical, parent, child, or physical device */
{
struct udevd_uevent_msg *loop_msg;
int childs_count = 0;
/* check exec-queue which may still contain delayed events we depend on */
/* skip ourself and all later events */
break;
/* check our old name */
return 2;
/* check identical, parent, or child device event */
return 3;
}
/* check for our major:minor number */
return 4;
}
/* check physical device event (special case of parent) */
return 5;
}
}
/* check run queue for still running events */
childs_count++;
if (childs_count++ >= limit) {
return 1;
}
/* check our old name */
return 2;
/* check identical, parent, or child device event */
return 3;
}
/* check for our major:minor number */
return 4;
}
/* check physical device event (special case of parent) */
return 5;
}
}
return 0;
}
/* serializes events for the identical and parent and child devices */
{
struct udevd_uevent_msg *loop_msg;
struct udevd_uevent_msg *tmp_msg;
if (list_empty(&exec_list))
return;
/* serialize and wait for parent or child events */
continue;
}
/* move event to run list */
}
}
static struct udevd_uevent_msg *get_msg_from_envbuf(struct udev *udev, const char *buf, int buf_size)
{
int bufpos;
int i;
struct udevd_uevent_msg *msg;
char *physdevdriver_key = NULL;
int maj = 0;
int min = 0;
return NULL;
/* copy environment buffer and reconstruct envp */
bufpos = 0;
int keylen;
char *key;
/* remember some keys for further processing */
}
/* for older kernels DRIVER is empty for a bus device, export PHYSDEVDRIVER as DRIVER */
}
return NULL;
}
return msg;
}
/* receive the udevd message from userspace */
{
struct udev_ctrl_msg *ctrl_msg;
const char *str;
int i;
return;
if (i >= 0) {
}
if (udev_ctrl_get_stop_exec_queue(ctrl_msg) > 0) {
stop_exec_q = 1;
}
if (udev_ctrl_get_start_exec_queue(ctrl_msg) > 0) {
stop_exec_q = 0;
}
if (udev_ctrl_get_reload_rules(ctrl_msg) > 0) {
reload_config = 1;
}
char *val;
val[0] = '\0';
if (val[0] == '\0') {
} else {
}
} else {
}
}
if (i >= 0) {
max_childs = i;
}
}
/* receive the kernel user event message and do some sanity checks */
{
struct udevd_uevent_msg *msg;
int bufpos;
char *pos;
if (size < 0) {
return NULL;
}
/* start of event payload */
return NULL;
/* validate message */
return NULL;
}
pos[0] = '\0';
return NULL;
}
return NULL;
}
return msg;
}
{
switch (signum) {
case SIGINT:
case SIGTERM:
udev_exit = 1;
break;
case SIGCHLD:
/* set flag, then write to pipe if needed */
sigchilds_waiting = 1;
break;
case SIGHUP:
reload_config = 1;
break;
}
/* write to pipe, which will wakeup select() in our mainloop */
}
{
/* find msg associated with pid and delete it */
struct udevd_uevent_msg *msg;
/* there may be events waiting with the same devpath */
run_exec_q = 1;
return;
}
}
}
static void reap_sigchilds(void)
{
int status;
while (1) {
if (pid <= 0)
break;
else if (WIFSIGNALED(status))
else
status = 0;
}
}
{
struct sockaddr_nl snl;
int retval;
if (uevent_netlink_sock == -1) {
return -1;
}
/* set receive buffersize */
if (retval < 0) {
uevent_netlink_sock = -1;
return -1;
}
return 0;
}
{
char filename[UTIL_PATH_SIZE];
int fd;
char seqnum[32];
if (fd >= 0) {
}
if (len <= 0) {
len = 3;
}
if (fd >= 0) {
}
}
{
int retval;
int fd;
const char *value;
int daemonize = 0;
{}
};
int rc = 1;
int maxfd;
goto exit;
logging_init("udevd");
while (1) {
int option;
if (option == -1)
break;
switch (option) {
case 'd':
daemonize = 1;
break;
case 't':
debug_trace = 1;
break;
case 'D':
debug = 1;
break;
case 'h':
printf("Usage: udevd [--help] [--daemon] [--debug-trace] [--debug] [--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 (fd > STDIN_FILENO)
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;
}
if (init_uevent_netlink_sock(udev) < 0) {
rc = 3;
goto exit;
}
if (retval < 0) {
goto exit;
}
if (retval < 0) {
goto exit;
}
if (retval < 0) {
goto exit;
}
if (retval < 0) {
goto exit;
}
if (retval < 0) {
goto exit;
}
/* parse the rules and keep them in memory */
sysfs_init();
if (daemonize) {
switch (pid) {
case 0:
break;
case -1:
rc = 4;
goto exit;
default:
rc = 0;
goto exit;
}
}
/* redirect std{out,err} */
if (!debug) {
}
if (fd > STDERR_FILENO)
/* set scheduling priority for the daemon */
chdir("/");
umask(022);
setsid();
/* OOM_DISABLE == -17 */
if (fd < 0)
else {
}
if (fd > 0) {
char path[UTIL_PATH_SIZE];
const char *depr_str = "<6>udev: deprecated sysfs layout "
"(CONFIG_SYSFS_DEPRECATED) is unsupported\n";
}
}
}
/* set signal handlers */
/* watch rules directory */
inotify_fd = inotify_init();
if (inotify_fd >= 0) {
} else {
/* watch dynamic rules directory */
}
else
/* maximum limit of forked childs */
if (value)
else {
int memsize = mem_size_mb();
if (memsize > 0)
else
}
/* clear environment for forked event processes */
clearenv();
/* export log_priority , as called programs may want to follow that setting */
if (debug_trace)
putenv("DEBUG=1");
while (!udev_exit) {
struct udevd_uevent_msg *msg;
int fdcount;
if (inotify_fd >= 0)
if (fdcount < 0) {
continue;
}
/* get control message */
/* get netlink message */
if (msg)
}
/* received a signal, clear our notification pipe */
char buf[256];
}
/* rules directory inotify watch */
int nbytes;
/* discard all possible events, we can just reload the config */
char *buf;
reload_config = 1;
inotify_fd = -1;
}
}
}
/* rules changed, set by inotify or a HUP signal */
if (reload_config) {
reload_config = 0;
}
/* forked child has returned */
if (sigchilds_waiting) {
sigchilds_waiting = 0;
}
if (run_exec_q) {
run_exec_q = 0;
if (!stop_exec_q)
}
}
rc = 0;
exit:
if (signal_pipe[READ_END] >= 0)
if (signal_pipe[WRITE_END] >= 0)
if (inotify_fd >= 0)
if (uevent_netlink_sock >= 0)
return rc;
}