2017-01-31 01:32:50 +01:00
|
|
|
// Copyright (c) 2013-2017 The btcsuite developers
|
2013-07-18 16:49:28 +02:00
|
|
|
// Use of this source code is governed by an ISC
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
2015-01-30 21:54:30 +01:00
|
|
|
package blockchain
|
2013-07-18 16:49:28 +02:00
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/binary"
|
|
|
|
"fmt"
|
2014-07-02 18:04:59 +02:00
|
|
|
"math"
|
|
|
|
"math/big"
|
|
|
|
"time"
|
|
|
|
|
2021-10-15 07:45:32 +02:00
|
|
|
"github.com/lbryio/lbcd/chaincfg"
|
|
|
|
"github.com/lbryio/lbcd/chaincfg/chainhash"
|
|
|
|
"github.com/lbryio/lbcd/txscript"
|
|
|
|
"github.com/lbryio/lbcd/wire"
|
|
|
|
btcutil "github.com/lbryio/lbcutil"
|
2013-07-18 16:49:28 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2014-06-30 17:27:44 +02:00
|
|
|
// MaxTimeOffsetSeconds is the maximum number of seconds a block time
|
|
|
|
// is allowed to be ahead of the current time. This is currently 2
|
|
|
|
// hours.
|
|
|
|
MaxTimeOffsetSeconds = 2 * 60 * 60
|
|
|
|
|
2014-03-15 21:01:48 +01:00
|
|
|
// MinCoinbaseScriptLen is the minimum length a coinbase script can be.
|
|
|
|
MinCoinbaseScriptLen = 2
|
2013-07-18 16:49:28 +02:00
|
|
|
|
2014-03-15 21:01:48 +01:00
|
|
|
// MaxCoinbaseScriptLen is the maximum length a coinbase script can be.
|
|
|
|
MaxCoinbaseScriptLen = 100
|
2013-07-18 16:49:28 +02:00
|
|
|
|
|
|
|
// medianTimeBlocks is the number of previous blocks which should be
|
|
|
|
// used to calculate the median time used to validate block timestamps.
|
|
|
|
medianTimeBlocks = 11
|
|
|
|
|
|
|
|
// serializedHeightVersion is the block version which changed block
|
|
|
|
// coinbases to start with the serialized block height.
|
|
|
|
serializedHeightVersion = 2
|
|
|
|
|
|
|
|
// baseSubsidy is the starting subsidy amount for mined blocks. This
|
2014-02-23 21:25:15 +01:00
|
|
|
// value is halved every SubsidyHalvingInterval blocks.
|
2018-06-13 06:11:42 +02:00
|
|
|
baseSubsidy = 500 * btcutil.SatoshiPerBitcoin
|
2013-07-18 16:49:28 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2016-08-08 21:04:33 +02:00
|
|
|
// zeroHash is the zero value for a chainhash.Hash and is defined as
|
2013-07-18 16:49:28 +02:00
|
|
|
// a package level variable to avoid the need to create a new instance
|
|
|
|
// every time a check is needed.
|
2017-09-06 03:24:52 +02:00
|
|
|
zeroHash chainhash.Hash
|
2013-07-18 16:49:28 +02:00
|
|
|
|
|
|
|
// block91842Hash is one of the two nodes which violate the rules
|
|
|
|
// set forth in BIP0030. It is defined as a package level variable to
|
|
|
|
// avoid the need to create a new instance every time a check is needed.
|
2016-08-08 21:04:33 +02:00
|
|
|
block91842Hash = newHashFromStr("00000000000a4d0a398161ffc163c503763b1f4360639393e0e4c8e300e0caec")
|
2013-07-18 16:49:28 +02:00
|
|
|
|
|
|
|
// block91880Hash is one of the two nodes which violate the rules
|
|
|
|
// set forth in BIP0030. It is defined as a package level variable to
|
|
|
|
// avoid the need to create a new instance every time a check is needed.
|
2016-08-08 21:04:33 +02:00
|
|
|
block91880Hash = newHashFromStr("00000000000743f190a18c5577a3c2d2a1f610ae9601ac046a38084ccb7cd721")
|
2013-07-18 16:49:28 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// isNullOutpoint determines whether or not a previous transaction output point
|
|
|
|
// is set.
|
2015-02-05 22:16:39 +01:00
|
|
|
func isNullOutpoint(outpoint *wire.OutPoint) bool {
|
2017-09-06 03:24:52 +02:00
|
|
|
if outpoint.Index == math.MaxUint32 && outpoint.Hash == zeroHash {
|
2013-07-18 16:49:28 +02:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2015-02-11 21:39:11 +01:00
|
|
|
// ShouldHaveSerializedBlockHeight determines if a block should have a
|
|
|
|
// serialized block height embedded within the scriptSig of its
|
|
|
|
// coinbase transaction. Judgement is based on the block version in the block
|
|
|
|
// header. Blocks with version 2 and above satisfy this criteria. See BIP0034
|
|
|
|
// for further information.
|
|
|
|
func ShouldHaveSerializedBlockHeight(header *wire.BlockHeader) bool {
|
|
|
|
return header.Version >= serializedHeightVersion
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsCoinBaseTx determines whether or not a transaction is a coinbase. A coinbase
|
|
|
|
// is a special transaction created by miners that has no inputs. This is
|
|
|
|
// represented in the block chain by a transaction with a single input that has
|
|
|
|
// a previous output transaction index set to the maximum value along with a
|
2013-07-18 16:49:28 +02:00
|
|
|
// zero hash.
|
2015-03-10 06:41:19 +01:00
|
|
|
//
|
|
|
|
// This function only differs from IsCoinBase in that it works with a raw wire
|
|
|
|
// transaction as opposed to a higher level util transaction.
|
|
|
|
func IsCoinBaseTx(msgTx *wire.MsgTx) bool {
|
2013-07-18 16:49:28 +02:00
|
|
|
// A coin base must only have one transaction input.
|
|
|
|
if len(msgTx.TxIn) != 1 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// The previous output of a coin base must have a max value index and
|
|
|
|
// a zero hash.
|
2015-03-10 06:41:19 +01:00
|
|
|
prevOut := &msgTx.TxIn[0].PreviousOutPoint
|
2017-09-06 03:24:52 +02:00
|
|
|
if prevOut.Index != math.MaxUint32 || prevOut.Hash != zeroHash {
|
2013-07-18 16:49:28 +02:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2015-03-10 06:41:19 +01:00
|
|
|
// IsCoinBase determines whether or not a transaction is a coinbase. A coinbase
|
|
|
|
// is a special transaction created by miners that has no inputs. This is
|
|
|
|
// represented in the block chain by a transaction with a single input that has
|
|
|
|
// a previous output transaction index set to the maximum value along with a
|
|
|
|
// zero hash.
|
|
|
|
//
|
|
|
|
// This function only differs from IsCoinBaseTx in that it works with a higher
|
|
|
|
// level util transaction as opposed to a raw wire transaction.
|
|
|
|
func IsCoinBase(tx *btcutil.Tx) bool {
|
|
|
|
return IsCoinBaseTx(tx.MsgTx())
|
|
|
|
}
|
|
|
|
|
2016-06-22 06:51:43 +02:00
|
|
|
// SequenceLockActive determines if a transaction's sequence locks have been
|
|
|
|
// met, meaning that all the inputs of a given transaction have reached a
|
|
|
|
// height or time sufficient for their relative lock-time maturity.
|
|
|
|
func SequenceLockActive(sequenceLock *SequenceLock, blockHeight int32,
|
|
|
|
medianTimePast time.Time) bool {
|
|
|
|
|
|
|
|
// If either the seconds, or height relative-lock time has not yet
|
|
|
|
// reached, then the transaction is not yet mature according to its
|
|
|
|
// sequence locks.
|
|
|
|
if sequenceLock.Seconds >= medianTimePast.Unix() ||
|
|
|
|
sequenceLock.BlockHeight >= blockHeight {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2015-06-29 17:12:35 +02:00
|
|
|
// IsFinalizedTransaction determines whether or not a transaction is finalized.
|
2015-08-08 04:20:49 +02:00
|
|
|
func IsFinalizedTransaction(tx *btcutil.Tx, blockHeight int32, blockTime time.Time) bool {
|
2015-06-29 17:12:35 +02:00
|
|
|
msgTx := tx.MsgTx()
|
|
|
|
|
|
|
|
// Lock time of zero means the transaction is finalized.
|
|
|
|
lockTime := msgTx.LockTime
|
|
|
|
if lockTime == 0 {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
// The lock time field of a transaction is either a block height at
|
|
|
|
// which the transaction is finalized or a timestamp depending on if the
|
|
|
|
// value is before the txscript.LockTimeThreshold. When it is under the
|
|
|
|
// threshold it is a block height.
|
|
|
|
blockTimeOrHeight := int64(0)
|
|
|
|
if lockTime < txscript.LockTimeThreshold {
|
2015-08-08 04:20:49 +02:00
|
|
|
blockTimeOrHeight = int64(blockHeight)
|
2015-06-29 17:12:35 +02:00
|
|
|
} else {
|
|
|
|
blockTimeOrHeight = blockTime.Unix()
|
|
|
|
}
|
|
|
|
if int64(lockTime) < blockTimeOrHeight {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2016-02-25 18:17:12 +01:00
|
|
|
// At this point, the transaction's lock time hasn't occurred yet, but
|
2015-06-29 17:12:35 +02:00
|
|
|
// the transaction might still be finalized if the sequence number
|
|
|
|
// for all transaction inputs is maxed out.
|
|
|
|
for _, txIn := range msgTx.TxIn {
|
|
|
|
if txIn.Sequence != math.MaxUint32 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2013-07-18 16:49:28 +02:00
|
|
|
// isBIP0030Node returns whether or not the passed node represents one of the
|
|
|
|
// two blocks that violate the BIP0030 rule which prevents transactions from
|
|
|
|
// overwriting old ones.
|
|
|
|
func isBIP0030Node(node *blockNode) bool {
|
|
|
|
if node.height == 91842 && node.hash.IsEqual(block91842Hash) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
if node.height == 91880 && node.hash.IsEqual(block91880Hash) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2014-02-25 07:31:58 +01:00
|
|
|
// CalcBlockSubsidy returns the subsidy amount a block at the provided height
|
2013-07-18 16:49:28 +02:00
|
|
|
// should have. This is mainly used for determining how much the coinbase for
|
|
|
|
// newly generated blocks awards as well as validating the coinbase for blocks
|
|
|
|
// has the expected value.
|
|
|
|
//
|
2016-08-10 23:02:23 +02:00
|
|
|
// The subsidy is halved every SubsidyReductionInterval blocks. Mathematically
|
|
|
|
// this is: baseSubsidy / 2^(height/SubsidyReductionInterval)
|
2013-07-18 16:49:28 +02:00
|
|
|
//
|
2014-02-25 07:31:58 +01:00
|
|
|
// At the target block generation rate for the main network, this is
|
|
|
|
// approximately every 4 years.
|
2015-08-08 04:20:49 +02:00
|
|
|
func CalcBlockSubsidy(height int32, chainParams *chaincfg.Params) int64 {
|
2018-06-13 06:11:42 +02:00
|
|
|
h := int64(height)
|
|
|
|
if h == 0 {
|
|
|
|
return btcutil.SatoshiPerBitcoin * 4e8
|
2014-05-26 17:27:50 +02:00
|
|
|
}
|
2018-06-13 06:11:42 +02:00
|
|
|
if h <= 5100 {
|
|
|
|
return btcutil.SatoshiPerBitcoin
|
|
|
|
}
|
|
|
|
if h <= 55000 {
|
|
|
|
return btcutil.SatoshiPerBitcoin * (1 + (h-5001)/100)
|
|
|
|
}
|
|
|
|
|
|
|
|
lv := (h - 55001) / int64(chainParams.SubsidyReductionInterval)
|
|
|
|
reduction := (int64(math.Sqrt((float64(8*lv))+1)) - 1) / 2
|
|
|
|
for !withinLevelBounds(reduction, lv) {
|
|
|
|
if ((reduction*reduction + reduction) >> 1) > lv {
|
|
|
|
reduction--
|
|
|
|
} else {
|
|
|
|
reduction++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
subsidyReduction := btcutil.SatoshiPerBitcoin * reduction
|
|
|
|
if subsidyReduction >= baseSubsidy {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
return baseSubsidy - subsidyReduction
|
|
|
|
}
|
2014-05-26 17:27:50 +02:00
|
|
|
|
2018-06-13 06:11:42 +02:00
|
|
|
func withinLevelBounds(reduction int64, lv int64) bool {
|
|
|
|
if ((reduction*reduction + reduction) >> 1) > lv {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
reduction++
|
2021-10-16 00:16:58 +02:00
|
|
|
return ((reduction*reduction + reduction) >> 1) > lv
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
2013-09-30 23:43:10 +02:00
|
|
|
// CheckTransactionSanity performs some preliminary checks on a transaction to
|
2018-06-15 22:06:26 +02:00
|
|
|
// ensure it is sane.
|
|
|
|
func CheckTransactionSanity(tx *btcutil.Tx, enforceSoftFork bool) error {
|
2013-07-18 16:49:28 +02:00
|
|
|
// A transaction must have at least one input.
|
2013-10-28 21:17:53 +01:00
|
|
|
msgTx := tx.MsgTx()
|
|
|
|
if len(msgTx.TxIn) == 0 {
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrNoTxInputs, "transaction has no inputs")
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// A transaction must have at least one output.
|
2013-10-28 21:17:53 +01:00
|
|
|
if len(msgTx.TxOut) == 0 {
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrNoTxOutputs, "transaction has no outputs")
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
2014-06-29 22:11:13 +02:00
|
|
|
// A transaction must not exceed the maximum allowed block payload when
|
|
|
|
// serialized.
|
2016-10-19 03:33:20 +02:00
|
|
|
serializedTxSize := tx.MsgTx().SerializeSizeStripped()
|
|
|
|
if serializedTxSize > MaxBlockBaseSize {
|
2014-06-29 22:11:13 +02:00
|
|
|
str := fmt.Sprintf("serialized transaction is too big - got "+
|
2016-10-19 03:33:20 +02:00
|
|
|
"%d, max %d", serializedTxSize, MaxBlockBaseSize)
|
2014-06-29 22:11:13 +02:00
|
|
|
return ruleError(ErrTxTooBig, str)
|
|
|
|
}
|
2013-07-18 16:49:28 +02:00
|
|
|
|
|
|
|
// Ensure the transaction amounts are in range. Each transaction
|
|
|
|
// output must not be negative or more than the max allowed per
|
|
|
|
// transaction. Also, the total of all outputs must abide by the same
|
|
|
|
// restrictions. All amounts in a transaction are in a unit value known
|
|
|
|
// as a satoshi. One bitcoin is a quantity of satoshi as defined by the
|
2013-10-29 19:57:31 +01:00
|
|
|
// SatoshiPerBitcoin constant.
|
2013-07-18 16:49:28 +02:00
|
|
|
var totalSatoshi int64
|
2013-10-28 21:17:53 +01:00
|
|
|
for _, txOut := range msgTx.TxOut {
|
2013-07-18 16:49:28 +02:00
|
|
|
satoshi := txOut.Value
|
|
|
|
if satoshi < 0 {
|
|
|
|
str := fmt.Sprintf("transaction output has negative "+
|
|
|
|
"value of %v", satoshi)
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrBadTxOutValue, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
2013-10-29 21:08:12 +01:00
|
|
|
if satoshi > btcutil.MaxSatoshi {
|
2013-07-18 16:49:28 +02:00
|
|
|
str := fmt.Sprintf("transaction output value of %v is "+
|
|
|
|
"higher than max allowed value of %v", satoshi,
|
2013-10-29 21:08:12 +01:00
|
|
|
btcutil.MaxSatoshi)
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrBadTxOutValue, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
2015-11-06 23:16:07 +01:00
|
|
|
// Two's complement int64 overflow guarantees that any overflow
|
|
|
|
// is detected and reported. This is impossible for Bitcoin, but
|
|
|
|
// perhaps possible if an alt increases the total money supply.
|
2013-07-18 16:49:28 +02:00
|
|
|
totalSatoshi += satoshi
|
|
|
|
if totalSatoshi < 0 {
|
|
|
|
str := fmt.Sprintf("total value of all transaction "+
|
2015-11-06 23:16:07 +01:00
|
|
|
"outputs exceeds max allowed value of %v",
|
|
|
|
btcutil.MaxSatoshi)
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrBadTxOutValue, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
2013-10-29 21:08:12 +01:00
|
|
|
if totalSatoshi > btcutil.MaxSatoshi {
|
2013-07-18 16:49:28 +02:00
|
|
|
str := fmt.Sprintf("total value of all transaction "+
|
|
|
|
"outputs is %v which is higher than max "+
|
2013-10-29 21:08:12 +01:00
|
|
|
"allowed value of %v", totalSatoshi,
|
|
|
|
btcutil.MaxSatoshi)
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrBadTxOutValue, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
2018-06-15 22:06:26 +02:00
|
|
|
|
|
|
|
err := txscript.AllClaimsAreSane(txOut.PkScript, enforceSoftFork)
|
|
|
|
if err != nil {
|
|
|
|
return ruleError(ErrBadTxOutValue, err.Error())
|
|
|
|
}
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check for duplicate transaction inputs.
|
2015-02-05 22:16:39 +01:00
|
|
|
existingTxOut := make(map[wire.OutPoint]struct{})
|
2013-10-28 21:17:53 +01:00
|
|
|
for _, txIn := range msgTx.TxIn {
|
2014-10-01 14:53:47 +02:00
|
|
|
if _, exists := existingTxOut[txIn.PreviousOutPoint]; exists {
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrDuplicateTxInputs, "transaction "+
|
|
|
|
"contains duplicate inputs")
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
2014-10-01 14:53:47 +02:00
|
|
|
existingTxOut[txIn.PreviousOutPoint] = struct{}{}
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Coinbase script length must be between min and max length.
|
2013-09-30 23:41:51 +02:00
|
|
|
if IsCoinBase(tx) {
|
2013-10-28 21:17:53 +01:00
|
|
|
slen := len(msgTx.TxIn[0].SignatureScript)
|
2014-03-15 21:01:48 +01:00
|
|
|
if slen < MinCoinbaseScriptLen || slen > MaxCoinbaseScriptLen {
|
2013-07-18 16:49:28 +02:00
|
|
|
str := fmt.Sprintf("coinbase transaction script length "+
|
|
|
|
"of %d is out of range (min: %d, max: %d)",
|
2014-03-15 21:01:48 +01:00
|
|
|
slen, MinCoinbaseScriptLen, MaxCoinbaseScriptLen)
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrBadCoinbaseScriptLen, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Previous transaction outputs referenced by the inputs to this
|
|
|
|
// transaction must not be null.
|
2013-10-28 21:17:53 +01:00
|
|
|
for _, txIn := range msgTx.TxIn {
|
multi: Rework utxoset/view to use outpoints.
This modifies the utxoset in the database and related UtxoViewpoint to
store and work with unspent transaction outputs on a per-output basis
instead of at a transaction level. This was inspired by similar recent
changes in Bitcoin Core.
The primary motivation is to simplify the code, pave the way for a
utxo cache, and generally focus on optimizing runtime performance.
The tradeoff is that this approach does somewhat increase the size of
the serialized utxoset since it means that the transaction hash is
duplicated for each output as a part of the key and some additional
details such as whether the containing transaction is a coinbase and the
block height it was a part of are duplicated in each output.
However, in practice, the size difference isn't all that large, disk
space is relatively cheap, certainly cheaper than memory, and it is much
more important to provide more efficient runtime operation since that is
the ultimate purpose of the daemon.
While performing this conversion, it also simplifies the code to remove
the transaction version information from the utxoset as well as the
spend journal. The logic for only serializing it under certain
circumstances is complicated and it isn't actually used anywhere aside
from the gettxout RPC where it also isn't used by anything important
either. Consequently, this also removes the version field of the
gettxout RPC result.
The utxos in the database are automatically migrated to the new format
with this commit and it is possible to interrupt and resume the
migration process.
Finally, it also updates the tests for the new format and adds a new
function to the tests to convert the old test data to the new format for
convenience. The data has already been converted and updated in the
commit.
An overview of the changes are as follows:
- Remove transaction version from both spent and unspent output entries
- Update utxo serialization format to exclude the version
- Modify the spend journal serialization format
- The old version field is now reserved and always stores zero and
ignores it when reading
- This allows old entries to be used by new code without having to
migrate the entire spend journal
- Remove version field from gettxout RPC result
- Convert UtxoEntry to represent a specific utxo instead of a
transaction with all remaining utxos
- Optimize for memory usage with an eye towards a utxo cache
- Combine details such as whether the txout was contained in a
coinbase, is spent, and is modified into a single packed field of
bit flags
- Align entry fields to eliminate extra padding since ultimately
there will be a lot of these in memory
- Introduce a free list for serializing an outpoint to the database
key format to significantly reduce pressure on the GC
- Update all related functions that previously dealt with transaction
hashes to accept outpoints instead
- Update all callers accordingly
- Only add individually requested outputs from the mempool when
constructing a mempool view
- Modify the spend journal to always store the block height and coinbase
information with every spent txout
- Introduce code to handle fetching the missing information from
another utxo from the same transaction in the event an old style
entry is encountered
- Make use of a database cursor with seek to do this much more
efficiently than testing every possible output
- Always decompress data loaded from the database now that a utxo entry
only consists of a specific output
- Introduce upgrade code to migrate the utxo set to the new format
- Store versions of the utxoset and spend journal buckets
- Allow migration process to be interrupted and resumed
- Update all tests to expect the correct encodings, remove tests that no
longer apply, and add new ones for the new expected behavior
- Convert old tests for the legacy utxo format deserialization code to
test the new function that is used during upgrade
- Update the utxostore test data and add function that was used to
convert it
- Introduce a few new functions on UtxoViewpoint
- AddTxOut for adding an individual txout versus all of them
- addTxOut to handle the common code between the new AddTxOut and
existing AddTxOuts
- RemoveEntry for removing an individual txout
- fetchEntryByHash for fetching any remaining utxo for a given
transaction hash
2017-09-03 09:59:15 +02:00
|
|
|
if isNullOutpoint(&txIn.PreviousOutPoint) {
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrBadTxInput, "transaction "+
|
|
|
|
"input refers to previous output that "+
|
|
|
|
"is null")
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-06-26 22:50:13 +02:00
|
|
|
// checkProofOfWork ensures the block header bits which indicate the target
|
2013-07-18 16:49:28 +02:00
|
|
|
// difficulty is in min/max range and that the block hash is less than the
|
|
|
|
// target difficulty as claimed.
|
2014-06-26 22:50:13 +02:00
|
|
|
//
|
|
|
|
// The flags modify the behavior of this function as follows:
|
|
|
|
// - BFNoPoWCheck: The check to ensure the block hash is less than the target
|
|
|
|
// difficulty is not performed.
|
2015-04-19 20:34:42 +02:00
|
|
|
func checkProofOfWork(header *wire.BlockHeader, powLimit *big.Int, flags BehaviorFlags) error {
|
2013-07-18 16:49:28 +02:00
|
|
|
// The target difficulty must be larger than zero.
|
2015-04-19 20:34:42 +02:00
|
|
|
target := CompactToBig(header.Bits)
|
2013-07-18 16:49:28 +02:00
|
|
|
if target.Sign() <= 0 {
|
|
|
|
str := fmt.Sprintf("block target difficulty of %064x is too low",
|
|
|
|
target)
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrUnexpectedDifficulty, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// The target difficulty must be less than the maximum allowed.
|
|
|
|
if target.Cmp(powLimit) > 0 {
|
|
|
|
str := fmt.Sprintf("block target difficulty of %064x is "+
|
|
|
|
"higher than max of %064x", target, powLimit)
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrUnexpectedDifficulty, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
2014-06-26 22:50:13 +02:00
|
|
|
// The block hash must be less than the claimed target unless the flag
|
|
|
|
// to avoid proof of work checks is set.
|
|
|
|
if flags&BFNoPoWCheck != BFNoPoWCheck {
|
|
|
|
// The block hash must be less than the claimed target.
|
2018-05-29 06:06:46 +02:00
|
|
|
hash := header.BlockPoWHash()
|
2016-08-08 21:04:33 +02:00
|
|
|
hashNum := HashToBig(&hash)
|
2014-06-26 22:50:13 +02:00
|
|
|
if hashNum.Cmp(target) > 0 {
|
|
|
|
str := fmt.Sprintf("block hash of %064x is higher than "+
|
|
|
|
"expected max of %064x", hashNum, target)
|
|
|
|
return ruleError(ErrHighHash, str)
|
|
|
|
}
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-06-26 22:50:13 +02:00
|
|
|
// CheckProofOfWork ensures the block header bits which indicate the target
|
|
|
|
// difficulty is in min/max range and that the block hash is less than the
|
|
|
|
// target difficulty as claimed.
|
|
|
|
func CheckProofOfWork(block *btcutil.Block, powLimit *big.Int) error {
|
2015-04-19 20:34:42 +02:00
|
|
|
return checkProofOfWork(&block.MsgBlock().Header, powLimit, BFNone)
|
2014-06-26 22:50:13 +02:00
|
|
|
}
|
|
|
|
|
2014-02-28 19:16:56 +01:00
|
|
|
// CountSigOps returns the number of signature operations for all transaction
|
2013-07-18 16:49:28 +02:00
|
|
|
// input and output scripts in the provided transaction. This uses the
|
|
|
|
// quicker, but imprecise, signature operation counting mechanism from
|
2015-01-30 19:08:47 +01:00
|
|
|
// txscript.
|
2014-02-28 19:16:56 +01:00
|
|
|
func CountSigOps(tx *btcutil.Tx) int {
|
2013-10-28 21:17:53 +01:00
|
|
|
msgTx := tx.MsgTx()
|
|
|
|
|
2013-07-18 16:49:28 +02:00
|
|
|
// Accumulate the number of signature operations in all transaction
|
2013-07-29 23:18:15 +02:00
|
|
|
// inputs.
|
2013-07-18 16:49:28 +02:00
|
|
|
totalSigOps := 0
|
2013-07-29 23:18:15 +02:00
|
|
|
for _, txIn := range msgTx.TxIn {
|
2015-01-30 19:08:47 +01:00
|
|
|
numSigOps := txscript.GetSigOpCount(txIn.SignatureScript)
|
2013-07-18 16:49:28 +02:00
|
|
|
totalSigOps += numSigOps
|
|
|
|
}
|
|
|
|
|
|
|
|
// Accumulate the number of signature operations in all transaction
|
|
|
|
// outputs.
|
|
|
|
for _, txOut := range msgTx.TxOut {
|
2015-01-30 19:08:47 +01:00
|
|
|
numSigOps := txscript.GetSigOpCount(txOut.PkScript)
|
2013-07-18 16:49:28 +02:00
|
|
|
totalSigOps += numSigOps
|
|
|
|
}
|
|
|
|
|
2013-07-29 23:18:15 +02:00
|
|
|
return totalSigOps
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
2014-03-02 04:50:11 +01:00
|
|
|
// CountP2SHSigOps returns the number of signature operations for all input
|
2013-07-18 16:49:28 +02:00
|
|
|
// transactions which are of the pay-to-script-hash type. This uses the
|
2015-01-30 19:08:47 +01:00
|
|
|
// precise, signature operation counting mechanism from the script engine which
|
|
|
|
// requires access to the input transaction scripts.
|
2015-08-26 06:03:18 +02:00
|
|
|
func CountP2SHSigOps(tx *btcutil.Tx, isCoinBaseTx bool, utxoView *UtxoViewpoint) (int, error) {
|
2013-07-18 16:49:28 +02:00
|
|
|
// Coinbase transactions have no interesting inputs.
|
|
|
|
if isCoinBaseTx {
|
|
|
|
return 0, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Accumulate the number of signature operations in all transaction
|
|
|
|
// inputs.
|
2013-10-28 21:17:53 +01:00
|
|
|
msgTx := tx.MsgTx()
|
2013-07-18 16:49:28 +02:00
|
|
|
totalSigOps := 0
|
2015-08-26 06:03:18 +02:00
|
|
|
for txInIndex, txIn := range msgTx.TxIn {
|
2013-07-18 16:49:28 +02:00
|
|
|
// Ensure the referenced input transaction is available.
|
multi: Rework utxoset/view to use outpoints.
This modifies the utxoset in the database and related UtxoViewpoint to
store and work with unspent transaction outputs on a per-output basis
instead of at a transaction level. This was inspired by similar recent
changes in Bitcoin Core.
The primary motivation is to simplify the code, pave the way for a
utxo cache, and generally focus on optimizing runtime performance.
The tradeoff is that this approach does somewhat increase the size of
the serialized utxoset since it means that the transaction hash is
duplicated for each output as a part of the key and some additional
details such as whether the containing transaction is a coinbase and the
block height it was a part of are duplicated in each output.
However, in practice, the size difference isn't all that large, disk
space is relatively cheap, certainly cheaper than memory, and it is much
more important to provide more efficient runtime operation since that is
the ultimate purpose of the daemon.
While performing this conversion, it also simplifies the code to remove
the transaction version information from the utxoset as well as the
spend journal. The logic for only serializing it under certain
circumstances is complicated and it isn't actually used anywhere aside
from the gettxout RPC where it also isn't used by anything important
either. Consequently, this also removes the version field of the
gettxout RPC result.
The utxos in the database are automatically migrated to the new format
with this commit and it is possible to interrupt and resume the
migration process.
Finally, it also updates the tests for the new format and adds a new
function to the tests to convert the old test data to the new format for
convenience. The data has already been converted and updated in the
commit.
An overview of the changes are as follows:
- Remove transaction version from both spent and unspent output entries
- Update utxo serialization format to exclude the version
- Modify the spend journal serialization format
- The old version field is now reserved and always stores zero and
ignores it when reading
- This allows old entries to be used by new code without having to
migrate the entire spend journal
- Remove version field from gettxout RPC result
- Convert UtxoEntry to represent a specific utxo instead of a
transaction with all remaining utxos
- Optimize for memory usage with an eye towards a utxo cache
- Combine details such as whether the txout was contained in a
coinbase, is spent, and is modified into a single packed field of
bit flags
- Align entry fields to eliminate extra padding since ultimately
there will be a lot of these in memory
- Introduce a free list for serializing an outpoint to the database
key format to significantly reduce pressure on the GC
- Update all related functions that previously dealt with transaction
hashes to accept outpoints instead
- Update all callers accordingly
- Only add individually requested outputs from the mempool when
constructing a mempool view
- Modify the spend journal to always store the block height and coinbase
information with every spent txout
- Introduce code to handle fetching the missing information from
another utxo from the same transaction in the event an old style
entry is encountered
- Make use of a database cursor with seek to do this much more
efficiently than testing every possible output
- Always decompress data loaded from the database now that a utxo entry
only consists of a specific output
- Introduce upgrade code to migrate the utxo set to the new format
- Store versions of the utxoset and spend journal buckets
- Allow migration process to be interrupted and resumed
- Update all tests to expect the correct encodings, remove tests that no
longer apply, and add new ones for the new expected behavior
- Convert old tests for the legacy utxo format deserialization code to
test the new function that is used during upgrade
- Update the utxostore test data and add function that was used to
convert it
- Introduce a few new functions on UtxoViewpoint
- AddTxOut for adding an individual txout versus all of them
- addTxOut to handle the common code between the new AddTxOut and
existing AddTxOuts
- RemoveEntry for removing an individual txout
- fetchEntryByHash for fetching any remaining utxo for a given
transaction hash
2017-09-03 09:59:15 +02:00
|
|
|
utxo := utxoView.LookupEntry(txIn.PreviousOutPoint)
|
|
|
|
if utxo == nil || utxo.IsSpent() {
|
2017-08-14 07:22:40 +02:00
|
|
|
str := fmt.Sprintf("output %v referenced from "+
|
|
|
|
"transaction %s:%d either does not exist or "+
|
|
|
|
"has already been spent", txIn.PreviousOutPoint,
|
|
|
|
tx.Hash(), txInIndex)
|
|
|
|
return 0, ruleError(ErrMissingTxOut, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// We're only interested in pay-to-script-hash types, so skip
|
|
|
|
// this input if it's not one.
|
multi: Rework utxoset/view to use outpoints.
This modifies the utxoset in the database and related UtxoViewpoint to
store and work with unspent transaction outputs on a per-output basis
instead of at a transaction level. This was inspired by similar recent
changes in Bitcoin Core.
The primary motivation is to simplify the code, pave the way for a
utxo cache, and generally focus on optimizing runtime performance.
The tradeoff is that this approach does somewhat increase the size of
the serialized utxoset since it means that the transaction hash is
duplicated for each output as a part of the key and some additional
details such as whether the containing transaction is a coinbase and the
block height it was a part of are duplicated in each output.
However, in practice, the size difference isn't all that large, disk
space is relatively cheap, certainly cheaper than memory, and it is much
more important to provide more efficient runtime operation since that is
the ultimate purpose of the daemon.
While performing this conversion, it also simplifies the code to remove
the transaction version information from the utxoset as well as the
spend journal. The logic for only serializing it under certain
circumstances is complicated and it isn't actually used anywhere aside
from the gettxout RPC where it also isn't used by anything important
either. Consequently, this also removes the version field of the
gettxout RPC result.
The utxos in the database are automatically migrated to the new format
with this commit and it is possible to interrupt and resume the
migration process.
Finally, it also updates the tests for the new format and adds a new
function to the tests to convert the old test data to the new format for
convenience. The data has already been converted and updated in the
commit.
An overview of the changes are as follows:
- Remove transaction version from both spent and unspent output entries
- Update utxo serialization format to exclude the version
- Modify the spend journal serialization format
- The old version field is now reserved and always stores zero and
ignores it when reading
- This allows old entries to be used by new code without having to
migrate the entire spend journal
- Remove version field from gettxout RPC result
- Convert UtxoEntry to represent a specific utxo instead of a
transaction with all remaining utxos
- Optimize for memory usage with an eye towards a utxo cache
- Combine details such as whether the txout was contained in a
coinbase, is spent, and is modified into a single packed field of
bit flags
- Align entry fields to eliminate extra padding since ultimately
there will be a lot of these in memory
- Introduce a free list for serializing an outpoint to the database
key format to significantly reduce pressure on the GC
- Update all related functions that previously dealt with transaction
hashes to accept outpoints instead
- Update all callers accordingly
- Only add individually requested outputs from the mempool when
constructing a mempool view
- Modify the spend journal to always store the block height and coinbase
information with every spent txout
- Introduce code to handle fetching the missing information from
another utxo from the same transaction in the event an old style
entry is encountered
- Make use of a database cursor with seek to do this much more
efficiently than testing every possible output
- Always decompress data loaded from the database now that a utxo entry
only consists of a specific output
- Introduce upgrade code to migrate the utxo set to the new format
- Store versions of the utxoset and spend journal buckets
- Allow migration process to be interrupted and resumed
- Update all tests to expect the correct encodings, remove tests that no
longer apply, and add new ones for the new expected behavior
- Convert old tests for the legacy utxo format deserialization code to
test the new function that is used during upgrade
- Update the utxostore test data and add function that was used to
convert it
- Introduce a few new functions on UtxoViewpoint
- AddTxOut for adding an individual txout versus all of them
- addTxOut to handle the common code between the new AddTxOut and
existing AddTxOuts
- RemoveEntry for removing an individual txout
- fetchEntryByHash for fetching any remaining utxo for a given
transaction hash
2017-09-03 09:59:15 +02:00
|
|
|
pkScript := utxo.PkScript()
|
2015-01-30 19:08:47 +01:00
|
|
|
if !txscript.IsPayToScriptHash(pkScript) {
|
2013-07-18 16:49:28 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// Count the precise number of signature operations in the
|
|
|
|
// referenced public key script.
|
|
|
|
sigScript := txIn.SignatureScript
|
2015-01-30 19:08:47 +01:00
|
|
|
numSigOps := txscript.GetPreciseSigOpCount(sigScript, pkScript,
|
2013-07-29 23:18:15 +02:00
|
|
|
true)
|
2013-07-18 16:49:28 +02:00
|
|
|
|
|
|
|
// We could potentially overflow the accumulator so check for
|
|
|
|
// overflow.
|
|
|
|
lastSigOps := totalSigOps
|
|
|
|
totalSigOps += numSigOps
|
|
|
|
if totalSigOps < lastSigOps {
|
2015-08-26 06:03:18 +02:00
|
|
|
str := fmt.Sprintf("the public key script from output "+
|
|
|
|
"%v contains too many signature operations - "+
|
|
|
|
"overflow", txIn.PreviousOutPoint)
|
2014-06-25 00:10:53 +02:00
|
|
|
return 0, ruleError(ErrTooManySigOps, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return totalSigOps, nil
|
|
|
|
}
|
|
|
|
|
2015-04-19 20:34:42 +02:00
|
|
|
// checkBlockHeaderSanity performs some preliminary checks on a block header to
|
|
|
|
// ensure it is sane before continuing with processing. These checks are
|
|
|
|
// context free.
|
2014-06-26 22:50:13 +02:00
|
|
|
//
|
|
|
|
// The flags do not modify the behavior of this function directly, however they
|
|
|
|
// are needed to pass along to checkProofOfWork.
|
2015-04-19 20:34:42 +02:00
|
|
|
func checkBlockHeaderSanity(header *wire.BlockHeader, powLimit *big.Int, timeSource MedianTimeSource, flags BehaviorFlags) error {
|
2013-07-18 16:49:28 +02:00
|
|
|
// Ensure the proof of work bits in the block header is in min/max range
|
|
|
|
// and the block hash is less than the target value described by the
|
|
|
|
// bits.
|
2015-04-19 20:34:42 +02:00
|
|
|
err := checkProofOfWork(header, powLimit, flags)
|
2013-07-18 16:49:28 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2014-02-24 17:17:13 +01:00
|
|
|
// A block timestamp must not have a greater precision than one second.
|
|
|
|
// This check is necessary because Go time.Time values support
|
|
|
|
// nanosecond precision whereas the consensus rules only apply to
|
|
|
|
// seconds and it's much nicer to deal with standard Go time values
|
|
|
|
// instead of converting to seconds everywhere.
|
|
|
|
if !header.Timestamp.Equal(time.Unix(header.Timestamp.Unix(), 0)) {
|
|
|
|
str := fmt.Sprintf("block timestamp of %v has a higher "+
|
|
|
|
"precision than one second", header.Timestamp)
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrInvalidTime, str)
|
2014-02-24 17:17:13 +01:00
|
|
|
}
|
|
|
|
|
2014-06-30 17:27:44 +02:00
|
|
|
// Ensure the block time is not too far in the future.
|
2014-10-03 22:29:25 +02:00
|
|
|
maxTimestamp := timeSource.AdjustedTime().Add(time.Second *
|
|
|
|
MaxTimeOffsetSeconds)
|
2014-06-30 17:27:44 +02:00
|
|
|
if header.Timestamp.After(maxTimestamp) {
|
2013-07-18 16:49:28 +02:00
|
|
|
str := fmt.Sprintf("block timestamp of %v is too far in the "+
|
|
|
|
"future", header.Timestamp)
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrTimeTooNew, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
2015-04-19 20:34:42 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// checkBlockSanity performs some preliminary checks on a block to ensure it is
|
|
|
|
// sane before continuing with block processing. These checks are context free.
|
|
|
|
//
|
|
|
|
// The flags do not modify the behavior of this function directly, however they
|
|
|
|
// are needed to pass along to checkBlockHeaderSanity.
|
|
|
|
func checkBlockSanity(block *btcutil.Block, powLimit *big.Int, timeSource MedianTimeSource, flags BehaviorFlags) error {
|
|
|
|
msgBlock := block.MsgBlock()
|
|
|
|
header := &msgBlock.Header
|
|
|
|
err := checkBlockHeaderSanity(header, powLimit, timeSource, flags)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// A block must have at least one transaction.
|
|
|
|
numTx := len(msgBlock.Transactions)
|
|
|
|
if numTx == 0 {
|
|
|
|
return ruleError(ErrNoTransactions, "block does not contain "+
|
|
|
|
"any transactions")
|
|
|
|
}
|
|
|
|
|
2017-09-06 05:42:54 +02:00
|
|
|
// A block must not have more transactions than the max block payload or
|
|
|
|
// else it is certainly over the weight limit.
|
|
|
|
if numTx > MaxBlockBaseSize {
|
2015-04-19 20:34:42 +02:00
|
|
|
str := fmt.Sprintf("block contains too many transactions - "+
|
2017-09-06 05:42:54 +02:00
|
|
|
"got %d, max %d", numTx, MaxBlockBaseSize)
|
|
|
|
return ruleError(ErrBlockTooBig, str)
|
2015-04-19 20:34:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// A block must not exceed the maximum allowed block payload when
|
|
|
|
// serialized.
|
2016-10-19 03:33:20 +02:00
|
|
|
serializedSize := msgBlock.SerializeSizeStripped()
|
|
|
|
if serializedSize > MaxBlockBaseSize {
|
2015-04-19 20:34:42 +02:00
|
|
|
str := fmt.Sprintf("serialized block is too big - got %d, "+
|
2016-10-19 03:33:20 +02:00
|
|
|
"max %d", serializedSize, MaxBlockBaseSize)
|
2015-04-19 20:34:42 +02:00
|
|
|
return ruleError(ErrBlockTooBig, str)
|
|
|
|
}
|
|
|
|
|
2013-07-18 16:49:28 +02:00
|
|
|
// The first transaction in a block must be a coinbase.
|
2014-06-29 22:11:13 +02:00
|
|
|
transactions := block.Transactions()
|
2013-09-30 23:41:51 +02:00
|
|
|
if !IsCoinBase(transactions[0]) {
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrFirstTxNotCoinbase, "first transaction in "+
|
|
|
|
"block is not a coinbase")
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// A block must not have more than one coinbase.
|
2013-07-30 00:02:42 +02:00
|
|
|
for i, tx := range transactions[1:] {
|
2013-09-30 23:41:51 +02:00
|
|
|
if IsCoinBase(tx) {
|
2013-07-30 00:02:42 +02:00
|
|
|
str := fmt.Sprintf("block contains second coinbase at "+
|
2015-08-26 06:03:18 +02:00
|
|
|
"index %d", i+1)
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrMultipleCoinbases, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do some preliminary checks on each transaction to ensure they are
|
|
|
|
// sane before continuing.
|
|
|
|
for _, tx := range transactions {
|
2021-08-12 17:35:25 +02:00
|
|
|
err := CheckTransactionSanity(tx, block.Height() >= param.ActiveParams.GrandForkHeight)
|
2013-07-18 16:49:28 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build merkle tree and ensure the calculated merkle root matches the
|
|
|
|
// entry in the block header. This also has the effect of caching all
|
|
|
|
// of the transaction hashes in the block to speed up future hash
|
|
|
|
// checks. Bitcoind builds the tree here and checks the merkle root
|
|
|
|
// after the following checks, but there is no reason not to check the
|
|
|
|
// merkle root matches here.
|
2016-10-19 03:33:20 +02:00
|
|
|
merkles := BuildMerkleTreeStore(block.Transactions(), false)
|
2013-07-18 16:49:28 +02:00
|
|
|
calculatedMerkleRoot := merkles[len(merkles)-1]
|
|
|
|
if !header.MerkleRoot.IsEqual(calculatedMerkleRoot) {
|
2013-07-30 02:50:00 +02:00
|
|
|
str := fmt.Sprintf("block merkle root is invalid - block "+
|
|
|
|
"header indicates %v, but calculated value is %v",
|
|
|
|
header.MerkleRoot, calculatedMerkleRoot)
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrBadMerkleRoot, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check for duplicate transactions. This check will be fairly quick
|
|
|
|
// since the transaction hashes are already cached due to building the
|
|
|
|
// merkle tree above.
|
2016-08-08 21:04:33 +02:00
|
|
|
existingTxHashes := make(map[chainhash.Hash]struct{})
|
2013-10-28 21:17:53 +01:00
|
|
|
for _, tx := range transactions {
|
2016-08-08 21:04:33 +02:00
|
|
|
hash := tx.Hash()
|
2013-07-18 16:49:28 +02:00
|
|
|
if _, exists := existingTxHashes[*hash]; exists {
|
|
|
|
str := fmt.Sprintf("block contains duplicate "+
|
|
|
|
"transaction %v", hash)
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrDuplicateTx, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
2014-07-02 17:56:22 +02:00
|
|
|
existingTxHashes[*hash] = struct{}{}
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// The number of signature operations must be less than the maximum
|
|
|
|
// allowed per block.
|
|
|
|
totalSigOps := 0
|
2013-07-29 23:18:15 +02:00
|
|
|
for _, tx := range transactions {
|
2013-07-18 16:49:28 +02:00
|
|
|
// We could potentially overflow the accumulator so check for
|
|
|
|
// overflow.
|
|
|
|
lastSigOps := totalSigOps
|
2016-10-19 03:33:20 +02:00
|
|
|
totalSigOps += (CountSigOps(tx) * WitnessScaleFactor)
|
|
|
|
if totalSigOps < lastSigOps || totalSigOps > MaxBlockSigOpsCost {
|
2013-07-18 16:49:28 +02:00
|
|
|
str := fmt.Sprintf("block contains too many signature "+
|
|
|
|
"operations - got %v, max %v", totalSigOps,
|
2016-10-19 03:33:20 +02:00
|
|
|
MaxBlockSigOpsCost)
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrTooManySigOps, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-06-26 22:50:13 +02:00
|
|
|
// CheckBlockSanity performs some preliminary checks on a block to ensure it is
|
|
|
|
// sane before continuing with block processing. These checks are context free.
|
2014-10-03 22:29:25 +02:00
|
|
|
func CheckBlockSanity(block *btcutil.Block, powLimit *big.Int, timeSource MedianTimeSource) error {
|
|
|
|
return checkBlockSanity(block, powLimit, timeSource, BFNone)
|
2014-06-26 22:50:13 +02:00
|
|
|
}
|
|
|
|
|
2015-08-26 06:03:18 +02:00
|
|
|
// ExtractCoinbaseHeight attempts to extract the height of the block from the
|
|
|
|
// scriptSig of a coinbase transaction. Coinbase heights are only present in
|
|
|
|
// blocks of version 2 or later. This was added as part of BIP0034.
|
|
|
|
func ExtractCoinbaseHeight(coinbaseTx *btcutil.Tx) (int32, error) {
|
|
|
|
sigScript := coinbaseTx.MsgTx().TxIn[0].SignatureScript
|
|
|
|
if len(sigScript) < 1 {
|
|
|
|
str := "the coinbase signature script for blocks of " +
|
|
|
|
"version %d or greater must start with the " +
|
|
|
|
"length of the serialized block height"
|
|
|
|
str = fmt.Sprintf(str, serializedHeightVersion)
|
|
|
|
return 0, ruleError(ErrMissingCoinbaseHeight, str)
|
|
|
|
}
|
|
|
|
|
2016-10-20 05:48:10 +02:00
|
|
|
// Detect the case when the block height is a small integer encoded with
|
|
|
|
// as single byte.
|
|
|
|
opcode := int(sigScript[0])
|
|
|
|
if opcode == txscript.OP_0 {
|
|
|
|
return 0, nil
|
|
|
|
}
|
|
|
|
if opcode >= txscript.OP_1 && opcode <= txscript.OP_16 {
|
|
|
|
return int32(opcode - (txscript.OP_1 - 1)), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, the opcode is the length of the following bytes which
|
|
|
|
// encode in the block height.
|
2015-08-26 06:03:18 +02:00
|
|
|
serializedLen := int(sigScript[0])
|
|
|
|
if len(sigScript[1:]) < serializedLen {
|
|
|
|
str := "the coinbase signature script for blocks of " +
|
|
|
|
"version %d or greater must start with the " +
|
|
|
|
"serialized block height"
|
|
|
|
str = fmt.Sprintf(str, serializedLen)
|
|
|
|
return 0, ruleError(ErrMissingCoinbaseHeight, str)
|
|
|
|
}
|
|
|
|
|
2017-03-22 21:18:32 +01:00
|
|
|
serializedHeightBytes := make([]byte, 8)
|
2015-08-26 06:03:18 +02:00
|
|
|
copy(serializedHeightBytes, sigScript[1:serializedLen+1])
|
|
|
|
serializedHeight := binary.LittleEndian.Uint64(serializedHeightBytes)
|
|
|
|
|
|
|
|
return int32(serializedHeight), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// checkSerializedHeight checks if the signature script in the passed
|
|
|
|
// transaction starts with the serialized block height of wantHeight.
|
|
|
|
func checkSerializedHeight(coinbaseTx *btcutil.Tx, wantHeight int32) error {
|
|
|
|
serializedHeight, err := ExtractCoinbaseHeight(coinbaseTx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if serializedHeight != wantHeight {
|
|
|
|
str := fmt.Sprintf("the coinbase signature script serialized "+
|
|
|
|
"block height is %d when %d was expected",
|
|
|
|
serializedHeight, wantHeight)
|
|
|
|
return ruleError(ErrBadCoinbaseHeight, str)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-05-30 16:59:51 +02:00
|
|
|
// checkBlockHeaderContext performs several validation checks on the block header
|
2015-04-19 20:34:42 +02:00
|
|
|
// which depend on its position within the block chain.
|
|
|
|
//
|
|
|
|
// The flags modify the behavior of this function as follows:
|
|
|
|
// - BFFastAdd: All checks except those involving comparing the header against
|
|
|
|
// the checkpoints are not performed.
|
2015-08-26 06:03:18 +02:00
|
|
|
//
|
|
|
|
// This function MUST be called with the chain state lock held (for writes).
|
2015-04-19 20:34:42 +02:00
|
|
|
func (b *BlockChain) checkBlockHeaderContext(header *wire.BlockHeader, prevNode *blockNode, flags BehaviorFlags) error {
|
|
|
|
fastAdd := flags&BFFastAdd == BFFastAdd
|
|
|
|
if !fastAdd {
|
|
|
|
// Ensure the difficulty specified in the block header matches
|
|
|
|
// the calculated difficulty based on the previous block and
|
|
|
|
// difficulty retarget rules.
|
|
|
|
expectedDifficulty, err := b.calcNextRequiredDifficulty(prevNode,
|
|
|
|
header.Timestamp)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
blockDifficulty := header.Bits
|
|
|
|
if blockDifficulty != expectedDifficulty {
|
|
|
|
str := "block difficulty of %d is not the expected value of %d"
|
|
|
|
str = fmt.Sprintf(str, blockDifficulty, expectedDifficulty)
|
|
|
|
return ruleError(ErrUnexpectedDifficulty, str)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure the timestamp for the block header is after the
|
|
|
|
// median time of the last several blocks (medianTimeBlocks).
|
2017-02-03 19:13:53 +01:00
|
|
|
medianTime := prevNode.CalcPastMedianTime()
|
2015-04-19 20:34:42 +02:00
|
|
|
if !header.Timestamp.After(medianTime) {
|
|
|
|
str := "block timestamp of %v is not after expected %v"
|
|
|
|
str = fmt.Sprintf(str, header.Timestamp, medianTime)
|
|
|
|
return ruleError(ErrTimeTooOld, str)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// The height of this block is one more than the referenced previous
|
|
|
|
// block.
|
|
|
|
blockHeight := prevNode.height + 1
|
|
|
|
|
|
|
|
// Ensure chain matches up to predetermined checkpoints.
|
2016-08-08 21:04:33 +02:00
|
|
|
blockHash := header.BlockHash()
|
2015-04-19 20:34:42 +02:00
|
|
|
if !b.verifyCheckpoint(blockHeight, &blockHash) {
|
|
|
|
str := fmt.Sprintf("block at height %d does not match "+
|
|
|
|
"checkpoint hash", blockHeight)
|
|
|
|
return ruleError(ErrBadCheckpoint, str)
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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.
|
2017-02-04 04:18:39 +01:00
|
|
|
checkpointNode, err := b.findPreviousCheckpoint()
|
2015-04-19 20:34:42 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-02-04 04:18:39 +01:00
|
|
|
if checkpointNode != nil && blockHeight < checkpointNode.height {
|
2015-04-19 20:34:42 +02:00
|
|
|
str := fmt.Sprintf("block at height %d forks the main chain "+
|
|
|
|
"before the previous checkpoint at height %d",
|
2017-02-04 04:18:39 +01:00
|
|
|
blockHeight, checkpointNode.height)
|
2015-04-19 20:34:42 +02:00
|
|
|
return ruleError(ErrForkTooOld, str)
|
|
|
|
}
|
|
|
|
|
2016-10-19 21:24:42 +02:00
|
|
|
// Reject outdated block versions once a majority of the network
|
|
|
|
// has upgraded. These were originally voted on by BIP0034,
|
|
|
|
// BIP0065, and BIP0066.
|
|
|
|
params := b.chainParams
|
|
|
|
if header.Version < 2 && blockHeight >= params.BIP0034Height ||
|
|
|
|
header.Version < 3 && blockHeight >= params.BIP0066Height ||
|
|
|
|
header.Version < 4 && blockHeight >= params.BIP0065Height {
|
2015-10-28 16:43:36 +01:00
|
|
|
|
2016-10-19 21:24:42 +02:00
|
|
|
str := "new blocks with version %d are no longer valid"
|
|
|
|
str = fmt.Sprintf(str, header.Version)
|
|
|
|
return ruleError(ErrBlockVersionTooOld, str)
|
2015-04-19 20:34:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// checkBlockContext peforms several validation checks on the block which depend
|
|
|
|
// on its position within the block chain.
|
|
|
|
//
|
|
|
|
// The flags modify the behavior of this function as follows:
|
|
|
|
// - BFFastAdd: The transaction are not checked to see if they are finalized
|
|
|
|
// and the somewhat expensive BIP0034 validation is not performed.
|
|
|
|
//
|
|
|
|
// The flags are also passed to checkBlockHeaderContext. See its documentation
|
|
|
|
// for how the flags modify its behavior.
|
2015-08-26 06:03:18 +02:00
|
|
|
//
|
|
|
|
// This function MUST be called with the chain state lock held (for writes).
|
2015-04-19 20:34:42 +02:00
|
|
|
func (b *BlockChain) checkBlockContext(block *btcutil.Block, prevNode *blockNode, flags BehaviorFlags) error {
|
|
|
|
// Perform all block header related validation checks.
|
|
|
|
header := &block.MsgBlock().Header
|
|
|
|
err := b.checkBlockHeaderContext(header, prevNode, flags)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
fastAdd := flags&BFFastAdd == BFFastAdd
|
|
|
|
if !fastAdd {
|
2016-09-27 04:00:28 +02:00
|
|
|
// Obtain the latest state of the deployed CSV soft-fork in
|
|
|
|
// order to properly guard the new validation behavior based on
|
|
|
|
// the current BIP 9 version bits state.
|
|
|
|
csvState, err := b.deploymentState(prevNode, chaincfg.DeploymentCSV)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Once the CSV soft-fork is fully active, we'll switch to
|
|
|
|
// using the current median time past of the past block's
|
|
|
|
// timestamps for all lock-time based checks.
|
|
|
|
blockTime := header.Timestamp
|
|
|
|
if csvState == ThresholdActive {
|
2017-02-03 19:13:53 +01:00
|
|
|
blockTime = prevNode.CalcPastMedianTime()
|
2016-09-27 04:00:28 +02:00
|
|
|
}
|
|
|
|
|
2015-04-19 20:34:42 +02:00
|
|
|
// The height of this block is one more than the referenced
|
|
|
|
// previous block.
|
|
|
|
blockHeight := prevNode.height + 1
|
|
|
|
|
|
|
|
// Ensure all transactions in the block are finalized.
|
|
|
|
for _, tx := range block.Transactions() {
|
2015-06-29 17:12:35 +02:00
|
|
|
if !IsFinalizedTransaction(tx, blockHeight,
|
2016-09-27 04:00:28 +02:00
|
|
|
blockTime) {
|
2015-04-19 20:34:42 +02:00
|
|
|
|
|
|
|
str := fmt.Sprintf("block contains unfinalized "+
|
2016-08-08 21:04:33 +02:00
|
|
|
"transaction %v", tx.Hash())
|
2015-04-19 20:34:42 +02:00
|
|
|
return ruleError(ErrUnfinalizedTx, str)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure coinbase starts with serialized block heights for
|
|
|
|
// blocks whose version is the serializedHeightVersion or newer
|
|
|
|
// once a majority of the network has upgraded. This is part of
|
|
|
|
// BIP0034.
|
|
|
|
if ShouldHaveSerializedBlockHeight(header) &&
|
2016-10-19 21:24:42 +02:00
|
|
|
blockHeight >= b.chainParams.BIP0034Height {
|
2015-04-19 20:34:42 +02:00
|
|
|
|
|
|
|
coinbaseTx := block.Transactions()[0]
|
|
|
|
err := checkSerializedHeight(coinbaseTx, blockHeight)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2016-10-19 03:33:20 +02:00
|
|
|
|
|
|
|
// Query for the Version Bits state for the segwit soft-fork
|
|
|
|
// deployment. If segwit is active, we'll switch over to
|
|
|
|
// enforcing all the new rules.
|
|
|
|
segwitState, err := b.deploymentState(prevNode,
|
|
|
|
chaincfg.DeploymentSegwit)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// If segwit is active, then we'll need to fully validate the
|
2017-10-29 20:56:03 +01:00
|
|
|
// new witness commitment for adherence to the rules.
|
2016-10-19 03:33:20 +02:00
|
|
|
if segwitState == ThresholdActive {
|
|
|
|
// Validate the witness commitment (if any) within the
|
|
|
|
// block. This involves asserting that if the coinbase
|
|
|
|
// contains the special commitment output, then this
|
|
|
|
// merkle root matches a computed merkle root of all
|
|
|
|
// the wtxid's of the transactions within the block. In
|
|
|
|
// addition, various other checks against the
|
|
|
|
// coinbase's witness stack.
|
|
|
|
if err := ValidateWitnessCommitment(block); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Once the witness commitment, witness nonce, and sig
|
|
|
|
// op cost have been validated, we can finally assert
|
|
|
|
// that the block's weight doesn't exceed the current
|
|
|
|
// consensus parameter.
|
|
|
|
blockWeight := GetBlockWeight(block)
|
|
|
|
if blockWeight > MaxBlockWeight {
|
|
|
|
str := fmt.Sprintf("block's weight metric is "+
|
|
|
|
"too high - got %v, max %v",
|
|
|
|
blockWeight, MaxBlockWeight)
|
2017-09-06 05:42:54 +02:00
|
|
|
return ruleError(ErrBlockWeightTooHigh, str)
|
2016-10-19 03:33:20 +02:00
|
|
|
}
|
|
|
|
}
|
2015-04-19 20:34:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2013-07-18 16:49:28 +02:00
|
|
|
// checkBIP0030 ensures blocks do not contain duplicate transactions which
|
|
|
|
// 'overwrite' older transactions that are not fully spent. This prevents an
|
|
|
|
// attack where a coinbase and all of its dependent transactions could be
|
|
|
|
// duplicated to effectively revert the overwritten transactions to a single
|
|
|
|
// confirmation thereby making them vulnerable to a double spend.
|
|
|
|
//
|
multi: Rework utxoset/view to use outpoints.
This modifies the utxoset in the database and related UtxoViewpoint to
store and work with unspent transaction outputs on a per-output basis
instead of at a transaction level. This was inspired by similar recent
changes in Bitcoin Core.
The primary motivation is to simplify the code, pave the way for a
utxo cache, and generally focus on optimizing runtime performance.
The tradeoff is that this approach does somewhat increase the size of
the serialized utxoset since it means that the transaction hash is
duplicated for each output as a part of the key and some additional
details such as whether the containing transaction is a coinbase and the
block height it was a part of are duplicated in each output.
However, in practice, the size difference isn't all that large, disk
space is relatively cheap, certainly cheaper than memory, and it is much
more important to provide more efficient runtime operation since that is
the ultimate purpose of the daemon.
While performing this conversion, it also simplifies the code to remove
the transaction version information from the utxoset as well as the
spend journal. The logic for only serializing it under certain
circumstances is complicated and it isn't actually used anywhere aside
from the gettxout RPC where it also isn't used by anything important
either. Consequently, this also removes the version field of the
gettxout RPC result.
The utxos in the database are automatically migrated to the new format
with this commit and it is possible to interrupt and resume the
migration process.
Finally, it also updates the tests for the new format and adds a new
function to the tests to convert the old test data to the new format for
convenience. The data has already been converted and updated in the
commit.
An overview of the changes are as follows:
- Remove transaction version from both spent and unspent output entries
- Update utxo serialization format to exclude the version
- Modify the spend journal serialization format
- The old version field is now reserved and always stores zero and
ignores it when reading
- This allows old entries to be used by new code without having to
migrate the entire spend journal
- Remove version field from gettxout RPC result
- Convert UtxoEntry to represent a specific utxo instead of a
transaction with all remaining utxos
- Optimize for memory usage with an eye towards a utxo cache
- Combine details such as whether the txout was contained in a
coinbase, is spent, and is modified into a single packed field of
bit flags
- Align entry fields to eliminate extra padding since ultimately
there will be a lot of these in memory
- Introduce a free list for serializing an outpoint to the database
key format to significantly reduce pressure on the GC
- Update all related functions that previously dealt with transaction
hashes to accept outpoints instead
- Update all callers accordingly
- Only add individually requested outputs from the mempool when
constructing a mempool view
- Modify the spend journal to always store the block height and coinbase
information with every spent txout
- Introduce code to handle fetching the missing information from
another utxo from the same transaction in the event an old style
entry is encountered
- Make use of a database cursor with seek to do this much more
efficiently than testing every possible output
- Always decompress data loaded from the database now that a utxo entry
only consists of a specific output
- Introduce upgrade code to migrate the utxo set to the new format
- Store versions of the utxoset and spend journal buckets
- Allow migration process to be interrupted and resumed
- Update all tests to expect the correct encodings, remove tests that no
longer apply, and add new ones for the new expected behavior
- Convert old tests for the legacy utxo format deserialization code to
test the new function that is used during upgrade
- Update the utxostore test data and add function that was used to
convert it
- Introduce a few new functions on UtxoViewpoint
- AddTxOut for adding an individual txout versus all of them
- addTxOut to handle the common code between the new AddTxOut and
existing AddTxOuts
- RemoveEntry for removing an individual txout
- fetchEntryByHash for fetching any remaining utxo for a given
transaction hash
2017-09-03 09:59:15 +02:00
|
|
|
// For more details, see
|
|
|
|
// https://github.com/bitcoin/bips/blob/master/bip-0030.mediawiki and
|
2013-07-18 16:49:28 +02:00
|
|
|
// http://r6.ca/blog/20120206T005236Z.html.
|
2015-08-26 06:03:18 +02:00
|
|
|
//
|
|
|
|
// This function MUST be called with the chain state lock held (for reads).
|
|
|
|
func (b *BlockChain) checkBIP0030(node *blockNode, block *btcutil.Block, view *UtxoViewpoint) error {
|
multi: Rework utxoset/view to use outpoints.
This modifies the utxoset in the database and related UtxoViewpoint to
store and work with unspent transaction outputs on a per-output basis
instead of at a transaction level. This was inspired by similar recent
changes in Bitcoin Core.
The primary motivation is to simplify the code, pave the way for a
utxo cache, and generally focus on optimizing runtime performance.
The tradeoff is that this approach does somewhat increase the size of
the serialized utxoset since it means that the transaction hash is
duplicated for each output as a part of the key and some additional
details such as whether the containing transaction is a coinbase and the
block height it was a part of are duplicated in each output.
However, in practice, the size difference isn't all that large, disk
space is relatively cheap, certainly cheaper than memory, and it is much
more important to provide more efficient runtime operation since that is
the ultimate purpose of the daemon.
While performing this conversion, it also simplifies the code to remove
the transaction version information from the utxoset as well as the
spend journal. The logic for only serializing it under certain
circumstances is complicated and it isn't actually used anywhere aside
from the gettxout RPC where it also isn't used by anything important
either. Consequently, this also removes the version field of the
gettxout RPC result.
The utxos in the database are automatically migrated to the new format
with this commit and it is possible to interrupt and resume the
migration process.
Finally, it also updates the tests for the new format and adds a new
function to the tests to convert the old test data to the new format for
convenience. The data has already been converted and updated in the
commit.
An overview of the changes are as follows:
- Remove transaction version from both spent and unspent output entries
- Update utxo serialization format to exclude the version
- Modify the spend journal serialization format
- The old version field is now reserved and always stores zero and
ignores it when reading
- This allows old entries to be used by new code without having to
migrate the entire spend journal
- Remove version field from gettxout RPC result
- Convert UtxoEntry to represent a specific utxo instead of a
transaction with all remaining utxos
- Optimize for memory usage with an eye towards a utxo cache
- Combine details such as whether the txout was contained in a
coinbase, is spent, and is modified into a single packed field of
bit flags
- Align entry fields to eliminate extra padding since ultimately
there will be a lot of these in memory
- Introduce a free list for serializing an outpoint to the database
key format to significantly reduce pressure on the GC
- Update all related functions that previously dealt with transaction
hashes to accept outpoints instead
- Update all callers accordingly
- Only add individually requested outputs from the mempool when
constructing a mempool view
- Modify the spend journal to always store the block height and coinbase
information with every spent txout
- Introduce code to handle fetching the missing information from
another utxo from the same transaction in the event an old style
entry is encountered
- Make use of a database cursor with seek to do this much more
efficiently than testing every possible output
- Always decompress data loaded from the database now that a utxo entry
only consists of a specific output
- Introduce upgrade code to migrate the utxo set to the new format
- Store versions of the utxoset and spend journal buckets
- Allow migration process to be interrupted and resumed
- Update all tests to expect the correct encodings, remove tests that no
longer apply, and add new ones for the new expected behavior
- Convert old tests for the legacy utxo format deserialization code to
test the new function that is used during upgrade
- Update the utxostore test data and add function that was used to
convert it
- Introduce a few new functions on UtxoViewpoint
- AddTxOut for adding an individual txout versus all of them
- addTxOut to handle the common code between the new AddTxOut and
existing AddTxOuts
- RemoveEntry for removing an individual txout
- fetchEntryByHash for fetching any remaining utxo for a given
transaction hash
2017-09-03 09:59:15 +02:00
|
|
|
// Fetch utxos for all of the transaction ouputs in this block.
|
|
|
|
// Typically, there will not be any utxos for any of the outputs.
|
|
|
|
fetchSet := make(map[wire.OutPoint]struct{})
|
2014-03-24 19:14:04 +01:00
|
|
|
for _, tx := range block.Transactions() {
|
multi: Rework utxoset/view to use outpoints.
This modifies the utxoset in the database and related UtxoViewpoint to
store and work with unspent transaction outputs on a per-output basis
instead of at a transaction level. This was inspired by similar recent
changes in Bitcoin Core.
The primary motivation is to simplify the code, pave the way for a
utxo cache, and generally focus on optimizing runtime performance.
The tradeoff is that this approach does somewhat increase the size of
the serialized utxoset since it means that the transaction hash is
duplicated for each output as a part of the key and some additional
details such as whether the containing transaction is a coinbase and the
block height it was a part of are duplicated in each output.
However, in practice, the size difference isn't all that large, disk
space is relatively cheap, certainly cheaper than memory, and it is much
more important to provide more efficient runtime operation since that is
the ultimate purpose of the daemon.
While performing this conversion, it also simplifies the code to remove
the transaction version information from the utxoset as well as the
spend journal. The logic for only serializing it under certain
circumstances is complicated and it isn't actually used anywhere aside
from the gettxout RPC where it also isn't used by anything important
either. Consequently, this also removes the version field of the
gettxout RPC result.
The utxos in the database are automatically migrated to the new format
with this commit and it is possible to interrupt and resume the
migration process.
Finally, it also updates the tests for the new format and adds a new
function to the tests to convert the old test data to the new format for
convenience. The data has already been converted and updated in the
commit.
An overview of the changes are as follows:
- Remove transaction version from both spent and unspent output entries
- Update utxo serialization format to exclude the version
- Modify the spend journal serialization format
- The old version field is now reserved and always stores zero and
ignores it when reading
- This allows old entries to be used by new code without having to
migrate the entire spend journal
- Remove version field from gettxout RPC result
- Convert UtxoEntry to represent a specific utxo instead of a
transaction with all remaining utxos
- Optimize for memory usage with an eye towards a utxo cache
- Combine details such as whether the txout was contained in a
coinbase, is spent, and is modified into a single packed field of
bit flags
- Align entry fields to eliminate extra padding since ultimately
there will be a lot of these in memory
- Introduce a free list for serializing an outpoint to the database
key format to significantly reduce pressure on the GC
- Update all related functions that previously dealt with transaction
hashes to accept outpoints instead
- Update all callers accordingly
- Only add individually requested outputs from the mempool when
constructing a mempool view
- Modify the spend journal to always store the block height and coinbase
information with every spent txout
- Introduce code to handle fetching the missing information from
another utxo from the same transaction in the event an old style
entry is encountered
- Make use of a database cursor with seek to do this much more
efficiently than testing every possible output
- Always decompress data loaded from the database now that a utxo entry
only consists of a specific output
- Introduce upgrade code to migrate the utxo set to the new format
- Store versions of the utxoset and spend journal buckets
- Allow migration process to be interrupted and resumed
- Update all tests to expect the correct encodings, remove tests that no
longer apply, and add new ones for the new expected behavior
- Convert old tests for the legacy utxo format deserialization code to
test the new function that is used during upgrade
- Update the utxostore test data and add function that was used to
convert it
- Introduce a few new functions on UtxoViewpoint
- AddTxOut for adding an individual txout versus all of them
- addTxOut to handle the common code between the new AddTxOut and
existing AddTxOuts
- RemoveEntry for removing an individual txout
- fetchEntryByHash for fetching any remaining utxo for a given
transaction hash
2017-09-03 09:59:15 +02:00
|
|
|
prevOut := wire.OutPoint{Hash: *tx.Hash()}
|
|
|
|
for txOutIdx := range tx.MsgTx().TxOut {
|
|
|
|
prevOut.Index = uint32(txOutIdx)
|
|
|
|
fetchSet[prevOut] = struct{}{}
|
|
|
|
}
|
2013-10-10 19:23:46 +02:00
|
|
|
}
|
2015-08-26 06:03:18 +02:00
|
|
|
err := view.fetchUtxos(b.db, fetchSet)
|
2013-07-18 16:49:28 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2015-08-26 06:03:18 +02:00
|
|
|
// Duplicate transactions are only allowed if the previous transaction
|
|
|
|
// is fully spent.
|
multi: Rework utxoset/view to use outpoints.
This modifies the utxoset in the database and related UtxoViewpoint to
store and work with unspent transaction outputs on a per-output basis
instead of at a transaction level. This was inspired by similar recent
changes in Bitcoin Core.
The primary motivation is to simplify the code, pave the way for a
utxo cache, and generally focus on optimizing runtime performance.
The tradeoff is that this approach does somewhat increase the size of
the serialized utxoset since it means that the transaction hash is
duplicated for each output as a part of the key and some additional
details such as whether the containing transaction is a coinbase and the
block height it was a part of are duplicated in each output.
However, in practice, the size difference isn't all that large, disk
space is relatively cheap, certainly cheaper than memory, and it is much
more important to provide more efficient runtime operation since that is
the ultimate purpose of the daemon.
While performing this conversion, it also simplifies the code to remove
the transaction version information from the utxoset as well as the
spend journal. The logic for only serializing it under certain
circumstances is complicated and it isn't actually used anywhere aside
from the gettxout RPC where it also isn't used by anything important
either. Consequently, this also removes the version field of the
gettxout RPC result.
The utxos in the database are automatically migrated to the new format
with this commit and it is possible to interrupt and resume the
migration process.
Finally, it also updates the tests for the new format and adds a new
function to the tests to convert the old test data to the new format for
convenience. The data has already been converted and updated in the
commit.
An overview of the changes are as follows:
- Remove transaction version from both spent and unspent output entries
- Update utxo serialization format to exclude the version
- Modify the spend journal serialization format
- The old version field is now reserved and always stores zero and
ignores it when reading
- This allows old entries to be used by new code without having to
migrate the entire spend journal
- Remove version field from gettxout RPC result
- Convert UtxoEntry to represent a specific utxo instead of a
transaction with all remaining utxos
- Optimize for memory usage with an eye towards a utxo cache
- Combine details such as whether the txout was contained in a
coinbase, is spent, and is modified into a single packed field of
bit flags
- Align entry fields to eliminate extra padding since ultimately
there will be a lot of these in memory
- Introduce a free list for serializing an outpoint to the database
key format to significantly reduce pressure on the GC
- Update all related functions that previously dealt with transaction
hashes to accept outpoints instead
- Update all callers accordingly
- Only add individually requested outputs from the mempool when
constructing a mempool view
- Modify the spend journal to always store the block height and coinbase
information with every spent txout
- Introduce code to handle fetching the missing information from
another utxo from the same transaction in the event an old style
entry is encountered
- Make use of a database cursor with seek to do this much more
efficiently than testing every possible output
- Always decompress data loaded from the database now that a utxo entry
only consists of a specific output
- Introduce upgrade code to migrate the utxo set to the new format
- Store versions of the utxoset and spend journal buckets
- Allow migration process to be interrupted and resumed
- Update all tests to expect the correct encodings, remove tests that no
longer apply, and add new ones for the new expected behavior
- Convert old tests for the legacy utxo format deserialization code to
test the new function that is used during upgrade
- Update the utxostore test data and add function that was used to
convert it
- Introduce a few new functions on UtxoViewpoint
- AddTxOut for adding an individual txout versus all of them
- addTxOut to handle the common code between the new AddTxOut and
existing AddTxOuts
- RemoveEntry for removing an individual txout
- fetchEntryByHash for fetching any remaining utxo for a given
transaction hash
2017-09-03 09:59:15 +02:00
|
|
|
for outpoint := range fetchSet {
|
|
|
|
utxo := view.LookupEntry(outpoint)
|
|
|
|
if utxo != nil && !utxo.IsSpent() {
|
2015-08-26 06:03:18 +02:00
|
|
|
str := fmt.Sprintf("tried to overwrite transaction %v "+
|
|
|
|
"at block height %d that is not fully spent",
|
multi: Rework utxoset/view to use outpoints.
This modifies the utxoset in the database and related UtxoViewpoint to
store and work with unspent transaction outputs on a per-output basis
instead of at a transaction level. This was inspired by similar recent
changes in Bitcoin Core.
The primary motivation is to simplify the code, pave the way for a
utxo cache, and generally focus on optimizing runtime performance.
The tradeoff is that this approach does somewhat increase the size of
the serialized utxoset since it means that the transaction hash is
duplicated for each output as a part of the key and some additional
details such as whether the containing transaction is a coinbase and the
block height it was a part of are duplicated in each output.
However, in practice, the size difference isn't all that large, disk
space is relatively cheap, certainly cheaper than memory, and it is much
more important to provide more efficient runtime operation since that is
the ultimate purpose of the daemon.
While performing this conversion, it also simplifies the code to remove
the transaction version information from the utxoset as well as the
spend journal. The logic for only serializing it under certain
circumstances is complicated and it isn't actually used anywhere aside
from the gettxout RPC where it also isn't used by anything important
either. Consequently, this also removes the version field of the
gettxout RPC result.
The utxos in the database are automatically migrated to the new format
with this commit and it is possible to interrupt and resume the
migration process.
Finally, it also updates the tests for the new format and adds a new
function to the tests to convert the old test data to the new format for
convenience. The data has already been converted and updated in the
commit.
An overview of the changes are as follows:
- Remove transaction version from both spent and unspent output entries
- Update utxo serialization format to exclude the version
- Modify the spend journal serialization format
- The old version field is now reserved and always stores zero and
ignores it when reading
- This allows old entries to be used by new code without having to
migrate the entire spend journal
- Remove version field from gettxout RPC result
- Convert UtxoEntry to represent a specific utxo instead of a
transaction with all remaining utxos
- Optimize for memory usage with an eye towards a utxo cache
- Combine details such as whether the txout was contained in a
coinbase, is spent, and is modified into a single packed field of
bit flags
- Align entry fields to eliminate extra padding since ultimately
there will be a lot of these in memory
- Introduce a free list for serializing an outpoint to the database
key format to significantly reduce pressure on the GC
- Update all related functions that previously dealt with transaction
hashes to accept outpoints instead
- Update all callers accordingly
- Only add individually requested outputs from the mempool when
constructing a mempool view
- Modify the spend journal to always store the block height and coinbase
information with every spent txout
- Introduce code to handle fetching the missing information from
another utxo from the same transaction in the event an old style
entry is encountered
- Make use of a database cursor with seek to do this much more
efficiently than testing every possible output
- Always decompress data loaded from the database now that a utxo entry
only consists of a specific output
- Introduce upgrade code to migrate the utxo set to the new format
- Store versions of the utxoset and spend journal buckets
- Allow migration process to be interrupted and resumed
- Update all tests to expect the correct encodings, remove tests that no
longer apply, and add new ones for the new expected behavior
- Convert old tests for the legacy utxo format deserialization code to
test the new function that is used during upgrade
- Update the utxostore test data and add function that was used to
convert it
- Introduce a few new functions on UtxoViewpoint
- AddTxOut for adding an individual txout versus all of them
- addTxOut to handle the common code between the new AddTxOut and
existing AddTxOuts
- RemoveEntry for removing an individual txout
- fetchEntryByHash for fetching any remaining utxo for a given
transaction hash
2017-09-03 09:59:15 +02:00
|
|
|
outpoint.Hash, utxo.BlockHeight())
|
2015-08-26 06:03:18 +02:00
|
|
|
return ruleError(ErrOverwriteTx, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2013-09-30 23:44:01 +02:00
|
|
|
// CheckTransactionInputs performs a series of checks on the inputs to a
|
2013-07-18 16:49:28 +02:00
|
|
|
// transaction to ensure they are valid. An example of some of the checks
|
|
|
|
// include verifying all inputs exist, ensuring the coinbase seasoning
|
2013-07-26 18:50:20 +02:00
|
|
|
// requirements are met, detecting double spends, validating all values and fees
|
|
|
|
// are in the legal range and the total output amount doesn't exceed the input
|
|
|
|
// amount, and verifying the signatures to prove the spender was the owner of
|
|
|
|
// the bitcoins and therefore allowed to spend them. As it checks the inputs,
|
|
|
|
// it also calculates the total fees for the transaction and returns that value.
|
2015-08-26 06:03:18 +02:00
|
|
|
//
|
|
|
|
// NOTE: The transaction MUST have already been sanity checked with the
|
|
|
|
// CheckTransactionSanity function prior to calling this function.
|
2016-08-10 23:02:23 +02:00
|
|
|
func CheckTransactionInputs(tx *btcutil.Tx, txHeight int32, utxoView *UtxoViewpoint, chainParams *chaincfg.Params) (int64, error) {
|
2013-07-18 16:49:28 +02:00
|
|
|
// Coinbase transactions have no inputs.
|
2013-09-30 23:41:51 +02:00
|
|
|
if IsCoinBase(tx) {
|
2013-07-18 16:49:28 +02:00
|
|
|
return 0, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var totalSatoshiIn int64
|
2015-08-26 06:03:18 +02:00
|
|
|
for txInIndex, txIn := range tx.MsgTx().TxIn {
|
|
|
|
// Ensure the referenced input transaction is available.
|
multi: Rework utxoset/view to use outpoints.
This modifies the utxoset in the database and related UtxoViewpoint to
store and work with unspent transaction outputs on a per-output basis
instead of at a transaction level. This was inspired by similar recent
changes in Bitcoin Core.
The primary motivation is to simplify the code, pave the way for a
utxo cache, and generally focus on optimizing runtime performance.
The tradeoff is that this approach does somewhat increase the size of
the serialized utxoset since it means that the transaction hash is
duplicated for each output as a part of the key and some additional
details such as whether the containing transaction is a coinbase and the
block height it was a part of are duplicated in each output.
However, in practice, the size difference isn't all that large, disk
space is relatively cheap, certainly cheaper than memory, and it is much
more important to provide more efficient runtime operation since that is
the ultimate purpose of the daemon.
While performing this conversion, it also simplifies the code to remove
the transaction version information from the utxoset as well as the
spend journal. The logic for only serializing it under certain
circumstances is complicated and it isn't actually used anywhere aside
from the gettxout RPC where it also isn't used by anything important
either. Consequently, this also removes the version field of the
gettxout RPC result.
The utxos in the database are automatically migrated to the new format
with this commit and it is possible to interrupt and resume the
migration process.
Finally, it also updates the tests for the new format and adds a new
function to the tests to convert the old test data to the new format for
convenience. The data has already been converted and updated in the
commit.
An overview of the changes are as follows:
- Remove transaction version from both spent and unspent output entries
- Update utxo serialization format to exclude the version
- Modify the spend journal serialization format
- The old version field is now reserved and always stores zero and
ignores it when reading
- This allows old entries to be used by new code without having to
migrate the entire spend journal
- Remove version field from gettxout RPC result
- Convert UtxoEntry to represent a specific utxo instead of a
transaction with all remaining utxos
- Optimize for memory usage with an eye towards a utxo cache
- Combine details such as whether the txout was contained in a
coinbase, is spent, and is modified into a single packed field of
bit flags
- Align entry fields to eliminate extra padding since ultimately
there will be a lot of these in memory
- Introduce a free list for serializing an outpoint to the database
key format to significantly reduce pressure on the GC
- Update all related functions that previously dealt with transaction
hashes to accept outpoints instead
- Update all callers accordingly
- Only add individually requested outputs from the mempool when
constructing a mempool view
- Modify the spend journal to always store the block height and coinbase
information with every spent txout
- Introduce code to handle fetching the missing information from
another utxo from the same transaction in the event an old style
entry is encountered
- Make use of a database cursor with seek to do this much more
efficiently than testing every possible output
- Always decompress data loaded from the database now that a utxo entry
only consists of a specific output
- Introduce upgrade code to migrate the utxo set to the new format
- Store versions of the utxoset and spend journal buckets
- Allow migration process to be interrupted and resumed
- Update all tests to expect the correct encodings, remove tests that no
longer apply, and add new ones for the new expected behavior
- Convert old tests for the legacy utxo format deserialization code to
test the new function that is used during upgrade
- Update the utxostore test data and add function that was used to
convert it
- Introduce a few new functions on UtxoViewpoint
- AddTxOut for adding an individual txout versus all of them
- addTxOut to handle the common code between the new AddTxOut and
existing AddTxOuts
- RemoveEntry for removing an individual txout
- fetchEntryByHash for fetching any remaining utxo for a given
transaction hash
2017-09-03 09:59:15 +02:00
|
|
|
utxo := utxoView.LookupEntry(txIn.PreviousOutPoint)
|
|
|
|
if utxo == nil || utxo.IsSpent() {
|
2017-08-14 07:22:40 +02:00
|
|
|
str := fmt.Sprintf("output %v referenced from "+
|
|
|
|
"transaction %s:%d either does not exist or "+
|
|
|
|
"has already been spent", txIn.PreviousOutPoint,
|
|
|
|
tx.Hash(), txInIndex)
|
|
|
|
return 0, ruleError(ErrMissingTxOut, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure the transaction is not spending coins which have not
|
|
|
|
// yet reached the required coinbase maturity.
|
multi: Rework utxoset/view to use outpoints.
This modifies the utxoset in the database and related UtxoViewpoint to
store and work with unspent transaction outputs on a per-output basis
instead of at a transaction level. This was inspired by similar recent
changes in Bitcoin Core.
The primary motivation is to simplify the code, pave the way for a
utxo cache, and generally focus on optimizing runtime performance.
The tradeoff is that this approach does somewhat increase the size of
the serialized utxoset since it means that the transaction hash is
duplicated for each output as a part of the key and some additional
details such as whether the containing transaction is a coinbase and the
block height it was a part of are duplicated in each output.
However, in practice, the size difference isn't all that large, disk
space is relatively cheap, certainly cheaper than memory, and it is much
more important to provide more efficient runtime operation since that is
the ultimate purpose of the daemon.
While performing this conversion, it also simplifies the code to remove
the transaction version information from the utxoset as well as the
spend journal. The logic for only serializing it under certain
circumstances is complicated and it isn't actually used anywhere aside
from the gettxout RPC where it also isn't used by anything important
either. Consequently, this also removes the version field of the
gettxout RPC result.
The utxos in the database are automatically migrated to the new format
with this commit and it is possible to interrupt and resume the
migration process.
Finally, it also updates the tests for the new format and adds a new
function to the tests to convert the old test data to the new format for
convenience. The data has already been converted and updated in the
commit.
An overview of the changes are as follows:
- Remove transaction version from both spent and unspent output entries
- Update utxo serialization format to exclude the version
- Modify the spend journal serialization format
- The old version field is now reserved and always stores zero and
ignores it when reading
- This allows old entries to be used by new code without having to
migrate the entire spend journal
- Remove version field from gettxout RPC result
- Convert UtxoEntry to represent a specific utxo instead of a
transaction with all remaining utxos
- Optimize for memory usage with an eye towards a utxo cache
- Combine details such as whether the txout was contained in a
coinbase, is spent, and is modified into a single packed field of
bit flags
- Align entry fields to eliminate extra padding since ultimately
there will be a lot of these in memory
- Introduce a free list for serializing an outpoint to the database
key format to significantly reduce pressure on the GC
- Update all related functions that previously dealt with transaction
hashes to accept outpoints instead
- Update all callers accordingly
- Only add individually requested outputs from the mempool when
constructing a mempool view
- Modify the spend journal to always store the block height and coinbase
information with every spent txout
- Introduce code to handle fetching the missing information from
another utxo from the same transaction in the event an old style
entry is encountered
- Make use of a database cursor with seek to do this much more
efficiently than testing every possible output
- Always decompress data loaded from the database now that a utxo entry
only consists of a specific output
- Introduce upgrade code to migrate the utxo set to the new format
- Store versions of the utxoset and spend journal buckets
- Allow migration process to be interrupted and resumed
- Update all tests to expect the correct encodings, remove tests that no
longer apply, and add new ones for the new expected behavior
- Convert old tests for the legacy utxo format deserialization code to
test the new function that is used during upgrade
- Update the utxostore test data and add function that was used to
convert it
- Introduce a few new functions on UtxoViewpoint
- AddTxOut for adding an individual txout versus all of them
- addTxOut to handle the common code between the new AddTxOut and
existing AddTxOuts
- RemoveEntry for removing an individual txout
- fetchEntryByHash for fetching any remaining utxo for a given
transaction hash
2017-09-03 09:59:15 +02:00
|
|
|
if utxo.IsCoinBase() {
|
|
|
|
originHeight := utxo.BlockHeight()
|
2013-07-18 16:49:28 +02:00
|
|
|
blocksSincePrev := txHeight - originHeight
|
2016-08-10 23:02:23 +02:00
|
|
|
coinbaseMaturity := int32(chainParams.CoinbaseMaturity)
|
2013-07-18 16:49:28 +02:00
|
|
|
if blocksSincePrev < coinbaseMaturity {
|
|
|
|
str := fmt.Sprintf("tried to spend coinbase "+
|
multi: Rework utxoset/view to use outpoints.
This modifies the utxoset in the database and related UtxoViewpoint to
store and work with unspent transaction outputs on a per-output basis
instead of at a transaction level. This was inspired by similar recent
changes in Bitcoin Core.
The primary motivation is to simplify the code, pave the way for a
utxo cache, and generally focus on optimizing runtime performance.
The tradeoff is that this approach does somewhat increase the size of
the serialized utxoset since it means that the transaction hash is
duplicated for each output as a part of the key and some additional
details such as whether the containing transaction is a coinbase and the
block height it was a part of are duplicated in each output.
However, in practice, the size difference isn't all that large, disk
space is relatively cheap, certainly cheaper than memory, and it is much
more important to provide more efficient runtime operation since that is
the ultimate purpose of the daemon.
While performing this conversion, it also simplifies the code to remove
the transaction version information from the utxoset as well as the
spend journal. The logic for only serializing it under certain
circumstances is complicated and it isn't actually used anywhere aside
from the gettxout RPC where it also isn't used by anything important
either. Consequently, this also removes the version field of the
gettxout RPC result.
The utxos in the database are automatically migrated to the new format
with this commit and it is possible to interrupt and resume the
migration process.
Finally, it also updates the tests for the new format and adds a new
function to the tests to convert the old test data to the new format for
convenience. The data has already been converted and updated in the
commit.
An overview of the changes are as follows:
- Remove transaction version from both spent and unspent output entries
- Update utxo serialization format to exclude the version
- Modify the spend journal serialization format
- The old version field is now reserved and always stores zero and
ignores it when reading
- This allows old entries to be used by new code without having to
migrate the entire spend journal
- Remove version field from gettxout RPC result
- Convert UtxoEntry to represent a specific utxo instead of a
transaction with all remaining utxos
- Optimize for memory usage with an eye towards a utxo cache
- Combine details such as whether the txout was contained in a
coinbase, is spent, and is modified into a single packed field of
bit flags
- Align entry fields to eliminate extra padding since ultimately
there will be a lot of these in memory
- Introduce a free list for serializing an outpoint to the database
key format to significantly reduce pressure on the GC
- Update all related functions that previously dealt with transaction
hashes to accept outpoints instead
- Update all callers accordingly
- Only add individually requested outputs from the mempool when
constructing a mempool view
- Modify the spend journal to always store the block height and coinbase
information with every spent txout
- Introduce code to handle fetching the missing information from
another utxo from the same transaction in the event an old style
entry is encountered
- Make use of a database cursor with seek to do this much more
efficiently than testing every possible output
- Always decompress data loaded from the database now that a utxo entry
only consists of a specific output
- Introduce upgrade code to migrate the utxo set to the new format
- Store versions of the utxoset and spend journal buckets
- Allow migration process to be interrupted and resumed
- Update all tests to expect the correct encodings, remove tests that no
longer apply, and add new ones for the new expected behavior
- Convert old tests for the legacy utxo format deserialization code to
test the new function that is used during upgrade
- Update the utxostore test data and add function that was used to
convert it
- Introduce a few new functions on UtxoViewpoint
- AddTxOut for adding an individual txout versus all of them
- addTxOut to handle the common code between the new AddTxOut and
existing AddTxOuts
- RemoveEntry for removing an individual txout
- fetchEntryByHash for fetching any remaining utxo for a given
transaction hash
2017-09-03 09:59:15 +02:00
|
|
|
"transaction output %v from height %v "+
|
|
|
|
"at height %v before required maturity "+
|
|
|
|
"of %v blocks", txIn.PreviousOutPoint,
|
2015-08-26 06:03:18 +02:00
|
|
|
originHeight, txHeight,
|
|
|
|
coinbaseMaturity)
|
2014-06-25 00:10:53 +02:00
|
|
|
return 0, ruleError(ErrImmatureSpend, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure the transaction amounts are in range. Each of the
|
|
|
|
// output values of the input transactions must not be negative
|
|
|
|
// or more than the max allowed per transaction. All amounts in
|
|
|
|
// a transaction are in a unit value known as a satoshi. One
|
|
|
|
// bitcoin is a quantity of satoshi as defined by the
|
2013-10-29 19:57:31 +01:00
|
|
|
// SatoshiPerBitcoin constant.
|
multi: Rework utxoset/view to use outpoints.
This modifies the utxoset in the database and related UtxoViewpoint to
store and work with unspent transaction outputs on a per-output basis
instead of at a transaction level. This was inspired by similar recent
changes in Bitcoin Core.
The primary motivation is to simplify the code, pave the way for a
utxo cache, and generally focus on optimizing runtime performance.
The tradeoff is that this approach does somewhat increase the size of
the serialized utxoset since it means that the transaction hash is
duplicated for each output as a part of the key and some additional
details such as whether the containing transaction is a coinbase and the
block height it was a part of are duplicated in each output.
However, in practice, the size difference isn't all that large, disk
space is relatively cheap, certainly cheaper than memory, and it is much
more important to provide more efficient runtime operation since that is
the ultimate purpose of the daemon.
While performing this conversion, it also simplifies the code to remove
the transaction version information from the utxoset as well as the
spend journal. The logic for only serializing it under certain
circumstances is complicated and it isn't actually used anywhere aside
from the gettxout RPC where it also isn't used by anything important
either. Consequently, this also removes the version field of the
gettxout RPC result.
The utxos in the database are automatically migrated to the new format
with this commit and it is possible to interrupt and resume the
migration process.
Finally, it also updates the tests for the new format and adds a new
function to the tests to convert the old test data to the new format for
convenience. The data has already been converted and updated in the
commit.
An overview of the changes are as follows:
- Remove transaction version from both spent and unspent output entries
- Update utxo serialization format to exclude the version
- Modify the spend journal serialization format
- The old version field is now reserved and always stores zero and
ignores it when reading
- This allows old entries to be used by new code without having to
migrate the entire spend journal
- Remove version field from gettxout RPC result
- Convert UtxoEntry to represent a specific utxo instead of a
transaction with all remaining utxos
- Optimize for memory usage with an eye towards a utxo cache
- Combine details such as whether the txout was contained in a
coinbase, is spent, and is modified into a single packed field of
bit flags
- Align entry fields to eliminate extra padding since ultimately
there will be a lot of these in memory
- Introduce a free list for serializing an outpoint to the database
key format to significantly reduce pressure on the GC
- Update all related functions that previously dealt with transaction
hashes to accept outpoints instead
- Update all callers accordingly
- Only add individually requested outputs from the mempool when
constructing a mempool view
- Modify the spend journal to always store the block height and coinbase
information with every spent txout
- Introduce code to handle fetching the missing information from
another utxo from the same transaction in the event an old style
entry is encountered
- Make use of a database cursor with seek to do this much more
efficiently than testing every possible output
- Always decompress data loaded from the database now that a utxo entry
only consists of a specific output
- Introduce upgrade code to migrate the utxo set to the new format
- Store versions of the utxoset and spend journal buckets
- Allow migration process to be interrupted and resumed
- Update all tests to expect the correct encodings, remove tests that no
longer apply, and add new ones for the new expected behavior
- Convert old tests for the legacy utxo format deserialization code to
test the new function that is used during upgrade
- Update the utxostore test data and add function that was used to
convert it
- Introduce a few new functions on UtxoViewpoint
- AddTxOut for adding an individual txout versus all of them
- addTxOut to handle the common code between the new AddTxOut and
existing AddTxOuts
- RemoveEntry for removing an individual txout
- fetchEntryByHash for fetching any remaining utxo for a given
transaction hash
2017-09-03 09:59:15 +02:00
|
|
|
originTxSatoshi := utxo.Amount()
|
2013-07-18 16:49:28 +02:00
|
|
|
if originTxSatoshi < 0 {
|
|
|
|
str := fmt.Sprintf("transaction output has negative "+
|
2015-08-26 06:03:18 +02:00
|
|
|
"value of %v", btcutil.Amount(originTxSatoshi))
|
2014-06-25 00:10:53 +02:00
|
|
|
return 0, ruleError(ErrBadTxOutValue, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
2013-10-29 21:08:12 +01:00
|
|
|
if originTxSatoshi > btcutil.MaxSatoshi {
|
2013-07-18 16:49:28 +02:00
|
|
|
str := fmt.Sprintf("transaction output value of %v is "+
|
|
|
|
"higher than max allowed value of %v",
|
2015-08-26 06:03:18 +02:00
|
|
|
btcutil.Amount(originTxSatoshi),
|
|
|
|
btcutil.MaxSatoshi)
|
2014-06-25 00:10:53 +02:00
|
|
|
return 0, ruleError(ErrBadTxOutValue, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// The total of all outputs must not be more than the max
|
|
|
|
// allowed per transaction. Also, we could potentially overflow
|
|
|
|
// the accumulator so check for overflow.
|
|
|
|
lastSatoshiIn := totalSatoshiIn
|
|
|
|
totalSatoshiIn += originTxSatoshi
|
2013-10-29 21:08:12 +01:00
|
|
|
if totalSatoshiIn < lastSatoshiIn ||
|
|
|
|
totalSatoshiIn > btcutil.MaxSatoshi {
|
2013-07-18 16:49:28 +02:00
|
|
|
str := fmt.Sprintf("total value of all transaction "+
|
|
|
|
"inputs is %v which is higher than max "+
|
|
|
|
"allowed value of %v", totalSatoshiIn,
|
2013-10-29 21:08:12 +01:00
|
|
|
btcutil.MaxSatoshi)
|
2014-06-25 00:10:53 +02:00
|
|
|
return 0, ruleError(ErrBadTxOutValue, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Calculate the total output amount for this transaction. It is safe
|
|
|
|
// to ignore overflow and out of range errors here because those error
|
|
|
|
// conditions would have already been caught by checkTransactionSanity.
|
|
|
|
var totalSatoshiOut int64
|
2013-10-28 21:17:53 +01:00
|
|
|
for _, txOut := range tx.MsgTx().TxOut {
|
2013-07-18 16:49:28 +02:00
|
|
|
totalSatoshiOut += txOut.Value
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure the transaction does not spend more than its inputs.
|
|
|
|
if totalSatoshiIn < totalSatoshiOut {
|
|
|
|
str := fmt.Sprintf("total value of all transaction inputs for "+
|
|
|
|
"transaction %v is %v which is less than the amount "+
|
2020-07-13 16:24:40 +02:00
|
|
|
"spent of %v", tx.Hash(), totalSatoshiIn, totalSatoshiOut)
|
2014-06-25 00:10:53 +02:00
|
|
|
return 0, ruleError(ErrSpendTooHigh, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE: bitcoind checks if the transaction fees are < 0 here, but that
|
|
|
|
// is an impossible condition because of the check above that ensures
|
|
|
|
// the inputs are >= the outputs.
|
|
|
|
txFeeInSatoshi := totalSatoshiIn - totalSatoshiOut
|
|
|
|
return txFeeInSatoshi, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// checkConnectBlock performs several checks to confirm connecting the passed
|
2015-08-26 06:03:18 +02:00
|
|
|
// block to the chain represented by the passed view does not violate any rules.
|
|
|
|
// In addition, the passed view is updated to spend all of the referenced
|
|
|
|
// outputs and add all of the new utxos created by block. Thus, the view will
|
|
|
|
// represent the state of the chain as if the block were actually connected and
|
|
|
|
// consequently the best hash for the view is also updated to passed block.
|
2014-03-11 19:18:27 +01:00
|
|
|
//
|
2017-09-15 04:09:32 +02:00
|
|
|
// An example of some of the checks performed are ensuring connecting the block
|
|
|
|
// would not cause any duplicate transaction hashes for old transactions that
|
|
|
|
// aren't already fully spent, double spends, exceeding the maximum allowed
|
|
|
|
// signature operations per block, invalid values in relation to the expected
|
|
|
|
// block subsidy, or fail transaction script validation.
|
2014-03-11 19:18:27 +01:00
|
|
|
//
|
2017-09-15 04:09:32 +02:00
|
|
|
// The CheckConnectBlockTemplate function makes use of this function to perform
|
|
|
|
// the bulk of its work. The only difference is this function accepts a node
|
|
|
|
// which may or may not require reorganization to connect it to the main chain
|
|
|
|
// whereas CheckConnectBlockTemplate creates a new node which specifically
|
|
|
|
// connects to the end of the current main chain and then calls this function
|
|
|
|
// with that node.
|
2015-08-26 06:03:18 +02:00
|
|
|
//
|
|
|
|
// This function MUST be called with the chain state lock held (for writes).
|
2018-05-29 06:13:59 +02:00
|
|
|
func (b *BlockChain) checkConnectBlock(node *blockNode, block *btcutil.Block, view *UtxoViewpoint, stxos *[]SpentTxOut) error {
|
2013-07-18 16:49:28 +02:00
|
|
|
// If the side chain blocks end up in the database, a call to
|
2014-06-25 00:10:53 +02:00
|
|
|
// CheckBlockSanity should be done here in case a previous version
|
2013-07-18 16:49:28 +02:00
|
|
|
// allowed a block that is no longer valid. However, since the
|
|
|
|
// implementation only currently uses memory for the side chain blocks,
|
|
|
|
// it isn't currently necessary.
|
|
|
|
|
|
|
|
// The coinbase for the Genesis block is not spendable, so just return
|
2015-08-26 06:03:18 +02:00
|
|
|
// an error now.
|
|
|
|
if node.hash.IsEqual(b.chainParams.GenesisHash) {
|
|
|
|
str := "the coinbase for the genesis block is not spendable"
|
2017-08-14 07:22:40 +02:00
|
|
|
return ruleError(ErrMissingTxOut, str)
|
2015-08-26 06:03:18 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure the view is for the node being checked.
|
2017-02-03 19:13:53 +01:00
|
|
|
parentHash := &block.MsgBlock().Header.PrevBlock
|
|
|
|
if !view.BestHash().IsEqual(parentHash) {
|
2015-08-26 06:03:18 +02:00
|
|
|
return AssertError(fmt.Sprintf("inconsistent view when "+
|
|
|
|
"checking block connection: best hash is %v instead "+
|
2017-02-03 19:13:53 +01:00
|
|
|
"of expected %v", view.BestHash(), parentHash))
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// BIP0030 added a rule to prevent blocks which contain duplicate
|
|
|
|
// transactions that 'overwrite' older transactions which are not fully
|
|
|
|
// spent. See the documentation for checkBIP0030 for more details.
|
|
|
|
//
|
2015-08-26 06:03:18 +02:00
|
|
|
// There are two blocks in the chain which violate this rule, so the
|
2016-10-19 21:30:40 +02:00
|
|
|
// check must be skipped for those blocks. The isBIP0030Node function
|
|
|
|
// is used to determine if this block is one of the two blocks that must
|
|
|
|
// be skipped.
|
|
|
|
//
|
|
|
|
// In addition, as of BIP0034, duplicate coinbases are no longer
|
|
|
|
// possible due to its requirement for including the block height in the
|
|
|
|
// coinbase and thus it is no longer possible to create transactions
|
|
|
|
// that 'overwrite' older ones. Therefore, only enforce the rule if
|
|
|
|
// BIP0034 is not yet active. This is a useful optimization because the
|
|
|
|
// BIP0030 check is expensive since it involves a ton of cache misses in
|
|
|
|
// the utxoset.
|
|
|
|
if !isBIP0030Node(node) && (node.height < b.chainParams.BIP0034Height) {
|
2015-08-26 06:03:18 +02:00
|
|
|
err := b.checkBIP0030(node, block, view)
|
2013-07-18 16:49:28 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-26 06:03:18 +02:00
|
|
|
// Load all of the utxos referenced by the inputs for all transactions
|
|
|
|
// in the block don't already exist in the utxo view from the database.
|
|
|
|
//
|
|
|
|
// These utxo entries are needed for verification of things such as
|
2013-07-18 16:49:28 +02:00
|
|
|
// transaction inputs, counting pay-to-script-hashes, and scripts.
|
2015-08-26 06:03:18 +02:00
|
|
|
err := view.fetchInputUtxos(b.db, block)
|
2013-07-18 16:49:28 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// BIP0016 describes a pay-to-script-hash type that is considered a
|
|
|
|
// "standard" type. The rules for this BIP only apply to transactions
|
2015-01-30 19:08:47 +01:00
|
|
|
// after the timestamp defined by txscript.Bip16Activation. See
|
2013-07-18 16:49:28 +02:00
|
|
|
// https://en.bitcoin.it/wiki/BIP_0016 for more details.
|
2017-01-31 01:32:50 +01:00
|
|
|
enforceBIP0016 := node.timestamp >= txscript.Bip16Activation.Unix()
|
2017-01-04 06:14:05 +01:00
|
|
|
|
|
|
|
// Query for the Version Bits state for the segwit soft-fork
|
|
|
|
// deployment. If segwit is active, we'll switch over to enforcing all
|
|
|
|
// the new rules.
|
|
|
|
segwitState, err := b.deploymentState(node.parent, chaincfg.DeploymentSegwit)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
enforceSegWit := segwitState == ThresholdActive
|
2013-07-18 16:49:28 +02:00
|
|
|
|
|
|
|
// The number of signature operations must be less than the maximum
|
|
|
|
// allowed per block. Note that the preliminary sanity checks on a
|
|
|
|
// block also include a check similar to this one, but this check
|
|
|
|
// expands the count to include a precise count of pay-to-script-hash
|
|
|
|
// signature operations in each of the input transaction public key
|
|
|
|
// scripts.
|
2013-10-28 21:17:53 +01:00
|
|
|
transactions := block.Transactions()
|
2016-10-19 03:33:20 +02:00
|
|
|
totalSigOpCost := 0
|
2013-07-18 16:49:28 +02:00
|
|
|
for i, tx := range transactions {
|
2016-10-19 03:33:20 +02:00
|
|
|
// Since the first (and only the first) transaction has
|
|
|
|
// already been verified to be a coinbase transaction,
|
|
|
|
// use i == 0 as an optimization for the flag to
|
|
|
|
// countP2SHSigOps for whether or not the transaction is
|
|
|
|
// a coinbase transaction rather than having to do a
|
|
|
|
// full coinbase check again.
|
|
|
|
sigOpCost, err := GetSigOpCost(tx, i == 0, view, enforceBIP0016,
|
|
|
|
enforceSegWit)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check for overflow or going over the limits. We have to do
|
2013-10-11 17:24:13 +02:00
|
|
|
// this on every loop iteration to avoid overflow.
|
2016-10-19 03:33:20 +02:00
|
|
|
lastSigOpCost := totalSigOpCost
|
|
|
|
totalSigOpCost += sigOpCost
|
|
|
|
if totalSigOpCost < lastSigOpCost || totalSigOpCost > MaxBlockSigOpsCost {
|
2013-07-18 16:49:28 +02:00
|
|
|
str := fmt.Sprintf("block contains too many "+
|
|
|
|
"signature operations - got %v, max %v",
|
2016-10-19 03:33:20 +02:00
|
|
|
totalSigOpCost, MaxBlockSigOpsCost)
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrTooManySigOps, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Perform several checks on the inputs for each transaction. Also
|
|
|
|
// accumulate the total fees. This could technically be combined with
|
|
|
|
// the loop above instead of running another loop over the transactions,
|
|
|
|
// but by separating it we can avoid running the more expensive (though
|
|
|
|
// still relatively cheap as compared to running the scripts) checks
|
|
|
|
// against all the inputs when the signature operations are out of
|
|
|
|
// bounds.
|
|
|
|
var totalFees int64
|
|
|
|
for _, tx := range transactions {
|
2016-08-10 23:02:23 +02:00
|
|
|
txFee, err := CheckTransactionInputs(tx, node.height, view,
|
|
|
|
b.chainParams)
|
2013-07-18 16:49:28 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sum the total fees and ensure we don't overflow the
|
|
|
|
// accumulator.
|
|
|
|
lastTotalFees := totalFees
|
|
|
|
totalFees += txFee
|
|
|
|
if totalFees < lastTotalFees {
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrBadFees, "total fees for block "+
|
|
|
|
"overflows accumulator")
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
2015-08-26 06:03:18 +02:00
|
|
|
|
|
|
|
// Add all of the outputs for this transaction which are not
|
|
|
|
// provably unspendable as available utxos. Also, the passed
|
|
|
|
// spent txos slice is updated to contain an entry for each
|
|
|
|
// spent txout in the order each transaction spends them.
|
|
|
|
err = view.connectTransaction(tx, node.height, stxos)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// The total output values of the coinbase transaction must not exceed
|
|
|
|
// the expected subsidy value plus total transaction fees gained from
|
|
|
|
// mining the block. It is safe to ignore overflow and out of range
|
|
|
|
// errors here because those error conditions would have already been
|
|
|
|
// caught by checkTransactionSanity.
|
|
|
|
var totalSatoshiOut int64
|
2013-10-28 21:17:53 +01:00
|
|
|
for _, txOut := range transactions[0].MsgTx().TxOut {
|
2013-07-18 16:49:28 +02:00
|
|
|
totalSatoshiOut += txOut.Value
|
|
|
|
}
|
2015-02-06 06:18:27 +01:00
|
|
|
expectedSatoshiOut := CalcBlockSubsidy(node.height, b.chainParams) +
|
2014-05-26 17:27:50 +02:00
|
|
|
totalFees
|
2013-07-18 16:49:28 +02:00
|
|
|
if totalSatoshiOut > expectedSatoshiOut {
|
|
|
|
str := fmt.Sprintf("coinbase transaction for block pays %v "+
|
|
|
|
"which is more than expected value of %v",
|
|
|
|
totalSatoshiOut, expectedSatoshiOut)
|
2014-06-25 00:10:53 +02:00
|
|
|
return ruleError(ErrBadCoinbaseValue, str)
|
2013-07-18 16:49:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Don't run scripts if this node is before the latest known good
|
|
|
|
// checkpoint since the validity is verified via the checkpoints (all
|
|
|
|
// transactions are included in the merkle root hash and any changes
|
|
|
|
// will therefore be detected by the next checkpoint). This is a huge
|
|
|
|
// optimization because running the scripts is the most time consuming
|
|
|
|
// portion of block handling.
|
2016-12-02 00:54:41 +01:00
|
|
|
checkpoint := b.LatestCheckpoint()
|
2017-08-21 11:07:52 +02:00
|
|
|
runScripts := true
|
2013-07-18 16:49:28 +02:00
|
|
|
if checkpoint != nil && node.height <= checkpoint.Height {
|
|
|
|
runScripts = false
|
|
|
|
}
|
|
|
|
|
2015-02-25 21:31:46 +01:00
|
|
|
// Blocks created after the BIP0016 activation time need to have the
|
|
|
|
// pay-to-script-hash checks enabled.
|
|
|
|
var scriptFlags txscript.ScriptFlags
|
2015-08-26 06:03:18 +02:00
|
|
|
if enforceBIP0016 {
|
2015-02-25 21:31:46 +01:00
|
|
|
scriptFlags |= txscript.ScriptBip16
|
|
|
|
}
|
|
|
|
|
2016-10-19 21:24:42 +02:00
|
|
|
// Enforce DER signatures for block versions 3+ once the historical
|
|
|
|
// activation threshold has been reached. This is part of BIP0066.
|
2015-02-25 21:32:37 +01:00
|
|
|
blockHeader := &block.MsgBlock().Header
|
2016-10-19 21:24:42 +02:00
|
|
|
if blockHeader.Version >= 3 && node.height >= b.chainParams.BIP0066Height {
|
2015-02-25 21:32:37 +01:00
|
|
|
scriptFlags |= txscript.ScriptVerifyDERSignatures
|
|
|
|
}
|
|
|
|
|
2016-10-19 21:24:42 +02:00
|
|
|
// Enforce CHECKLOCKTIMEVERIFY for block versions 4+ once the historical
|
|
|
|
// activation threshold has been reached. This is part of BIP0065.
|
|
|
|
if blockHeader.Version >= 4 && node.height >= b.chainParams.BIP0065Height {
|
2015-10-28 16:43:36 +01:00
|
|
|
scriptFlags |= txscript.ScriptVerifyCheckLockTimeVerify
|
|
|
|
}
|
|
|
|
|
2016-09-27 04:00:28 +02:00
|
|
|
// Enforce CHECKSEQUENCEVERIFY during all block validation checks once
|
|
|
|
// the soft-fork deployment is fully active.
|
|
|
|
csvState, err := b.deploymentState(node.parent, chaincfg.DeploymentCSV)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if csvState == ThresholdActive {
|
|
|
|
// If the CSV soft-fork is now active, then modify the
|
|
|
|
// scriptFlags to ensure that the CSV op code is properly
|
|
|
|
// validated during the script checks bleow.
|
|
|
|
scriptFlags |= txscript.ScriptVerifyCheckSequenceVerify
|
|
|
|
|
|
|
|
// We obtain the MTP of the *previous* block in order to
|
|
|
|
// determine if transactions in the current block are final.
|
2017-02-03 19:13:53 +01:00
|
|
|
medianTime := node.parent.CalcPastMedianTime()
|
2016-09-27 04:00:28 +02:00
|
|
|
|
|
|
|
// Additionally, if the CSV soft-fork package is now active,
|
|
|
|
// then we also enforce the relative sequence number based
|
|
|
|
// lock-times within the inputs of all transactions in this
|
|
|
|
// candidate block.
|
|
|
|
for _, tx := range block.Transactions() {
|
|
|
|
// A transaction can only be included within a block
|
|
|
|
// once the sequence locks of *all* its inputs are
|
|
|
|
// active.
|
|
|
|
sequenceLock, err := b.calcSequenceLock(node, tx, view,
|
|
|
|
false)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !SequenceLockActive(sequenceLock, node.height,
|
|
|
|
medianTime) {
|
|
|
|
str := fmt.Sprintf("block contains " +
|
|
|
|
"transaction whose input sequence " +
|
|
|
|
"locks are not met")
|
|
|
|
return ruleError(ErrUnfinalizedTx, str)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-04 06:14:05 +01:00
|
|
|
// Enforce the segwit soft-fork package once the soft-fork has shifted
|
|
|
|
// into the "active" version bits state.
|
|
|
|
if enforceSegWit {
|
|
|
|
scriptFlags |= txscript.ScriptVerifyWitness
|
|
|
|
scriptFlags |= txscript.ScriptStrictMultiSig
|
|
|
|
}
|
2016-10-19 03:03:51 +02:00
|
|
|
|
2013-07-18 16:49:28 +02:00
|
|
|
// Now that the inexpensive checks are done and have passed, verify the
|
|
|
|
// transactions are actually allowed to spend the coins by running the
|
|
|
|
// expensive ECDSA signature check scripts. Doing this last helps
|
|
|
|
// prevent CPU exhaustion attacks.
|
|
|
|
if runScripts {
|
2016-10-19 03:33:20 +02:00
|
|
|
err := checkBlockScripts(block, view, scriptFlags, b.sigCache,
|
|
|
|
b.hashCache)
|
2013-07-18 16:49:28 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-26 06:03:18 +02:00
|
|
|
// Update the best hash for view to include this block since all of its
|
|
|
|
// transactions have been connected.
|
2017-02-01 22:25:25 +01:00
|
|
|
view.SetBestHash(&node.hash)
|
2015-08-26 06:03:18 +02:00
|
|
|
|
2013-07-18 16:49:28 +02:00
|
|
|
return nil
|
|
|
|
}
|
2014-03-11 19:18:27 +01:00
|
|
|
|
2017-09-15 04:09:32 +02:00
|
|
|
// CheckConnectBlockTemplate fully validates that connecting the passed block to
|
2017-09-15 05:35:11 +02:00
|
|
|
// the main chain does not violate any consensus rules, aside from the proof of
|
|
|
|
// work requirement. The block must connect to the current tip of the main chain.
|
2014-03-11 19:18:27 +01:00
|
|
|
//
|
2015-08-26 06:03:18 +02:00
|
|
|
// This function is safe for concurrent access.
|
2017-09-15 04:09:32 +02:00
|
|
|
func (b *BlockChain) CheckConnectBlockTemplate(block *btcutil.Block) error {
|
2015-08-26 06:03:18 +02:00
|
|
|
b.chainLock.Lock()
|
|
|
|
defer b.chainLock.Unlock()
|
2014-03-11 19:18:27 +01:00
|
|
|
|
2017-09-15 04:09:32 +02:00
|
|
|
// Skip the proof of work check as this is just a block template.
|
|
|
|
flags := BFNoPoWCheck
|
|
|
|
|
|
|
|
// This only checks whether the block can be connected to the tip of the
|
|
|
|
// current chain.
|
|
|
|
tip := b.bestChain.Tip()
|
|
|
|
header := block.MsgBlock().Header
|
|
|
|
if tip.hash != header.PrevBlock {
|
|
|
|
str := fmt.Sprintf("previous block must be the current chain tip %v, "+
|
|
|
|
"instead got %v", tip.hash, header.PrevBlock)
|
|
|
|
return ruleError(ErrPrevBlockNotBest, str)
|
|
|
|
}
|
|
|
|
|
|
|
|
err := checkBlockSanity(block, b.chainParams.PowLimit, b.timeSource, flags)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = b.checkBlockContext(block, tip, flags)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2015-08-26 06:03:18 +02:00
|
|
|
|
|
|
|
// Leave the spent txouts entry nil in the state since the information
|
|
|
|
// is not needed and thus extra work can be avoided.
|
|
|
|
view := NewUtxoViewpoint()
|
2017-09-15 04:09:32 +02:00
|
|
|
view.SetBestHash(&tip.hash)
|
2017-09-15 04:32:36 +02:00
|
|
|
newNode := newBlockNode(&header, tip)
|
2015-08-26 06:03:18 +02:00
|
|
|
return b.checkConnectBlock(newNode, block, view, nil)
|
2014-03-11 19:18:27 +01:00
|
|
|
}
|