mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2024-12-22 11:32:09 +00:00
Copy All Field Tries For Late Blocks (#12461)
* add new thing * only have it for late blocks * comments * change to lock * add test * Update beacon-chain/state/state-native/state_test.go --------- Co-authored-by: Radosław Kapka <rkapka@wp.pl>
This commit is contained in:
parent
8aa688729d
commit
70152bf476
@ -713,6 +713,9 @@ func (s *Service) lateBlockTasks(ctx context.Context) {
|
||||
if lastState == nil {
|
||||
lastRoot, lastState = headRoot[:], headState
|
||||
}
|
||||
// Copy all the field tries in our cached state in the event of late
|
||||
// blocks.
|
||||
lastState.CopyAllTries()
|
||||
if err = transition.UpdateNextSlotCache(ctx, lastRoot, lastState); err != nil {
|
||||
log.WithError(err).Debug("could not update next slot state cache")
|
||||
}
|
||||
|
@ -20,6 +20,7 @@ type BeaconState interface {
|
||||
ReadOnlyBeaconState
|
||||
WriteOnlyBeaconState
|
||||
Copy() BeaconState
|
||||
CopyAllTries()
|
||||
HashTreeRoot(ctx context.Context) ([32]byte, error)
|
||||
StateProver
|
||||
}
|
||||
|
@ -2,11 +2,13 @@ package state_native
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"strconv"
|
||||
"sync"
|
||||
"testing"
|
||||
|
||||
"github.com/prysmaticlabs/go-bitfield"
|
||||
"github.com/prysmaticlabs/prysm/v4/beacon-chain/state"
|
||||
"github.com/prysmaticlabs/prysm/v4/beacon-chain/state/state-native/types"
|
||||
"github.com/prysmaticlabs/prysm/v4/beacon-chain/state/stateutil"
|
||||
fieldparams "github.com/prysmaticlabs/prysm/v4/config/fieldparams"
|
||||
@ -271,6 +273,94 @@ func TestBeaconState_NoDeadlock_Capella(t *testing.T) {
|
||||
}
|
||||
|
||||
func TestBeaconState_AppendBalanceWithTrie(t *testing.T) {
|
||||
|
||||
newState := generateState(t)
|
||||
st, ok := newState.(*BeaconState)
|
||||
require.Equal(t, true, ok)
|
||||
_, err := st.HashTreeRoot(context.Background())
|
||||
assert.NoError(t, err)
|
||||
|
||||
for i := 0; i < 100; i++ {
|
||||
if i%2 == 0 {
|
||||
assert.NoError(t, st.UpdateBalancesAtIndex(primitives.ValidatorIndex(i), 1000))
|
||||
}
|
||||
if i%3 == 0 {
|
||||
assert.NoError(t, st.AppendBalance(1000))
|
||||
}
|
||||
}
|
||||
_, err = st.HashTreeRoot(context.Background())
|
||||
assert.NoError(t, err)
|
||||
newRt := bytesutil.ToBytes32(st.merkleLayers[0][types.Balances])
|
||||
wantedRt, err := stateutil.Uint64ListRootWithRegistryLimit(st.Balances())
|
||||
assert.NoError(t, err)
|
||||
assert.Equal(t, wantedRt, newRt, "state roots are unequal")
|
||||
}
|
||||
|
||||
func TestBeaconState_ModifyPreviousParticipationBits(t *testing.T) {
|
||||
st, err := InitializeFromProtoUnsafePhase0(ðpb.BeaconState{})
|
||||
assert.NoError(t, err)
|
||||
assert.ErrorContains(t, "ModifyPreviousParticipationBits is not supported", st.ModifyPreviousParticipationBits(func(val []byte) ([]byte, error) {
|
||||
return nil, nil
|
||||
}))
|
||||
}
|
||||
|
||||
func TestBeaconState_ModifyCurrentParticipationBits(t *testing.T) {
|
||||
st, err := InitializeFromProtoUnsafePhase0(ðpb.BeaconState{})
|
||||
assert.NoError(t, err)
|
||||
assert.ErrorContains(t, "ModifyCurrentParticipationBits is not supported", st.ModifyCurrentParticipationBits(func(val []byte) ([]byte, error) {
|
||||
return nil, nil
|
||||
}))
|
||||
}
|
||||
|
||||
func TestCopyAllTries(t *testing.T) {
|
||||
newState := generateState(t)
|
||||
_, err := newState.HashTreeRoot(context.Background())
|
||||
assert.NoError(t, err)
|
||||
|
||||
assert.NoError(t, newState.UpdateBalancesAtIndex(0, 10000))
|
||||
assert.NoError(t, newState.UpdateBlockRootAtIndex(0, [32]byte{'a'}))
|
||||
|
||||
_, err = newState.HashTreeRoot(context.Background())
|
||||
assert.NoError(t, err)
|
||||
|
||||
st, ok := newState.(*BeaconState)
|
||||
require.Equal(t, true, ok)
|
||||
|
||||
obj := st.stateFieldLeaves[types.Balances]
|
||||
|
||||
fieldAddr := fmt.Sprintf("%p", obj)
|
||||
|
||||
nState, ok := st.Copy().(*BeaconState)
|
||||
require.Equal(t, true, ok)
|
||||
|
||||
obj = nState.stateFieldLeaves[types.Balances]
|
||||
|
||||
newFieldAddr := fmt.Sprintf("%p", obj)
|
||||
assert.Equal(t, fieldAddr, newFieldAddr)
|
||||
assert.Equal(t, 2, int(obj.FieldReference().Refs()))
|
||||
|
||||
nState.CopyAllTries()
|
||||
|
||||
obj = nState.stateFieldLeaves[types.Balances]
|
||||
updatedFieldAddr := fmt.Sprintf("%p", obj)
|
||||
|
||||
assert.NotEqual(t, fieldAddr, updatedFieldAddr)
|
||||
assert.Equal(t, 1, int(obj.FieldReference().Refs()))
|
||||
|
||||
assert.NoError(t, nState.UpdateBalancesAtIndex(20, 10000))
|
||||
|
||||
_, err = nState.HashTreeRoot(context.Background())
|
||||
assert.NoError(t, err)
|
||||
|
||||
rt, err := st.stateFieldLeaves[types.Balances].TrieRoot()
|
||||
assert.NoError(t, err)
|
||||
|
||||
newRt, err := nState.stateFieldLeaves[types.Balances].TrieRoot()
|
||||
assert.NoError(t, err)
|
||||
assert.NotEqual(t, rt, newRt)
|
||||
}
|
||||
|
||||
func generateState(t *testing.T) state.BeaconState {
|
||||
count := uint64(100)
|
||||
vals := make([]*ethpb.Validator, 0, count)
|
||||
bals := make([]uint64, 0, count)
|
||||
@ -334,39 +424,5 @@ func TestBeaconState_AppendBalanceWithTrie(t *testing.T) {
|
||||
Slashings: make([]uint64, params.BeaconConfig().EpochsPerSlashingsVector),
|
||||
})
|
||||
assert.NoError(t, err)
|
||||
st, ok := newState.(*BeaconState)
|
||||
require.Equal(t, true, ok)
|
||||
_, err = st.HashTreeRoot(context.Background())
|
||||
assert.NoError(t, err)
|
||||
|
||||
for i := 0; i < 100; i++ {
|
||||
if i%2 == 0 {
|
||||
assert.NoError(t, st.UpdateBalancesAtIndex(primitives.ValidatorIndex(i), 1000))
|
||||
}
|
||||
if i%3 == 0 {
|
||||
assert.NoError(t, st.AppendBalance(1000))
|
||||
}
|
||||
}
|
||||
_, err = st.HashTreeRoot(context.Background())
|
||||
assert.NoError(t, err)
|
||||
newRt := bytesutil.ToBytes32(st.merkleLayers[0][types.Balances])
|
||||
wantedRt, err := stateutil.Uint64ListRootWithRegistryLimit(st.Balances())
|
||||
assert.NoError(t, err)
|
||||
assert.Equal(t, wantedRt, newRt, "state roots are unequal")
|
||||
}
|
||||
|
||||
func TestBeaconState_ModifyPreviousParticipationBits(t *testing.T) {
|
||||
st, err := InitializeFromProtoUnsafePhase0(ðpb.BeaconState{})
|
||||
assert.NoError(t, err)
|
||||
assert.ErrorContains(t, "ModifyPreviousParticipationBits is not supported", st.ModifyPreviousParticipationBits(func(val []byte) ([]byte, error) {
|
||||
return nil, nil
|
||||
}))
|
||||
}
|
||||
|
||||
func TestBeaconState_ModifyCurrentParticipationBits(t *testing.T) {
|
||||
st, err := InitializeFromProtoUnsafePhase0(ðpb.BeaconState{})
|
||||
assert.NoError(t, err)
|
||||
assert.ErrorContains(t, "ModifyCurrentParticipationBits is not supported", st.ModifyCurrentParticipationBits(func(val []byte) ([]byte, error) {
|
||||
return nil, nil
|
||||
}))
|
||||
return newState
|
||||
}
|
||||
|
@ -840,6 +840,26 @@ func (b *BeaconState) rootSelector(ctx context.Context, field types.FieldIndex)
|
||||
return [32]byte{}, errors.New("invalid field index provided")
|
||||
}
|
||||
|
||||
// CopyAllTries copies our field tries from the state. This is used to
|
||||
// remove shared field tries which have references to other states and
|
||||
// only have this copied set referencing to the current state.
|
||||
func (b *BeaconState) CopyAllTries() {
|
||||
b.lock.Lock()
|
||||
defer b.lock.Unlock()
|
||||
|
||||
for fldIdx, fieldTrie := range b.stateFieldLeaves {
|
||||
if fieldTrie.FieldReference() != nil {
|
||||
fieldTrie.Lock()
|
||||
if fieldTrie.FieldReference().Refs() > 1 {
|
||||
fieldTrie.FieldReference().MinusRef()
|
||||
newTrie := fieldTrie.CopyTrie()
|
||||
b.stateFieldLeaves[fldIdx] = newTrie
|
||||
}
|
||||
fieldTrie.Unlock()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func (b *BeaconState) recomputeFieldTrie(index types.FieldIndex, elements interface{}) ([32]byte, error) {
|
||||
fTrie := b.stateFieldLeaves[index]
|
||||
fTrieMutex := fTrie.RWMutex
|
||||
|
Loading…
Reference in New Issue
Block a user