diff options
Diffstat (limited to 'kdm/backend/process.c')
-rw-r--r-- | kdm/backend/process.c | 762 |
1 files changed, 0 insertions, 762 deletions
diff --git a/kdm/backend/process.c b/kdm/backend/process.c deleted file mode 100644 index f9d34fe7f..000000000 --- a/kdm/backend/process.c +++ /dev/null @@ -1,762 +0,0 @@ -/* - -Copyright 1988, 1998 The Open Group -Copyright 2001-2004 Oswald Buddenhagen <[email protected]> - -Permission to use, copy, modify, distribute, and sell this software and its -documentation for any purpose is hereby granted without fee, provided that -the above copyright notice appear in all copies and that both that -copyright notice and this permission notice appear in supporting -documentation. - -The above copyright notice and this permission notice shall be included -in all copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS -OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. -IN NO EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR -OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, -ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR -OTHER DEALINGS IN THE SOFTWARE. - -Except as contained in this notice, the name of a copyright holder shall -not be used in advertising or otherwise to promote the sale, use or -other dealings in this Software without prior written authorization -from the copyright holder. - -*/ - -/* - * xdm - display manager daemon - * Author: Keith Packard, MIT X Consortium - * - * subdaemon and external process management and communication - */ - -#include "dm.h" -#include "dm_error.h" - -#include <ctype.h> -#include <stdio.h> -#include <stdarg.h> -#include <signal.h> -#include <unistd.h> -#ifdef _POSIX_PRIORITY_SCHEDULING -# include <sched.h> -#endif - -extern char **environ; - - -SIGFUNC Signal( int sig, SIGFUNC handler ) -{ -#ifndef __EMX__ - struct sigaction sigact, osigact; - sigact.sa_handler = handler; - sigemptyset( &sigact.sa_mask ); -# ifdef SA_RESTART - sigact.sa_flags = SA_RESTART; -# else - sigact.sa_flags = 0; -# endif - sigaction( sig, &sigact, &osigact ); - return osigact.sa_handler; -#else - return signal( sig, handler ); -#endif -} - - -void -TerminateProcess( int pid, int sig ) -{ - kill( pid, sig ); -#ifdef SIGCONT - kill( pid, SIGCONT ); -#endif -} - - -static FD_TYPE CloseMask; -static int max = -1; - -void -RegisterCloseOnFork( int fd ) -{ - FD_SET( fd, &CloseMask ); - if (fd > max) - max = fd; -} - -void -ClearCloseOnFork( int fd ) -{ - FD_CLR( fd, &CloseMask ); -} - -void -CloseNClearCloseOnFork( int fd ) -{ - close( fd ); - FD_CLR( fd, &CloseMask ); -} - -static void -CloseOnFork( void ) -{ - int fd; - - for (fd = 0; fd <= max; fd++) - if (FD_ISSET( fd, &CloseMask )) - close( fd ); - FD_ZERO( &CloseMask ); - max = -1; -} - -int -Fork() -{ - int pid; - - sigset_t ss, oss; - sigfillset( &ss ); - sigprocmask( SIG_SETMASK, &ss, &oss ); - - if (!(pid = fork())) { -#ifdef SIGCHLD - (void)Signal( SIGCHLD, SIG_DFL ); -#endif - (void)Signal( SIGTERM, SIG_DFL ); - (void)Signal( SIGINT, SIG_IGN ); /* for -nodaemon */ - (void)Signal( SIGPIPE, SIG_DFL ); - (void)Signal( SIGALRM, SIG_DFL ); - (void)Signal( SIGHUP, SIG_DFL ); - sigemptyset( &ss ); - sigprocmask( SIG_SETMASK, &ss, NULL ); - CloseOnFork(); - return 0; - } - - sigprocmask( SIG_SETMASK, &oss, 0 ); - - return pid; -} - -int -Wait4( int pid ) -{ - waitType result; - - while (waitpid( pid, &result, 0 ) < 0) - if (errno != EINTR) { - Debug( "Wait4(%d) failed: %m\n", pid ); - return 0; - } - return waitVal( result ); -} - - -void -execute( char **argv, char **env ) -{ - Debug( "execute: %[s ; %[s\n", argv, env ); - execve( argv[0], argv, env ); - /* - * In case this is a shell script which hasn't been - * made executable (or this is a SYSV box), do - * a reasonable thing - */ - if (errno != ENOENT) { - char **newargv; - FILE *f; - int nu; - char program[1024]; - - /* - * emulate BSD kernel behaviour -- read - * the first line; check if it starts - * with "#!", in which case it uses - * the rest of the line as the name of - * program to run. Else use "/bin/sh". - */ - if (!(f = fopen( argv[0], "r" ))) - return; - if (!fGets( program, sizeof(program), f )) { - fclose( f ); - return; - } - fclose( f ); - if (!strncmp( program, "#!", 2 )) - newargv = parseArgs( 0, program + 2 ); - else - newargv = addStrArr( 0, "/bin/sh", 7 ); - if (!newargv) - return; - nu = arrLen( newargv ); - if (!(argv = xCopyStrArr( nu, argv ))) - return; - memcpy( argv, newargv, sizeof(char *) * nu ); - Debug( "shell script execution: %[s\n", argv ); - execve( argv[0], argv, env ); - } -} - -int -runAndWait( char **args, char **env ) -{ - int pid, ret; - - switch (pid = Fork()) { - case 0: - execute( args, env ); - LogError( "Can't execute %\"s: %m\n", args[0] ); - exit( 127 ); - case -1: - LogError( "Can't fork to execute %\"s: %m\n", args[0] ); - return 1; - } - ret = Wait4( pid ); - return waitVal( ret ); -} - -FILE * -pOpen( char **what, char m, int *pid ) -{ - int dp[2]; - - if (pipe( dp )) - return 0; - switch ((*pid = Fork())) { - case 0: - if (m == 'r') - dup2( dp[1], 1 ); - else - dup2( dp[0], 0 ); - close( dp[0] ); - close( dp[1] ); - execute( what, environ ); - LogError( "Can't execute %\"s: %m\n", what[0] ); - exit( 127 ); - case -1: - close( dp[0] ); - close( dp[1] ); - LogError( "Can't fork to execute %\"s: %m\n", what[0] ); - return 0; - } - if (m == 'r') { - close( dp[1] ); - return fdopen( dp[0], "r" ); - } else { - close( dp[0] ); - return fdopen( dp[1], "w" ); - } -} - -int -pClose( FILE *f, int pid ) -{ - fclose( f ); - return Wait4( pid ); -} - -char * -locate( const char *exe ) -{ - int len; - char *path, *name, *thenam, nambuf[PATH_MAX+1]; - char *pathe; - - if (!(path = getenv( "PATH" ))) { - LogError( "Can't execute %'s: $PATH not set.\n", exe ); - return 0; - } - len = strlen( exe ); - name = nambuf + PATH_MAX - len; - memcpy( name, exe, len + 1 ); - *--name = '/'; - do { - if (!(pathe = strchr( path, ':' ))) - pathe = path + strlen( path ); - len = pathe - path; - if (len && !(len == 1 && *path == '.')) { - thenam = name - len; - if (thenam >= nambuf) { - memcpy( thenam, path, len ); - if (!access( thenam, X_OK )) { - StrDup( &name, thenam ); - return name; - } - } - } - path = pathe; - } while (*path++ != '\0'); - LogError( "Can't execute %'s: not in $PATH.\n", exe ); - return 0; -} - - -static GTalk *curtalk; - -void -GSet( GTalk *tlk ) -{ - curtalk = tlk; -} - -int -GFork( GPipe *pajp, const char *pname, char *cname, - GPipe *ogp, char *cgname ) -{ - int opipe[2], ipipe[2], ogpipe[2], igpipe[2], pid; - - if (pipe( opipe )) - goto badp1; - if (pipe( ipipe )) - goto badp2; - if (ogp) { - if (pipe( ogpipe )) - goto badp3; - if (pipe( igpipe )) { - close( ogpipe[0] ); - close( ogpipe[1] ); - badp3: - close( ipipe[0] ); - close( ipipe[1] ); - badp2: - close( opipe[0] ); - close( opipe[1] ); - badp1: - LogError( "Cannot start %s, pipe() failed", cname ); - if (cname) - free( cname ); - return -1; - } - } - RegisterCloseOnFork( opipe[1] ); - RegisterCloseOnFork( ipipe[0] ); - if (ogp) { - RegisterCloseOnFork( ogpipe[1] ); - RegisterCloseOnFork( igpipe[0] ); - } - switch (pid = Fork()) { - case -1: - close( opipe[0] ); - close( ipipe[1] ); - CloseNClearCloseOnFork( opipe[1] ); - CloseNClearCloseOnFork( ipipe[0] ); - if (ogp) { - close( ogpipe[0] ); - close( igpipe[1] ); - CloseNClearCloseOnFork( ogpipe[1] ); - CloseNClearCloseOnFork( igpipe[0] ); - } - LogError( "Cannot start %s, fork() failed\n", cname ); - if (cname) - free( cname ); - return -1; - case 0: - pajp->wfd = ipipe[1]; - RegisterCloseOnFork( ipipe[1] ); - pajp->rfd = opipe[0]; - RegisterCloseOnFork( opipe[0] ); - pajp->who = (char *)pname; - if (ogp) { - ogp->wfd = igpipe[1]; - RegisterCloseOnFork( igpipe[1] ); - ogp->rfd = ogpipe[0]; - RegisterCloseOnFork( ogpipe[0] ); - ogp->who = (char *)pname; - } - break; - default: - close( opipe[0] ); - close( ipipe[1] ); - pajp->rfd = ipipe[0]; - pajp->wfd = opipe[1]; - pajp->who = cname; - if (ogp) { - close( ogpipe[0] ); - close( igpipe[1] ); - ogp->rfd = igpipe[0]; - ogp->wfd = ogpipe[1]; - ogp->who = cgname; - } - break; - } - return pid; -} - -int -GOpen( GProc *proc, char **argv, const char *what, char **env, char *cname, - GPipe *gp ) -{ - char **margv; - int pip[2]; - char coninfo[32]; - -/* ### GSet (proc->pipe); */ - if (proc->pid) { - LogError( "%s already running\n", cname ); - if (cname) - free( cname ); - return -1; - } - if (!(margv = xCopyStrArr( 1, argv ))) { - if (cname) - free( cname ); - return -1; - } - if (!StrApp( margv, progpath, what, (char *)0 )) { - free( margv ); - if (cname) - free( cname ); - return -1; - } - if (pipe( pip )) { - LogError( "Cannot start %s, pipe() failed\n", cname ); - if (cname) - free( cname ); - goto fail; - } - if (gp) { - ClearCloseOnFork( gp->rfd ); - ClearCloseOnFork( gp->wfd ); - } - proc->pid = GFork( &proc->pipe, 0, cname, 0, 0 ); - if (proc->pid) { - close( pip[1] ); - if (gp) { - RegisterCloseOnFork( gp->rfd ); - RegisterCloseOnFork( gp->wfd ); - } - } - switch (proc->pid) { - case -1: - fail1: - close( pip[0] ); - fail: - free( margv[0] ); - free( margv ); - return -1; - case 0: - (void)Signal( SIGPIPE, SIG_IGN ); - close( pip[0] ); - fcntl( pip[1], F_SETFD, FD_CLOEXEC ); - if (gp) - sprintf( coninfo, "CONINFO=%d %d %d %d", - proc->pipe.rfd, proc->pipe.wfd, gp->rfd, gp->wfd ); - else - sprintf( coninfo, "CONINFO=%d %d", - proc->pipe.rfd, proc->pipe.wfd ); - env = putEnv( coninfo, env ); - if (debugLevel & DEBUG_VALGRIND) { - char **nmargv = xCopyStrArr( 1, margv ); - nmargv[0] = locate( "valgrind" ); - execute( nmargv, env ); - } else if (debugLevel & DEBUG_STRACE) { - char **nmargv = xCopyStrArr( 1, margv ); - nmargv[0] = locate( "strace" ); - execute( nmargv, env ); - } else - execute( margv, env ); - write( pip[1], "", 1 ); - exit( 1 ); - default: - (void)Signal( SIGPIPE, SIG_IGN ); - if (Reader( pip[0], coninfo, 1 )) { - Wait4( proc->pid ); - LogError( "Cannot execute %\"s (%s)\n", margv[0], cname ); - GClosen (&proc->pipe); - goto fail1; - } - close( pip[0] ); - Debug( "started %s (%\"s), pid %d\n", cname, margv[0], proc->pid ); - free( margv[0] ); - free( margv ); - GSendInt( debugLevel ); - return 0; - } -} - -static void -iGClosen( GPipe *pajp ) -{ - CloseNClearCloseOnFork( pajp->rfd ); - CloseNClearCloseOnFork( pajp->wfd ); - pajp->rfd = pajp->wfd = -1; -} - -void -GClosen (GPipe *pajp) -{ - iGClosen( pajp ); - if (pajp->who) - free( pajp->who ); - pajp->who = 0; -} - -int -GClose (GProc *proc, GPipe *gp, int force) -{ - int ret; - - if (!proc->pid) { - Debug( "whoops, GClose while helper not running\n" ); - return 0; - } - iGClosen( &proc->pipe ); - if (gp) - GClosen (gp); - if (force) - TerminateProcess( proc->pid, SIGTERM ); - ret = Wait4( proc->pid ); - proc->pid = 0; - if (WaitSig( ret ) ? WaitSig( ret ) != SIGTERM : - (WaitCode( ret ) < EX_NORMAL || WaitCode( ret ) > EX_MAX)) - LogError( "Abnormal termination of %s, code %d, signal %d\n", - proc->pipe.who, WaitCode( ret ), WaitSig( ret ) ); - Debug( "closed %s\n", proc->pipe.who ); - if (proc->pipe.who) - free( proc->pipe.who ); - proc->pipe.who = 0; - return ret; -} - -static void ATTR_NORETURN -GErr( void ) -{ - Longjmp( curtalk->errjmp, 1 ); -} - -static void -GRead( void *buf, int len ) -{ - if (Reader( curtalk->pipe->rfd, buf, len ) != len) { - LogError( "Cannot read from %s\n", curtalk->pipe->who ); - GErr(); - } -} - -static void -GWrite( const void *buf, int len ) -{ - if (Writer( curtalk->pipe->wfd, buf, len ) != len) { - LogError( "Cannot write to %s\n", curtalk->pipe->who ); - GErr(); - } -#ifdef _POSIX_PRIORITY_SCHEDULING - if ((debugLevel & DEBUG_HLPCON)) - sched_yield(); -#endif -} - -void -GSendInt( int val ) -{ - GDebug( "sending int %d (%#x) to %s\n", val, val, curtalk->pipe->who ); - GWrite( &val, sizeof(val) ); -} - -int -GRecvInt() -{ - int val; - - GDebug( "receiving int from %s ...\n", curtalk->pipe->who ); - GRead( &val, sizeof(val) ); - GDebug( " -> %d (%#x)\n", val, val ); - return val; -} - -int -GRecvCmd( int *cmd ) -{ - GDebug( "receiving command from %s ...\n", curtalk->pipe->who ); - if (Reader( curtalk->pipe->rfd, cmd, sizeof(*cmd) ) == sizeof(*cmd)) { - GDebug( " -> %d\n", *cmd ); - return 1; - } - GDebug( " -> no data\n" ); - return 0; -} - -void -GSendArr( int len, const char *data ) -{ - GDebug( "sending array[%d] %02[*{hhx to %s\n", - len, len, data, curtalk->pipe->who ); - GWrite( &len, sizeof(len) ); - GWrite( data, len ); -} - -static char * -iGRecvArr( int *rlen ) -{ - int len; - char *buf; - - GRead( &len, sizeof(len) ); - *rlen = len; - GDebug( " -> %d bytes\n", len ); - if (!len) - return (char *)0; - if (!(buf = Malloc( len ))) - GErr(); - GRead( buf, len ); - return buf; -} - -char * -GRecvArr( int *rlen ) -{ - char *buf; - - GDebug( "receiving array from %s ...\n", curtalk->pipe->who ); - buf = iGRecvArr( rlen ); - GDebug( " -> %02[*{hhx\n", *rlen, buf ); - return buf; -} - -static int -iGRecvArrBuf( char *buf ) -{ - int len; - - GRead( &len, sizeof(len) ); - GDebug( " -> %d bytes\n", len ); - if (len) - GRead( buf, len ); - return len; -} - -int -GRecvArrBuf( char *buf ) -{ - int len; - - GDebug( "receiving already allocated array from %s ...\n", - curtalk->pipe->who ); - len = iGRecvArrBuf( buf ); - GDebug( " -> %02[*{hhx\n", len, buf ); - return len; -} - -int -GRecvStrBuf( char *buf ) -{ - int len; - - GDebug( "receiving already allocated string from %s ...\n", - curtalk->pipe->who ); - len = iGRecvArrBuf( buf ); - GDebug( " -> %\".*s\n", len, buf ); - return len; -} - -void -GSendStr( const char *buf ) -{ - int len; - - GDebug( "sending string %\"s to %s\n", buf, curtalk->pipe->who ); - if (buf) { - len = strlen( buf ) + 1; - GWrite( &len, sizeof(len) ); - GWrite( buf, len ); - } else - GWrite( &buf, sizeof(int) ); -} - -void -GSendNStr( const char *buf, int len ) -{ - int tlen = len + 1; - GDebug( "sending string %\".*s to %s\n", len, buf, curtalk->pipe->who ); - GWrite( &tlen, sizeof(tlen) ); - GWrite( buf, len ); - GWrite( "", 1 ); -} - -void -GSendStrN( const char *buf, int len ) -{ - if (buf) - GSendNStr( buf, StrNLen( buf, len ) ); - else - GSendStr( buf ); -} - -char * -GRecvStr() -{ - int len; - char *buf; - - GDebug( "receiving string from %s ...\n", curtalk->pipe->who ); - buf = iGRecvArr( &len ); - GDebug( " -> %\".*s\n", len, buf ); - return buf; -} - -static void -iGSendStrArr( int num, char **data ) -{ - char **cdata; - - GWrite( &num, sizeof(num) ); - for (cdata = data; --num >= 0; cdata++) - GSendStr( *cdata ); -} - -/* -void -GSendStrArr (int num, char **data) -{ - GDebug( "sending string array[%d] to %s\n", num, curtalk->pipe->who ); - iGSendStrArr( num, data ); -} -*/ - -char ** -GRecvStrArr( int *rnum ) -{ - int num; - char **argv, **cargv; - - GDebug( "receiving string array from %s ...\n", curtalk->pipe->who ); - GRead( &num, sizeof(num) ); - GDebug( " -> %d strings\n", num ); - *rnum = num; - if (!num) - return (char **)0; - if (!(argv = Malloc( num * sizeof(char *) ))) - GErr(); - for (cargv = argv; --num >= 0; cargv++) - *cargv = GRecvStr(); - return argv; -} - -void -GSendArgv( char **argv ) -{ - int num; - - if (argv) { - for (num = 0; argv[num]; num++); - GDebug( "sending argv[%d] to %s ...\n", num, curtalk->pipe->who ); - iGSendStrArr( num + 1, argv ); - } else { - GDebug( "sending NULL argv to %s\n", curtalk->pipe->who ); - GWrite( &argv, sizeof(int) ); - } -} - -char ** -GRecvArgv() -{ - int num; - - return GRecvStrArr( &num ); -} - |