golang.org/x/sys@v0.9.0/windows/svc/mgr/service.go (about) 1 // Copyright 2012 The Go Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style 3 // license that can be found in the LICENSE file. 4 5 //go:build windows 6 // +build windows 7 8 package mgr 9 10 import ( 11 "syscall" 12 "unsafe" 13 14 "golang.org/x/sys/windows" 15 "golang.org/x/sys/windows/svc" 16 ) 17 18 // Service is used to access Windows service. 19 type Service struct { 20 Name string 21 Handle windows.Handle 22 } 23 24 // Delete marks service s for deletion from the service control manager database. 25 func (s *Service) Delete() error { 26 return windows.DeleteService(s.Handle) 27 } 28 29 // Close relinquish access to the service s. 30 func (s *Service) Close() error { 31 return windows.CloseServiceHandle(s.Handle) 32 } 33 34 // Start starts service s. 35 // args will be passed to svc.Handler.Execute. 36 func (s *Service) Start(args ...string) error { 37 var p **uint16 38 if len(args) > 0 { 39 vs := make([]*uint16, len(args)) 40 for i := range vs { 41 vs[i] = syscall.StringToUTF16Ptr(args[i]) 42 } 43 p = &vs[0] 44 } 45 return windows.StartService(s.Handle, uint32(len(args)), p) 46 } 47 48 // Control sends state change request c to the service s. It returns the most 49 // recent status the service reported to the service control manager, and an 50 // error if the state change request was not accepted. 51 // Note that the returned service status is only set if the status change 52 // request succeeded, or if it failed with error ERROR_INVALID_SERVICE_CONTROL, 53 // ERROR_SERVICE_CANNOT_ACCEPT_CTRL, or ERROR_SERVICE_NOT_ACTIVE. 54 func (s *Service) Control(c svc.Cmd) (svc.Status, error) { 55 var t windows.SERVICE_STATUS 56 err := windows.ControlService(s.Handle, uint32(c), &t) 57 if err != nil && 58 err != windows.ERROR_INVALID_SERVICE_CONTROL && 59 err != windows.ERROR_SERVICE_CANNOT_ACCEPT_CTRL && 60 err != windows.ERROR_SERVICE_NOT_ACTIVE { 61 return svc.Status{}, err 62 } 63 return svc.Status{ 64 State: svc.State(t.CurrentState), 65 Accepts: svc.Accepted(t.ControlsAccepted), 66 }, err 67 } 68 69 // Query returns current status of service s. 70 func (s *Service) Query() (svc.Status, error) { 71 var t windows.SERVICE_STATUS_PROCESS 72 var needed uint32 73 err := windows.QueryServiceStatusEx(s.Handle, windows.SC_STATUS_PROCESS_INFO, (*byte)(unsafe.Pointer(&t)), uint32(unsafe.Sizeof(t)), &needed) 74 if err != nil { 75 return svc.Status{}, err 76 } 77 return svc.Status{ 78 State: svc.State(t.CurrentState), 79 Accepts: svc.Accepted(t.ControlsAccepted), 80 ProcessId: t.ProcessId, 81 Win32ExitCode: t.Win32ExitCode, 82 ServiceSpecificExitCode: t.ServiceSpecificExitCode, 83 }, nil 84 } 85 86 // ListDependentServices returns the names of the services dependent on service s, which match the given status. 87 func (s *Service) ListDependentServices(status svc.ActivityStatus) ([]string, error) { 88 var bytesNeeded, returnedServiceCount uint32 89 var services []windows.ENUM_SERVICE_STATUS 90 for { 91 var servicesPtr *windows.ENUM_SERVICE_STATUS 92 if len(services) > 0 { 93 servicesPtr = &services[0] 94 } 95 allocatedBytes := uint32(len(services)) * uint32(unsafe.Sizeof(windows.ENUM_SERVICE_STATUS{})) 96 err := windows.EnumDependentServices(s.Handle, uint32(status), servicesPtr, allocatedBytes, &bytesNeeded, 97 &returnedServiceCount) 98 if err == nil { 99 break 100 } 101 if err != syscall.ERROR_MORE_DATA { 102 return nil, err 103 } 104 if bytesNeeded <= allocatedBytes { 105 return nil, err 106 } 107 // ERROR_MORE_DATA indicates the provided buffer was too small, run the call again after resizing the buffer 108 requiredSliceLen := bytesNeeded / uint32(unsafe.Sizeof(windows.ENUM_SERVICE_STATUS{})) 109 if bytesNeeded%uint32(unsafe.Sizeof(windows.ENUM_SERVICE_STATUS{})) != 0 { 110 requiredSliceLen += 1 111 } 112 services = make([]windows.ENUM_SERVICE_STATUS, requiredSliceLen) 113 } 114 if returnedServiceCount == 0 { 115 return nil, nil 116 } 117 118 // The slice mutated by EnumDependentServices may have a length greater than returnedServiceCount, any elements 119 // past that should be ignored. 120 var dependents []string 121 for i := 0; i < int(returnedServiceCount); i++ { 122 dependents = append(dependents, windows.UTF16PtrToString(services[i].ServiceName)) 123 } 124 return dependents, nil 125 }