mirror of https://github.com/poanetwork/quorum.git
109 lines
3.2 KiB
Go
109 lines
3.2 KiB
Go
// Copyright 2017 The go-ethereum Authors
|
|
// This file is part of the go-ethereum library.
|
|
//
|
|
// The go-ethereum library is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// The go-ethereum library is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Lesser General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
package core
|
|
|
|
import (
|
|
"time"
|
|
|
|
"github.com/ethereum/go-ethereum/consensus"
|
|
"github.com/ethereum/go-ethereum/consensus/istanbul"
|
|
)
|
|
|
|
func (c *core) sendPreprepare(request *istanbul.Request) {
|
|
logger := c.logger.New("state", c.state)
|
|
|
|
// If I'm the proposer and I have the same sequence with the proposal
|
|
if c.current.Sequence().Cmp(request.Proposal.Number()) == 0 && c.isProposer() {
|
|
curView := c.currentView()
|
|
preprepare, err := Encode(&istanbul.Preprepare{
|
|
View: curView,
|
|
Proposal: request.Proposal,
|
|
})
|
|
if err != nil {
|
|
logger.Error("Failed to encode", "view", curView)
|
|
return
|
|
}
|
|
|
|
c.broadcast(&message{
|
|
Code: msgPreprepare,
|
|
Msg: preprepare,
|
|
})
|
|
}
|
|
}
|
|
|
|
func (c *core) handlePreprepare(msg *message, src istanbul.Validator) error {
|
|
logger := c.logger.New("from", src, "state", c.state)
|
|
|
|
// Decode PRE-PREPARE
|
|
var preprepare *istanbul.Preprepare
|
|
err := msg.Decode(&preprepare)
|
|
if err != nil {
|
|
return errFailedDecodePreprepare
|
|
}
|
|
|
|
// Ensure we have the same view with the PRE-PREPARE message
|
|
if err := c.checkMessage(msgPreprepare, preprepare.View); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Check if the message comes from current proposer
|
|
if !c.valSet.IsProposer(src.Address()) {
|
|
logger.Warn("Ignore preprepare messages from non-proposer")
|
|
return errNotFromProposer
|
|
}
|
|
|
|
// Verify the proposal we received
|
|
if duration, err := c.backend.Verify(preprepare.Proposal); err != nil {
|
|
logger.Warn("Failed to verify proposal", "err", err, "duration", duration)
|
|
// if it's a future block, we will handle it again after the duration
|
|
if err == consensus.ErrFutureBlock {
|
|
c.stopFuturePreprepareTimer()
|
|
c.futurePreprepareTimer = time.AfterFunc(duration, func() {
|
|
c.sendEvent(backlogEvent{
|
|
src: src,
|
|
msg: msg,
|
|
})
|
|
})
|
|
} else {
|
|
c.sendNextRoundChange()
|
|
}
|
|
return err
|
|
}
|
|
|
|
// Here is about to accept the PRE-PREPARE
|
|
if c.state == StateAcceptRequest {
|
|
// Send ROUND CHANGE if the locked proposal and the received proposal are different
|
|
if c.current.IsHashLocked() && preprepare.Proposal.Hash() != c.current.GetLockedHash() {
|
|
c.sendNextRoundChange()
|
|
} else {
|
|
// Either
|
|
// 1. the locked proposal and the received proposal match
|
|
// 2. we have no locked proposal
|
|
c.acceptPreprepare(preprepare)
|
|
c.setState(StatePreprepared)
|
|
c.sendPrepare()
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (c *core) acceptPreprepare(preprepare *istanbul.Preprepare) {
|
|
c.consensusTimestamp = time.Now()
|
|
c.current.SetPreprepare(preprepare)
|
|
}
|