| |
| |
| |
|
|
| |
| |
|
|
| |
| |
| |
|
|
| package main |
|
|
| import ( |
| "fmt" |
| "io" |
| "os" |
| "os/exec" |
| "os/signal" |
| "runtime" |
| "sync" |
| "syscall" |
| "time" |
| ) |
|
|
| func init() { |
| register("SignalDuringExec", SignalDuringExec) |
| register("SignalDuringExecPgrp", SignalDuringExecPgrp) |
| register("Nop", Nop) |
| } |
|
|
| func SignalDuringExec() { |
| |
| cmd := exec.Command(os.Args[0], "SignalDuringExecPgrp") |
| cmd.Stdout = os.Stdout |
| cmd.Stderr = os.Stderr |
| cmd.SysProcAttr = &syscall.SysProcAttr{ |
| Setpgid: true, |
| } |
|
|
| |
| |
| rp, wp, err := os.Pipe() |
| if err != nil { |
| fmt.Printf("Failed to create pipe: %v", err) |
| return |
| } |
| cmd.ExtraFiles = []*os.File{rp} |
|
|
| |
| if err := cmd.Run(); err != nil { |
| fmt.Printf("Run failed: %v", err) |
| } |
|
|
| |
| |
| |
| runtime.KeepAlive(wp) |
| } |
|
|
| func SignalDuringExecPgrp() { |
| |
| f := os.NewFile(3, "pipe") |
| go func() { |
| |
| |
| |
| io.ReadAll(f) |
| os.Exit(1) |
| }() |
|
|
| |
| pgrp := syscall.Getpgrp() |
|
|
| const tries = 10 |
|
|
| var wg sync.WaitGroup |
| c := make(chan os.Signal, tries) |
| signal.Notify(c, syscall.SIGWINCH) |
| wg.Add(1) |
| go func() { |
| defer wg.Done() |
| for range c { |
| } |
| }() |
|
|
| for i := 0; i < tries; i++ { |
| time.Sleep(time.Microsecond) |
| wg.Add(2) |
| go func() { |
| defer wg.Done() |
| cmd := exec.Command(os.Args[0], "Nop") |
| cmd.Stdout = os.Stdout |
| cmd.Stderr = os.Stderr |
| if err := cmd.Run(); err != nil { |
| fmt.Printf("Start failed: %v", err) |
| } |
| }() |
| go func() { |
| defer wg.Done() |
| syscall.Kill(-pgrp, syscall.SIGWINCH) |
| }() |
| } |
|
|
| signal.Stop(c) |
| close(c) |
| wg.Wait() |
|
|
| fmt.Println("OK") |
| } |
|
|
| func Nop() { |
| |
| } |
|
|