mirror of
https://github.com/netbirdio/gvisor.git
synced 2026-05-22 17:12:49 -07:00
0721ca2fe3
Also add the missing HugePage equivalents of Page functions. The fix for golang/go#56280, which first appears in Go 1.20, is needed to prevent this CL from regressing performance: Before fix: ``` TEXT gvisor/pkg/sentry/mm/mm.(*MemoryManager).SetNumaPolicy(SB) gvisor/pkg/sentry/mm/syscalls.go ... addr.go:75 0x7f000054e35e 488d053bc2b100 LEAQ gvisor/pkg/hostarch/hostarch..dict.IsPageAligned[gvisor/pkg/hostarch/hostarch.Addr](SB), AX addr.go:75 0x7f000054e365 e8961cc4ff CALL gvisor/pkg/hostarch/hostarch.IsPageAligned[go.shape.uintptr](SB) ``` After fix: ``` TEXT gvisor/pkg/sentry/mm/mm.(*MemoryManager).SetNumaPolicy(SB) gvisor/pkg/sentry/mm/syscalls.go ... addr.go:75 0x7f00004da61a 90 NOPL addr.go:75 0x7f00004da61b 0f1f440000 NOPL 0(AX)(AX*1) sizes_util.go:57 0x7f00004da620 48f7c3ff0f0000 TESTQ $0xfff, BX syscalls.go:1021 0x7f00004da627 0f855f010000 JNE 0x7f00004da78c ``` PiperOrigin-RevId: 507608080
114 lines
3.1 KiB
Go
114 lines
3.1 KiB
Go
// Copyright 2022 The gVisor Authors.
|
|
//
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package hostarch
|
|
|
|
// Masks often used when working with alignment in constant expressions.
|
|
const (
|
|
PageMask = PageSize - 1
|
|
HugePageMask = HugePageSize - 1
|
|
CacheLineMask = CacheLineSize - 1
|
|
)
|
|
|
|
type bytecount interface {
|
|
~uint | ~uint16 | ~uint32 | ~uint64 | ~uintptr
|
|
}
|
|
|
|
type hugebytecount interface {
|
|
~uint | ~uint32 | ~uint64 | ~uintptr
|
|
}
|
|
|
|
// PageRoundDown returns x rounded down to the nearest multiple of PageSize.
|
|
func PageRoundDown[T bytecount](x T) T {
|
|
return x &^ PageMask
|
|
}
|
|
|
|
// PageRoundUp returns x rounded up to the nearest multiple of PageSize. ok is
|
|
// true iff rounding up does not overflow the range of T.
|
|
func PageRoundUp[T bytecount](x T) (val T, ok bool) {
|
|
val = PageRoundDown(x + PageMask)
|
|
ok = val >= x
|
|
return
|
|
}
|
|
|
|
// MustPageRoundUp is equivalent to PageRoundUp, but panics if rounding up
|
|
// overflows.
|
|
func MustPageRoundUp[T bytecount](x T) T {
|
|
val, ok := PageRoundUp(x)
|
|
if !ok {
|
|
panic("PageRoundUp overflows")
|
|
}
|
|
return val
|
|
}
|
|
|
|
// PageOffset returns the offset of x into its containing page.
|
|
func PageOffset[T bytecount](x T) T {
|
|
return x & PageMask
|
|
}
|
|
|
|
// IsPageAligned returns true if x is a multiple of PageSize.
|
|
func IsPageAligned[T bytecount](x T) bool {
|
|
return PageOffset(x) == 0
|
|
}
|
|
|
|
// ToPagesRoundUp returns (the number of pages equal to x bytes rounded up,
|
|
// true). If rounding x up to a multiple of PageSize overflows the range of T,
|
|
// ToPagesRoundUp returns (unspecified, false).
|
|
func ToPagesRoundUp[T bytecount](x T) (T, bool) {
|
|
y := x + PageMask
|
|
if y < x {
|
|
return x, false
|
|
}
|
|
return y / PageSize, true
|
|
}
|
|
|
|
// HugePageRoundDown returns x rounded down to the nearest multiple of
|
|
// HugePageSize.
|
|
func HugePageRoundDown[T hugebytecount](x T) T {
|
|
return x &^ HugePageMask
|
|
}
|
|
|
|
// HugePageRoundUp returns x rounded up to the nearest multiple of
|
|
// HugePageSize. ok is true iff rounding up does not overflow the range of T.
|
|
func HugePageRoundUp[T hugebytecount](x T) (val T, ok bool) {
|
|
val = HugePageRoundDown(x + HugePageMask)
|
|
ok = val >= x
|
|
return
|
|
}
|
|
|
|
// MustHugePageRoundUp is equivalent to HugePageRoundUp, but panics if rounding
|
|
// up overflows.
|
|
func MustHugePageRoundUp[T hugebytecount](x T) T {
|
|
val, ok := HugePageRoundUp(x)
|
|
if !ok {
|
|
panic("HugePageRoundUp overflows")
|
|
}
|
|
return val
|
|
}
|
|
|
|
// HugePageOffset returns the offset of x into its containing page.
|
|
func HugePageOffset[T hugebytecount](x T) T {
|
|
return x & HugePageMask
|
|
}
|
|
|
|
// IsHugePageAligned returns true if x is a multiple of HugePageSize.
|
|
func IsHugePageAligned[T hugebytecount](x T) bool {
|
|
return HugePageOffset(x) == 0
|
|
}
|
|
|
|
// CacheLineRoundDown returns the offset rounded down to the nearest multiple
|
|
// of CacheLineSize.
|
|
func CacheLineRoundDown[T bytecount](x T) T {
|
|
return x &^ CacheLineMask
|
|
}
|
|
|
|
// CacheLineRoundUp returns the offset rounded up to the nearest multiple of
|
|
// CacheLineSize. ok is true iff rounding up does not overflow the range of T.
|
|
func CacheLineRoundUp[T bytecount](x T) (val T, ok bool) {
|
|
val = CacheLineRoundDown(x + CacheLineMask)
|
|
ok = val >= x
|
|
return
|
|
}
|