From a069af4787256779fa6001620ab0a6b807e4d95c Mon Sep 17 00:00:00 2001 From: wwqgtxx Date: Wed, 13 Mar 2024 14:47:26 +0800 Subject: [PATCH] Fix TypedValue --- common/atomic/typed.go | 21 ++++++++++++++++----- 1 file changed, 16 insertions(+), 5 deletions(-) diff --git a/common/atomic/typed.go b/common/atomic/typed.go index 23f77fb..19934f8 100644 --- a/common/atomic/typed.go +++ b/common/atomic/typed.go @@ -10,26 +10,37 @@ type TypedValue[T any] struct { 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 { value := t.value.Load() if value == nil { return common.DefaultValue[T]() } - return value.(T) + return value.(typedValue[T]).value } 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 { - old := t.value.Swap(new) + old := t.value.Swap(typedValue[T]{new}) if old == nil { return common.DefaultValue[T]() } - return old.(T) + return old.(typedValue[T]).value } 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}) }