X-Git-Url: https://arthur.barton.de/cgi-bin/gitweb.cgi?p=ngircd-alex.git;a=blobdiff_plain;f=src%2Fngircd%2Fproc.c;h=4220f1ab16f9101550e40df8e351caa777545b21;hp=3eb3d8042edc0967bd19f1a7251a710850e4139d;hb=259c314d142abd6f9295047c116235cfdd119563;hpb=7b5e2fe38e7af696155e687924462c4b9fe951bc diff --git a/src/ngircd/proc.c b/src/ngircd/proc.c index 3eb3d804..4220f1ab 100644 --- a/src/ngircd/proc.c +++ b/src/ngircd/proc.c @@ -1,19 +1,21 @@ /* * ngIRCd -- The Next Generation IRC Daemon - * Copyright (c)2001-2010 Alexander Barton (alex@barton.de) + * Copyright (c)2001-2011 Alexander Barton (alex@barton.de) and Contributors. * * This program is free software; you can redistribute it and/or modify * 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. * Please read the file COPYING, README and AUTHORS for more information. - * - * Process management */ #include "portab.h" -#include "imp.h" +/** + * @file + * Process management + */ + #include #include #include @@ -23,8 +25,8 @@ #include "log.h" #include "io.h" - -#include "exp.h" +#include "conn.h" +#include "sighandlers.h" #include "proc.h" /** @@ -42,9 +44,12 @@ Proc_InitStruct (PROC_STAT *proc) * Fork a child process. */ GLOBAL pid_t -Proc_Fork(PROC_STAT *proc, int *pipefds, void (*cbfunc)(int, short)) +Proc_Fork(PROC_STAT *proc, int *pipefds, void (*cbfunc)(int, short), int timeout) { pid_t pid; +#ifndef HAVE_ARC4RANDOM + unsigned int seed; +#endif assert(proc != NULL); assert(pipefds != NULL); @@ -56,6 +61,9 @@ Proc_Fork(PROC_STAT *proc, int *pipefds, void (*cbfunc)(int, short)) return -1; } +#ifndef HAVE_ARC4RANDOM + seed = (unsigned int)rand(); +#endif pid = fork(); switch (pid) { case -1: @@ -66,7 +74,17 @@ Proc_Fork(PROC_STAT *proc, int *pipefds, void (*cbfunc)(int, short)) return -1; case 0: /* New child process: */ +#ifdef HAVE_ARC4RANDOM_STIR + arc4random_stir(); +#endif +#ifndef HAVE_ARC4RANDOM + srand(seed ^ (unsigned int)time(NULL) ^ getpid()); +#endif + Signals_Exit(); + signal(SIGTERM, Proc_GenericSignalHandler); + signal(SIGALRM, Proc_GenericSignalHandler); close(pipefds[0]); + alarm(timeout); return 0; } @@ -86,21 +104,6 @@ Proc_Fork(PROC_STAT *proc, int *pipefds, void (*cbfunc)(int, short)) return pid; } -/** - * Kill forked child process. - */ -GLOBAL void -Proc_Kill(PROC_STAT *proc) -{ - assert(proc != NULL); - - if (proc->pipe_fd > 0) - io_close(proc->pipe_fd); - if (proc->pid > 0) - kill(proc->pid, SIGTERM); - Proc_InitStruct(proc); -} - /** * Generic signal handler for forked child processes. */ @@ -113,7 +116,55 @@ Proc_GenericSignalHandler(int Signal) Log_Subprocess(LOG_DEBUG, "Child got TERM signal, exiting."); #endif exit(1); + case SIGALRM: +#ifdef DEBUG + Log_Subprocess(LOG_DEBUG, "Child got ALARM signal, exiting."); +#endif + exit(1); + } +} + +/** + * Read bytes from a pipe of a forked child process. + * In addition, this function makes sure that the child process is ignored + * after all data has been read or a fatal error occurred. + */ +GLOBAL size_t +Proc_Read(PROC_STAT *proc, void *buffer, size_t buflen) +{ + ssize_t bytes_read = 0; + + assert(buffer != NULL); + assert(buflen > 0); + + bytes_read = read(proc->pipe_fd, buffer, buflen); + if (bytes_read < 0) { + if (errno == EAGAIN) + return 0; + Log(LOG_CRIT, "Can't read from child process %ld: %s", + proc->pid, strerror(errno)); + Proc_Close(proc); + bytes_read = 0; + } else if (bytes_read == 0) { + /* EOF: clean up */ + LogDebug("Child process %ld: EOF reached, closing pipe.", + proc->pid); + Proc_Close(proc); } + return (size_t)bytes_read; +} + +/** + * Close pipe to a forked child process. + */ +GLOBAL void +Proc_Close(PROC_STAT *proc) +{ + /* Close socket, if it exists */ + if (proc->pipe_fd >= 0) + io_close(proc->pipe_fd); + + Proc_InitStruct(proc); } /* -eof- */