/* * Copyright (c) 2004-2005 The Trustees of Indiana University. * All rights reserved. * Copyright (c) 2004-2005 The Trustees of the University of Tennessee. * All rights reserved. * Copyright (c) 2004-2005 High Performance Computing Center Stuttgart, * University of Stuttgart. All rights reserved. * Copyright (c) 2004-2005 The Regents of the University of California. * All rights reserved. * $COPYRIGHT$ * * Additional copyrights may follow * * $HEADER$ * * These symbols are in a file by themselves to provide nice linker * semantics. Since linkers generally pull in symbols by object * files, keeping these symbols as the only symbols in this file * prevents utility programs such as "ompi_info" from having to import * entire components just to query their version and parameters. */ #include "ompi_config.h" #include #ifdef HAVE_UNISTD_H #include #endif #include #include #ifdef HAVE_SYS_WAIT_H #include #endif #include #ifdef HAVE_UTIL_H #include #endif #ifdef HAVE_PTY_H #include #endif #ifdef HAVE_FCNTL_H #include #endif #ifdef HAVE_TERMIOS_H #include # ifdef HAVE_TERMIO_H # include # endif #endif #ifdef HAVE_LIBUTIL_H #include #endif #include "mca/iof/base/iof_base_setup.h" #include "include/orte_constants.h" #include "opal/util/output.h" #include "mca/errmgr/errmgr.h" #include "mca/iof/iof.h" #include "mca/ns/ns.h" int orte_iof_base_setup_prefork(orte_iof_base_io_conf_t *opts) { int ret; /* first check to make sure we can do ptys */ #if (! defined(HAVE_OPENPTY)) || (OMPI_ENABLE_PTY_SUPPORT == 0) opts->usepty = 0; #endif fflush(stdout); #if defined(HAVE_OPENPTY) && OMPI_ENABLE_PTY_SUPPORT if (opts->usepty) { ret = openpty(&(opts->p_stdout[0]), &(opts->p_stdout[1]), NULL, NULL, NULL); } else { ret = -1; } #else ret = -1; #endif #if defined(WIN32) /* Windows doesn't have a 'pipe' function. * So we need to do something a bit more complex */ /* XXX Implement this properly JJH * http://www-106.ibm.com/developerworks/linux/library/l-rt4/?open&t=grl,l=252,p=pipes */ #else if (ret < 0) { if (pipe(opts->p_stdout) < 0) { ORTE_ERROR_LOG(ORTE_ERR_OUT_OF_RESOURCE); return ORTE_ERR_OUT_OF_RESOURCE; } if (pipe(opts->p_stdin) < 0) { ORTE_ERROR_LOG(ORTE_ERR_OUT_OF_RESOURCE); return ORTE_ERR_OUT_OF_RESOURCE; } } if (pipe(opts->p_stderr) < 0) { ORTE_ERROR_LOG(ORTE_ERR_OUT_OF_RESOURCE); return ORTE_ERR_OUT_OF_RESOURCE; } #endif return ORTE_SUCCESS; } int orte_iof_base_setup_child(orte_iof_base_io_conf_t *opts) { int ret; if (! opts->usepty) { close(opts->p_stdout[0]); close(opts->p_stdin[1]); } close(opts->p_stderr[0]); if (opts->usepty) { if (opts->connect_stdin) { #ifndef WIN32 /* disable echo */ struct termios term_attrs; if (tcgetattr(opts->p_stdout[1], &term_attrs) < 0) { return ORTE_ERROR; } term_attrs.c_lflag &= ~ (ECHO | ECHOE | ECHOK | ECHOCTL | ECHOKE | ECHONL); if (tcsetattr(opts->p_stdout[1], TCSANOW, &term_attrs) == -1) { return ORTE_ERROR; } /* and connect the pty to stdin */ ret = dup2(opts->p_stdout[1], fileno(stdin)); if (ret < 0) return ORTE_ERROR; #endif } else { int fd; /* connect input to /dev/null */ fd = open("/dev/null", O_RDONLY); if(fd > fileno(stdin)) { dup2(fd, fileno(stdin)); close(fd); } } ret = dup2(opts->p_stdout[1], fileno(stdout)); if (ret < 0) return ORTE_ERROR; } else { if(opts->p_stdout[1] != fileno(stdout)) { ret = dup2(opts->p_stdout[1], fileno(stdout)); if (ret < 0) return ORTE_ERROR; close(opts->p_stdout[1]); } if (opts->connect_stdin) { if(opts->p_stdin[0] != fileno(stdin)) { ret = dup2(opts->p_stdin[1], fileno(stdin)); if (ret < 0) return ORTE_ERROR; close(opts->p_stdin[1]); } } else { int fd; close(opts->p_stdin[0]); /* connect input to /dev/null */ fd = open("/dev/null", O_RDONLY); if(fd > fileno(stdin)) { dup2(fd, fileno(stdin)); close(fd); } } } if(opts->p_stderr[1] != fileno(stderr)) { ret = dup2(opts->p_stderr[1], fileno(stderr)); if (ret < 0) return ORTE_ERROR; close(opts->p_stderr[1]); } return ORTE_SUCCESS; } int orte_iof_base_setup_parent(const orte_process_name_t* name, orte_iof_base_io_conf_t *opts) { int ret; if (! opts->usepty) { close(opts->p_stdout[1]); close(opts->p_stdin[0]); } close(opts->p_stderr[1]); /* connect stdin endpoint */ if (opts->connect_stdin) { ret = orte_iof.iof_publish(name, ORTE_IOF_SINK, ORTE_IOF_STDIN, opts->usepty ? opts->p_stdout[0] : opts->p_stdin[1]); if(ORTE_SUCCESS != ret) { ORTE_ERROR_LOG(ret); return ret; } } else { if (! opts->usepty) { close(opts->p_stdin[0]); } } /* connect read end to IOF */ ret = orte_iof.iof_publish(name, ORTE_IOF_SOURCE, ORTE_IOF_STDOUT, opts->p_stdout[0]); if(ORTE_SUCCESS != ret) { ORTE_ERROR_LOG(ret); return ret; } ret = orte_iof.iof_publish(name, ORTE_IOF_SOURCE, ORTE_IOF_STDERR, opts->p_stderr[0]); if(ORTE_SUCCESS != ret) { ORTE_ERROR_LOG(ret); return ret; } return ORTE_SUCCESS; }