2015-05-01 01:28:01 -05:00
|
|
|
// Copyright (c) 2013-2015 The btcsuite developers
|
2013-07-18 09:49:28 -05:00
|
|
|
// Use of this source code is governed by an ISC
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
2015-01-30 14:54:30 -06:00
|
|
|
package blockchain
|
2013-07-18 09:49:28 -05:00
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2014-07-02 18:04:59 +02:00
|
|
|
|
2015-01-15 10:23:47 -06:00
|
|
|
"github.com/btcsuite/btcutil"
|
2013-07-18 09:49:28 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
// maybeAcceptBlock potentially accepts a block into the memory block chain.
|
|
|
|
// It performs several validation checks which depend on its position within
|
|
|
|
// the block chain before adding it. The block is expected to have already gone
|
|
|
|
// through ProcessBlock before calling this function with it.
|
2014-06-26 15:50:13 -05:00
|
|
|
//
|
|
|
|
// The flags modify the behavior of this function as follows:
|
|
|
|
// - BFFastAdd: The somewhat expensive BIP0034 validation is not performed.
|
2014-06-29 15:11:13 -05:00
|
|
|
// - BFDryRun: The memory chain index will not be pruned and no accept
|
|
|
|
// notification will be sent since the block is not being accepted.
|
2014-06-26 15:50:13 -05:00
|
|
|
func (b *BlockChain) maybeAcceptBlock(block *btcutil.Block, flags BehaviorFlags) error {
|
|
|
|
fastAdd := flags&BFFastAdd == BFFastAdd
|
2014-06-29 15:11:13 -05:00
|
|
|
dryRun := flags&BFDryRun == BFDryRun
|
2014-06-26 15:50:13 -05:00
|
|
|
|
2013-07-18 09:49:28 -05:00
|
|
|
// Get a block node for the block previous to this one. Will be nil
|
|
|
|
// if this is the genesis block.
|
|
|
|
prevNode, err := b.getPrevNodeFromBlock(block)
|
|
|
|
if err != nil {
|
2013-07-24 12:31:14 -05:00
|
|
|
log.Errorf("getPrevNodeFromBlock: %v", err)
|
2013-07-18 09:49:28 -05:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2013-10-11 10:24:13 -05:00
|
|
|
// The height of this block is one more than the referenced previous
|
|
|
|
// block.
|
2013-07-18 09:49:28 -05:00
|
|
|
blockHeight := int64(0)
|
|
|
|
if prevNode != nil {
|
|
|
|
blockHeight = prevNode.height + 1
|
|
|
|
}
|
2013-07-29 16:58:48 -05:00
|
|
|
block.SetHeight(blockHeight)
|
2013-07-18 09:49:28 -05:00
|
|
|
|
2014-01-19 12:38:31 -06:00
|
|
|
blockHeader := &block.MsgBlock().Header
|
2013-11-18 16:23:51 -05:00
|
|
|
if !fastAdd {
|
|
|
|
// Ensure the difficulty specified in the block header matches
|
|
|
|
// the calculated difficulty based on the previous block and
|
2014-01-09 11:36:25 -06:00
|
|
|
// difficulty retarget rules.
|
2014-03-02 12:17:36 -06:00
|
|
|
expectedDifficulty, err := b.calcNextRequiredDifficulty(prevNode,
|
|
|
|
block.MsgBlock().Header.Timestamp)
|
2013-11-18 16:23:51 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
blockDifficulty := blockHeader.Bits
|
|
|
|
if blockDifficulty != expectedDifficulty {
|
|
|
|
str := "block difficulty of %d is not the expected value of %d"
|
|
|
|
str = fmt.Sprintf(str, blockDifficulty, expectedDifficulty)
|
2014-06-24 17:10:53 -05:00
|
|
|
return ruleError(ErrUnexpectedDifficulty, str)
|
2013-11-18 16:23:51 -05:00
|
|
|
}
|
2013-07-18 09:49:28 -05:00
|
|
|
|
2013-11-18 16:23:51 -05:00
|
|
|
// Ensure the timestamp for the block header is after the
|
|
|
|
// median time of the last several blocks (medianTimeBlocks).
|
|
|
|
medianTime, err := b.calcPastMedianTime(prevNode)
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("calcPastMedianTime: %v", err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !blockHeader.Timestamp.After(medianTime) {
|
|
|
|
str := "block timestamp of %v is not after expected %v"
|
|
|
|
str = fmt.Sprintf(str, blockHeader.Timestamp,
|
|
|
|
medianTime)
|
2014-06-24 17:10:53 -05:00
|
|
|
return ruleError(ErrTimeTooOld, str)
|
2013-07-18 09:49:28 -05:00
|
|
|
}
|
|
|
|
|
2013-11-18 16:23:51 -05:00
|
|
|
// Ensure all transactions in the block are finalized.
|
|
|
|
for _, tx := range block.Transactions() {
|
|
|
|
if !IsFinalizedTransaction(tx, blockHeight,
|
|
|
|
blockHeader.Timestamp) {
|
|
|
|
str := fmt.Sprintf("block contains "+
|
|
|
|
"unfinalized transaction %v", tx.Sha())
|
2014-06-24 17:10:53 -05:00
|
|
|
return ruleError(ErrUnfinalizedTx, str)
|
2013-11-18 16:23:51 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2014-01-09 11:36:25 -06:00
|
|
|
|
2013-07-18 09:49:28 -05:00
|
|
|
// Ensure chain matches up to predetermined checkpoints.
|
2015-04-17 00:44:15 -05:00
|
|
|
blockHash := block.Sha()
|
2013-07-18 09:49:28 -05:00
|
|
|
if !b.verifyCheckpoint(blockHeight, blockHash) {
|
|
|
|
str := fmt.Sprintf("block at height %d does not match "+
|
|
|
|
"checkpoint hash", blockHeight)
|
2014-06-24 17:10:53 -05:00
|
|
|
return ruleError(ErrBadCheckpoint, str)
|
2013-07-18 09:49:28 -05:00
|
|
|
}
|
|
|
|
|
2014-02-21 15:03:44 -06:00
|
|
|
// Find the previous checkpoint and prevent blocks which fork the main
|
|
|
|
// chain before it. This prevents storage of new, otherwise valid,
|
|
|
|
// blocks which build off of old blocks that are likely at a much easier
|
|
|
|
// difficulty and therefore could be used to waste cache and disk space.
|
|
|
|
checkpointBlock, err := b.findPreviousCheckpoint()
|
2014-02-21 13:02:59 -06:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if checkpointBlock != nil && blockHeight < checkpointBlock.Height() {
|
|
|
|
str := fmt.Sprintf("block at height %d forks the main chain "+
|
|
|
|
"before the previous checkpoint at height %d",
|
|
|
|
blockHeight, checkpointBlock.Height())
|
2014-06-24 17:10:53 -05:00
|
|
|
return ruleError(ErrForkTooOld, str)
|
2014-02-21 13:02:59 -06:00
|
|
|
}
|
|
|
|
|
2013-11-18 16:23:51 -05:00
|
|
|
if !fastAdd {
|
2015-02-25 14:32:37 -06:00
|
|
|
// Reject version 2 blocks once a majority of the network has
|
|
|
|
// upgraded. This is part of BIP0066.
|
|
|
|
if blockHeader.Version < 3 && b.isMajorityVersion(3, prevNode,
|
|
|
|
b.chainParams.BlockRejectNumRequired) {
|
|
|
|
|
|
|
|
str := "new blocks with version %d are no longer valid"
|
|
|
|
str = fmt.Sprintf(str, blockHeader.Version)
|
|
|
|
return ruleError(ErrBlockVersionTooOld, str)
|
|
|
|
}
|
|
|
|
|
2013-11-18 16:23:51 -05:00
|
|
|
// Reject version 1 blocks once a majority of the network has
|
2014-05-26 10:27:50 -05:00
|
|
|
// upgraded. This is part of BIP0034.
|
2015-02-25 14:31:46 -06:00
|
|
|
if blockHeader.Version < 2 && b.isMajorityVersion(2, prevNode,
|
|
|
|
b.chainParams.BlockRejectNumRequired) {
|
|
|
|
|
|
|
|
str := "new blocks with version %d are no longer valid"
|
|
|
|
str = fmt.Sprintf(str, blockHeader.Version)
|
|
|
|
return ruleError(ErrBlockVersionTooOld, str)
|
2013-07-18 09:49:28 -05:00
|
|
|
}
|
|
|
|
|
2013-11-18 16:23:51 -05:00
|
|
|
// Ensure coinbase starts with serialized block heights for
|
|
|
|
// blocks whose version is the serializedHeightVersion or
|
2014-05-26 10:27:50 -05:00
|
|
|
// newer once a majority of the network has upgraded. This is
|
|
|
|
// part of BIP0034.
|
2015-02-11 12:39:11 -08:00
|
|
|
if ShouldHaveSerializedBlockHeight(blockHeader) &&
|
2015-02-25 14:31:46 -06:00
|
|
|
b.isMajorityVersion(serializedHeightVersion, prevNode,
|
2015-02-18 15:09:54 -05:00
|
|
|
b.chainParams.BlockEnforceNumRequired) {
|
2013-11-18 16:23:51 -05:00
|
|
|
|
2015-02-25 14:31:46 -06:00
|
|
|
expectedHeight := int64(0)
|
|
|
|
if prevNode != nil {
|
|
|
|
expectedHeight = prevNode.height + 1
|
|
|
|
}
|
|
|
|
coinbaseTx := block.Transactions()[0]
|
|
|
|
err := checkSerializedHeight(coinbaseTx, expectedHeight)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2013-07-18 09:49:28 -05:00
|
|
|
}
|
|
|
|
}
|
2014-01-28 13:45:49 -06:00
|
|
|
}
|
2013-07-18 09:49:28 -05:00
|
|
|
|
2014-01-28 13:45:49 -06:00
|
|
|
// Prune block nodes which are no longer needed before creating
|
|
|
|
// a new node.
|
2014-06-29 15:11:13 -05:00
|
|
|
if !dryRun {
|
|
|
|
err = b.pruneBlockNodes()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2013-08-01 11:31:42 -05:00
|
|
|
}
|
|
|
|
|
2013-07-18 09:49:28 -05:00
|
|
|
// Create a new block node for the block and add it to the in-memory
|
|
|
|
// block chain (could be either a side chain or the main chain).
|
2014-01-19 13:00:12 -06:00
|
|
|
newNode := newBlockNode(blockHeader, blockHash, blockHeight)
|
2013-07-18 09:49:28 -05:00
|
|
|
if prevNode != nil {
|
|
|
|
newNode.parent = prevNode
|
|
|
|
newNode.height = blockHeight
|
|
|
|
newNode.workSum.Add(prevNode.workSum, newNode.workSum)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Connect the passed block to the chain while respecting proper chain
|
|
|
|
// selection according to the chain with the most proof of work. This
|
|
|
|
// also handles validation of the transaction scripts.
|
2014-06-26 15:50:13 -05:00
|
|
|
err = b.connectBestChain(newNode, block, flags)
|
2013-07-18 09:49:28 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Notify the caller that the new block was accepted into the block
|
|
|
|
// chain. The caller would typically want to react by relaying the
|
|
|
|
// inventory to other peers.
|
2014-06-29 15:11:13 -05:00
|
|
|
if !dryRun {
|
|
|
|
b.sendNotification(NTBlockAccepted, block)
|
|
|
|
}
|
2013-07-18 09:49:28 -05:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|