mirror of
https://github.com/SagerNet/sing.git
synced 2025-04-03 20:07:38 +03:00
Fix TypedValue
This commit is contained in:
parent
807a51bb81
commit
a069af4787
1 changed files with 16 additions and 5 deletions
|
@ -10,26 +10,37 @@ type TypedValue[T any] struct {
|
||||||
value atomic.Value
|
value atomic.Value
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// typedValue is a struct with determined type to resolve atomic.Value usages with interface types
|
||||||
|
// https://github.com/golang/go/issues/22550
|
||||||
|
//
|
||||||
|
// The intention to have an atomic value store for errors. However, running this code panics:
|
||||||
|
// panic: sync/atomic: store of inconsistently typed value into Value
|
||||||
|
// This is because atomic.Value requires that the underlying concrete type be the same (which is a reasonable expectation for its implementation).
|
||||||
|
// When going through the atomic.Value.Store method call, the fact that both these are of the error interface is lost.
|
||||||
|
type typedValue[T any] struct {
|
||||||
|
value T
|
||||||
|
}
|
||||||
|
|
||||||
func (t *TypedValue[T]) Load() T {
|
func (t *TypedValue[T]) Load() T {
|
||||||
value := t.value.Load()
|
value := t.value.Load()
|
||||||
if value == nil {
|
if value == nil {
|
||||||
return common.DefaultValue[T]()
|
return common.DefaultValue[T]()
|
||||||
}
|
}
|
||||||
return value.(T)
|
return value.(typedValue[T]).value
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *TypedValue[T]) Store(value T) {
|
func (t *TypedValue[T]) Store(value T) {
|
||||||
t.value.Store(value)
|
t.value.Store(typedValue[T]{value})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *TypedValue[T]) Swap(new T) T {
|
func (t *TypedValue[T]) Swap(new T) T {
|
||||||
old := t.value.Swap(new)
|
old := t.value.Swap(typedValue[T]{new})
|
||||||
if old == nil {
|
if old == nil {
|
||||||
return common.DefaultValue[T]()
|
return common.DefaultValue[T]()
|
||||||
}
|
}
|
||||||
return old.(T)
|
return old.(typedValue[T]).value
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *TypedValue[T]) CompareAndSwap(old, new T) bool {
|
func (t *TypedValue[T]) CompareAndSwap(old, new T) bool {
|
||||||
return t.value.CompareAndSwap(old, new)
|
return t.value.CompareAndSwap(typedValue[T]{old}, typedValue[T]{new})
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue