2018-11-22 19:18:06 +00:00
|
|
|
package state
|
2018-10-15 13:17:07 +00:00
|
|
|
|
|
|
|
import (
|
2018-10-18 01:41:47 +00:00
|
|
|
"encoding/binary"
|
2018-10-17 06:32:26 +00:00
|
|
|
|
2018-11-21 06:44:04 +00:00
|
|
|
v "github.com/prysmaticlabs/prysm/beacon-chain/core/validators"
|
2018-10-15 13:17:07 +00:00
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
2018-11-18 16:39:35 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
2018-10-15 13:17:07 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// FinalizeAndJustifySlots justifies slots and sets the justified streak according to Casper FFG
|
|
|
|
// conditions. It also finalizes slots when the conditions are fulfilled.
|
|
|
|
func FinalizeAndJustifySlots(
|
|
|
|
slot uint64, justifiedSlot uint64, finalizedSlot uint64,
|
|
|
|
justifiedStreak uint64, blockVoteBalance uint64, totalDeposits uint64) (uint64, uint64, uint64) {
|
2018-11-18 16:39:35 +00:00
|
|
|
cycleLength := params.BeaconConfig().CycleLength
|
2018-10-15 13:17:07 +00:00
|
|
|
|
|
|
|
if 3*blockVoteBalance >= 2*totalDeposits {
|
|
|
|
if slot > justifiedSlot {
|
|
|
|
justifiedSlot = slot
|
|
|
|
}
|
|
|
|
justifiedStreak++
|
|
|
|
} else {
|
|
|
|
justifiedStreak = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
newFinalizedSlot := slot - cycleLength - 1
|
|
|
|
|
|
|
|
if slot > cycleLength && justifiedStreak >= cycleLength+1 && newFinalizedSlot > finalizedSlot {
|
|
|
|
finalizedSlot = newFinalizedSlot
|
|
|
|
}
|
|
|
|
|
|
|
|
return justifiedSlot, finalizedSlot, justifiedStreak
|
|
|
|
}
|
|
|
|
|
2018-12-03 03:10:47 +00:00
|
|
|
// UpdateLatestCrosslinks checks the vote balances and if there is a supermajority it sets the crosslink
|
2018-10-15 13:17:07 +00:00
|
|
|
// for that shard.
|
2018-12-03 03:10:47 +00:00
|
|
|
func UpdateLatestCrosslinks(slot uint64, voteBalance uint64, totalBalance uint64,
|
2018-10-15 13:17:07 +00:00
|
|
|
attestation *pb.AggregatedAttestation, crosslinkRecords []*pb.CrosslinkRecord) []*pb.CrosslinkRecord {
|
|
|
|
// if 2/3 of committee voted on this crosslink, update the crosslink
|
|
|
|
// with latest dynasty number, shard block hash, and slot number.
|
|
|
|
voteMajority := 3*voteBalance >= 2*totalBalance
|
2018-11-20 19:43:07 +00:00
|
|
|
if voteMajority {
|
2018-10-15 13:17:07 +00:00
|
|
|
crosslinkRecords[attestation.Shard] = &pb.CrosslinkRecord{
|
2018-12-14 19:35:53 +00:00
|
|
|
ShardBlockRootHash32: attestation.ShardBlockHash,
|
|
|
|
Slot: slot,
|
2018-10-15 13:17:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return crosslinkRecords
|
|
|
|
}
|
2018-10-17 06:32:26 +00:00
|
|
|
|
|
|
|
// ProcessSpecialRecords processes the pending special record objects,
|
|
|
|
// this is called during crystallized state transition.
|
2018-11-22 19:18:06 +00:00
|
|
|
func ProcessSpecialRecords(slotNumber uint64, validators []*pb.ValidatorRecord,
|
|
|
|
pendingSpecials []*pb.SpecialRecord) ([]*pb.ValidatorRecord, error) {
|
2018-10-17 06:32:26 +00:00
|
|
|
// For each special record object in active state.
|
|
|
|
for _, specialRecord := range pendingSpecials {
|
|
|
|
// Covers validators submitted logouts from last cycle.
|
|
|
|
if specialRecord.Kind == uint32(params.Logout) {
|
2018-10-18 01:41:47 +00:00
|
|
|
validatorIndex := binary.BigEndian.Uint64(specialRecord.Data[0])
|
2018-11-21 06:44:04 +00:00
|
|
|
exitedValidator := v.ExitValidator(validators[validatorIndex], slotNumber, false)
|
2018-10-17 06:32:26 +00:00
|
|
|
validators[validatorIndex] = exitedValidator
|
2018-11-22 19:18:06 +00:00
|
|
|
// TODO(#633): Verify specialRecord.Data[1] as signature.
|
|
|
|
// BLSVerify(pubkey=validator.pubkey, msg=hash(LOGOUT_MESSAGE + bytes8(version))
|
2018-10-17 06:32:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return validators, nil
|
|
|
|
}
|