diff options
author | stevel@tonic-gate <none@none> | 2005-06-14 00:00:00 -0700 |
---|---|---|
committer | stevel@tonic-gate <none@none> | 2005-06-14 00:00:00 -0700 |
commit | 7c478bd95313f5f23a4c958a745db2134aa03244 (patch) | |
tree | c871e58545497667cbb4b0a4f2daf204743e1fe7 /usr/src/lib/libproc/common/proc_stdio.c | |
download | illumos-gate-7c478bd95313f5f23a4c958a745db2134aa03244.tar.gz |
OpenSolaris Launch
Diffstat (limited to 'usr/src/lib/libproc/common/proc_stdio.c')
-rw-r--r-- | usr/src/lib/libproc/common/proc_stdio.c | 171 |
1 files changed, 171 insertions, 0 deletions
diff --git a/usr/src/lib/libproc/common/proc_stdio.c b/usr/src/lib/libproc/common/proc_stdio.c new file mode 100644 index 0000000000..b8e74bfe5a --- /dev/null +++ b/usr/src/lib/libproc/common/proc_stdio.c @@ -0,0 +1,171 @@ +/* + * CDDL HEADER START + * + * The contents of this file are subject to the terms of the + * Common Development and Distribution License, Version 1.0 only + * (the "License"). You may not use this file except in compliance + * with the License. + * + * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE + * or http://www.opensolaris.org/os/licensing. + * See the License for the specific language governing permissions + * and limitations under the License. + * + * When distributing Covered Code, include this CDDL HEADER in each + * file and include the License file at usr/src/OPENSOLARIS.LICENSE. + * If applicable, add the following below this CDDL HEADER, with the + * fields enclosed by brackets "[]" replaced with your own identifying + * information: Portions Copyright [yyyy] [name of copyright owner] + * + * CDDL HEADER END + */ +/* + * Copyright 2004 Sun Microsystems, Inc. All rights reserved. + * Use is subject to license terms. + */ + +#pragma ident "%Z%%M% %I% %E% SMI" + +/* + * Utility functions for buffering output to stdout, stderr while + * process is grabbed. Prevents infamous deadlocks due to pfiles `pgrep xterm` + * and other varients. + */ + +#include "libproc.h" +#include <stdio.h> + +static int cached_stdout_fd = -1; +static int cached_stderr_fd = -1; +static int initialized = 0; + +static char stdout_name[] = "/tmp/.stdoutXXXXXX"; +static char stderr_name[] = "/tmp/.stderrXXXXXX"; + +int +proc_initstdio(void) +{ + int fd; + + (void) fflush(stdout); + (void) fflush(stderr); + + if ((cached_stdout_fd = dup(1)) < 0) { + return (-1); + } + + if ((cached_stderr_fd = dup(2)) < 0) { + (void) close(cached_stdout_fd); + return (-1); + } + + if ((fd = mkstemp(stdout_name)) < 0) { + (void) close(cached_stdout_fd); + (void) close(cached_stderr_fd); + return (-1); + } + + (void) unlink(stdout_name); + + if (dup2(fd, 1) < 0) { + (void) close(fd); + (void) close(cached_stdout_fd); + (void) close(cached_stderr_fd); + return (-1); + } + + (void) close(fd); + + + if ((fd = mkstemp(stderr_name)) < 0) { + (void) dup2(cached_stdout_fd, 1); + (void) close(cached_stdout_fd); + (void) close(cached_stderr_fd); + return (-1); + } + + (void) unlink(stderr_name); + + if (dup2(fd, 2) < 0) { + (void) close(fd); + (void) dup2(cached_stdout_fd, 1); + (void) close(cached_stdout_fd); + (void) dup2(cached_stderr_fd, 2); + (void) close(cached_stderr_fd); + (void) close(fd); + return (-1); + } + + (void) close(fd); + + initialized = 1; + + return (0); +} + +static int +copy_fd(int out, FILE *in, size_t len) +{ + char buffer[8192]; + int rlen, alen; + int errors = 0; + + rewind(in); + while (len > 0 && !errors) { + rlen = (len > sizeof (buffer)) ? sizeof (buffer) : len; + alen = read(fileno(in), buffer, rlen); + if (alen == rlen) { + if (write(out, buffer, alen) < alen) + errors++; + else + len -= alen; + } + else + errors++; + } + rewind(in); + return (errors); +} + +int +proc_flushstdio(void) +{ + size_t len; + int errors = 0; + + /* + * flush any pending IO + */ + + if (!initialized) + return (-1); + + (void) fflush(stdout); + (void) fflush(stderr); + + if ((len = ftell(stdout)) > 0) + errors += copy_fd(cached_stdout_fd, stdout, len); + + + if ((len = ftell(stderr)) > 0) + errors += copy_fd(cached_stderr_fd, stderr, len); + + return (errors?-1:0); +} + +int +proc_finistdio(void) +{ + if (!initialized) + return (-1); + + if (proc_flushstdio() != 0) + return (-1); + + (void) dup2(cached_stdout_fd, 1); + (void) close(cached_stdout_fd); + (void) dup2(cached_stderr_fd, 2); + (void) close(cached_stderr_fd); + + return (0); +} |