import str::sbuf; import vec::vbuf; export program; export run_program; export start_program; export program_output; export spawn_process; native "rust" mod rustrt { fn rust_run_program(argv: vbuf, in_fd: int, out_fd: int, err_fd: int) -> int; } fn arg_vec(prog: str, args: vec[str]) -> vec[sbuf] { let argptrs = [str::buf(prog)]; for arg: str in args { vec::push[sbuf](argptrs, str::buf(arg)); } vec::push[sbuf](argptrs, 0 as sbuf); ret argptrs; } fn spawn_process(prog: str, args: vec[str], in_fd: int, out_fd: int, err_fd: int) -> int { // Note: we have to hold on to this vector reference while we hold a // pointer to its buffer let argv = arg_vec(prog, args); let pid = rustrt::rust_run_program(vec::buf(argv), in_fd, out_fd, err_fd); ret pid; } fn run_program(prog: str, args: vec[str]) -> int { ret os::waitpid(spawn_process(prog, args, 0, 0, 0)); } type program = obj { fn get_id() -> int ; fn input() -> io::writer ; fn output() -> io::reader ; fn close_input() ; fn finish() -> int ; }; fn start_program(prog: str, args: vec[str]) -> @program { let pipe_input = os::pipe(); let pipe_output = os::pipe(); let pid = spawn_process(prog, args, pipe_input.in, pipe_output.out, 0); if pid == -1 { fail; } os::libc::close(pipe_input.in); os::libc::close(pipe_output.out); obj new_program(pid: int, mutable in_fd: int, out_file: os::libc::FILE, mutable finished: bool) { fn get_id() -> int { ret pid; } fn input() -> io::writer { ret io::new_writer(io::fd_buf_writer(in_fd, false)); } fn output() -> io::reader { ret io::new_reader(io::FILE_buf_reader(out_file, false)); } fn close_input() { let invalid_fd = -1; if in_fd != invalid_fd { os::libc::close(in_fd); in_fd = invalid_fd; } } fn finish() -> int { if finished { ret 0; } finished = true; self.close_input(); ret os::waitpid(pid); }drop { self.close_input(); if !finished { os::waitpid(pid); } os::libc::fclose(out_file); } } ret @new_program(pid, pipe_input.out, os::fd_FILE(pipe_output.in), false); } fn program_output(prog: str, args: vec[str]) -> {status: int, out: str} { let pr = start_program(prog, args); pr.close_input(); let out = pr.output(); let buf = ""; while !out.eof() { let bytes = out.read_bytes(4096u); buf += str::unsafe_from_bytes(bytes); } ret {status: pr.finish(), out: buf}; } // Local Variables: // mode: rust // fill-column: 78; // indent-tabs-mode: nil // c-basic-offset: 4 // buffer-file-coding-system: utf-8-unix // compile-command: "make -k -C $RBUILD 2>&1 | sed -e 's/\\/x\\//x:\\//g'"; // End: