// Copyright 2009 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. package main import ( "exec" "fmt" "go/token" "io/ioutil" "os" ) // A ByteReaderAt implements io.ReadAt using a slice of bytes. type ByteReaderAt []byte func (r ByteReaderAt) ReadAt(p []byte, off int64) (n int, err os.Error) { if off >= int64(len(r)) || off < 0 { return 0, os.EOF } return copy(p, r[off:]), nil } // run runs the command argv, feeding in stdin on standard input. // It returns the output to standard output and standard error. // ok indicates whether the command exited successfully. func run(stdin []byte, argv []string) (stdout, stderr []byte, ok bool) { cmd, err := exec.LookPath(argv[0]) if err != nil { fatal("exec %s: %s", argv[0], err) } r0, w0, err := os.Pipe() if err != nil { fatal("%s", err) } r1, w1, err := os.Pipe() if err != nil { fatal("%s", err) } r2, w2, err := os.Pipe() if err != nil { fatal("%s", err) } pid, err := os.ForkExec(cmd, argv, os.Environ(), "", []*os.File{r0, w1, w2}) if err != nil { fatal("%s", err) } r0.Close() w1.Close() w2.Close() c := make(chan bool) go func() { w0.Write(stdin) w0.Close() c <- true }() var xstdout []byte // TODO(rsc): delete after 6g can take address of out parameter go func() { xstdout, _ = ioutil.ReadAll(r1) r1.Close() c <- true }() stderr, _ = ioutil.ReadAll(r2) r2.Close() <-c <-c stdout = xstdout w, err := os.Wait(pid, 0) if err != nil { fatal("%s", err) } ok = w.Exited() && w.ExitStatus() == 0 return } // Die with an error message. func fatal(msg string, args ...interface{}) { fmt.Fprintf(os.Stderr, msg+"\n", args) os.Exit(2) } var nerrors int var noPos token.Position func error(pos token.Position, msg string, args ...interface{}) { nerrors++ if pos.IsValid() { fmt.Fprintf(os.Stderr, "%s: ", pos) } fmt.Fprintf(os.Stderr, msg, args) fmt.Fprintf(os.Stderr, "\n") }