mirror of
https://github.com/netbirdio/gvisor.git
synced 2026-05-22 17:12:49 -07:00
f6ed4523dc
PiperOrigin-RevId: 449358041
232 lines
5.7 KiB
Go
232 lines
5.7 KiB
Go
// Copyright 2021 The gVisor Authors.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
//go:build arm || mips || mipsle || 386
|
|
// +build arm mips mipsle 386
|
|
|
|
package atomicbitops
|
|
|
|
import (
|
|
"sync/atomic"
|
|
"unsafe"
|
|
|
|
"gvisor.dev/gvisor/pkg/sync"
|
|
)
|
|
|
|
// Int64 is an atomic int64 that is guaranteed to be 64-bit
|
|
// aligned, even on 32-bit systems.
|
|
//
|
|
// Don't add fields to this struct. It is important that it remain the same
|
|
// size as its builtin analogue.
|
|
//
|
|
// Per https://golang.org/pkg/sync/atomic/#pkg-note-BUG:
|
|
//
|
|
// "On ARM, 386, and 32-bit MIPS, it is the caller's responsibility to arrange
|
|
// for 64-bit alignment of 64-bit words accessed atomically. The first word in
|
|
// a variable or in an allocated struct, array, or slice can be relied upon to
|
|
// be 64-bit aligned."
|
|
//
|
|
// +stateify savable
|
|
type Int64 struct {
|
|
_ sync.NoCopy
|
|
value int64
|
|
value32 int32
|
|
}
|
|
|
|
//go:nosplit
|
|
func (i *Int64) ptr() *int64 {
|
|
// On 32-bit systems, i.value is guaranteed to be 32-bit aligned. It means
|
|
// that in the 12-byte i.value, there are guaranteed to be 8 contiguous bytes
|
|
// with 64-bit alignment.
|
|
return (*int64)(unsafe.Pointer((uintptr(unsafe.Pointer(&i.value)) + 4) &^ 7))
|
|
}
|
|
|
|
// FromInt64 returns an Int64 initialized to value v.
|
|
//
|
|
//go:nosplit
|
|
func FromInt64(v int64) Int64 {
|
|
var i Int64
|
|
*i.ptr() = v
|
|
return i
|
|
}
|
|
|
|
// Load is analogous to atomic.LoadInt64.
|
|
//
|
|
//go:nosplit
|
|
func (i *Int64) Load() int64 {
|
|
return atomic.LoadInt64(i.ptr())
|
|
}
|
|
|
|
// RacyLoad is analogous to reading an atomic value without using
|
|
// synchronization.
|
|
//
|
|
// It may be helpful to document why a racy operation is permitted.
|
|
//
|
|
//go:nosplit
|
|
func (i *Int64) RacyLoad() int64 {
|
|
return *i.ptr()
|
|
}
|
|
|
|
// Store is analogous to atomic.StoreInt64.
|
|
//
|
|
//go:nosplit
|
|
func (i *Int64) Store(v int64) {
|
|
atomic.StoreInt64(i.ptr(), v)
|
|
}
|
|
|
|
// RacyStore is analogous to setting an atomic value without using
|
|
// synchronization.
|
|
//
|
|
// It may be helpful to document why a racy operation is permitted.
|
|
//
|
|
//go:nosplit
|
|
func (i *Int64) RacyStore(v int64) {
|
|
*i.ptr() = v
|
|
}
|
|
|
|
// Add is analogous to atomic.AddInt64.
|
|
//
|
|
//go:nosplit
|
|
func (i *Int64) Add(v int64) int64 {
|
|
return atomic.AddInt64(i.ptr(), v)
|
|
}
|
|
|
|
// RacyAdd is analogous to adding to an atomic value without using
|
|
// synchronization.
|
|
//
|
|
// It may be helpful to document why a racy operation is permitted.
|
|
//
|
|
//go:nosplit
|
|
func (i *Int64) RacyAdd(v int64) int64 {
|
|
*i.ptr() += v
|
|
return *i.ptr()
|
|
}
|
|
|
|
// Swap is analogous to atomic.SwapInt64.
|
|
//
|
|
//go:nosplit
|
|
func (i *Int64) Swap(v int64) int64 {
|
|
return atomic.SwapInt64(i.ptr(), v)
|
|
}
|
|
|
|
// CompareAndSwap is analogous to atomic.CompareAndSwapInt64.
|
|
//
|
|
//go:nosplit
|
|
func (i *Int64) CompareAndSwap(oldVal, newVal int64) bool {
|
|
return atomic.CompareAndSwapInt64(&i.value, oldVal, newVal)
|
|
}
|
|
|
|
// Uint64 is an atomic uint64 that is guaranteed to be 64-bit
|
|
// aligned, even on 32-bit systems.
|
|
//
|
|
// Don't add fields to this struct. It is important that it remain the same
|
|
// size as its builtin analogue.
|
|
//
|
|
// Per https://golang.org/pkg/sync/atomic/#pkg-note-BUG:
|
|
//
|
|
// "On ARM, 386, and 32-bit MIPS, it is the caller's responsibility to arrange
|
|
// for 64-bit alignment of 64-bit words accessed atomically. The first word in
|
|
// a variable or in an allocated struct, array, or slice can be relied upon to
|
|
// be 64-bit aligned."
|
|
//
|
|
// +stateify savable
|
|
type Uint64 struct {
|
|
_ sync.NoCopy
|
|
value uint64
|
|
value32 uint32
|
|
}
|
|
|
|
//go:nosplit
|
|
func (u *Uint64) ptr() *uint64 {
|
|
// On 32-bit systems, i.value is guaranteed to be 32-bit aligned. It means
|
|
// that in the 12-byte i.value, there are guaranteed to be 8 contiguous bytes
|
|
// with 64-bit alignment.
|
|
return (*uint64)(unsafe.Pointer((uintptr(unsafe.Pointer(&u.value)) + 4) &^ 7))
|
|
}
|
|
|
|
// FromUint64 returns an Uint64 initialized to value v.
|
|
//
|
|
//go:nosplit
|
|
func FromUint64(v uint64) Uint64 {
|
|
var u Uint64
|
|
*u.ptr() = v
|
|
return u
|
|
}
|
|
|
|
// Load is analogous to atomic.LoadUint64.
|
|
//
|
|
//go:nosplit
|
|
func (u *Uint64) Load() uint64 {
|
|
return atomic.LoadUint64(u.ptr())
|
|
}
|
|
|
|
// RacyLoad is analogous to reading an atomic value without using
|
|
// synchronization.
|
|
//
|
|
// It may be helpful to document why a racy operation is permitted.
|
|
//
|
|
//go:nosplit
|
|
func (u *Uint64) RacyLoad() uint64 {
|
|
return *u.ptr()
|
|
}
|
|
|
|
// Store is analogous to atomic.StoreUint64.
|
|
//
|
|
//go:nosplit
|
|
func (u *Uint64) Store(v uint64) {
|
|
atomic.StoreUint64(u.ptr(), v)
|
|
}
|
|
|
|
// RacyStore is analogous to setting an atomic value without using
|
|
// synchronization.
|
|
//
|
|
// It may be helpful to document why a racy operation is permitted.
|
|
//
|
|
//go:nosplit
|
|
func (u *Uint64) RacyStore(v uint64) {
|
|
*u.ptr() = v
|
|
}
|
|
|
|
// Add is analogous to atomic.AddUint64.
|
|
//
|
|
//go:nosplit
|
|
func (u *Uint64) Add(v uint64) uint64 {
|
|
return atomic.AddUint64(u.ptr(), v)
|
|
}
|
|
|
|
// RacyAdd is analogous to adding to an atomic value without using
|
|
// synchronization.
|
|
//
|
|
// It may be helpful to document why a racy operation is permitted.
|
|
//
|
|
//go:nosplit
|
|
func (u *Uint64) RacyAdd(v uint64) uint64 {
|
|
*u.ptr() += v
|
|
return *u.ptr()
|
|
}
|
|
|
|
// Swap is analogous to atomic.SwapUint64.
|
|
//
|
|
//go:nosplit
|
|
func (u *Uint64) Swap(v uint64) uint64 {
|
|
return atomic.SwapUint64(u.ptr(), v)
|
|
}
|
|
|
|
// CompareAndSwap is analogous to atomic.CompareAndSwapUint64.
|
|
//
|
|
//go:nosplit
|
|
func (u *Uint64) CompareAndSwap(oldVal, newVal uint64) bool {
|
|
return atomic.CompareAndSwapUint64(u.ptr(), oldVal, newVal)
|
|
}
|