mirror of https://github.com/k3s-io/k3s
89 lines
2.8 KiB
Go
89 lines
2.8 KiB
Go
|
package hcs
|
||
|
|
||
|
import (
|
||
|
"sync"
|
||
|
"syscall"
|
||
|
|
||
|
"github.com/Microsoft/hcsshim/internal/interop"
|
||
|
"github.com/sirupsen/logrus"
|
||
|
)
|
||
|
|
||
|
var (
|
||
|
nextCallback uintptr
|
||
|
callbackMap = map[uintptr]*notifcationWatcherContext{}
|
||
|
callbackMapLock = sync.RWMutex{}
|
||
|
|
||
|
notificationWatcherCallback = syscall.NewCallback(notificationWatcher)
|
||
|
|
||
|
// Notifications for HCS_SYSTEM handles
|
||
|
hcsNotificationSystemExited hcsNotification = 0x00000001
|
||
|
hcsNotificationSystemCreateCompleted hcsNotification = 0x00000002
|
||
|
hcsNotificationSystemStartCompleted hcsNotification = 0x00000003
|
||
|
hcsNotificationSystemPauseCompleted hcsNotification = 0x00000004
|
||
|
hcsNotificationSystemResumeCompleted hcsNotification = 0x00000005
|
||
|
|
||
|
// Notifications for HCS_PROCESS handles
|
||
|
hcsNotificationProcessExited hcsNotification = 0x00010000
|
||
|
|
||
|
// Common notifications
|
||
|
hcsNotificationInvalid hcsNotification = 0x00000000
|
||
|
hcsNotificationServiceDisconnect hcsNotification = 0x01000000
|
||
|
)
|
||
|
|
||
|
type hcsNotification uint32
|
||
|
type notificationChannel chan error
|
||
|
|
||
|
type notifcationWatcherContext struct {
|
||
|
channels notificationChannels
|
||
|
handle hcsCallback
|
||
|
}
|
||
|
|
||
|
type notificationChannels map[hcsNotification]notificationChannel
|
||
|
|
||
|
func newChannels() notificationChannels {
|
||
|
channels := make(notificationChannels)
|
||
|
|
||
|
channels[hcsNotificationSystemExited] = make(notificationChannel, 1)
|
||
|
channels[hcsNotificationSystemCreateCompleted] = make(notificationChannel, 1)
|
||
|
channels[hcsNotificationSystemStartCompleted] = make(notificationChannel, 1)
|
||
|
channels[hcsNotificationSystemPauseCompleted] = make(notificationChannel, 1)
|
||
|
channels[hcsNotificationSystemResumeCompleted] = make(notificationChannel, 1)
|
||
|
channels[hcsNotificationProcessExited] = make(notificationChannel, 1)
|
||
|
channels[hcsNotificationServiceDisconnect] = make(notificationChannel, 1)
|
||
|
return channels
|
||
|
}
|
||
|
func closeChannels(channels notificationChannels) {
|
||
|
close(channels[hcsNotificationSystemExited])
|
||
|
close(channels[hcsNotificationSystemCreateCompleted])
|
||
|
close(channels[hcsNotificationSystemStartCompleted])
|
||
|
close(channels[hcsNotificationSystemPauseCompleted])
|
||
|
close(channels[hcsNotificationSystemResumeCompleted])
|
||
|
close(channels[hcsNotificationProcessExited])
|
||
|
close(channels[hcsNotificationServiceDisconnect])
|
||
|
}
|
||
|
|
||
|
func notificationWatcher(notificationType hcsNotification, callbackNumber uintptr, notificationStatus uintptr, notificationData *uint16) uintptr {
|
||
|
var result error
|
||
|
if int32(notificationStatus) < 0 {
|
||
|
result = interop.Win32FromHresult(notificationStatus)
|
||
|
}
|
||
|
|
||
|
callbackMapLock.RLock()
|
||
|
context := callbackMap[callbackNumber]
|
||
|
callbackMapLock.RUnlock()
|
||
|
|
||
|
if context == nil {
|
||
|
return 0
|
||
|
}
|
||
|
|
||
|
if channel, ok := context.channels[notificationType]; ok {
|
||
|
channel <- result
|
||
|
} else {
|
||
|
logrus.WithFields(logrus.Fields{
|
||
|
"notification-type": notificationType,
|
||
|
}).Warn("Received a callback of an unsupported type")
|
||
|
}
|
||
|
|
||
|
return 0
|
||
|
}
|