1
0
mirror of git://sourceware.org/git/lvm2.git synced 2025-01-17 06:04:23 +03:00

Add cmdline debug & version options to dmeventd.

Fix oom_adj handling.
This commit is contained in:
Alasdair Kergon 2007-01-19 17:22:17 +00:00
parent 866a1a0836
commit fcdf2e09a2
2 changed files with 113 additions and 41 deletions

View File

@ -1,5 +1,6 @@
Version 1.02.16 - Version 1.02.16 -
=================================== ===================================
Add cmdline debug & version options to dmeventd.
Add DM_LIB_VERSION definition to configure.h. Add DM_LIB_VERSION definition to configure.h.
Suppress 'Unrecognised field' error if report field is 'help'. Suppress 'Unrecognised field' error if report field is 'help'.
Add --separator and --sort to dmsetup (unused). Add --separator and --sort to dmsetup (unused).

View File

@ -39,7 +39,12 @@
#include <arpa/inet.h> /* for htonl, ntohl */ #include <arpa/inet.h> /* for htonl, ntohl */
#ifdef linux #ifdef linux
#include <malloc.h> # include <malloc.h>
/* From linux/oom.h */
# define OOM_DISABLE (-17)
# define OOM_ADJUST_MIN (-16)
#endif #endif
/* FIXME We use syslog for now, because multilog is not yet implemented */ /* FIXME We use syslog for now, because multilog is not yet implemented */
@ -47,6 +52,7 @@
static volatile sig_atomic_t _exit_now = 0; /* set to '1' when signal is given to exit */ static volatile sig_atomic_t _exit_now = 0; /* set to '1' when signal is given to exit */
static volatile sig_atomic_t _thread_registries_empty = 1; /* registries are empty initially */ static volatile sig_atomic_t _thread_registries_empty = 1; /* registries are empty initially */
static int _debug = 0;
/* List (un)link macros. */ /* List (un)link macros. */
#define LINK(x, head) list_add(head, &(x)->list) #define LINK(x, head) list_add(head, &(x)->list)
@ -58,9 +64,10 @@ static volatile sig_atomic_t _thread_registries_empty = 1; /* registries are emp
#define UNLINK_THREAD(x) UNLINK(x) #define UNLINK_THREAD(x) UNLINK(x)
#define DAEMON_NAME "dmeventd" #define DAEMON_NAME "dmeventd"
#define OOM_ADJ_FILE "/proc/self/oom_adj"
/* /*
Global mutex for thread list access. Has to be held when: Global mutex for thread list access. Has to be held when:
- iterating thread list - iterating thread list
- adding or removing elements from thread list - adding or removing elements from thread list
- changing or reading thread_status's fields: - changing or reading thread_status's fields:
@ -75,6 +82,8 @@ static pthread_mutex_t _global_mutex;
#define THREAD_STACK_SIZE (300*1024) #define THREAD_STACK_SIZE (300*1024)
#define DEBUGLOG(fmt, args...) _debuglog(fmt, ## args)
/* Data kept about a DSO. */ /* Data kept about a DSO. */
struct dso_data { struct dso_data {
struct list list; struct list list;
@ -172,6 +181,24 @@ static LIST_INIT(_timeout_registry);
static pthread_mutex_t _timeout_mutex = PTHREAD_MUTEX_INITIALIZER; static pthread_mutex_t _timeout_mutex = PTHREAD_MUTEX_INITIALIZER;
static pthread_cond_t _timeout_cond = PTHREAD_COND_INITIALIZER; static pthread_cond_t _timeout_cond = PTHREAD_COND_INITIALIZER;
static void _debuglog(const char *fmt, ...)
{
time_t P;
va_list ap;
if (!_debug)
return;
va_start(ap,fmt);
time(&P);
fprintf(stderr, "dmeventd[%x]: %.15s ", (int)pthread_self(), ctime(&P)+4 );
vfprintf(stderr, fmt, ap);
fprintf(stderr, "\n");
va_end(ap);
}
/* Allocate/free the status structure for a monitoring thread. */ /* Allocate/free the status structure for a monitoring thread. */
static struct thread_status *_alloc_thread_status(struct message_data *data, static struct thread_status *_alloc_thread_status(struct message_data *data,
struct dso_data *dso_data) struct dso_data *dso_data)
@ -883,11 +910,11 @@ static int _register_for_event(struct message_data *message_data)
almost as good as dead already... */ almost as good as dead already... */
if (thread_new->events & DM_EVENT_TIMEOUT) { if (thread_new->events & DM_EVENT_TIMEOUT) {
ret = -_register_for_timeout(thread); ret = -_register_for_timeout(thread);
if (ret) { if (ret) {
_unlock_mutex(); _unlock_mutex();
goto out; goto out;
} }
} }
if (!(thread = _lookup_thread_status(message_data))) { if (!(thread = _lookup_thread_status(message_data))) {
_unlock_mutex(); _unlock_mutex();
@ -1361,7 +1388,7 @@ static void _cleanup_unused_threads(void)
if (thread->status == DM_THREAD_RUNNING) { if (thread->status == DM_THREAD_RUNNING) {
thread->status = DM_THREAD_SHUTDOWN; thread->status = DM_THREAD_SHUTDOWN;
break; break;
} }
if (thread->status == DM_THREAD_SHUTDOWN) { if (thread->status == DM_THREAD_SHUTDOWN) {
if (!thread->events) { if (!thread->events) {
@ -1377,7 +1404,7 @@ static void _cleanup_unused_threads(void)
stack; stack;
} }
break; break;
} }
list_del(l); list_del(l);
syslog(LOG_ERR, syslog(LOG_ERR,
@ -1465,31 +1492,39 @@ static int _lock_pidfile(void)
return 0; return 0;
} }
/*
* Protection against OOM killer if kernel supports it
*/
static int _set_oom_adj(int val) static int _set_oom_adj(int val)
{ {
FILE *fp; FILE *fp;
struct stat st; struct stat st;
if (stat("/proc/self/oom_adj", &st) == -1) if (stat(OOM_ADJ_FILE, &st) == -1) {
return -errno; if (errno == ENOENT)
DEBUGLOG(OOM_ADJ_FILE " not found");
else
perror(OOM_ADJ_FILE ": stat failed");
return 1;
}
fp = fopen("/proc/self/oom_adj", "w"); if (!(fp = fopen(OOM_ADJ_FILE, "w"))) {
perror(OOM_ADJ_FILE ": fopen failed");
if (!fp) return 0;
return -1; }
fprintf(fp, "%i", val); fprintf(fp, "%i", val);
fclose(fp); fclose(fp);
return 0; return 1;
} }
static void _daemonize(void) static void _daemonize(void)
{ {
int status; int child_status;
int pid;
int fd; int fd;
pid_t pid;
struct rlimit rlim; struct rlimit rlim;
struct timeval tval; struct timeval tval;
sigset_t my_sigset; sigset_t my_sigset;
@ -1501,14 +1536,17 @@ static void _daemonize(void)
} }
signal(SIGTERM, &_exit_handler); signal(SIGTERM, &_exit_handler);
pid = fork(); switch (pid = fork()) {
case -1:
if (pid < 0) perror("fork failed:");
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
if (pid) { case 0: /* Child */
break;
default:
/* Wait for response from child */ /* Wait for response from child */
while (!waitpid(pid, &status, WNOHANG) && !_exit_now) { while (!waitpid(pid, &child_status, WNOHANG) && !_exit_now) {
tval.tv_sec = 0; tval.tv_sec = 0;
tval.tv_usec = 250000; /* .25 sec */ tval.tv_usec = 250000; /* .25 sec */
select(0, NULL, NULL, NULL, &tval); select(0, NULL, NULL, NULL, &tval);
@ -1518,7 +1556,7 @@ static void _daemonize(void)
exit(EXIT_SUCCESS); exit(EXIT_SUCCESS);
/* Problem with child. Determine what it is by exit code */ /* Problem with child. Determine what it is by exit code */
switch (WEXITSTATUS(status)) { switch (WEXITSTATUS(child_status)) {
case EXIT_LOCKFILE_INUSE: case EXIT_LOCKFILE_INUSE:
break; break;
case EXIT_DESC_CLOSE_FAILURE: case EXIT_DESC_CLOSE_FAILURE:
@ -1535,10 +1573,9 @@ static void _daemonize(void)
break; break;
} }
exit(EXIT_FAILURE); /* Redundant */ exit(child_status);
} }
setsid();
if (chdir("/")) if (chdir("/"))
exit(EXIT_CHDIR_FAILURE); exit(EXIT_CHDIR_FAILURE);
@ -1555,6 +1592,51 @@ static void _daemonize(void)
(open("/dev/null", O_WRONLY) < 0)) (open("/dev/null", O_WRONLY) < 0))
exit(EXIT_DESC_OPEN_FAILURE); exit(EXIT_DESC_OPEN_FAILURE);
setsid();
}
static void usage(char *prog, FILE *file)
{
fprintf(file, "Usage:\n");
fprintf(file, "%s [Vhd]\n", prog);
fprintf(file, "\n");
fprintf(file, " -V Show version of dmeventd\n");
fprintf(file, " -h Show this help information\n");
fprintf(file, " -d Don't fork, run in the foreground\n");
fprintf(file, "\n");
}
int main(int argc, char *argv[])
{
int ret;
signed char opt;
struct dm_event_fifos fifos;
//struct sys_log logdata = {DAEMON_NAME, LOG_DAEMON};
opterr = 0;
optind = 0;
while ((opt = getopt(argc, argv, "?hVd")) != EOF) {
switch (opt) {
case 'h':
usage(argv[0], stdout);
exit(0);
case '?':
usage(argv[0], stderr);
exit(0);
case 'd':
_debug++;
break;
case 'V':
printf("dmeventd version: %s\n", DM_LIB_VERSION);
exit(1);
break;
}
}
if (!_debug)
_daemonize();
openlog("dmeventd", LOG_PID, LOG_DAEMON); openlog("dmeventd", LOG_PID, LOG_DAEMON);
_lock_pidfile(); /* exits if failure */ _lock_pidfile(); /* exits if failure */
@ -1563,20 +1645,8 @@ static void _daemonize(void)
signal(SIGINT, &_exit_handler); signal(SIGINT, &_exit_handler);
signal(SIGHUP, &_exit_handler); signal(SIGHUP, &_exit_handler);
signal(SIGQUIT, &_exit_handler); signal(SIGQUIT, &_exit_handler);
}
int main(int argc, char *argv[]) if (!_set_oom_adj(OOM_DISABLE) && !_set_oom_adj(OOM_ADJUST_MIN))
{
int ret;
struct dm_event_fifos fifos;
//struct sys_log logdata = {DAEMON_NAME, LOG_DAEMON};
_daemonize();
/*
* ENOENT means the kernel does not support oom_adj
*/
if (_set_oom_adj(-16) != -ENOENT)
syslog(LOG_ERR, "Failed to set oom_adj to protect against OOM killer"); syslog(LOG_ERR, "Failed to set oom_adj to protect against OOM killer");
_init_thread_signals(); _init_thread_signals();
@ -1600,7 +1670,7 @@ int main(int argc, char *argv[])
/* Signal parent, letting them know we are ready to go. */ /* Signal parent, letting them know we are ready to go. */
kill(getppid(), SIGTERM); kill(getppid(), SIGTERM);
syslog(LOG_INFO, "dmeventd ready for processing."); syslog(LOG_NOTICE, "dmeventd ready for processing.");
while (!_exit_now) { while (!_exit_now) {
_process_request(&fifos); _process_request(&fifos);
@ -1619,7 +1689,8 @@ int main(int argc, char *argv[])
#endif #endif
pthread_mutex_destroy(&_global_mutex); pthread_mutex_destroy(&_global_mutex);
syslog(LOG_INFO, "dmeventd shutting down."); syslog(LOG_NOTICE, "dmeventd shutting down.");
closelog(); closelog();
exit(EXIT_SUCCESS); exit(EXIT_SUCCESS);
} }