Merge pull request #3753 from greatroar/indexmap-alloc

repository: Re-tune indexmap allocation strategy
This commit is contained in:
MichaelEischer 2022-05-14 15:44:08 +02:00 committed by GitHub
commit b2a2e5f727
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23

View file

@ -130,24 +130,32 @@ func (m *indexMap) init() {
func (m *indexMap) len() uint { return m.numentries } func (m *indexMap) len() uint { return m.numentries }
func (m *indexMap) newEntry() *indexEntry { func (m *indexMap) newEntry() *indexEntry {
// Allocating in batches means that we get closer to optimal space usage, // We keep a free list of objects to speed up allocation and GC.
// as Go's malloc will overallocate for structures of size 60 (indexEntry // There's an obvious trade-off here: allocating in larger batches
// on amd64). // means we allocate faster and the GC has to keep fewer bits to track
// what we have in use, but it means we waste some space.
// //
// 128*60 and 128*60 both have low malloc overhead among reasonable sizes. // Then again, allocating each indexEntry separately also wastes space
// See src/runtime/sizeclasses.go in the standard library. // on 32-bit platforms, because the Go malloc has no size class for
const entryAllocBatch = 128 // exactly 52 bytes, so it puts the indexEntry in a 64-byte slot instead.
// See src/runtime/sizeclasses.go in the Go source repo.
if m.free == nil { //
free := new([entryAllocBatch]indexEntry) // The batch size of 4 means we hit the size classes for 4×64=256 bytes
for i := range free[:len(free)-1] { // (64-bit) and 4×52=208 bytes (32-bit), wasting nothing in malloc on
free[i].next = &free[i+1] // 64-bit and relatively little on 32-bit.
} const entryAllocBatch = 4
m.free = &free[0]
}
e := m.free e := m.free
m.free = m.free.next if e != nil {
m.free = e.next
} else {
free := new([entryAllocBatch]indexEntry)
e = &free[0]
for i := 1; i < len(free)-1; i++ {
free[i].next = &free[i+1]
}
m.free = &free[1]
}
return e return e
} }