2018-02-05 21:05:59 +00:00
|
|
|
package daemon // import "github.com/docker/docker/daemon"
|
2015-07-07 01:58:53 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"os"
|
2016-10-14 00:00:58 +00:00
|
|
|
"unsafe"
|
2015-07-07 01:58:53 +00:00
|
|
|
|
2016-10-14 00:00:58 +00:00
|
|
|
winio "github.com/Microsoft/go-winio"
|
2016-03-28 00:23:34 +00:00
|
|
|
"github.com/docker/docker/pkg/signal"
|
2017-07-26 21:42:13 +00:00
|
|
|
"github.com/sirupsen/logrus"
|
2017-05-23 14:22:32 +00:00
|
|
|
"golang.org/x/sys/windows"
|
2015-07-07 01:58:53 +00:00
|
|
|
)
|
|
|
|
|
2016-10-12 22:29:47 +00:00
|
|
|
func (d *Daemon) setupDumpStackTrap(root string) {
|
2015-07-07 01:58:53 +00:00
|
|
|
// Windows does not support signals like *nix systems. So instead of
|
|
|
|
// trapping on SIGUSR1 to dump stacks, we wait on a Win32 event to be
|
2016-10-14 00:00:58 +00:00
|
|
|
// signaled. ACL'd to builtin administrators and local system
|
2019-02-15 23:26:56 +00:00
|
|
|
event := "Global\\stackdump-" + fmt.Sprint(os.Getpid())
|
2017-10-19 18:09:29 +00:00
|
|
|
ev, _ := windows.UTF16PtrFromString(event)
|
2016-10-14 00:00:58 +00:00
|
|
|
sd, err := winio.SddlToSecurityDescriptor("D:P(A;;GA;;;BA)(A;;GA;;;SY)")
|
|
|
|
if err != nil {
|
2017-10-19 18:09:29 +00:00
|
|
|
logrus.Errorf("failed to get security descriptor for debug stackdump event %s: %s", event, err.Error())
|
2016-10-14 00:00:58 +00:00
|
|
|
return
|
|
|
|
}
|
2017-05-23 14:22:32 +00:00
|
|
|
var sa windows.SecurityAttributes
|
2016-10-14 00:00:58 +00:00
|
|
|
sa.Length = uint32(unsafe.Sizeof(sa))
|
|
|
|
sa.InheritHandle = 1
|
|
|
|
sa.SecurityDescriptor = uintptr(unsafe.Pointer(&sd[0]))
|
2017-08-21 10:58:09 +00:00
|
|
|
h, err := windows.CreateEvent(&sa, 0, 0, ev)
|
2016-10-14 00:00:58 +00:00
|
|
|
if h == 0 || err != nil {
|
2017-10-19 18:09:29 +00:00
|
|
|
logrus.Errorf("failed to create debug stackdump event %s: %s", event, err.Error())
|
2016-10-14 00:00:58 +00:00
|
|
|
return
|
|
|
|
}
|
2015-07-07 01:58:53 +00:00
|
|
|
go func() {
|
2017-10-19 18:09:29 +00:00
|
|
|
logrus.Debugf("Stackdump - waiting signal at %s", event)
|
2016-10-14 00:00:58 +00:00
|
|
|
for {
|
2017-05-23 14:22:32 +00:00
|
|
|
windows.WaitForSingleObject(h, windows.INFINITE)
|
2016-11-01 14:32:55 +00:00
|
|
|
path, err := signal.DumpStacks(root)
|
|
|
|
if err != nil {
|
|
|
|
logrus.WithError(err).Error("failed to write goroutines dump")
|
2016-10-12 22:29:47 +00:00
|
|
|
} else {
|
|
|
|
logrus.Infof("goroutine stacks written to %s", path)
|
|
|
|
}
|
2015-07-07 01:58:53 +00:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|