2015-10-11 18:02:48 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2015-10-11 18:51:56 +02:00
|
|
|
"os"
|
|
|
|
"os/exec"
|
2015-10-11 18:02:48 +02:00
|
|
|
"os/signal"
|
2015-10-11 18:51:56 +02:00
|
|
|
"syscall"
|
2016-06-05 14:26:16 +02:00
|
|
|
|
2017-05-07 22:15:01 +02:00
|
|
|
"github.com/rfjakob/gocryptfs/internal/exitcodes"
|
2016-06-15 23:30:44 +02:00
|
|
|
"github.com/rfjakob/gocryptfs/internal/tlog"
|
2015-10-11 18:02:48 +02:00
|
|
|
)
|
|
|
|
|
2016-11-26 18:18:14 +01:00
|
|
|
// The child sends us USR1 if the mount was successful. Exit with error code
|
|
|
|
// 0 if we get it.
|
2015-11-14 21:25:10 +01:00
|
|
|
func exitOnUsr1() {
|
2015-10-11 18:02:48 +02:00
|
|
|
c := make(chan os.Signal, 1)
|
|
|
|
signal.Notify(c, syscall.SIGUSR1)
|
2016-11-26 18:18:14 +01:00
|
|
|
go func() {
|
|
|
|
<-c
|
|
|
|
os.Exit(0)
|
|
|
|
}()
|
2015-10-11 18:02:48 +02:00
|
|
|
}
|
|
|
|
|
2016-11-01 19:04:49 +01:00
|
|
|
// forkChild - execute ourselves once again, this time with the "-fg" flag, and
|
2015-11-14 21:25:10 +01:00
|
|
|
// wait for SIGUSR1 or child exit.
|
|
|
|
// This is a workaround for the missing true fork function in Go.
|
2016-09-20 19:49:44 +02:00
|
|
|
func forkChild() int {
|
2015-10-11 18:02:48 +02:00
|
|
|
name := os.Args[0]
|
2016-11-01 18:59:34 +01:00
|
|
|
newArgs := []string{"-fg", fmt.Sprintf("-notifypid=%d", os.Getpid())}
|
2015-10-11 18:02:48 +02:00
|
|
|
newArgs = append(newArgs, os.Args[1:]...)
|
|
|
|
c := exec.Command(name, newArgs...)
|
|
|
|
c.Stdout = os.Stdout
|
|
|
|
c.Stderr = os.Stderr
|
|
|
|
c.Stdin = os.Stdin
|
2016-11-26 18:18:14 +01:00
|
|
|
exitOnUsr1()
|
2015-10-11 18:02:48 +02:00
|
|
|
err := c.Start()
|
|
|
|
if err != nil {
|
2016-06-15 23:30:44 +02:00
|
|
|
tlog.Fatal.Printf("forkChild: starting %s failed: %v\n", name, err)
|
2017-05-07 22:15:01 +02:00
|
|
|
return exitcodes.ForkChild
|
2015-10-11 18:02:48 +02:00
|
|
|
}
|
|
|
|
err = c.Wait()
|
|
|
|
if err != nil {
|
|
|
|
if exiterr, ok := err.(*exec.ExitError); ok {
|
|
|
|
if waitstat, ok := exiterr.Sys().(syscall.WaitStatus); ok {
|
2015-10-11 18:51:56 +02:00
|
|
|
os.Exit(waitstat.ExitStatus())
|
|
|
|
}
|
2015-10-11 18:02:48 +02:00
|
|
|
}
|
2016-06-15 23:30:44 +02:00
|
|
|
tlog.Fatal.Printf("forkChild: wait returned an unknown error: %v\n", err)
|
2017-05-07 22:15:01 +02:00
|
|
|
return exitcodes.ForkChild
|
2015-10-11 18:02:48 +02:00
|
|
|
}
|
|
|
|
// The child exited with 0 - let's do the same.
|
2016-09-20 19:49:44 +02:00
|
|
|
return 0
|
2015-10-11 18:02:48 +02:00
|
|
|
}
|