openldap/servers/slapd/back-shell/fork.c

119 lines
2.8 KiB
C
Raw Normal View History

1998-08-09 08:43:13 +08:00
/* fork.c - fork and exec a process, connecting stdin/out w/pipes */
/* $OpenLDAP$ */
2003-11-27 14:35:49 +08:00
/* This work is part of OpenLDAP Software <http://www.openldap.org/>.
*
2009-01-22 08:40:04 +08:00
* Copyright 1998-2009 The OpenLDAP Foundation.
2003-11-27 14:35:49 +08:00
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted only as authorized by the OpenLDAP
* Public License.
*
* A copy of this license is available in the file LICENSE in the
* top-level directory of the distribution or, alternatively, at
* <http://www.OpenLDAP.org/license.html>.
*/
/* Portions Copyright (c) 1995 Regents of the University of Michigan.
* All rights reserved.
*
* Redistribution and use in source and binary forms are permitted
* provided that this notice is preserved and that due credit is given
* to the University of Michigan at Ann Arbor. The name of the University
* may not be used to endorse or promote products derived from this
* software without specific prior written permission. This software
* is provided ``as is'' without express or implied warranty.
*/
/* ACKNOWLEDGEMENTS:
* This work was originally developed by the University of Michigan
* (as part of U-MICH LDAP).
2000-05-16 00:35:48 +08:00
*/
1998-08-09 08:43:13 +08:00
1998-10-25 09:41:42 +08:00
#include "portable.h"
1998-08-09 08:43:13 +08:00
#include <stdio.h>
1998-10-25 09:41:42 +08:00
#include <ac/errno.h>
1998-10-25 09:41:42 +08:00
#include <ac/string.h>
#include <ac/socket.h>
#include <ac/unistd.h>
1998-10-25 09:41:42 +08:00
1998-08-09 08:43:13 +08:00
#include "slap.h"
#include "shell.h"
1998-08-09 08:43:13 +08:00
pid_t
1998-08-09 08:43:13 +08:00
forkandexec(
char **args,
1998-08-09 08:43:13 +08:00
FILE **rfp,
FILE **wfp
)
{
int p2c[2] = { -1, -1 }, c2p[2];
pid_t pid;
1998-08-09 08:43:13 +08:00
if ( pipe( p2c ) != 0 || pipe( c2p ) != 0 ) {
Debug( LDAP_DEBUG_ANY, "pipe failed\n", 0, 0, 0 );
close( p2c[0] );
close( p2c[1] );
1998-08-09 08:43:13 +08:00
return( -1 );
}
/*
* what we're trying to set up looks like this:
* parent *wfp -> p2c[1] | p2c[0] -> stdin child
* parent *rfp <- c2p[0] | c2p[1] <- stdout child
*/
fflush( NULL );
# ifdef HAVE_THR
pid = fork1();
# else
pid = fork();
# endif
if ( pid == 0 ) { /* child */
/*
* child could deadlock here due to resources locked
* by our parent
*
* If so, configure --without-threads.
*/
1998-08-09 08:43:13 +08:00
if ( dup2( p2c[0], 0 ) == -1 || dup2( c2p[1], 1 ) == -1 ) {
Debug( LDAP_DEBUG_ANY, "dup2 failed\n", 0, 0, 0 );
exit( EXIT_FAILURE );
1998-08-09 08:43:13 +08:00
}
}
close( p2c[0] );
close( c2p[1] );
if ( pid <= 0 ) {
close( p2c[1] );
close( c2p[0] );
}
switch ( pid ) {
case 0:
1998-08-09 08:43:13 +08:00
execv( args[0], args );
Debug( LDAP_DEBUG_ANY, "execv failed\n", 0, 0, 0 );
exit( EXIT_FAILURE );
1998-08-09 08:43:13 +08:00
case -1: /* trouble */
Debug( LDAP_DEBUG_ANY, "fork failed\n", 0, 0, 0 );
return( -1 );
}
/* parent */
1998-08-09 08:43:13 +08:00
if ( (*rfp = fdopen( c2p[0], "r" )) == NULL || (*wfp = fdopen( p2c[1],
"w" )) == NULL ) {
Debug( LDAP_DEBUG_ANY, "fdopen failed\n", 0, 0, 0 );
2006-04-04 18:12:55 +08:00
if ( *rfp ) {
fclose( *rfp );
*rfp = NULL;
} else {
close( c2p[0] );
}
1998-08-09 08:43:13 +08:00
close( p2c[1] );
return( -1 );
}
return( pid );
}