2020-04-16 13:14:08 -04:00
|
|
|
// Copyright (c) 2009-2020 The Bitcoin Core developers
|
2019-02-13 14:25:49 -05:00
|
|
|
// Distributed under the MIT software license, see the accompanying
|
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
2019-10-27 21:14:42 +00:00
|
|
|
#include <pubkey.h>
|
2019-02-13 14:25:49 -05:00
|
|
|
#include <script/interpreter.h>
|
|
|
|
#include <streams.h>
|
2019-10-27 21:14:42 +00:00
|
|
|
#include <util/memory.h>
|
2019-02-13 14:25:49 -05:00
|
|
|
#include <version.h>
|
|
|
|
|
|
|
|
#include <test/fuzz/fuzz.h>
|
|
|
|
|
|
|
|
/** Flags that are not forbidden by an assert */
|
|
|
|
static bool IsValidFlagCombination(unsigned flags);
|
|
|
|
|
2020-12-03 16:42:49 +01:00
|
|
|
void initialize_script_flags()
|
2019-10-27 21:14:42 +00:00
|
|
|
{
|
2020-03-10 12:55:41 +00:00
|
|
|
static const ECCVerifyHandle verify_handle;
|
2019-10-27 21:14:42 +00:00
|
|
|
}
|
|
|
|
|
2020-12-03 16:42:49 +01:00
|
|
|
FUZZ_TARGET_INIT(script_flags, initialize_script_flags)
|
2019-02-13 14:25:49 -05:00
|
|
|
{
|
|
|
|
CDataStream ds(buffer, SER_NETWORK, INIT_PROTO_VERSION);
|
|
|
|
try {
|
|
|
|
int nVersion;
|
|
|
|
ds >> nVersion;
|
|
|
|
ds.SetVersion(nVersion);
|
|
|
|
} catch (const std::ios_base::failure&) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
const CTransaction tx(deserialize, ds);
|
|
|
|
|
|
|
|
unsigned int verify_flags;
|
|
|
|
ds >> verify_flags;
|
|
|
|
|
|
|
|
if (!IsValidFlagCombination(verify_flags)) return;
|
|
|
|
|
|
|
|
unsigned int fuzzed_flags;
|
|
|
|
ds >> fuzzed_flags;
|
|
|
|
|
2020-10-26 09:28:11 +01:00
|
|
|
std::vector<CTxOut> spent_outputs;
|
2019-02-13 14:25:49 -05:00
|
|
|
for (unsigned i = 0; i < tx.vin.size(); ++i) {
|
|
|
|
CTxOut prevout;
|
|
|
|
ds >> prevout;
|
2020-10-26 09:28:11 +01:00
|
|
|
spent_outputs.push_back(prevout);
|
|
|
|
}
|
|
|
|
PrecomputedTransactionData txdata;
|
|
|
|
txdata.Init(tx, std::move(spent_outputs));
|
2019-02-13 14:25:49 -05:00
|
|
|
|
2020-10-26 09:28:11 +01:00
|
|
|
for (unsigned i = 0; i < tx.vin.size(); ++i) {
|
|
|
|
const CTxOut& prevout = txdata.m_spent_outputs.at(i);
|
2019-02-13 14:25:49 -05:00
|
|
|
const TransactionSignatureChecker checker{&tx, i, prevout.nValue, txdata};
|
|
|
|
|
|
|
|
ScriptError serror;
|
|
|
|
const bool ret = VerifyScript(tx.vin.at(i).scriptSig, prevout.scriptPubKey, &tx.vin.at(i).scriptWitness, verify_flags, checker, &serror);
|
|
|
|
assert(ret == (serror == SCRIPT_ERR_OK));
|
|
|
|
|
|
|
|
// Verify that removing flags from a passing test or adding flags to a failing test does not change the result
|
|
|
|
if (ret) {
|
|
|
|
verify_flags &= ~fuzzed_flags;
|
|
|
|
} else {
|
|
|
|
verify_flags |= fuzzed_flags;
|
|
|
|
}
|
|
|
|
if (!IsValidFlagCombination(verify_flags)) return;
|
|
|
|
|
|
|
|
ScriptError serror_fuzzed;
|
|
|
|
const bool ret_fuzzed = VerifyScript(tx.vin.at(i).scriptSig, prevout.scriptPubKey, &tx.vin.at(i).scriptWitness, verify_flags, checker, &serror_fuzzed);
|
|
|
|
assert(ret_fuzzed == (serror_fuzzed == SCRIPT_ERR_OK));
|
|
|
|
|
|
|
|
assert(ret_fuzzed == ret);
|
|
|
|
}
|
|
|
|
} catch (const std::ios_base::failure&) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool IsValidFlagCombination(unsigned flags)
|
|
|
|
{
|
|
|
|
if (flags & SCRIPT_VERIFY_CLEANSTACK && ~flags & (SCRIPT_VERIFY_P2SH | SCRIPT_VERIFY_WITNESS)) return false;
|
|
|
|
if (flags & SCRIPT_VERIFY_WITNESS && ~flags & SCRIPT_VERIFY_P2SH) return false;
|
|
|
|
return true;
|
|
|
|
}
|