validation.h
1 // Copyright (c) 2009-2010 Satoshi Nakamoto 2 // Copyright (c) 2009-present The Bitcoin Core developers 3 // Distributed under the MIT software license, see the accompanying 4 // file COPYING or http://www.opensource.org/licenses/mit-license.php. 5 6 #ifndef BITCOIN_CONSENSUS_VALIDATION_H 7 #define BITCOIN_CONSENSUS_VALIDATION_H 8 9 #include <string> 10 #include <consensus/consensus.h> 11 #include <primitives/transaction.h> 12 #include <primitives/block.h> 13 14 /** Index marker for when no witness commitment is present in a coinbase transaction. */ 15 static constexpr int NO_WITNESS_COMMITMENT{-1}; 16 17 /** Minimum size of a witness commitment structure. Defined in BIP 141. **/ 18 static constexpr size_t MINIMUM_WITNESS_COMMITMENT{38}; 19 20 /** A "reason" why a transaction was invalid, suitable for determining whether the 21 * provider of the transaction should be banned/ignored/disconnected/etc. 22 */ 23 enum class TxValidationResult { 24 TX_RESULT_UNSET = 0, //!< initial value. Tx has not yet been rejected 25 TX_CONSENSUS, //!< invalid by consensus rules 26 TX_INPUTS_NOT_STANDARD, //!< inputs (covered by txid) failed policy rules 27 TX_NOT_STANDARD, //!< otherwise didn't meet our local policy rules 28 TX_MISSING_INPUTS, //!< transaction was missing some of its inputs 29 TX_PREMATURE_SPEND, //!< transaction spends a coinbase too early, or violates locktime/sequence locks 30 /** 31 * Transaction might have a witness prior to SegWit 32 * activation, or witness may have been malleated (which includes 33 * non-standard witnesses). 34 */ 35 TX_WITNESS_MUTATED, 36 /** 37 * Transaction is missing a witness. 38 */ 39 TX_WITNESS_STRIPPED, 40 /** 41 * Tx already in mempool or conflicts with a tx in the chain 42 * (if it conflicts with another tx in mempool, we use MEMPOOL_POLICY as it failed to reach the RBF threshold) 43 * Currently this is only used if the transaction already exists in the mempool or on chain. 44 */ 45 TX_CONFLICT, 46 TX_MEMPOOL_POLICY, //!< violated mempool's fee/size/descendant/RBF/etc limits 47 TX_NO_MEMPOOL, //!< this node does not have a mempool so can't validate the transaction 48 TX_RECONSIDERABLE, //!< fails some policy, but might be acceptable if submitted in a (different) package 49 TX_UNKNOWN, //!< transaction was not validated because package failed 50 }; 51 52 /** A "reason" why a block was invalid, suitable for determining whether the 53 * provider of the block should be banned/ignored/disconnected/etc. 54 * These are much more granular than the rejection codes, which may be more 55 * useful for some other use-cases. 56 */ 57 enum class BlockValidationResult { 58 BLOCK_RESULT_UNSET = 0, //!< initial value. Block has not yet been rejected 59 BLOCK_CONSENSUS, //!< invalid by consensus rules (excluding any below reasons) 60 BLOCK_CACHED_INVALID, //!< this block was cached as being invalid and we didn't store the reason why 61 BLOCK_INVALID_HEADER, //!< invalid proof of work or time too old 62 BLOCK_MUTATED, //!< the block's data didn't match the data committed to by the PoW 63 BLOCK_MISSING_PREV, //!< We don't have the previous block the checked one is built on 64 BLOCK_INVALID_PREV, //!< A block this one builds on is invalid 65 BLOCK_TIME_FUTURE, //!< block timestamp was > 2 hours in the future (or our clock is bad) 66 BLOCK_HEADER_LOW_WORK //!< the block header may be on a too-little-work chain 67 }; 68 69 70 71 /** Template for capturing information about block/transaction validation. This is instantiated 72 * by TxValidationState and BlockValidationState for validation information on transactions 73 * and blocks respectively. */ 74 template <typename Result> 75 class ValidationState 76 { 77 private: 78 enum class ModeState { 79 M_VALID, //!< everything ok 80 M_INVALID, //!< network rule violation (DoS value may be set) 81 M_ERROR, //!< run-time error 82 } m_mode{ModeState::M_VALID}; 83 Result m_result{}; 84 std::string m_reject_reason; 85 std::string m_debug_message; 86 87 public: 88 bool Invalid(Result result, 89 const std::string& reject_reason = "", 90 const std::string& debug_message = "") 91 { 92 m_result = result; 93 m_reject_reason = reject_reason; 94 m_debug_message = debug_message; 95 if (m_mode != ModeState::M_ERROR) m_mode = ModeState::M_INVALID; 96 return false; 97 } 98 bool Error(const std::string& reject_reason) 99 { 100 if (m_mode == ModeState::M_VALID) 101 m_reject_reason = reject_reason; 102 m_mode = ModeState::M_ERROR; 103 return false; 104 } 105 bool IsValid() const { return m_mode == ModeState::M_VALID; } 106 bool IsInvalid() const { return m_mode == ModeState::M_INVALID; } 107 bool IsError() const { return m_mode == ModeState::M_ERROR; } 108 Result GetResult() const { return m_result; } 109 std::string GetRejectReason() const { return m_reject_reason; } 110 std::string GetDebugMessage() const { return m_debug_message; } 111 std::string ToString() const 112 { 113 if (IsValid()) { 114 return "Valid"; 115 } 116 117 if (!m_debug_message.empty()) { 118 return m_reject_reason + ", " + m_debug_message; 119 } 120 121 return m_reject_reason; 122 } 123 }; 124 125 class TxValidationState : public ValidationState<TxValidationResult> {}; 126 class BlockValidationState : public ValidationState<BlockValidationResult> {}; 127 128 // These implement the weight = (stripped_size * 4) + witness_size formula, 129 // using only serialization with and without witness data. As witness_size 130 // is equal to total_size - stripped_size, this formula is identical to: 131 // weight = (stripped_size * 3) + total_size. 132 static inline int32_t GetTransactionWeight(const CTransaction& tx) 133 { 134 return ::GetSerializeSize(TX_NO_WITNESS(tx)) * (WITNESS_SCALE_FACTOR - 1) + ::GetSerializeSize(TX_WITH_WITNESS(tx)); 135 } 136 static inline int64_t GetBlockWeight(const CBlock& block) 137 { 138 return ::GetSerializeSize(TX_NO_WITNESS(block)) * (WITNESS_SCALE_FACTOR - 1) + ::GetSerializeSize(TX_WITH_WITNESS(block)); 139 } 140 static inline int64_t GetTransactionInputWeight(const CTxIn& txin) 141 { 142 // scriptWitness size is added here because witnesses and txins are split up in segwit serialization. 143 return ::GetSerializeSize(TX_NO_WITNESS(txin)) * (WITNESS_SCALE_FACTOR - 1) + ::GetSerializeSize(TX_WITH_WITNESS(txin)) + ::GetSerializeSize(txin.scriptWitness.stack); 144 } 145 146 /** Compute at which vout of the block's coinbase transaction the witness commitment occurs, or -1 if not found */ 147 inline int GetWitnessCommitmentIndex(const CBlock& block) 148 { 149 int commitpos = NO_WITNESS_COMMITMENT; 150 if (!block.vtx.empty()) { 151 for (size_t o = 0; o < block.vtx[0]->vout.size(); o++) { 152 const CTxOut& vout = block.vtx[0]->vout[o]; 153 if (vout.scriptPubKey.size() >= MINIMUM_WITNESS_COMMITMENT && 154 vout.scriptPubKey[0] == OP_RETURN && 155 vout.scriptPubKey[1] == 0x24 && 156 vout.scriptPubKey[2] == 0xaa && 157 vout.scriptPubKey[3] == 0x21 && 158 vout.scriptPubKey[4] == 0xa9 && 159 vout.scriptPubKey[5] == 0xed) { 160 commitpos = o; 161 } 162 } 163 } 164 return commitpos; 165 } 166 167 #endif // BITCOIN_CONSENSUS_VALIDATION_H