|
|
@ -471,38 +471,53 @@ unsigned int LimitOrphanTxSize(unsigned int nMaxOrphans) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool IsStandardTx(const CTransaction& tx) |
|
|
|
bool IsStandardTx(const CTransaction& tx, string& reason) |
|
|
|
{ |
|
|
|
{ |
|
|
|
if (tx.nVersion > CTransaction::CURRENT_VERSION) |
|
|
|
if (tx.nVersion > CTransaction::CURRENT_VERSION) { |
|
|
|
|
|
|
|
reason = "version"; |
|
|
|
return false; |
|
|
|
return false; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
if (!IsFinalTx(tx)) |
|
|
|
if (!IsFinalTx(tx)) { |
|
|
|
|
|
|
|
reason = "non-final"; |
|
|
|
return false; |
|
|
|
return false; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// Extremely large transactions with lots of inputs can cost the network
|
|
|
|
// Extremely large transactions with lots of inputs can cost the network
|
|
|
|
// almost as much to process as they cost the sender in fees, because
|
|
|
|
// almost as much to process as they cost the sender in fees, because
|
|
|
|
// computing signature hashes is O(ninputs*txsize). Limiting transactions
|
|
|
|
// computing signature hashes is O(ninputs*txsize). Limiting transactions
|
|
|
|
// to MAX_STANDARD_TX_SIZE mitigates CPU exhaustion attacks.
|
|
|
|
// to MAX_STANDARD_TX_SIZE mitigates CPU exhaustion attacks.
|
|
|
|
unsigned int sz = tx.GetSerializeSize(SER_NETWORK, CTransaction::CURRENT_VERSION); |
|
|
|
unsigned int sz = tx.GetSerializeSize(SER_NETWORK, CTransaction::CURRENT_VERSION); |
|
|
|
if (sz >= MAX_STANDARD_TX_SIZE) |
|
|
|
if (sz >= MAX_STANDARD_TX_SIZE) { |
|
|
|
|
|
|
|
reason = "tx-size"; |
|
|
|
return false; |
|
|
|
return false; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
BOOST_FOREACH(const CTxIn& txin, tx.vin) |
|
|
|
BOOST_FOREACH(const CTxIn& txin, tx.vin) |
|
|
|
{ |
|
|
|
{ |
|
|
|
// Biggest 'standard' txin is a 3-signature 3-of-3 CHECKMULTISIG
|
|
|
|
// Biggest 'standard' txin is a 3-signature 3-of-3 CHECKMULTISIG
|
|
|
|
// pay-to-script-hash, which is 3 ~80-byte signatures, 3
|
|
|
|
// pay-to-script-hash, which is 3 ~80-byte signatures, 3
|
|
|
|
// ~65-byte public keys, plus a few script ops.
|
|
|
|
// ~65-byte public keys, plus a few script ops.
|
|
|
|
if (txin.scriptSig.size() > 500) |
|
|
|
if (txin.scriptSig.size() > 500) { |
|
|
|
|
|
|
|
reason = "scriptsig-size"; |
|
|
|
return false; |
|
|
|
return false; |
|
|
|
if (!txin.scriptSig.IsPushOnly()) |
|
|
|
} |
|
|
|
|
|
|
|
if (!txin.scriptSig.IsPushOnly()) { |
|
|
|
|
|
|
|
reason = "scriptsig-not-pushonly"; |
|
|
|
return false; |
|
|
|
return false; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
BOOST_FOREACH(const CTxOut& txout, tx.vout) { |
|
|
|
BOOST_FOREACH(const CTxOut& txout, tx.vout) { |
|
|
|
if (!::IsStandard(txout.scriptPubKey)) |
|
|
|
if (!::IsStandard(txout.scriptPubKey)) { |
|
|
|
|
|
|
|
reason = "scriptpubkey"; |
|
|
|
return false; |
|
|
|
return false; |
|
|
|
if (txout.IsDust(CTransaction::nMinRelayTxFee)) |
|
|
|
} |
|
|
|
|
|
|
|
if (txout.IsDust(CTransaction::nMinRelayTxFee)) { |
|
|
|
|
|
|
|
reason = "dust"; |
|
|
|
return false; |
|
|
|
return false; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
return true; |
|
|
|
return true; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
@ -796,8 +811,10 @@ bool CTxMemPool::accept(CValidationState &state, CTransaction &tx, bool fLimitFr |
|
|
|
return error("CTxMemPool::accept() : not accepting nLockTime beyond 2038 yet"); |
|
|
|
return error("CTxMemPool::accept() : not accepting nLockTime beyond 2038 yet"); |
|
|
|
|
|
|
|
|
|
|
|
// Rather not work on nonstandard transactions (unless -testnet)
|
|
|
|
// Rather not work on nonstandard transactions (unless -testnet)
|
|
|
|
if (!TestNet() && !IsStandardTx(tx)) |
|
|
|
string reason; |
|
|
|
return error("CTxMemPool::accept() : nonstandard transaction type"); |
|
|
|
if (!TestNet() && !IsStandardTx(tx, reason)) |
|
|
|
|
|
|
|
return error("CTxMemPool::accept() : nonstandard transaction: %s", |
|
|
|
|
|
|
|
reason.c_str()); |
|
|
|
|
|
|
|
|
|
|
|
// is it already in the memory pool?
|
|
|
|
// is it already in the memory pool?
|
|
|
|
uint256 hash = tx.GetHash(); |
|
|
|
uint256 hash = tx.GetHash(); |
|
|
|