2012-08-14 14:11:15 -05:00
|
|
|
// NB: transitionary, de-mode-ing.
|
|
|
|
#[forbid(deprecated_mode)];
|
|
|
|
#[forbid(deprecated_pattern)];
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
//! Process spawning
|
2012-09-04 13:12:17 -05:00
|
|
|
use option::{Some, None};
|
|
|
|
use libc::{pid_t, c_void, c_int};
|
|
|
|
use io::ReaderUtil;
|
2011-03-11 06:30:18 -06:00
|
|
|
|
2012-08-14 15:38:35 -05:00
|
|
|
export Program;
|
2011-07-18 23:07:33 -05:00
|
|
|
export run_program;
|
|
|
|
export start_program;
|
|
|
|
export program_output;
|
2011-07-25 23:25:32 -05:00
|
|
|
export spawn_process;
|
2011-09-08 14:03:07 -05:00
|
|
|
export waitpid;
|
2011-07-18 23:07:33 -05:00
|
|
|
|
2011-11-16 22:49:38 -06:00
|
|
|
#[abi = "cdecl"]
|
2012-07-03 18:11:00 -05:00
|
|
|
extern mod rustrt {
|
2012-03-14 17:10:34 -05:00
|
|
|
fn rust_run_program(argv: **libc::c_char, envp: *c_void,
|
|
|
|
dir: *libc::c_char,
|
2012-03-12 22:04:27 -05:00
|
|
|
in_fd: c_int, out_fd: c_int, err_fd: c_int)
|
2012-02-07 20:55:02 -06:00
|
|
|
-> pid_t;
|
2011-03-11 06:30:18 -06:00
|
|
|
}
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// A value representing a child process
|
2012-08-14 15:38:35 -05:00
|
|
|
trait Program {
|
2012-07-04 16:53:12 -05:00
|
|
|
/// Returns the process id of the program
|
2011-11-15 00:29:40 -06:00
|
|
|
fn get_id() -> pid_t;
|
2011-10-26 18:24:31 -05:00
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// Returns an io::writer that can be used to write to stdin
|
2012-08-14 15:38:35 -05:00
|
|
|
fn input() -> io::Writer;
|
2011-10-26 18:24:31 -05:00
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// Returns an io::reader that can be used to read from stdout
|
2012-08-14 15:38:35 -05:00
|
|
|
fn output() -> io::Reader;
|
2011-10-26 18:24:31 -05:00
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// Returns an io::reader that can be used to read from stderr
|
2012-08-14 15:38:35 -05:00
|
|
|
fn err() -> io::Reader;
|
2011-10-26 18:24:31 -05:00
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// Closes the handle to the child processes standard input
|
2011-10-26 18:24:31 -05:00
|
|
|
fn close_input();
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/**
|
|
|
|
* Waits for the child process to terminate. Closes the handle
|
|
|
|
* to stdin if necessary.
|
|
|
|
*/
|
2011-10-26 18:24:31 -05:00
|
|
|
fn finish() -> int;
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// Closes open handles
|
2011-10-26 18:24:31 -05:00
|
|
|
fn destroy();
|
2012-01-11 06:27:46 -06:00
|
|
|
}
|
2011-10-26 18:24:31 -05:00
|
|
|
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/**
|
|
|
|
* Run a program, providing stdin, stdout and stderr handles
|
|
|
|
*
|
|
|
|
* # Arguments
|
|
|
|
*
|
|
|
|
* * prog - The path to an executable
|
|
|
|
* * args - Vector of arguments to pass to the child process
|
|
|
|
* * env - optional env-modification for child
|
|
|
|
* * dir - optional dir to run child in (default current dir)
|
|
|
|
* * in_fd - A file descriptor for the child to use as std input
|
|
|
|
* * out_fd - A file descriptor for the child to use as std output
|
|
|
|
* * err_fd - A file descriptor for the child to use as std error
|
|
|
|
*
|
|
|
|
* # Return value
|
|
|
|
*
|
|
|
|
* The process id of the spawned process
|
|
|
|
*/
|
2012-08-14 14:11:15 -05:00
|
|
|
fn spawn_process(prog: &str, args: &[~str],
|
2012-08-20 14:23:37 -05:00
|
|
|
env: &Option<~[(~str,~str)]>,
|
|
|
|
dir: &Option<~str>,
|
2012-03-12 22:04:27 -05:00
|
|
|
in_fd: c_int, out_fd: c_int, err_fd: c_int)
|
2012-06-24 22:18:18 -05:00
|
|
|
-> pid_t {
|
2012-06-30 18:19:07 -05:00
|
|
|
do with_argv(prog, args) |argv| {
|
|
|
|
do with_envp(env) |envp| {
|
|
|
|
do with_dirp(dir) |dirp| {
|
2012-02-07 20:55:02 -06:00
|
|
|
rustrt::rust_run_program(argv, envp, dirp,
|
|
|
|
in_fd, out_fd, err_fd)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-14 14:11:15 -05:00
|
|
|
fn with_argv<T>(prog: &str, args: &[~str],
|
2012-06-24 22:18:18 -05:00
|
|
|
cb: fn(**libc::c_char) -> T) -> T {
|
2012-06-30 18:19:07 -05:00
|
|
|
let mut argptrs = str::as_c_str(prog, |b| ~[b]);
|
2012-06-29 18:26:56 -05:00
|
|
|
let mut tmps = ~[];
|
2012-06-30 18:19:07 -05:00
|
|
|
for vec::each(args) |arg| {
|
2012-09-02 17:37:15 -05:00
|
|
|
let t = @copy arg;
|
2012-06-27 17:21:50 -05:00
|
|
|
vec::push(tmps, t);
|
2012-06-30 18:19:07 -05:00
|
|
|
vec::push_all(argptrs, str::as_c_str(*t, |b| ~[b]));
|
2012-02-07 20:55:02 -06:00
|
|
|
}
|
2012-06-27 17:21:50 -05:00
|
|
|
vec::push(argptrs, ptr::null());
|
2012-09-13 13:46:10 -05:00
|
|
|
vec::as_imm_buf(argptrs, |buf, _len| cb(buf))
|
2012-02-07 20:55:02 -06:00
|
|
|
}
|
|
|
|
|
2012-06-07 23:38:25 -05:00
|
|
|
#[cfg(unix)]
|
2012-08-20 14:23:37 -05:00
|
|
|
fn with_envp<T>(env: &Option<~[(~str,~str)]>,
|
2012-06-24 22:18:18 -05:00
|
|
|
cb: fn(*c_void) -> T) -> T {
|
2012-02-07 20:55:02 -06:00
|
|
|
// On posixy systems we can pass a char** for envp, which is
|
|
|
|
// a null-terminated array of "k=v\n" strings.
|
2012-08-14 14:11:15 -05:00
|
|
|
match *env {
|
2012-08-20 14:23:37 -05:00
|
|
|
Some(es) if !vec::is_empty(es) => {
|
2012-06-29 18:26:56 -05:00
|
|
|
let mut tmps = ~[];
|
|
|
|
let mut ptrs = ~[];
|
2012-02-07 20:55:02 -06:00
|
|
|
|
2012-06-30 18:19:07 -05:00
|
|
|
for vec::each(es) |e| {
|
2012-09-02 17:37:15 -05:00
|
|
|
let (k,v) = copy e;
|
2012-08-22 19:24:52 -05:00
|
|
|
let t = @(fmt!("%s=%s", k, v));
|
2012-02-07 20:55:02 -06:00
|
|
|
vec::push(tmps, t);
|
2012-06-30 18:19:07 -05:00
|
|
|
vec::push_all(ptrs, str::as_c_str(*t, |b| ~[b]));
|
2012-02-07 20:55:02 -06:00
|
|
|
}
|
2012-06-27 17:21:50 -05:00
|
|
|
vec::push(ptrs, ptr::null());
|
2012-09-13 13:46:10 -05:00
|
|
|
vec::as_imm_buf(ptrs, |p, _len|
|
2012-09-18 19:34:08 -05:00
|
|
|
unsafe { cb(::cast::reinterpret_cast(&p)) }
|
2012-06-30 18:19:07 -05:00
|
|
|
)
|
2012-02-07 20:55:02 -06:00
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
_ => cb(ptr::null())
|
2012-02-07 20:55:02 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-06-07 23:38:25 -05:00
|
|
|
#[cfg(windows)]
|
2012-08-20 14:23:37 -05:00
|
|
|
fn with_envp<T>(env: &Option<~[(~str,~str)]>,
|
2012-06-24 22:18:18 -05:00
|
|
|
cb: fn(*c_void) -> T) -> T {
|
2012-02-07 20:55:02 -06:00
|
|
|
// On win32 we pass an "environment block" which is not a char**, but
|
|
|
|
// rather a concatenation of null-terminated k=v\0 sequences, with a final
|
|
|
|
// \0 to terminate.
|
2012-06-24 22:18:18 -05:00
|
|
|
unsafe {
|
2012-08-14 17:58:21 -05:00
|
|
|
match *env {
|
2012-08-20 14:23:37 -05:00
|
|
|
Some(es) if !vec::is_empty(es) => {
|
2012-06-29 18:26:56 -05:00
|
|
|
let mut blk : ~[u8] = ~[];
|
2012-06-30 18:19:07 -05:00
|
|
|
for vec::each(es) |e| {
|
2012-06-24 22:18:18 -05:00
|
|
|
let (k,v) = e;
|
2012-08-22 19:24:52 -05:00
|
|
|
let t = fmt!("%s=%s", k, v);
|
2012-08-29 18:00:36 -05:00
|
|
|
let mut v : ~[u8] = ::unsafe::reinterpret_cast(&t);
|
2012-06-24 22:18:18 -05:00
|
|
|
blk += v;
|
|
|
|
::unsafe::forget(v);
|
|
|
|
}
|
2012-06-29 18:26:56 -05:00
|
|
|
blk += ~[0_u8];
|
2012-09-13 13:46:10 -05:00
|
|
|
vec::as_imm_buf(blk, |p, _len| cb(::unsafe::reinterpret_cast(&p)))
|
2012-06-24 22:18:18 -05:00
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
_ => cb(ptr::null())
|
2012-02-07 20:55:02 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-20 14:23:37 -05:00
|
|
|
fn with_dirp<T>(d: &Option<~str>,
|
2012-06-24 22:18:18 -05:00
|
|
|
cb: fn(*libc::c_char) -> T) -> T {
|
2012-08-14 14:11:15 -05:00
|
|
|
match *d {
|
2012-08-20 14:23:37 -05:00
|
|
|
Some(dir) => str::as_c_str(dir, cb),
|
|
|
|
None => cb(ptr::null())
|
2012-02-07 20:55:02 -06:00
|
|
|
}
|
2011-07-18 23:07:33 -05:00
|
|
|
}
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/**
|
|
|
|
* Spawns a process and waits for it to terminate
|
|
|
|
*
|
|
|
|
* # Arguments
|
|
|
|
*
|
|
|
|
* * prog - The path to an executable
|
|
|
|
* * args - Vector of arguments to pass to the child process
|
|
|
|
*
|
|
|
|
* # Return value
|
|
|
|
*
|
|
|
|
* The process id
|
|
|
|
*/
|
2012-08-14 14:11:15 -05:00
|
|
|
fn run_program(prog: &str, args: &[~str]) -> int {
|
2012-08-20 14:23:37 -05:00
|
|
|
let pid = spawn_process(prog, args, &None, &None,
|
2012-03-20 14:38:57 -05:00
|
|
|
0i32, 0i32, 0i32);
|
|
|
|
if pid == -1 as pid_t { fail; }
|
2012-08-01 19:30:05 -05:00
|
|
|
return waitpid(pid);
|
2011-03-11 06:30:18 -06:00
|
|
|
}
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/**
|
|
|
|
* Spawns a process and returns a program
|
|
|
|
*
|
|
|
|
* The returned value is a boxed class containing a <program> object that can
|
|
|
|
* be used for sending and receiving data over the standard file descriptors.
|
|
|
|
* The class will ensure that file descriptors are closed properly.
|
|
|
|
*
|
|
|
|
* # Arguments
|
|
|
|
*
|
|
|
|
* * prog - The path to an executable
|
|
|
|
* * args - Vector of arguments to pass to the child process
|
|
|
|
*
|
|
|
|
* # Return value
|
|
|
|
*
|
|
|
|
* A class with a <program> field
|
|
|
|
*/
|
2012-08-14 15:38:35 -05:00
|
|
|
fn start_program(prog: &str, args: &[~str]) -> Program {
|
2011-07-27 07:19:39 -05:00
|
|
|
let pipe_input = os::pipe();
|
|
|
|
let pipe_output = os::pipe();
|
2011-07-28 23:15:56 -05:00
|
|
|
let pipe_err = os::pipe();
|
2011-08-19 17:16:48 -05:00
|
|
|
let pid =
|
2012-08-20 14:23:37 -05:00
|
|
|
spawn_process(prog, args, &None, &None,
|
2012-02-07 20:55:02 -06:00
|
|
|
pipe_input.in, pipe_output.out,
|
2011-08-19 17:16:48 -05:00
|
|
|
pipe_err.out);
|
2011-07-18 23:07:33 -05:00
|
|
|
|
2012-03-20 14:38:57 -05:00
|
|
|
if pid == -1 as pid_t { fail; }
|
2012-03-12 22:04:27 -05:00
|
|
|
libc::close(pipe_input.in);
|
|
|
|
libc::close(pipe_output.out);
|
|
|
|
libc::close(pipe_err.out);
|
2012-01-11 06:27:46 -06:00
|
|
|
|
2012-08-14 15:38:35 -05:00
|
|
|
type ProgRepr = {pid: pid_t,
|
|
|
|
mut in_fd: c_int,
|
|
|
|
out_file: *libc::FILE,
|
|
|
|
err_file: *libc::FILE,
|
|
|
|
mut finished: bool};
|
2012-01-11 06:27:46 -06:00
|
|
|
|
2012-08-14 15:38:35 -05:00
|
|
|
fn close_repr_input(r: &ProgRepr) {
|
2012-01-11 06:27:46 -06:00
|
|
|
let invalid_fd = -1i32;
|
|
|
|
if r.in_fd != invalid_fd {
|
2012-03-12 22:04:27 -05:00
|
|
|
libc::close(r.in_fd);
|
2012-01-11 06:27:46 -06:00
|
|
|
r.in_fd = invalid_fd;
|
|
|
|
}
|
|
|
|
}
|
2012-08-14 15:38:35 -05:00
|
|
|
fn finish_repr(r: &ProgRepr) -> int {
|
2012-08-01 19:30:05 -05:00
|
|
|
if r.finished { return 0; }
|
2012-01-11 06:27:46 -06:00
|
|
|
r.finished = true;
|
|
|
|
close_repr_input(r);
|
2012-08-01 19:30:05 -05:00
|
|
|
return waitpid(r.pid);
|
2012-01-11 06:27:46 -06:00
|
|
|
}
|
2012-08-14 15:38:35 -05:00
|
|
|
fn destroy_repr(r: &ProgRepr) {
|
2012-01-11 06:27:46 -06:00
|
|
|
finish_repr(r);
|
2012-03-12 22:04:27 -05:00
|
|
|
libc::fclose(r.out_file);
|
|
|
|
libc::fclose(r.err_file);
|
2012-01-11 06:27:46 -06:00
|
|
|
}
|
2012-08-15 20:46:55 -05:00
|
|
|
struct ProgRes {
|
2012-09-06 21:40:15 -05:00
|
|
|
r: ProgRepr,
|
2012-08-14 14:11:15 -05:00
|
|
|
drop { destroy_repr(&self.r); }
|
2012-06-21 23:46:43 -05:00
|
|
|
}
|
2012-01-11 06:27:46 -06:00
|
|
|
|
2012-09-04 17:23:28 -05:00
|
|
|
fn ProgRes(+r: ProgRepr) -> ProgRes {
|
|
|
|
ProgRes {
|
|
|
|
r: r
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-14 15:38:35 -05:00
|
|
|
impl ProgRes: Program {
|
2012-08-01 19:30:05 -05:00
|
|
|
fn get_id() -> pid_t { return self.r.pid; }
|
2012-08-14 15:38:35 -05:00
|
|
|
fn input() -> io::Writer { io::fd_writer(self.r.in_fd, false) }
|
|
|
|
fn output() -> io::Reader { io::FILE_reader(self.r.out_file, false) }
|
|
|
|
fn err() -> io::Reader { io::FILE_reader(self.r.err_file, false) }
|
2012-08-14 14:11:15 -05:00
|
|
|
fn close_input() { close_repr_input(&self.r); }
|
|
|
|
fn finish() -> int { finish_repr(&self.r) }
|
|
|
|
fn destroy() { destroy_repr(&self.r); }
|
2011-03-11 06:30:18 -06:00
|
|
|
}
|
2012-01-11 06:27:46 -06:00
|
|
|
let repr = {pid: pid,
|
2012-03-26 20:35:18 -05:00
|
|
|
mut in_fd: pipe_input.out,
|
2012-03-12 22:04:27 -05:00
|
|
|
out_file: os::fdopen(pipe_output.in),
|
|
|
|
err_file: os::fdopen(pipe_err.in),
|
2012-03-26 20:35:18 -05:00
|
|
|
mut finished: false};
|
2012-08-14 15:38:35 -05:00
|
|
|
return ProgRes(move repr) as Program;
|
2011-03-11 06:30:18 -06:00
|
|
|
}
|
|
|
|
|
2012-08-14 15:38:35 -05:00
|
|
|
fn read_all(rd: io::Reader) -> ~str {
|
2012-07-14 00:57:48 -05:00
|
|
|
let mut buf = ~"";
|
2011-07-28 23:15:56 -05:00
|
|
|
while !rd.eof() {
|
2011-08-11 20:49:36 -05:00
|
|
|
let bytes = rd.read_bytes(4096u);
|
2012-01-25 02:53:17 -06:00
|
|
|
buf += str::from_bytes(bytes);
|
2011-03-11 06:30:18 -06:00
|
|
|
}
|
2012-09-10 18:31:00 -05:00
|
|
|
move buf
|
2011-07-28 23:15:56 -05:00
|
|
|
}
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/**
|
|
|
|
* Spawns a process, waits for it to exit, and returns the exit code, and
|
|
|
|
* contents of stdout and stderr.
|
|
|
|
*
|
|
|
|
* # Arguments
|
|
|
|
*
|
|
|
|
* * prog - The path to an executable
|
|
|
|
* * args - Vector of arguments to pass to the child process
|
|
|
|
*
|
|
|
|
* # Return value
|
|
|
|
*
|
|
|
|
* A record, {status: int, out: str, err: str} containing the exit code,
|
|
|
|
* the contents of stdout and the contents of stderr.
|
|
|
|
*/
|
2012-08-14 14:11:15 -05:00
|
|
|
fn program_output(prog: &str, args: &[~str]) ->
|
2012-07-14 00:57:48 -05:00
|
|
|
{status: int, out: ~str, err: ~str} {
|
2012-06-09 00:03:26 -05:00
|
|
|
|
|
|
|
let pipe_in = os::pipe();
|
|
|
|
let pipe_out = os::pipe();
|
|
|
|
let pipe_err = os::pipe();
|
2012-08-20 14:23:37 -05:00
|
|
|
let pid = spawn_process(prog, args, &None, &None,
|
2012-06-09 00:03:26 -05:00
|
|
|
pipe_in.in, pipe_out.out, pipe_err.out);
|
|
|
|
|
|
|
|
os::close(pipe_in.in);
|
|
|
|
os::close(pipe_out.out);
|
|
|
|
os::close(pipe_err.out);
|
|
|
|
if pid == -1i32 {
|
|
|
|
os::close(pipe_in.out);
|
|
|
|
os::close(pipe_out.in);
|
|
|
|
os::close(pipe_err.in);
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
os::close(pipe_in.out);
|
|
|
|
|
|
|
|
// Spawn two entire schedulers to read both stdout and sterr
|
|
|
|
// in parallel so we don't deadlock while blocking on one
|
2012-06-21 18:44:10 -05:00
|
|
|
// or the other. FIXME (#2625): Surely there's a much more
|
|
|
|
// clever way to do this.
|
2012-08-27 16:22:25 -05:00
|
|
|
let p = comm::Port();
|
|
|
|
let ch = comm::Chan(p);
|
2012-08-15 16:10:46 -05:00
|
|
|
do task::spawn_sched(task::SingleThreaded) {
|
2012-06-09 00:03:26 -05:00
|
|
|
let errput = readclose(pipe_err.in);
|
2012-09-10 18:31:00 -05:00
|
|
|
comm::send(ch, (2, move errput));
|
2012-06-09 00:03:26 -05:00
|
|
|
};
|
2012-08-15 16:10:46 -05:00
|
|
|
do task::spawn_sched(task::SingleThreaded) {
|
2012-06-09 00:03:26 -05:00
|
|
|
let output = readclose(pipe_out.in);
|
2012-09-10 18:31:00 -05:00
|
|
|
comm::send(ch, (1, move output));
|
2012-06-09 00:03:26 -05:00
|
|
|
};
|
|
|
|
let status = run::waitpid(pid);
|
2012-07-14 00:57:48 -05:00
|
|
|
let mut errs = ~"";
|
|
|
|
let mut outs = ~"";
|
2012-06-09 00:03:26 -05:00
|
|
|
let mut count = 2;
|
|
|
|
while count > 0 {
|
|
|
|
let stream = comm::recv(p);
|
2012-08-15 13:55:17 -05:00
|
|
|
match stream {
|
2012-08-03 21:59:04 -05:00
|
|
|
(1, s) => {
|
2012-09-02 17:37:15 -05:00
|
|
|
outs = copy s;
|
2012-06-09 00:03:26 -05:00
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
(2, s) => {
|
2012-09-02 17:37:15 -05:00
|
|
|
errs = copy s;
|
2012-06-09 00:03:26 -05:00
|
|
|
}
|
2012-08-15 13:55:17 -05:00
|
|
|
(n, _) => {
|
2012-08-27 15:50:40 -05:00
|
|
|
fail(fmt!("program_output received an unexpected file \
|
|
|
|
number: %u", n));
|
2012-08-15 13:55:17 -05:00
|
|
|
}
|
2012-06-09 00:03:26 -05:00
|
|
|
};
|
|
|
|
count -= 1;
|
|
|
|
};
|
2012-09-10 18:31:00 -05:00
|
|
|
return {status: status, out: move outs, err: move errs};
|
2012-06-09 00:03:26 -05:00
|
|
|
}
|
|
|
|
|
2012-08-14 14:11:15 -05:00
|
|
|
fn writeclose(fd: c_int, s: &str) {
|
2012-09-07 20:08:21 -05:00
|
|
|
use io::WriterUtil;
|
2012-06-09 00:03:26 -05:00
|
|
|
|
2012-08-22 19:24:52 -05:00
|
|
|
error!("writeclose %d, %s", fd as int, s);
|
2012-06-09 00:03:26 -05:00
|
|
|
let writer = io::fd_writer(fd, false);
|
|
|
|
writer.write_str(s);
|
|
|
|
|
|
|
|
os::close(fd);
|
|
|
|
}
|
|
|
|
|
2012-07-14 00:57:48 -05:00
|
|
|
fn readclose(fd: c_int) -> ~str {
|
2012-06-09 00:03:26 -05:00
|
|
|
let file = os::fdopen(fd);
|
|
|
|
let reader = io::FILE_reader(file, false);
|
2012-07-14 00:57:48 -05:00
|
|
|
let mut buf = ~"";
|
2012-06-09 00:03:26 -05:00
|
|
|
while !reader.eof() {
|
|
|
|
let bytes = reader.read_bytes(4096u);
|
|
|
|
buf += str::from_bytes(bytes);
|
|
|
|
}
|
|
|
|
os::fclose(file);
|
2012-09-10 18:31:00 -05:00
|
|
|
move buf
|
2011-03-11 06:30:18 -06:00
|
|
|
}
|
2011-09-08 14:03:07 -05:00
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// Waits for a process to exit and returns the exit code
|
2011-11-15 00:29:40 -06:00
|
|
|
fn waitpid(pid: pid_t) -> int {
|
2012-08-01 19:30:05 -05:00
|
|
|
return waitpid_os(pid);
|
2011-09-08 14:03:07 -05:00
|
|
|
|
2012-06-07 23:38:25 -05:00
|
|
|
#[cfg(windows)]
|
2011-11-15 18:49:25 -06:00
|
|
|
fn waitpid_os(pid: pid_t) -> int {
|
|
|
|
os::waitpid(pid) as int
|
2011-09-08 14:03:07 -05:00
|
|
|
}
|
|
|
|
|
2012-06-07 23:38:25 -05:00
|
|
|
#[cfg(unix)]
|
2011-11-15 00:29:40 -06:00
|
|
|
fn waitpid_os(pid: pid_t) -> int {
|
2011-10-26 18:24:31 -05:00
|
|
|
#[cfg(target_os = "linux")]
|
2011-11-15 00:29:40 -06:00
|
|
|
fn WIFEXITED(status: i32) -> bool {
|
|
|
|
(status & 0xffi32) == 0i32
|
2011-10-26 18:24:31 -05:00
|
|
|
}
|
2011-09-08 14:03:07 -05:00
|
|
|
|
2011-10-26 18:24:31 -05:00
|
|
|
#[cfg(target_os = "macos")]
|
2011-12-30 02:18:55 -06:00
|
|
|
#[cfg(target_os = "freebsd")]
|
2011-11-15 00:29:40 -06:00
|
|
|
fn WIFEXITED(status: i32) -> bool {
|
|
|
|
(status & 0x7fi32) == 0i32
|
2011-10-26 18:24:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(target_os = "linux")]
|
2011-11-15 00:29:40 -06:00
|
|
|
fn WEXITSTATUS(status: i32) -> i32 {
|
2011-11-15 16:27:55 -06:00
|
|
|
(status >> 8i32) & 0xffi32
|
2011-10-26 18:24:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(target_os = "macos")]
|
2011-12-30 02:18:55 -06:00
|
|
|
#[cfg(target_os = "freebsd")]
|
2011-11-15 00:29:40 -06:00
|
|
|
fn WEXITSTATUS(status: i32) -> i32 {
|
|
|
|
status >> 8i32
|
2011-10-26 18:24:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
let status = os::waitpid(pid);
|
2012-08-01 19:30:05 -05:00
|
|
|
return if WIFEXITED(status) {
|
2011-11-15 00:29:40 -06:00
|
|
|
WEXITSTATUS(status) as int
|
2011-10-26 18:24:31 -05:00
|
|
|
} else {
|
|
|
|
1
|
|
|
|
};
|
|
|
|
}
|
2011-09-08 14:03:07 -05:00
|
|
|
}
|
|
|
|
|
2012-01-17 21:05:07 -06:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
|
2012-09-07 20:08:21 -05:00
|
|
|
use io::WriterUtil;
|
2012-01-17 21:05:07 -06:00
|
|
|
|
|
|
|
// Regression test for memory leaks
|
2012-06-14 21:13:17 -05:00
|
|
|
#[ignore(cfg(windows))] // FIXME (#2626)
|
2012-01-17 21:05:07 -06:00
|
|
|
fn test_leaks() {
|
2012-08-14 14:11:15 -05:00
|
|
|
run::run_program("echo", []);
|
|
|
|
run::start_program("echo", []);
|
|
|
|
run::program_output("echo", []);
|
2012-01-17 21:05:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_pipes() {
|
|
|
|
let pipe_in = os::pipe();
|
|
|
|
let pipe_out = os::pipe();
|
|
|
|
let pipe_err = os::pipe();
|
|
|
|
|
|
|
|
let pid =
|
|
|
|
run::spawn_process(
|
2012-08-20 14:23:37 -05:00
|
|
|
"cat", [], &None, &None,
|
2012-02-07 20:55:02 -06:00
|
|
|
pipe_in.in, pipe_out.out, pipe_err.out);
|
2012-01-17 21:05:07 -06:00
|
|
|
os::close(pipe_in.in);
|
|
|
|
os::close(pipe_out.out);
|
|
|
|
os::close(pipe_err.out);
|
|
|
|
|
|
|
|
if pid == -1i32 { fail; }
|
2012-07-14 00:57:48 -05:00
|
|
|
let expected = ~"test";
|
2012-01-17 21:05:07 -06:00
|
|
|
writeclose(pipe_in.out, expected);
|
|
|
|
let actual = readclose(pipe_out.in);
|
|
|
|
readclose(pipe_err.in);
|
|
|
|
os::waitpid(pid);
|
|
|
|
|
|
|
|
log(debug, expected);
|
|
|
|
log(debug, actual);
|
|
|
|
assert (expected == actual);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn waitpid() {
|
2012-08-14 14:11:15 -05:00
|
|
|
let pid = run::spawn_process("false", [],
|
2012-08-20 14:23:37 -05:00
|
|
|
&None, &None,
|
2012-02-07 20:55:02 -06:00
|
|
|
0i32, 0i32, 0i32);
|
2012-01-17 21:05:07 -06:00
|
|
|
let status = run::waitpid(pid);
|
|
|
|
assert status == 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-03-11 06:30:18 -06:00
|
|
|
// Local Variables:
|
|
|
|
// mode: rust
|
|
|
|
// fill-column: 78;
|
|
|
|
// indent-tabs-mode: nil
|
|
|
|
// c-basic-offset: 4
|
|
|
|
// buffer-file-coding-system: utf-8-unix
|
|
|
|
// End:
|