// efaceWords is interface{} internal representation. type efaceWords struct { typ unsafe.Pointer data unsafe.Pointer }
其他
其他具体的数据结构有
Bool
Int32
Int64
Uint32
Uint64
Uintptr
Pointer
举个例子,Pointer类型的数据结构如下
1 2 3 4 5 6 7 8 9 10
// A Pointer is an atomic pointer of type *T. The zero value is a nil *T. type Pointer[T any] struct { // Mention *T in a field to disallow conversion between Pointer types. // See go.dev/issue/56603 for more details. // Use *T, not T, to avoid spurious recursive type definition errors. _ [0]*T
// Load returns the value set by the most recent Store. // It returns nil if there has been no call to Store for this Value. func(v *Value) Load() (val any) { vp := (*efaceWords)(unsafe.Pointer(v)) typ := LoadPointer(&vp.typ) if typ == nil || typ == unsafe.Pointer(&firstStoreInProgress) { // First store not yet completed. returnnil } data := LoadPointer(&vp.data) vlp := (*efaceWords)(unsafe.Pointer(&val)) vlp.typ = typ vlp.data = data return }
// Store sets the value of the [Value] v to val. // All calls to Store for a given Value must use values of the same concrete type. // Store of an inconsistent type panics, as does Store(nil). func(v *Value) Store(val any) { if val == nil { panic("sync/atomic: store of nil value into Value") } vp := (*efaceWords)(unsafe.Pointer(v)) vlp := (*efaceWords)(unsafe.Pointer(&val)) for { typ := LoadPointer(&vp.typ) if typ == nil { // Attempt to start first store. // Disable preemption so that other goroutines can use // active spin wait to wait for completion. runtime_procPin() if !CompareAndSwapPointer(&vp.typ, nil, unsafe.Pointer(&firstStoreInProgress)) { runtime_procUnpin() continue } // Complete first store. StorePointer(&vp.data, vlp.data) StorePointer(&vp.typ, vlp.typ) runtime_procUnpin() return } if typ == unsafe.Pointer(&firstStoreInProgress) { // First store in progress. Wait. // Since we disable preemption around the first store, // we can wait with active spinning. continue } // First store completed. Check type and overwrite data. if typ != vlp.typ { panic("sync/atomic: store of inconsistently typed value into Value") } StorePointer(&vp.data, vlp.data) return } }
// Swap stores new into Value and returns the previous value. It returns nil if // the Value is empty. // // All calls to Swap for a given Value must use values of the same concrete // type. Swap of an inconsistent type panics, as does Swap(nil). func(v *Value) Swap(new any) (old any) { ifnew == nil { panic("sync/atomic: swap of nil value into Value") } vp := (*efaceWords)(unsafe.Pointer(v)) np := (*efaceWords)(unsafe.Pointer(&new)) for { typ := LoadPointer(&vp.typ) if typ == nil { // Attempt to start first store. // Disable preemption so that other goroutines can use // active spin wait to wait for completion; and so that // GC does not see the fake type accidentally. runtime_procPin() if !CompareAndSwapPointer(&vp.typ, nil, unsafe.Pointer(&firstStoreInProgress)) { runtime_procUnpin() continue } // Complete first store. StorePointer(&vp.data, np.data) StorePointer(&vp.typ, np.typ) runtime_procUnpin() returnnil } if typ == unsafe.Pointer(&firstStoreInProgress) { // First store in progress. Wait. // Since we disable preemption around the first store, // we can wait with active spinning. continue } // First store completed. Check type and overwrite data. if typ != np.typ { panic("sync/atomic: swap of inconsistently typed value into Value") } op := (*efaceWords)(unsafe.Pointer(&old)) op.typ, op.data = np.typ, SwapPointer(&vp.data, np.data) return old } }
// CompareAndSwap executes the compare-and-swap operation for the [Value]. // // All calls to CompareAndSwap for a given Value must use values of the same // concrete type. CompareAndSwap of an inconsistent type panics, as does // CompareAndSwap(old, nil). func(v *Value) CompareAndSwap(old, new any) (swapped bool) { ifnew == nil { panic("sync/atomic: compare and swap of nil value into Value") } vp := (*efaceWords)(unsafe.Pointer(v)) np := (*efaceWords)(unsafe.Pointer(&new)) op := (*efaceWords)(unsafe.Pointer(&old)) if op.typ != nil && np.typ != op.typ { panic("sync/atomic: compare and swap of inconsistently typed values") } for { typ := LoadPointer(&vp.typ) if typ == nil { if old != nil { returnfalse } // Attempt to start first store. // Disable preemption so that other goroutines can use // active spin wait to wait for completion; and so that // GC does not see the fake type accidentally. runtime_procPin() if !CompareAndSwapPointer(&vp.typ, nil, unsafe.Pointer(&firstStoreInProgress)) { runtime_procUnpin() continue } // Complete first store. StorePointer(&vp.data, np.data) StorePointer(&vp.typ, np.typ) runtime_procUnpin() returntrue } if typ == unsafe.Pointer(&firstStoreInProgress) { // First store in progress. Wait. // Since we disable preemption around the first store, // we can wait with active spinning. continue } // First store completed. Check type and overwrite data. if typ != np.typ { panic("sync/atomic: compare and swap of inconsistently typed value into Value") } // Compare old and current via runtime equality check. // This allows value types to be compared, something // not offered by the package functions. // CompareAndSwapPointer below only ensures vp.data // has not changed since LoadPointer. data := LoadPointer(&vp.data) var i any (*efaceWords)(unsafe.Pointer(&i)).typ = typ (*efaceWords)(unsafe.Pointer(&i)).data = data if i != old { returnfalse } return CompareAndSwapPointer(&vp.data, data, np.data) } }
// And atomically performs a bitwise AND operation on x using the bitmask // provided as mask and returns the old value. func(x *Int64) And(mask int64) (old int64) { return AndInt64(&x.v, mask) }
AndInt64汇编代码如下
1 2 3 4 5
TEXT sync∕atomic·AndInt64(SB), NOSPLIT|NOFRAME, $0-24 // GO_ARGS MOVQ $__tsan_go_atomic64_fetch_and(SB), AX CALL racecallatomic<>(SB) RET
相关注释如下,只展示与AddInt64汇编代码不同的部份
字段
注释
GO_ARGS
这是一个宏,表示处理传入的 Go 参数,具体的参数内容会根据函数调用的上下文进行调整。这是 Go 编译器用来处理函数调用时的一些约定。它会将传入的参数从 Go 栈中提取到寄存器中
Go 的一个内建函数,用来执行原子操作并同时为 ThreadSanitizer(TSan)提供监控支持。它会根据 AX 寄存器中存储的函数地址(即 __tsan_go_atomic64_fetch_and)执行对应的操作
<>
表示泛型类型参数,在 Go 汇编中表示函数的参数类型。racecallatomic 函数会根据这些类型参数来处理实际的原子操作。
Or
原子位或(|)操作,新结果写入变量,返回旧的数据
1 2 3
// Or atomically performs a bitwise OR operation on x using the bitmask // provided as mask and returns the old value. func(x *Int64) Or(mask int64) (old int64) { return OrInt64(&x.v, mask) }
OrInt64的汇编代码如下,不再展示注释
1 2 3 4 5
TEXT sync∕atomic·OrInt64(SB), NOSPLIT|NOFRAME, $0-24 GO_ARGS MOVQ $__tsan_go_atomic64_fetch_or(SB), AX CALL racecallatomic<>(SB) RET