veyron/lib/modules: allow Shutdown to return an error status and avoid races.
Change-Id: I36c45fc1944aed97220eb627276f58b770b4c382
diff --git a/lib/modules/func.go b/lib/modules/func.go
index e5fd73b..5a7c149 100644
--- a/lib/modules/func.go
+++ b/lib/modules/func.go
@@ -17,6 +17,9 @@
main Main
stdin, stderr, stdout pipe
bufferedStdout *bufio.Reader
+ err error
+ sh *Shell
+ wg sync.WaitGroup
}
func newFunctionHandle(main Main) command {
@@ -35,15 +38,23 @@
return fh.stderr.r
}
-func (fh *functionHandle) Stdin() io.WriteCloser {
+func (fh *functionHandle) Stdin() io.Writer {
fh.mu.Lock()
defer fh.mu.Unlock()
return fh.stdin.w
}
+func (fh *functionHandle) CloseStdin() {
+ fh.mu.Lock()
+ fd := fh.stdin.w.Fd()
+ fh.mu.Unlock()
+ syscall.Close(int(fd))
+}
+
func (fh *functionHandle) start(sh *Shell, args ...string) (Handle, error) {
fh.mu.Lock()
defer fh.mu.Unlock()
+ fh.sh = sh
for _, p := range []*pipe{&fh.stdin, &fh.stdout, &fh.stderr} {
var err error
if p.r, p.w, err = os.Pipe(); err != nil {
@@ -51,30 +62,43 @@
}
}
fh.bufferedStdout = bufio.NewReader(fh.stdout.r)
+ fh.wg.Add(1)
+
go func() {
err := fh.main(fh.stdin.r, fh.stdout.w, fh.stderr.w, sh.mergeOSEnv(), args...)
if err != nil {
fmt.Fprintf(fh.stderr.w, "%s\n", err)
}
- // See the comment below in Shutdown.
- syscall.Close(int(fh.stdin.r.Fd()))
- syscall.Close(int(fh.stdout.w.Fd()))
- syscall.Close(int(fh.stderr.w.Fd()))
+ fh.mu.Lock()
+ // We close these files using the Close system call since there
+ // may be an oustanding read on them that would otherwise trigger
+ // a test failure with go test -race
+ syscall.Close(int(fh.stdin.w.Fd()))
+ syscall.Close(int(fh.stdout.r.Fd()))
+ syscall.Close(int(fh.stderr.r.Fd()))
+ fh.err = err
+ fh.mu.Unlock()
+ fh.wg.Done()
}()
return fh, nil
}
-func (fh *functionHandle) Shutdown(output io.Writer) {
+func (fh *functionHandle) Shutdown(output io.Writer) error {
fh.mu.Lock()
- defer fh.mu.Unlock()
- scanner := bufio.NewScanner(fh.stderr.r)
- for scanner.Scan() {
- fmt.Fprintf(output, "%s\n", scanner.Text())
- }
- // We close these files using the Close system call since there
- // may be an oustanding read on them that would otherwise trigger
- // a test failure with go test -race
syscall.Close(int(fh.stdin.w.Fd()))
- syscall.Close(int(fh.stdout.r.Fd()))
- syscall.Close(int(fh.stderr.r.Fd()))
+ if output != nil {
+ scanner := bufio.NewScanner(fh.stderr.r)
+ for scanner.Scan() {
+ fmt.Fprintf(output, "%s\n", scanner.Text())
+ }
+ }
+ fh.mu.Unlock()
+
+ fh.wg.Wait()
+
+ fh.mu.Lock()
+ err := fh.err
+ fh.sh.forget(fh)
+ fh.mu.Unlock()
+ return err
}