debugtrap_windows.go 1.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445
  1. package daemon
  2. import (
  3. "fmt"
  4. "os"
  5. "unsafe"
  6. winio "github.com/Microsoft/go-winio"
  7. "github.com/docker/docker/pkg/signal"
  8. "github.com/sirupsen/logrus"
  9. "golang.org/x/sys/windows"
  10. )
  11. func (d *Daemon) setupDumpStackTrap(root string) {
  12. // Windows does not support signals like *nix systems. So instead of
  13. // trapping on SIGUSR1 to dump stacks, we wait on a Win32 event to be
  14. // signaled. ACL'd to builtin administrators and local system
  15. ev, _ := windows.UTF16PtrFromString("Global\\docker-daemon-" + fmt.Sprint(os.Getpid()))
  16. sd, err := winio.SddlToSecurityDescriptor("D:P(A;;GA;;;BA)(A;;GA;;;SY)")
  17. if err != nil {
  18. logrus.Errorf("failed to get security descriptor for debug stackdump event %s: %s", ev, err.Error())
  19. return
  20. }
  21. var sa windows.SecurityAttributes
  22. sa.Length = uint32(unsafe.Sizeof(sa))
  23. sa.InheritHandle = 1
  24. sa.SecurityDescriptor = uintptr(unsafe.Pointer(&sd[0]))
  25. h, err := windows.CreateEvent(&sa, 0, 0, ev)
  26. if h == 0 || err != nil {
  27. logrus.Errorf("failed to create debug stackdump event %s: %s", ev, err.Error())
  28. return
  29. }
  30. go func() {
  31. logrus.Debugf("Stackdump - waiting signal at %s", ev)
  32. for {
  33. windows.WaitForSingleObject(h, windows.INFINITE)
  34. path, err := signal.DumpStacks(root)
  35. if err != nil {
  36. logrus.WithError(err).Error("failed to write goroutines dump")
  37. } else {
  38. logrus.Infof("goroutine stacks written to %s", path)
  39. }
  40. }
  41. }()
  42. }