diff options
| author | Austin Clements <austin@google.com> | 2018-01-15 00:00:02 -0500 |
|---|---|---|
| committer | Austin Clements <austin@google.com> | 2018-02-13 16:34:45 +0000 |
| commit | 245310883dcae717bb662b22d5b1fd07fdd59b76 (patch) | |
| tree | 4b4af5e2a49dc9ea1f5a9cb3eed3fc2231a3f526 /src/runtime/atomic_pointer.go | |
| parent | 2ae1e1ae2f8726057914f26d5360c3403b8f049a (diff) | |
| download | go-245310883dcae717bb662b22d5b1fd07fdd59b76.tar.xz | |
runtime: eliminate all writebarrierptr* calls
Calls to writebarrierptr can simply be actual pointer writes. Calls to
writebarrierptr_prewrite need to go through the write barrier buffer.
Updates #22460.
Change-Id: I92cee4da98c5baa499f1977563757c76f95bf0ca
Reviewed-on: https://go-review.googlesource.com/92704
Run-TryBot: Austin Clements <austin@google.com>
Reviewed-by: Rick Hudson <rlh@golang.org>
Diffstat (limited to 'src/runtime/atomic_pointer.go')
| -rw-r--r-- | src/runtime/atomic_pointer.go | 31 |
1 files changed, 26 insertions, 5 deletions
diff --git a/src/runtime/atomic_pointer.go b/src/runtime/atomic_pointer.go index 292b3517ad..09cfbda9b1 100644 --- a/src/runtime/atomic_pointer.go +++ b/src/runtime/atomic_pointer.go @@ -16,11 +16,24 @@ import ( // Instead, these are wrappers around the actual atomics (casp1 and so on) // that use noescape to convey which arguments do not escape. +// atomicwb performs a write barrier before an atomic pointer write. +// The caller should guard the call with "if writeBarrier.enabled". +// +//go:nosplit +func atomicwb(ptr *unsafe.Pointer, new unsafe.Pointer) { + slot := (*uintptr)(unsafe.Pointer(ptr)) + if !getg().m.p.ptr().wbBuf.putFast(*slot, uintptr(new)) { + wbBufFlush(slot, uintptr(new)) + } +} + // atomicstorep performs *ptr = new atomically and invokes a write barrier. // //go:nosplit func atomicstorep(ptr unsafe.Pointer, new unsafe.Pointer) { - writebarrierptr_prewrite((*uintptr)(ptr), uintptr(new)) + if writeBarrier.enabled { + atomicwb((*unsafe.Pointer)(ptr), new) + } atomic.StorepNoWB(noescape(ptr), new) } @@ -29,7 +42,9 @@ func casp(ptr *unsafe.Pointer, old, new unsafe.Pointer) bool { // The write barrier is only necessary if the CAS succeeds, // but since it needs to happen before the write becomes // public, we have to do it conservatively all the time. - writebarrierptr_prewrite((*uintptr)(unsafe.Pointer(ptr)), uintptr(new)) + if writeBarrier.enabled { + atomicwb(ptr, new) + } return atomic.Casp1((*unsafe.Pointer)(noescape(unsafe.Pointer(ptr))), noescape(old), new) } @@ -43,7 +58,9 @@ func sync_atomic_StoreUintptr(ptr *uintptr, new uintptr) //go:linkname sync_atomic_StorePointer sync/atomic.StorePointer //go:nosplit func sync_atomic_StorePointer(ptr *unsafe.Pointer, new unsafe.Pointer) { - writebarrierptr_prewrite((*uintptr)(unsafe.Pointer(ptr)), uintptr(new)) + if writeBarrier.enabled { + atomicwb(ptr, new) + } sync_atomic_StoreUintptr((*uintptr)(unsafe.Pointer(ptr)), uintptr(new)) } @@ -53,7 +70,9 @@ func sync_atomic_SwapUintptr(ptr *uintptr, new uintptr) uintptr //go:linkname sync_atomic_SwapPointer sync/atomic.SwapPointer //go:nosplit func sync_atomic_SwapPointer(ptr *unsafe.Pointer, new unsafe.Pointer) unsafe.Pointer { - writebarrierptr_prewrite((*uintptr)(unsafe.Pointer(ptr)), uintptr(new)) + if writeBarrier.enabled { + atomicwb(ptr, new) + } old := unsafe.Pointer(sync_atomic_SwapUintptr((*uintptr)(noescape(unsafe.Pointer(ptr))), uintptr(new))) return old } @@ -64,6 +83,8 @@ func sync_atomic_CompareAndSwapUintptr(ptr *uintptr, old, new uintptr) bool //go:linkname sync_atomic_CompareAndSwapPointer sync/atomic.CompareAndSwapPointer //go:nosplit func sync_atomic_CompareAndSwapPointer(ptr *unsafe.Pointer, old, new unsafe.Pointer) bool { - writebarrierptr_prewrite((*uintptr)(unsafe.Pointer(ptr)), uintptr(new)) + if writeBarrier.enabled { + atomicwb(ptr, new) + } return sync_atomic_CompareAndSwapUintptr((*uintptr)(noescape(unsafe.Pointer(ptr))), uintptr(old), uintptr(new)) } |
