//go:build windows // +build windows package windowskext import ( "fmt" "syscall" "time" "golang.org/x/sys/windows" ) type KextService struct { handle windows.Handle } func createKextService(driverName string, driverPath string) (*KextService, error) { // Open the service manager: manager, err := windows.OpenSCManager(nil, nil, windows.SC_MANAGER_ALL_ACCESS) if err != nil { return nil, fmt.Errorf("failed to open service manager: %d", err) } defer windows.CloseServiceHandle(manager) driverNameU16, err := syscall.UTF16FromString(driverName) if err != nil { return nil, fmt.Errorf("failed to convert driver name to UTF16 string: %w", err) } // Check if it's already created service, err := windows.OpenService(manager, &driverNameU16[0], windows.SERVICE_ALL_ACCESS) if err == nil { return &KextService{handle: service}, nil // service was already created } driverPathU16, err := syscall.UTF16FromString(driverPath) // Create the service service, err = windows.CreateService(manager, &driverNameU16[0], &driverNameU16[0], windows.SERVICE_ALL_ACCESS, windows.SERVICE_KERNEL_DRIVER, windows.SERVICE_DEMAND_START, windows.SERVICE_ERROR_NORMAL, &driverPathU16[0], nil, nil, nil, nil, nil) if err != nil { return nil, err } return &KextService{handle: service}, nil } func (s *KextService) isValid() bool { return s != nil && s.handle != winInvalidHandleValue && s.handle != 0 } func (s *KextService) isRunning() (bool, error) { if !s.isValid() { return false, fmt.Errorf("kext service not initialized") } var status windows.SERVICE_STATUS err := windows.QueryServiceStatus(s.handle, &status) if err != nil { return false, err } return status.CurrentState == windows.SERVICE_RUNNING, nil } func waitForServiceStatus(handle windows.Handle, neededStatus uint32, timeLimit time.Duration) (bool, error) { var status windows.SERVICE_STATUS status.CurrentState = windows.SERVICE_NO_CHANGE start := time.Now() for status.CurrentState == neededStatus { err := windows.QueryServiceStatus(handle, &status) if err != nil { return false, fmt.Errorf("failed while waiting for service to start: %w", err) } if time.Since(start) > timeLimit { return false, fmt.Errorf("time limit reached") } // Sleep for 1/10 of the wait hint, recommended time from microsoft time.Sleep(time.Duration((status.WaitHint / 10)) * time.Millisecond) } return true, nil } func (s *KextService) start(wait bool) error { if !s.isValid() { return fmt.Errorf("kext service not initialized") } // Start the service: err := windows.StartService(s.handle, 0, nil) if err != nil { err = windows.GetLastError() if err != windows.ERROR_SERVICE_ALREADY_RUNNING { // Failed to start service; clean-up: var status windows.SERVICE_STATUS _ = windows.ControlService(s.handle, windows.SERVICE_CONTROL_STOP, &status) _ = windows.DeleteService(s.handle) _ = windows.CloseServiceHandle(s.handle) s.handle = winInvalidHandleValue return err } } // Wait for service to start if wait { success, err := waitForServiceStatus(s.handle, windows.SERVICE_RUNNING, time.Duration(10*time.Second)) if err != nil || !success { return fmt.Errorf("service did not start: %w", err) } } return nil } func (s *KextService) stop(wait bool) error { if !s.isValid() { return fmt.Errorf("kext service not initialized") } // Stop the service var status windows.SERVICE_STATUS err := windows.ControlService(s.handle, windows.SERVICE_CONTROL_STOP, &status) if err != nil { return fmt.Errorf("service failed to stop: %w", err) } // Wait for service to stop if wait { success, err := waitForServiceStatus(s.handle, windows.SERVICE_STOPPED, time.Duration(10*time.Second)) if err != nil || !success { return fmt.Errorf("service did not stop: %w", err) } } return nil } func (s *KextService) delete() error { if !s.isValid() { return fmt.Errorf("kext service not initialized") } err := windows.DeleteService(s.handle) if err != nil { return fmt.Errorf("failed to delete service: %s", err) } return nil } func (s *KextService) closeHandle() error { if !s.isValid() { return fmt.Errorf("kext service not initialized") } err := windows.CloseServiceHandle(s.handle) if err != nil { return fmt.Errorf("failed to close service handle: %s", err) } return nil }