--- libelwix/src/pio.c 2015/06/25 17:53:50 1.4 +++ libelwix/src/pio.c 2024/08/19 15:39:40 1.8.30.1 @@ -3,7 +3,7 @@ * by Michael Pounov * * $Author: misho $ -* $Id: pio.c,v 1.4 2015/06/25 17:53:50 misho Exp $ +* $Id: pio.c,v 1.8.30.1 2024/08/19 15:39:40 misho Exp $ * ************************************************************************** The ELWIX and AITNET software is distributed under the following @@ -12,7 +12,7 @@ terms: All of the documentation and software included in the ELWIX and AITNET Releases is copyrighted by ELWIX - Sofia/Bulgaria -Copyright 2004 - 2015 +Copyright 2004 - 2024 by Michael Pounov . All rights reserved. Redistribution and use in source and binary forms, with or without @@ -46,21 +46,34 @@ SUCH DAMAGE. #include "global.h" +#ifdef __linux__ +#define __USE_GNU + +#ifndef O_CLOEXEC +#define O_CLOEXEC 02000000 +#endif +#endif + extern char **environ; pio_pid_t pio_pidlist = SLIST_HEAD_INITIALIZER(pio_pidlist); +#ifdef HAVE_LIBPTHREAD static pthread_mutex_t pidlist_mutex = PTHREAD_MUTEX_INITIALIZER; -#define THREAD_LOCK() if (__isthreaded) pthread_mutex_lock(&pidlist_mutex) -#define THREAD_UNLOCK() if (__isthreaded) pthread_mutex_unlock(&pidlist_mutex) +#define THREAD_LOCK() pthread_mutex_lock(&pidlist_mutex) +#define THREAD_UNLOCK() pthread_mutex_unlock(&pidlist_mutex) +#else +#define THREAD_LOCK() +#define THREAD_UNLOCK() +#endif - /* * e_popen() - ELWIX replacement of standard popen * * @command = command * @type = type * @ppid = return pid of child program + * If value of *ppid is -1 when invoke routine then child will be session leader * return: NULL error or !=NULL open program */ #ifdef POPEN_STREAM @@ -70,8 +83,29 @@ int #endif e_popen(const char *command, const char *type, pid_t *ppid) { + return e_popen2(command, type, ppid, NULL, 0); +} + +/* + * e_popen2() - ELWIX replacement of standard popen with post close of chosen handles + * + * @command = command + * @type = type + * @ppid = return pid of child program + * If value of *ppid is -1 when invoke routine then child will be session leader + * @fds = file descriptor array for close when fork + * @fdslen = fds number of descriptors + * return: NULL error or !=NULL open program + */ +#ifdef POPEN_STREAM +FILE * +#else +int +#endif +e_popen2(const char *command, const char *type, pid_t *ppid, int *fds, size_t fdslen) +{ struct tagPIOPID *cur, *p; - int pdes[2], pid, twoway, cloexec; + int pdes[2], pid, twoway, cloexec, i; char *argv[4]; if (!command || !type) @@ -139,6 +173,14 @@ e_popen(const char *command, const char *type, pid_t * #endif /* NOTREACHED */ case 0: /* Child. */ + if (ppid && *ppid == -1) + setsid(); + + if (fds && fdslen) { + for (i = 0; i < fdslen; i++) + close(fds[i]); + } + if (*type == 'r') { /* * The _dup2() to STDIN_FILENO is repeated to avoid @@ -231,7 +273,7 @@ e_pclose(int iop) #endif { struct tagPIOPID *cur, *last = NULL; - int pstat; + int pstat = 0; pid_t pid; if (!iop)