Skip to content

Commit

Permalink
fix allocated memory misalignment
Browse files Browse the repository at this point in the history
ref: #16

Signed-off-by: Miguel Ángel Ortuño <[email protected]>
  • Loading branch information
ortuman committed Mar 6, 2024
1 parent f56c9fa commit 9af0d70
Show file tree
Hide file tree
Showing 5 changed files with 38 additions and 21 deletions.
7 changes: 4 additions & 3 deletions arena.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,8 @@ import (
// Arena is an interface that describes a memory allocation arena.
type Arena interface {
// Alloc allocates memory of the given size and returns a pointer to it.
Alloc(size int) unsafe.Pointer
// The alignment parameter specifies the alignment of the allocated memory.
Alloc(size, alignment uintptr) unsafe.Pointer

// Reset resets the arena's state, optionally releasing the memory.
// After invoking this method any pointer previously returned by Alloc becomes immediately invalid.
Expand All @@ -22,7 +23,7 @@ type Arena interface {
func New[T any](a Arena) *T {
if a != nil {
var x T
if ptr := a.Alloc(int(unsafe.Sizeof(x))); ptr != nil {
if ptr := a.Alloc(unsafe.Sizeof(x), unsafe.Alignof(x)); ptr != nil {
return (*T)(ptr)
}
}
Expand All @@ -37,7 +38,7 @@ func MakeSlice[T any](a Arena, len, cap int) []T {
if a != nil {
var x T
bufSize := int(unsafe.Sizeof(x)) * cap
if ptr := (*T)(a.Alloc(bufSize)); ptr != nil {
if ptr := (*T)(a.Alloc(uintptr(bufSize), unsafe.Alignof(x))); ptr != nil {
s := unsafe.Slice(ptr, cap)
return s[:len]
}
Expand Down
4 changes: 2 additions & 2 deletions concurrent_arena.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,9 +19,9 @@ func NewConcurrentArena(a Arena) Arena {
}

// Alloc satisfies the Arena interface.
func (a *concurrentArena) Alloc(size int) unsafe.Pointer {
func (a *concurrentArena) Alloc(size, alignment uintptr) unsafe.Pointer {
a.mtx.Lock()
ptr := a.a.Alloc(size)
ptr := a.a.Alloc(size, alignment)
a.mtx.Unlock()
return ptr
}
Expand Down
26 changes: 16 additions & 10 deletions monotonic_arena.go
Original file line number Diff line number Diff line change
Expand Up @@ -12,24 +12,30 @@ type monotonicArena struct {

type monotonicBuffer struct {
ptr unsafe.Pointer
offset int
size int
offset uintptr
size uintptr
}

func newMonotonicBuffer(size int) *monotonicBuffer {
return &monotonicBuffer{size: size}
return &monotonicBuffer{size: uintptr(size)}
}

func (s *monotonicBuffer) alloc(size int) (unsafe.Pointer, bool) {
func (s *monotonicBuffer) alloc(size, alignment uintptr) (unsafe.Pointer, bool) {
if s.ptr == nil {
buf := make([]byte, s.size) // allocate monotonic buffer lazily
s.ptr = unsafe.Pointer(unsafe.SliceData(buf))
}
if s.availableBytes() < size {
alignOffset := uintptr(0)
for alignedPtr := uintptr(s.ptr) + s.offset; alignedPtr%alignment != 0; alignedPtr++ {
alignOffset++
}
allocSize := size + alignOffset

if s.availableBytes() < allocSize {
return nil, false
}
ptr := unsafe.Pointer(uintptr(s.ptr) + uintptr(s.offset))
s.offset += size
ptr := unsafe.Pointer(uintptr(s.ptr) + s.offset + alignOffset)
s.offset += allocSize

return ptr, true
}
Expand Down Expand Up @@ -59,7 +65,7 @@ func (s *monotonicBuffer) zeroOutBuffer() {
}
}

func (s *monotonicBuffer) availableBytes() int {
func (s *monotonicBuffer) availableBytes() uintptr {
return s.size - s.offset
}

Expand All @@ -73,9 +79,9 @@ func NewMonotonicArena(bufferSize, bufferCount int) Arena {
}

// Alloc satisfies the Arena interface.
func (a *monotonicArena) Alloc(size int) unsafe.Pointer {
func (a *monotonicArena) Alloc(size, alignment uintptr) unsafe.Pointer {
for i := 0; i < len(a.buffers); i++ {
ptr, ok := a.buffers[i].alloc(size)
ptr, ok := a.buffers[i].alloc(size, alignment)
if ok {
return ptr
}
Expand Down
20 changes: 15 additions & 5 deletions monotonic_arena_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -83,14 +83,24 @@ func TestMonotonicArenaReset(t *testing.T) {
_ = New[int](arena)
}

func TestMonotonicArenaMultipleTypes(t *testing.T) {
arena := NewMonotonicArena(8182, 1) // 8KB

var b = New[byte](arena)
var p = New[*int](arena)

require.Equal(t, *b, byte(0))
require.True(t, *p == nil)
}

func isMonotonicArenaPtr(a Arena, ptr unsafe.Pointer) bool {
ma := a.(*monotonicArena)
for _, s := range ma.buffers {
if s.ptr == nil {
break
}
beginPtr := uintptr(s.ptr)
endPtr := uintptr(s.ptr) + uintptr(s.size)
endPtr := uintptr(s.ptr) + s.size

if uintptr(ptr) >= beginPtr && uintptr(ptr) < endPtr {
return true
Expand All @@ -114,7 +124,7 @@ func BenchmarkRuntimeNewObject(b *testing.B) {
}

func BenchmarkMonotonicArenaNewObject(b *testing.B) {
monotonicArena := NewMonotonicArena(1024*1024, 128) // 1Mb buffer size (128 MB)
monotonicArena := NewMonotonicArena(2*1024*1024, 32) // 2Mb buffer size (64Mb max size)

a := newArenaAllocator[int](monotonicArena)
for _, objectCount := range []int{100, 1_000, 10_000, 100_000} {
Expand All @@ -131,7 +141,7 @@ func BenchmarkMonotonicArenaNewObject(b *testing.B) {
}

func BenchmarkConcurrentMonotonicArenaNewObject(b *testing.B) {
monotonicArena := NewMonotonicArena(1024*1024, 128) // 1Mb buffer size (128 MB)
monotonicArena := NewMonotonicArena(2*1024*1024, 32) // 2Mb buffer size (64Mb max size)

a := newArenaAllocator[int](NewConcurrentArena(monotonicArena))
for _, objectCount := range []int{100, 1_000, 10_000, 100_000} {
Expand Down Expand Up @@ -162,7 +172,7 @@ func BenchmarkRuntimeMakeSlice(b *testing.B) {
}

func BenchmarkMonotonicArenaMakeSlice(b *testing.B) {
monotonicArena := NewMonotonicArena(1024*1024, 128) // 1Mb buffer size (128 MB)
monotonicArena := NewMonotonicArena(2*1024*1024, 32) // 2Mb buffer size (64Mb max size)

a := newArenaAllocator[int](monotonicArena)
for _, objectCount := range []int{100, 1_000, 10_000, 100_000} {
Expand All @@ -179,7 +189,7 @@ func BenchmarkMonotonicArenaMakeSlice(b *testing.B) {
}

func BenchmarkConcurrentMonotonicArenaMakeSlice(b *testing.B) {
monotonicArena := NewMonotonicArena(1024*1024, 128) // 1Mb buffer size (128 MB)
monotonicArena := NewMonotonicArena(2*1024*1024, 32) // 2Mb buffer size (64Mb max size)

a := newArenaAllocator[int](NewConcurrentArena(monotonicArena))
for _, objectCount := range []int{100, 1_000, 10_000, 100_000} {
Expand Down
2 changes: 1 addition & 1 deletion slice_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ import (
// It simply allocates memory using Go's built-in make function.
type mockArena struct{}

func (m *mockArena) Alloc(size int) unsafe.Pointer {
func (m *mockArena) Alloc(size, _ uintptr) unsafe.Pointer {
return unsafe.Pointer(&make([]byte, size)[0])
}

Expand Down

0 comments on commit 9af0d70

Please sign in to comment.