Skip to content

Commit

Permalink
Fix slab size when resetting mutable storable in OrderedMap
Browse files Browse the repository at this point in the history
Also extracted map data slab prefix computation to a new function.
  • Loading branch information
fxamacker committed Sep 4, 2023
1 parent dda0495 commit cc19ec9
Show file tree
Hide file tree
Showing 2 changed files with 82 additions and 17 deletions.
43 changes: 26 additions & 17 deletions map.go
Original file line number Diff line number Diff line change
Expand Up @@ -1374,8 +1374,6 @@ func (e *hkeyElements) Set(storage SlabStorage, address Address, b DigesterBuild
}
}

oldElemSize := elem.Size()

elem, existingValue, err := elem.Set(storage, address, b, digester, level, hkey, comparator, hip, key, value)
if err != nil {
// Don't need to wrap error as external error because err is already categorized by element.Set().
Expand All @@ -1384,7 +1382,14 @@ func (e *hkeyElements) Set(storage SlabStorage, address Address, b DigesterBuild

e.elems[equalIndex] = elem

e.size += elem.Size() - oldElemSize
// Recompute slab size by adding all element sizes instead of using the size diff of old and new element because
// oldElem can be the same storable when the same value is reset and oldElem.ByteSize() can equal storable.ByteSize().
// Given this, size diff of the old and new element can be 0 even when its actual size changed.
size := uint32(hkeyElementsPrefixSize)
for _, element := range e.elems {
size += element.Size() + digestSize
}
e.size = size

return existingValue, nil
}
Expand Down Expand Up @@ -1879,8 +1884,6 @@ func (e *singleElements) Set(storage SlabStorage, address Address, _ DigesterBui
if equal {
existingValue := elem.value

oldSize := elem.Size()

vs, err := value.Storable(storage, address, maxInlineMapValueSize(uint64(elem.key.ByteSize())))
if err != nil {
// Wrap err as external error (if needed) because err is returned by Value interface.
Expand All @@ -1890,7 +1893,14 @@ func (e *singleElements) Set(storage SlabStorage, address Address, _ DigesterBui
elem.value = vs
elem.size = singleElementPrefixSize + elem.key.ByteSize() + elem.value.ByteSize()

e.size += elem.Size() - oldSize
// Recompute slab size by adding all element sizes instead of using the size diff of old and new element because
// oldElem can be the same storable when the same value is reset and oldElem.ByteSize() can equal storable.ByteSize().
// Given this, size diff of the old and new element can be 0 even when its actual size changed.
size := uint32(singleElementsPrefixSize)
for _, element := range e.elems {
size += element.Size()
}
e.size = size

return existingValue, nil
}
Expand Down Expand Up @@ -2372,6 +2382,13 @@ func (m *MapDataSlab) ChildStorables() []Storable {
return elementsStorables(m.elements, nil)
}

func (m *MapDataSlab) getPrefixSize() uint32 {
if m.extraData != nil {
return mapRootDataSlabPrefixSize
}
return mapDataSlabPrefixSize
}

func elementsStorables(elems elements, childStorables []Storable) []Storable {

switch v := elems.(type) {
Expand Down Expand Up @@ -2436,11 +2453,7 @@ func (m *MapDataSlab) Set(storage SlabStorage, b DigesterBuilder, digester Diges
m.header.firstKey = m.elements.firstKey()

// Adjust header's slab size
if m.extraData == nil {
m.header.size = mapDataSlabPrefixSize + m.elements.Size()
} else {
m.header.size = mapRootDataSlabPrefixSize + m.elements.Size()
}
m.header.size = m.getPrefixSize() + m.elements.Size()

// Store modified slab
err = storage.Store(m.header.slabID, m)
Expand All @@ -2464,11 +2477,7 @@ func (m *MapDataSlab) Remove(storage SlabStorage, digester Digester, level uint,
m.header.firstKey = m.elements.firstKey()

// Adjust header's slab size
if m.extraData == nil {
m.header.size = mapDataSlabPrefixSize + m.elements.Size()
} else {
m.header.size = mapRootDataSlabPrefixSize + m.elements.Size()
}
m.header.size = m.getPrefixSize() + m.elements.Size()

// Store modified slab
err = storage.Store(m.header.slabID, m)
Expand Down Expand Up @@ -2668,7 +2677,7 @@ func (m *MapDataSlab) PopIterate(storage SlabStorage, fn MapPopIterationFunc) er
}

// Reset data slab
m.header.size = mapDataSlabPrefixSize + hkeyElementsPrefixSize
m.header.size = m.getPrefixSize() + hkeyElementsPrefixSize
m.header.firstKey = 0
return nil
}
Expand Down
56 changes: 56 additions & 0 deletions map_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -6817,3 +6817,59 @@ func TestMapID(t *testing.T) {
require.Equal(t, sid.address[:], id[:8])
require.Equal(t, sid.index[:], id[8:])
}

func TestSlabSizeWhenResettingMutableStorableInMap(t *testing.T) {
const (
mapSize = 3
keyStringSize = 16
initialStorableSize = 1
mutatedStorableSize = 5
)

keyValues := make(map[Value]*mutableValue, mapSize)
for i := 0; i < mapSize; i++ {
k := Uint64Value(i)
v := newMutableValue(initialStorableSize)
keyValues[k] = v
}

typeInfo := testTypeInfo{42}
address := Address{1, 2, 3, 4, 5, 6, 7, 8}
storage := newTestPersistentStorage(t)

m, err := NewMap(storage, address, newBasicDigesterBuilder(), typeInfo)
require.NoError(t, err)

for k, v := range keyValues {
existingStorable, err := m.Set(compare, hashInputProvider, k, v)
require.NoError(t, err)
require.Nil(t, existingStorable)
}

require.True(t, m.root.IsData())

expectedElementSize := singleElementPrefixSize + digestSize + Uint64Value(0).ByteSize() + initialStorableSize
expectedMapRootDataSlabSize := mapRootDataSlabPrefixSize + hkeyElementsPrefixSize + expectedElementSize*mapSize
require.Equal(t, expectedMapRootDataSlabSize, m.root.ByteSize())

err = ValidMap(m, typeInfo, typeInfoComparator, hashInputProvider)
require.NoError(t, err)

// Reset mutable values after changing its storable size
for k, v := range keyValues {
v.updateStorableSize(mutatedStorableSize)

existingStorable, err := m.Set(compare, hashInputProvider, k, v)
require.NoError(t, err)
require.NotNil(t, existingStorable)
}

require.True(t, m.root.IsData())

expectedElementSize = singleElementPrefixSize + digestSize + Uint64Value(0).ByteSize() + mutatedStorableSize
expectedMapRootDataSlabSize = mapRootDataSlabPrefixSize + hkeyElementsPrefixSize + expectedElementSize*mapSize
require.Equal(t, expectedMapRootDataSlabSize, m.root.ByteSize())

err = ValidMap(m, typeInfo, typeInfoComparator, hashInputProvider)
require.NoError(t, err)
}

0 comments on commit cc19ec9

Please sign in to comment.