tinygo/testdata/atomic.go
Ayke van Laethem e1052f921c compiler: define atomic intrinsic functions directly
This changes the compiler from treating calls to sync/atomic.* functions
as special calls (emitted directly at the call site) to actually
defining their declarations when there is no Go SSA implementation. And
rely on the inliner to inline these very small functions.
This works a bit better in practice. For example, this makes it possible
to use these functions in deferred function calls.

This commit is a bit large because it also needs to refactor a few
things to make it possible to define such intrinsic functions.
2022-06-24 11:10:24 +02:00

106 строки
3,4 КиБ
Go

package main
import (
"sync/atomic"
"unsafe"
)
func main() {
i32 := int32(-5)
println("AddInt32:", atomic.AddInt32(&i32, 8), i32)
i64 := int64(-5)
println("AddInt64:", atomic.AddInt64(&i64, 8), i64)
u32 := uint32(5)
println("AddUint32:", atomic.AddUint32(&u32, 8), u32)
u64 := uint64(5)
println("AddUint64:", atomic.AddUint64(&u64, 8), u64)
uptr := uintptr(5)
println("AddUintptr:", uint64(atomic.AddUintptr(&uptr, 8)), uint64(uptr))
println("SwapInt32:", atomic.SwapInt32(&i32, 33), i32)
println("SwapInt64:", atomic.SwapInt64(&i64, 33), i64)
println("SwapUint32:", atomic.SwapUint32(&u32, 33), u32)
println("SwapUint64:", atomic.SwapUint64(&u64, 33), u64)
println("SwapUintptr:", uint64(atomic.SwapUintptr(&uptr, 33)), uint64(uptr))
ptr := unsafe.Pointer(&i32)
println("SwapPointer:", atomic.SwapPointer(&ptr, unsafe.Pointer(&u32)) == unsafe.Pointer(&i32), ptr == unsafe.Pointer(&u32))
i32 = int32(-5)
println("CompareAndSwapInt32:", atomic.CompareAndSwapInt32(&i32, 5, 3), i32)
println("CompareAndSwapInt32:", atomic.CompareAndSwapInt32(&i32, -5, 3), i32)
i64 = int64(-5)
println("CompareAndSwapInt64:", atomic.CompareAndSwapInt64(&i64, 5, 3), i64)
println("CompareAndSwapInt64:", atomic.CompareAndSwapInt64(&i64, -5, 3), i64)
u32 = uint32(5)
println("CompareAndSwapUint32:", atomic.CompareAndSwapUint32(&u32, 4, 3), u32)
println("CompareAndSwapUint32:", atomic.CompareAndSwapUint32(&u32, 5, 3), u32)
u64 = uint64(5)
println("CompareAndSwapUint64:", atomic.CompareAndSwapUint64(&u64, 4, 3), u64)
println("CompareAndSwapUint64:", atomic.CompareAndSwapUint64(&u64, 5, 3), u64)
uptr = uintptr(5)
println("CompareAndSwapUintptr:", atomic.CompareAndSwapUintptr(&uptr, 4, 3), uint64(uptr))
println("CompareAndSwapUintptr:", atomic.CompareAndSwapUintptr(&uptr, 5, 3), uint64(uptr))
ptr = unsafe.Pointer(&i32)
println("CompareAndSwapPointer:", atomic.CompareAndSwapPointer(&ptr, unsafe.Pointer(&u32), unsafe.Pointer(&i64)), ptr == unsafe.Pointer(&i32))
println("CompareAndSwapPointer:", atomic.CompareAndSwapPointer(&ptr, unsafe.Pointer(&i32), unsafe.Pointer(&i64)), ptr == unsafe.Pointer(&i64))
println("LoadInt32:", atomic.LoadInt32(&i32))
println("LoadInt64:", atomic.LoadInt64(&i64))
println("LoadUint32:", atomic.LoadUint32(&u32))
println("LoadUint64:", atomic.LoadUint64(&u64))
println("LoadUintptr:", uint64(atomic.LoadUintptr(&uptr)))
println("LoadPointer:", atomic.LoadPointer(&ptr) == unsafe.Pointer(&i64))
atomic.StoreInt32(&i32, -20)
println("StoreInt32:", i32)
atomic.StoreInt64(&i64, -20)
println("StoreInt64:", i64)
atomic.StoreUint32(&u32, 20)
println("StoreUint32:", u32)
atomic.StoreUint64(&u64, 20)
println("StoreUint64:", u64)
atomic.StoreUintptr(&uptr, 20)
println("StoreUintptr:", uint64(uptr))
atomic.StorePointer(&ptr, unsafe.Pointer(&uptr))
println("StorePointer:", ptr == unsafe.Pointer(&uptr))
// test atomic.Value load/store operations
testValue(int(3), int(-2))
testValue("", "foobar", "baz")
// Test atomic operations as deferred values.
testDefer()
}
func testValue(values ...interface{}) {
var av atomic.Value
for _, val := range values {
av.Store(val)
loadedVal := av.Load()
if loadedVal != val {
println("val store/load didn't work, expected", val, "but got", loadedVal)
}
}
}
func testDefer() {
n1 := int32(5)
defer func() {
println("deferred atomic add:", n1)
}()
defer atomic.AddInt32(&n1, 3)
}