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
|
|
|
|
|
|
|
"github.com/rfjakob/gocryptfs/internal/toggledlog"
|
2015-10-11 18:02:48 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// The child sends us USR1 if the mount was successful
|
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)
|
|
|
|
<-c
|
|
|
|
os.Exit(0)
|
|
|
|
}
|
|
|
|
|
2015-11-14 21:25:10 +01:00
|
|
|
// forkChild - execute ourselves once again, this time with the "-f" flag, and
|
|
|
|
// wait for SIGUSR1 or child exit.
|
|
|
|
// This is a workaround for the missing true fork function in Go.
|
|
|
|
func forkChild() {
|
|
|
|
go exitOnUsr1()
|
2015-10-11 18:02:48 +02:00
|
|
|
name := os.Args[0]
|
2016-01-20 22:31:15 +01:00
|
|
|
newArgs := []string{"-f", 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
|
|
|
|
err := c.Start()
|
|
|
|
if err != nil {
|
2016-06-05 14:26:16 +02:00
|
|
|
toggledlog.Fatal.Printf("forkChild: starting %s failed: %v\n", name, err)
|
2015-10-11 18:02:48 +02:00
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
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-05 14:26:16 +02:00
|
|
|
toggledlog.Fatal.Printf("forkChild: wait returned an unknown error: %v\n", err)
|
2015-10-11 18:02:48 +02:00
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
// The child exited with 0 - let's do the same.
|
|
|
|
os.Exit(0)
|
|
|
|
}
|