mirror of
https://github.com/SagerNet/sing.git
synced 2025-04-05 12:57:38 +03:00
Improve winpowrprof callback
This commit is contained in:
parent
e422e3d048
commit
7beca62e4f
3 changed files with 165 additions and 87 deletions
17
common/oncefunc.go
Normal file
17
common/oncefunc.go
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
//go:build go1.21
|
||||||
|
|
||||||
|
package common
|
||||||
|
|
||||||
|
import "sync"
|
||||||
|
|
||||||
|
func OnceFunc(f func()) func() {
|
||||||
|
return sync.OnceFunc(f)
|
||||||
|
}
|
||||||
|
|
||||||
|
func OnceValue[T any](f func() T) func() T {
|
||||||
|
return sync.OnceValue(f)
|
||||||
|
}
|
||||||
|
|
||||||
|
func OnceValues[T1, T2 any](f func() (T1, T2)) func() (T1, T2) {
|
||||||
|
return sync.OnceValues(f)
|
||||||
|
}
|
106
common/oncefunc_compat.go
Normal file
106
common/oncefunc_compat.go
Normal file
|
@ -0,0 +1,106 @@
|
||||||
|
//go:build !go1.21
|
||||||
|
|
||||||
|
// Copyright 2022 The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
// copied from go1.22.5
|
||||||
|
|
||||||
|
package common
|
||||||
|
|
||||||
|
import "sync"
|
||||||
|
|
||||||
|
// OnceFunc returns a function that invokes f only once. The returned function
|
||||||
|
// may be called concurrently.
|
||||||
|
//
|
||||||
|
// If f panics, the returned function will panic with the same value on every call.
|
||||||
|
func OnceFunc(f func()) func() {
|
||||||
|
var (
|
||||||
|
once sync.Once
|
||||||
|
valid bool
|
||||||
|
p any
|
||||||
|
)
|
||||||
|
// Construct the inner closure just once to reduce costs on the fast path.
|
||||||
|
g := func() {
|
||||||
|
defer func() {
|
||||||
|
p = recover()
|
||||||
|
if !valid {
|
||||||
|
// Re-panic immediately so on the first call the user gets a
|
||||||
|
// complete stack trace into f.
|
||||||
|
panic(p)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
f()
|
||||||
|
f = nil // Do not keep f alive after invoking it.
|
||||||
|
valid = true // Set only if f does not panic.
|
||||||
|
}
|
||||||
|
return func() {
|
||||||
|
once.Do(g)
|
||||||
|
if !valid {
|
||||||
|
panic(p)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// OnceValue returns a function that invokes f only once and returns the value
|
||||||
|
// returned by f. The returned function may be called concurrently.
|
||||||
|
//
|
||||||
|
// If f panics, the returned function will panic with the same value on every call.
|
||||||
|
func OnceValue[T any](f func() T) func() T {
|
||||||
|
var (
|
||||||
|
once sync.Once
|
||||||
|
valid bool
|
||||||
|
p any
|
||||||
|
result T
|
||||||
|
)
|
||||||
|
g := func() {
|
||||||
|
defer func() {
|
||||||
|
p = recover()
|
||||||
|
if !valid {
|
||||||
|
panic(p)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
result = f()
|
||||||
|
f = nil
|
||||||
|
valid = true
|
||||||
|
}
|
||||||
|
return func() T {
|
||||||
|
once.Do(g)
|
||||||
|
if !valid {
|
||||||
|
panic(p)
|
||||||
|
}
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// OnceValues returns a function that invokes f only once and returns the values
|
||||||
|
// returned by f. The returned function may be called concurrently.
|
||||||
|
//
|
||||||
|
// If f panics, the returned function will panic with the same value on every call.
|
||||||
|
func OnceValues[T1, T2 any](f func() (T1, T2)) func() (T1, T2) {
|
||||||
|
var (
|
||||||
|
once sync.Once
|
||||||
|
valid bool
|
||||||
|
p any
|
||||||
|
r1 T1
|
||||||
|
r2 T2
|
||||||
|
)
|
||||||
|
g := func() {
|
||||||
|
defer func() {
|
||||||
|
p = recover()
|
||||||
|
if !valid {
|
||||||
|
panic(p)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
r1, r2 = f()
|
||||||
|
f = nil
|
||||||
|
valid = true
|
||||||
|
}
|
||||||
|
return func() (T1, T2) {
|
||||||
|
once.Do(g)
|
||||||
|
if !valid {
|
||||||
|
panic(p)
|
||||||
|
}
|
||||||
|
return r1, r2
|
||||||
|
}
|
||||||
|
}
|
|
@ -3,77 +3,23 @@ package winpowrprof
|
||||||
// modify from https://github.com/golang/go/blob/b634f6fdcbebee23b7da709a243f3db217b64776/src/runtime/os_windows.go#L257
|
// modify from https://github.com/golang/go/blob/b634f6fdcbebee23b7da709a243f3db217b64776/src/runtime/os_windows.go#L257
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"sync"
|
|
||||||
"syscall"
|
"syscall"
|
||||||
"unsafe"
|
"unsafe"
|
||||||
|
|
||||||
"github.com/sagernet/sing/common/x/list"
|
"github.com/sagernet/sing/common"
|
||||||
|
|
||||||
"golang.org/x/sys/windows"
|
"golang.org/x/sys/windows"
|
||||||
)
|
)
|
||||||
|
|
||||||
type powerEventListener struct {
|
|
||||||
element *list.Element[EventCallback]
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewEventListener(callback EventCallback) (EventListener, error) {
|
|
||||||
err := initCallback()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
access.Lock()
|
|
||||||
defer access.Unlock()
|
|
||||||
return &powerEventListener{
|
|
||||||
element: callbackList.PushBack(callback),
|
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *powerEventListener) Start() error {
|
|
||||||
access.Lock()
|
|
||||||
defer access.Unlock()
|
|
||||||
if handle != 0 {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return startListener()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *powerEventListener) Close() error {
|
|
||||||
access.Lock()
|
|
||||||
defer access.Unlock()
|
|
||||||
if l.element != nil {
|
|
||||||
callbackList.Remove(l.element)
|
|
||||||
}
|
|
||||||
if callbackList.Len() > 0 {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return closeListener()
|
|
||||||
}
|
|
||||||
|
|
||||||
var (
|
var (
|
||||||
modpowerprof = windows.NewLazySystemDLL("powrprof.dll")
|
modpowerprof = windows.NewLazySystemDLL("powrprof.dll")
|
||||||
procPowerRegisterSuspendResumeNotification = modpowerprof.NewProc("PowerRegisterSuspendResumeNotification")
|
procPowerRegisterSuspendResumeNotification = modpowerprof.NewProc("PowerRegisterSuspendResumeNotification")
|
||||||
procPowerUnregisterSuspendResumeNotification = modpowerprof.NewProc("PowerUnregisterSuspendResumeNotification")
|
procPowerUnregisterSuspendResumeNotification = modpowerprof.NewProc("PowerUnregisterSuspendResumeNotification")
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var suspendResumeNotificationCallback = common.OnceValue(func() uintptr {
|
||||||
access sync.Mutex
|
return windows.NewCallback(func(context *EventCallback, changeType uint32, setting uintptr) uintptr {
|
||||||
callbackList list.List[EventCallback]
|
callback := *context
|
||||||
initCallbackOnce sync.Once
|
|
||||||
rawCallback uintptr
|
|
||||||
handle uintptr
|
|
||||||
)
|
|
||||||
|
|
||||||
func initCallback() error {
|
|
||||||
err := procPowerRegisterSuspendResumeNotification.Find()
|
|
||||||
if err != nil {
|
|
||||||
return err // Running on Windows 7, where we don't need it anyway.
|
|
||||||
}
|
|
||||||
err = procPowerUnregisterSuspendResumeNotification.Find()
|
|
||||||
if err != nil {
|
|
||||||
return err // Running on Windows 7, where we don't need it anyway.
|
|
||||||
}
|
|
||||||
initCallbackOnce.Do(func() {
|
|
||||||
rawCallback = windows.NewCallback(func(context uintptr, changeType uint32, setting uintptr) uintptr {
|
|
||||||
const (
|
const (
|
||||||
PBT_APMSUSPEND uint32 = 4
|
PBT_APMSUSPEND uint32 = 4
|
||||||
PBT_APMRESUMESUSPEND uint32 = 7
|
PBT_APMRESUMESUSPEND uint32 = 7
|
||||||
|
@ -90,32 +36,45 @@ func initCallback() error {
|
||||||
default:
|
default:
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
access.Lock()
|
|
||||||
callbacks := callbackList.Array()
|
|
||||||
access.Unlock()
|
|
||||||
for _, callback := range callbacks {
|
|
||||||
callback(event)
|
callback(event)
|
||||||
}
|
|
||||||
return 0
|
return 0
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
return nil
|
|
||||||
|
type powerEventListener struct {
|
||||||
|
callback EventCallback
|
||||||
|
handle uintptr
|
||||||
}
|
}
|
||||||
|
|
||||||
func startListener() error {
|
func NewEventListener(callback EventCallback) (EventListener, error) {
|
||||||
|
err := procPowerRegisterSuspendResumeNotification.Find()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err // Running on Windows 7, where we don't need it anyway.
|
||||||
|
}
|
||||||
|
err = procPowerUnregisterSuspendResumeNotification.Find()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err // Running on Windows 7, where we don't need it anyway.
|
||||||
|
}
|
||||||
|
return &powerEventListener{
|
||||||
|
callback: callback,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *powerEventListener) Start() error {
|
||||||
type DEVICE_NOTIFY_SUBSCRIBE_PARAMETERS struct {
|
type DEVICE_NOTIFY_SUBSCRIBE_PARAMETERS struct {
|
||||||
callback uintptr
|
callback uintptr
|
||||||
context uintptr
|
context unsafe.Pointer
|
||||||
}
|
}
|
||||||
const DEVICE_NOTIFY_CALLBACK = 2
|
const DEVICE_NOTIFY_CALLBACK = 2
|
||||||
params := DEVICE_NOTIFY_SUBSCRIBE_PARAMETERS{
|
params := DEVICE_NOTIFY_SUBSCRIBE_PARAMETERS{
|
||||||
callback: rawCallback,
|
callback: suspendResumeNotificationCallback(),
|
||||||
|
context: unsafe.Pointer(&l.callback),
|
||||||
}
|
}
|
||||||
_, _, errno := syscall.SyscallN(
|
_, _, errno := syscall.SyscallN(
|
||||||
procPowerRegisterSuspendResumeNotification.Addr(),
|
procPowerRegisterSuspendResumeNotification.Addr(),
|
||||||
DEVICE_NOTIFY_CALLBACK,
|
DEVICE_NOTIFY_CALLBACK,
|
||||||
uintptr(unsafe.Pointer(¶ms)),
|
uintptr(unsafe.Pointer(¶ms)),
|
||||||
uintptr(unsafe.Pointer(&handle)),
|
uintptr(unsafe.Pointer(&l.handle)),
|
||||||
)
|
)
|
||||||
if errno != 0 {
|
if errno != 0 {
|
||||||
return errno
|
return errno
|
||||||
|
@ -123,14 +82,10 @@ func startListener() error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func closeListener() error {
|
func (l *powerEventListener) Close() error {
|
||||||
if handle == 0 {
|
_, _, errno := syscall.SyscallN(procPowerUnregisterSuspendResumeNotification.Addr(), uintptr(unsafe.Pointer(&l.handle)))
|
||||||
return nil
|
|
||||||
}
|
|
||||||
_, _, errno := syscall.SyscallN(procPowerUnregisterSuspendResumeNotification.Addr(), uintptr(unsafe.Pointer(&handle)))
|
|
||||||
if errno != 0 {
|
if errno != 0 {
|
||||||
return errno
|
return errno
|
||||||
}
|
}
|
||||||
handle = 0
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue