Merge #13309: Directly operate with CMutableTransaction in SignSignature
6b8b63af14
Generic TransactionSignatureCreator works with both CTransaction and CMutableTransaction (Martin Ankerl)
Pull request description:
Refactored `TransactionSignatureCreator` into a templated `GenericTransactionSignatureCreator` that works with both `CMutableTransaction` and `CTransaction`.
The advantage is that now in `SignSignature`, the `MutableTransactionSignatureCreator` can now operate directly with the `CMutableTransaction` without the need to copy the data into a `CTransaction`.
Running all unit tests brings a very noticable speedup on my machine:
48.4 sec before this change
36.4 sec with this change
--------
12.0 seconds saved
running only `--run_test=transaction_tests/test_big_witness_transaction`:
16.7 sec before this change
5.9 sec with this change
--------
10.8 seconds saved
This relates to my first attempt with the const_cast hack #13202, and to the slow unit test issue #10026.
Also see #13050 which modifies the tests but not the production code (like this PR) to get a speedup.
Tree-SHA512: 2cff0e9699f484f26120a40e431a24c8bc8f9e780fd89cb0ecf20c5be3eab6c43f9c359cde244abd9f3620d06c7c354e3b9dd3da41fa2ca1ac1e09386fea25fb
This commit is contained in:
commit
36fc8052f6
5 changed files with 55 additions and 46 deletions
|
@ -1089,9 +1089,11 @@ namespace {
|
||||||
* Wrapper that serializes like CTransaction, but with the modifications
|
* Wrapper that serializes like CTransaction, but with the modifications
|
||||||
* required for the signature hash done in-place
|
* required for the signature hash done in-place
|
||||||
*/
|
*/
|
||||||
class CTransactionSignatureSerializer {
|
template <class T>
|
||||||
|
class CTransactionSignatureSerializer
|
||||||
|
{
|
||||||
private:
|
private:
|
||||||
const CTransaction& txTo; //!< reference to the spending transaction (the one being serialized)
|
const T& txTo; //!< reference to the spending transaction (the one being serialized)
|
||||||
const CScript& scriptCode; //!< output script being consumed
|
const CScript& scriptCode; //!< output script being consumed
|
||||||
const unsigned int nIn; //!< input index of txTo being signed
|
const unsigned int nIn; //!< input index of txTo being signed
|
||||||
const bool fAnyoneCanPay; //!< whether the hashtype has the SIGHASH_ANYONECANPAY flag set
|
const bool fAnyoneCanPay; //!< whether the hashtype has the SIGHASH_ANYONECANPAY flag set
|
||||||
|
@ -1099,7 +1101,7 @@ private:
|
||||||
const bool fHashNone; //!< whether the hashtype is SIGHASH_NONE
|
const bool fHashNone; //!< whether the hashtype is SIGHASH_NONE
|
||||||
|
|
||||||
public:
|
public:
|
||||||
CTransactionSignatureSerializer(const CTransaction &txToIn, const CScript &scriptCodeIn, unsigned int nInIn, int nHashTypeIn) :
|
CTransactionSignatureSerializer(const T& txToIn, const CScript& scriptCodeIn, unsigned int nInIn, int nHashTypeIn) :
|
||||||
txTo(txToIn), scriptCode(scriptCodeIn), nIn(nInIn),
|
txTo(txToIn), scriptCode(scriptCodeIn), nIn(nInIn),
|
||||||
fAnyoneCanPay(!!(nHashTypeIn & SIGHASH_ANYONECANPAY)),
|
fAnyoneCanPay(!!(nHashTypeIn & SIGHASH_ANYONECANPAY)),
|
||||||
fHashSingle((nHashTypeIn & 0x1f) == SIGHASH_SINGLE),
|
fHashSingle((nHashTypeIn & 0x1f) == SIGHASH_SINGLE),
|
||||||
|
@ -1180,7 +1182,9 @@ public:
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
uint256 GetPrevoutHash(const CTransaction& txTo) {
|
template <class T>
|
||||||
|
uint256 GetPrevoutHash(const T& txTo)
|
||||||
|
{
|
||||||
CHashWriter ss(SER_GETHASH, 0);
|
CHashWriter ss(SER_GETHASH, 0);
|
||||||
for (const auto& txin : txTo.vin) {
|
for (const auto& txin : txTo.vin) {
|
||||||
ss << txin.prevout;
|
ss << txin.prevout;
|
||||||
|
@ -1188,7 +1192,9 @@ uint256 GetPrevoutHash(const CTransaction& txTo) {
|
||||||
return ss.GetHash();
|
return ss.GetHash();
|
||||||
}
|
}
|
||||||
|
|
||||||
uint256 GetSequenceHash(const CTransaction& txTo) {
|
template <class T>
|
||||||
|
uint256 GetSequenceHash(const T& txTo)
|
||||||
|
{
|
||||||
CHashWriter ss(SER_GETHASH, 0);
|
CHashWriter ss(SER_GETHASH, 0);
|
||||||
for (const auto& txin : txTo.vin) {
|
for (const auto& txin : txTo.vin) {
|
||||||
ss << txin.nSequence;
|
ss << txin.nSequence;
|
||||||
|
@ -1196,7 +1202,9 @@ uint256 GetSequenceHash(const CTransaction& txTo) {
|
||||||
return ss.GetHash();
|
return ss.GetHash();
|
||||||
}
|
}
|
||||||
|
|
||||||
uint256 GetOutputsHash(const CTransaction& txTo) {
|
template <class T>
|
||||||
|
uint256 GetOutputsHash(const T& txTo)
|
||||||
|
{
|
||||||
CHashWriter ss(SER_GETHASH, 0);
|
CHashWriter ss(SER_GETHASH, 0);
|
||||||
for (const auto& txout : txTo.vout) {
|
for (const auto& txout : txTo.vout) {
|
||||||
ss << txout;
|
ss << txout;
|
||||||
|
@ -1206,7 +1214,8 @@ uint256 GetOutputsHash(const CTransaction& txTo) {
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
PrecomputedTransactionData::PrecomputedTransactionData(const CTransaction& txTo)
|
template <class T>
|
||||||
|
PrecomputedTransactionData::PrecomputedTransactionData(const T& txTo)
|
||||||
{
|
{
|
||||||
// Cache is calculated only for transactions with witness
|
// Cache is calculated only for transactions with witness
|
||||||
if (txTo.HasWitness()) {
|
if (txTo.HasWitness()) {
|
||||||
|
@ -1217,7 +1226,12 @@ PrecomputedTransactionData::PrecomputedTransactionData(const CTransaction& txTo)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
uint256 SignatureHash(const CScript& scriptCode, const CTransaction& txTo, unsigned int nIn, int nHashType, const CAmount& amount, SigVersion sigversion, const PrecomputedTransactionData* cache)
|
// explicit instantiation
|
||||||
|
template PrecomputedTransactionData::PrecomputedTransactionData(const CTransaction& txTo);
|
||||||
|
template PrecomputedTransactionData::PrecomputedTransactionData(const CMutableTransaction& txTo);
|
||||||
|
|
||||||
|
template <class T>
|
||||||
|
uint256 SignatureHash(const CScript& scriptCode, const T& txTo, unsigned int nIn, int nHashType, const CAmount& amount, SigVersion sigversion, const PrecomputedTransactionData* cache)
|
||||||
{
|
{
|
||||||
assert(nIn < txTo.vin.size());
|
assert(nIn < txTo.vin.size());
|
||||||
|
|
||||||
|
@ -1278,7 +1292,7 @@ uint256 SignatureHash(const CScript& scriptCode, const CTransaction& txTo, unsig
|
||||||
}
|
}
|
||||||
|
|
||||||
// Wrapper to serialize only the necessary parts of the transaction being signed
|
// Wrapper to serialize only the necessary parts of the transaction being signed
|
||||||
CTransactionSignatureSerializer txTmp(txTo, scriptCode, nIn, nHashType);
|
CTransactionSignatureSerializer<T> txTmp(txTo, scriptCode, nIn, nHashType);
|
||||||
|
|
||||||
// Serialize and hash
|
// Serialize and hash
|
||||||
CHashWriter ss(SER_GETHASH, 0);
|
CHashWriter ss(SER_GETHASH, 0);
|
||||||
|
@ -1286,12 +1300,14 @@ uint256 SignatureHash(const CScript& scriptCode, const CTransaction& txTo, unsig
|
||||||
return ss.GetHash();
|
return ss.GetHash();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool TransactionSignatureChecker::VerifySignature(const std::vector<unsigned char>& vchSig, const CPubKey& pubkey, const uint256& sighash) const
|
template <class T>
|
||||||
|
bool GenericTransactionSignatureChecker<T>::VerifySignature(const std::vector<unsigned char>& vchSig, const CPubKey& pubkey, const uint256& sighash) const
|
||||||
{
|
{
|
||||||
return pubkey.Verify(sighash, vchSig);
|
return pubkey.Verify(sighash, vchSig);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool TransactionSignatureChecker::CheckSig(const std::vector<unsigned char>& vchSigIn, const std::vector<unsigned char>& vchPubKey, const CScript& scriptCode, SigVersion sigversion) const
|
template <class T>
|
||||||
|
bool GenericTransactionSignatureChecker<T>::CheckSig(const std::vector<unsigned char>& vchSigIn, const std::vector<unsigned char>& vchPubKey, const CScript& scriptCode, SigVersion sigversion) const
|
||||||
{
|
{
|
||||||
CPubKey pubkey(vchPubKey);
|
CPubKey pubkey(vchPubKey);
|
||||||
if (!pubkey.IsValid())
|
if (!pubkey.IsValid())
|
||||||
|
@ -1312,7 +1328,8 @@ bool TransactionSignatureChecker::CheckSig(const std::vector<unsigned char>& vch
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool TransactionSignatureChecker::CheckLockTime(const CScriptNum& nLockTime) const
|
template <class T>
|
||||||
|
bool GenericTransactionSignatureChecker<T>::CheckLockTime(const CScriptNum& nLockTime) const
|
||||||
{
|
{
|
||||||
// There are two kinds of nLockTime: lock-by-blockheight
|
// There are two kinds of nLockTime: lock-by-blockheight
|
||||||
// and lock-by-blocktime, distinguished by whether
|
// and lock-by-blocktime, distinguished by whether
|
||||||
|
@ -1348,7 +1365,8 @@ bool TransactionSignatureChecker::CheckLockTime(const CScriptNum& nLockTime) con
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool TransactionSignatureChecker::CheckSequence(const CScriptNum& nSequence) const
|
template <class T>
|
||||||
|
bool GenericTransactionSignatureChecker<T>::CheckSequence(const CScriptNum& nSequence) const
|
||||||
{
|
{
|
||||||
// Relative lock times are supported by comparing the passed
|
// Relative lock times are supported by comparing the passed
|
||||||
// in operand to the sequence number of the input.
|
// in operand to the sequence number of the input.
|
||||||
|
@ -1394,6 +1412,10 @@ bool TransactionSignatureChecker::CheckSequence(const CScriptNum& nSequence) con
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// explicit instantiation
|
||||||
|
template class GenericTransactionSignatureChecker<CTransaction>;
|
||||||
|
template class GenericTransactionSignatureChecker<CMutableTransaction>;
|
||||||
|
|
||||||
static bool VerifyWitnessProgram(const CScriptWitness& witness, int witversion, const std::vector<unsigned char>& program, unsigned int flags, const BaseSignatureChecker& checker, ScriptError* serror)
|
static bool VerifyWitnessProgram(const CScriptWitness& witness, int witversion, const std::vector<unsigned char>& program, unsigned int flags, const BaseSignatureChecker& checker, ScriptError* serror)
|
||||||
{
|
{
|
||||||
std::vector<std::vector<unsigned char> > stack;
|
std::vector<std::vector<unsigned char> > stack;
|
||||||
|
|
|
@ -124,7 +124,8 @@ struct PrecomputedTransactionData
|
||||||
uint256 hashPrevouts, hashSequence, hashOutputs;
|
uint256 hashPrevouts, hashSequence, hashOutputs;
|
||||||
bool ready = false;
|
bool ready = false;
|
||||||
|
|
||||||
explicit PrecomputedTransactionData(const CTransaction& tx);
|
template <class T>
|
||||||
|
explicit PrecomputedTransactionData(const T& tx);
|
||||||
};
|
};
|
||||||
|
|
||||||
enum class SigVersion
|
enum class SigVersion
|
||||||
|
@ -137,7 +138,8 @@ enum class SigVersion
|
||||||
static constexpr size_t WITNESS_V0_SCRIPTHASH_SIZE = 32;
|
static constexpr size_t WITNESS_V0_SCRIPTHASH_SIZE = 32;
|
||||||
static constexpr size_t WITNESS_V0_KEYHASH_SIZE = 20;
|
static constexpr size_t WITNESS_V0_KEYHASH_SIZE = 20;
|
||||||
|
|
||||||
uint256 SignatureHash(const CScript &scriptCode, const CTransaction& txTo, unsigned int nIn, int nHashType, const CAmount& amount, SigVersion sigversion, const PrecomputedTransactionData* cache = nullptr);
|
template <class T>
|
||||||
|
uint256 SignatureHash(const CScript& scriptCode, const T& txTo, unsigned int nIn, int nHashType, const CAmount& amount, SigVersion sigversion, const PrecomputedTransactionData* cache = nullptr);
|
||||||
|
|
||||||
class BaseSignatureChecker
|
class BaseSignatureChecker
|
||||||
{
|
{
|
||||||
|
@ -160,10 +162,11 @@ public:
|
||||||
virtual ~BaseSignatureChecker() {}
|
virtual ~BaseSignatureChecker() {}
|
||||||
};
|
};
|
||||||
|
|
||||||
class TransactionSignatureChecker : public BaseSignatureChecker
|
template <class T>
|
||||||
|
class GenericTransactionSignatureChecker : public BaseSignatureChecker
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
const CTransaction* txTo;
|
const T* txTo;
|
||||||
unsigned int nIn;
|
unsigned int nIn;
|
||||||
const CAmount amount;
|
const CAmount amount;
|
||||||
const PrecomputedTransactionData* txdata;
|
const PrecomputedTransactionData* txdata;
|
||||||
|
@ -172,21 +175,15 @@ protected:
|
||||||
virtual bool VerifySignature(const std::vector<unsigned char>& vchSig, const CPubKey& vchPubKey, const uint256& sighash) const;
|
virtual bool VerifySignature(const std::vector<unsigned char>& vchSig, const CPubKey& vchPubKey, const uint256& sighash) const;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
TransactionSignatureChecker(const CTransaction* txToIn, unsigned int nInIn, const CAmount& amountIn) : txTo(txToIn), nIn(nInIn), amount(amountIn), txdata(nullptr) {}
|
GenericTransactionSignatureChecker(const T* txToIn, unsigned int nInIn, const CAmount& amountIn) : txTo(txToIn), nIn(nInIn), amount(amountIn), txdata(nullptr) {}
|
||||||
TransactionSignatureChecker(const CTransaction* txToIn, unsigned int nInIn, const CAmount& amountIn, const PrecomputedTransactionData& txdataIn) : txTo(txToIn), nIn(nInIn), amount(amountIn), txdata(&txdataIn) {}
|
GenericTransactionSignatureChecker(const T* txToIn, unsigned int nInIn, const CAmount& amountIn, const PrecomputedTransactionData& txdataIn) : txTo(txToIn), nIn(nInIn), amount(amountIn), txdata(&txdataIn) {}
|
||||||
bool CheckSig(const std::vector<unsigned char>& scriptSig, const std::vector<unsigned char>& vchPubKey, const CScript& scriptCode, SigVersion sigversion) const override;
|
bool CheckSig(const std::vector<unsigned char>& scriptSig, const std::vector<unsigned char>& vchPubKey, const CScript& scriptCode, SigVersion sigversion) const override;
|
||||||
bool CheckLockTime(const CScriptNum& nLockTime) const override;
|
bool CheckLockTime(const CScriptNum& nLockTime) const override;
|
||||||
bool CheckSequence(const CScriptNum& nSequence) const override;
|
bool CheckSequence(const CScriptNum& nSequence) const override;
|
||||||
};
|
};
|
||||||
|
|
||||||
class MutableTransactionSignatureChecker : public TransactionSignatureChecker
|
using TransactionSignatureChecker = GenericTransactionSignatureChecker<CTransaction>;
|
||||||
{
|
using MutableTransactionSignatureChecker = GenericTransactionSignatureChecker<CMutableTransaction>;
|
||||||
private:
|
|
||||||
const CTransaction txTo;
|
|
||||||
|
|
||||||
public:
|
|
||||||
MutableTransactionSignatureChecker(const CMutableTransaction* txToIn, unsigned int nInIn, const CAmount& amountIn) : TransactionSignatureChecker(&txTo, nInIn, amountIn), txTo(*txToIn) {}
|
|
||||||
};
|
|
||||||
|
|
||||||
bool EvalScript(std::vector<std::vector<unsigned char> >& stack, const CScript& script, unsigned int flags, const BaseSignatureChecker& checker, SigVersion sigversion, ScriptError* error = nullptr);
|
bool EvalScript(std::vector<std::vector<unsigned char> >& stack, const CScript& script, unsigned int flags, const BaseSignatureChecker& checker, SigVersion sigversion, ScriptError* error = nullptr);
|
||||||
bool VerifyScript(const CScript& scriptSig, const CScript& scriptPubKey, const CScriptWitness* witness, unsigned int flags, const BaseSignatureChecker& checker, ScriptError* serror = nullptr);
|
bool VerifyScript(const CScript& scriptSig, const CScript& scriptPubKey, const CScriptWitness* witness, unsigned int flags, const BaseSignatureChecker& checker, ScriptError* serror = nullptr);
|
||||||
|
|
|
@ -14,9 +14,9 @@
|
||||||
|
|
||||||
typedef std::vector<unsigned char> valtype;
|
typedef std::vector<unsigned char> valtype;
|
||||||
|
|
||||||
TransactionSignatureCreator::TransactionSignatureCreator(const CTransaction* txToIn, unsigned int nInIn, const CAmount& amountIn, int nHashTypeIn) : txTo(txToIn), nIn(nInIn), nHashType(nHashTypeIn), amount(amountIn), checker(txTo, nIn, amountIn) {}
|
MutableTransactionSignatureCreator::MutableTransactionSignatureCreator(const CMutableTransaction* txToIn, unsigned int nInIn, const CAmount& amountIn, int nHashTypeIn) : txTo(txToIn), nIn(nInIn), nHashType(nHashTypeIn), amount(amountIn), checker(txTo, nIn, amountIn) {}
|
||||||
|
|
||||||
bool TransactionSignatureCreator::CreateSig(const SigningProvider& provider, std::vector<unsigned char>& vchSig, const CKeyID& address, const CScript& scriptCode, SigVersion sigversion) const
|
bool MutableTransactionSignatureCreator::CreateSig(const SigningProvider& provider, std::vector<unsigned char>& vchSig, const CKeyID& address, const CScript& scriptCode, SigVersion sigversion) const
|
||||||
{
|
{
|
||||||
CKey key;
|
CKey key;
|
||||||
if (!provider.GetKey(address, key))
|
if (!provider.GetKey(address, key))
|
||||||
|
@ -209,8 +209,7 @@ bool SignSignature(const SigningProvider &provider, const CScript& fromPubKey, C
|
||||||
{
|
{
|
||||||
assert(nIn < txTo.vin.size());
|
assert(nIn < txTo.vin.size());
|
||||||
|
|
||||||
CTransaction txToConst(txTo);
|
MutableTransactionSignatureCreator creator(&txTo, nIn, amount, nHashType);
|
||||||
TransactionSignatureCreator creator(&txToConst, nIn, amount, nHashType);
|
|
||||||
|
|
||||||
SignatureData sigdata;
|
SignatureData sigdata;
|
||||||
bool ret = ProduceSignature(provider, creator, fromPubKey, sigdata);
|
bool ret = ProduceSignature(provider, creator, fromPubKey, sigdata);
|
||||||
|
|
|
@ -37,26 +37,19 @@ public:
|
||||||
};
|
};
|
||||||
|
|
||||||
/** A signature creator for transactions. */
|
/** A signature creator for transactions. */
|
||||||
class TransactionSignatureCreator : public BaseSignatureCreator {
|
class MutableTransactionSignatureCreator : public BaseSignatureCreator {
|
||||||
const CTransaction* txTo;
|
const CMutableTransaction* txTo;
|
||||||
unsigned int nIn;
|
unsigned int nIn;
|
||||||
int nHashType;
|
int nHashType;
|
||||||
CAmount amount;
|
CAmount amount;
|
||||||
const TransactionSignatureChecker checker;
|
const MutableTransactionSignatureChecker checker;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
TransactionSignatureCreator(const CTransaction* txToIn, unsigned int nInIn, const CAmount& amountIn, int nHashTypeIn=SIGHASH_ALL);
|
MutableTransactionSignatureCreator(const CMutableTransaction* txToIn, unsigned int nInIn, const CAmount& amountIn, int nHashTypeIn = SIGHASH_ALL);
|
||||||
const BaseSignatureChecker& Checker() const override { return checker; }
|
const BaseSignatureChecker& Checker() const override { return checker; }
|
||||||
bool CreateSig(const SigningProvider& provider, std::vector<unsigned char>& vchSig, const CKeyID& keyid, const CScript& scriptCode, SigVersion sigversion) const override;
|
bool CreateSig(const SigningProvider& provider, std::vector<unsigned char>& vchSig, const CKeyID& keyid, const CScript& scriptCode, SigVersion sigversion) const override;
|
||||||
};
|
};
|
||||||
|
|
||||||
class MutableTransactionSignatureCreator : public TransactionSignatureCreator {
|
|
||||||
CTransaction tx;
|
|
||||||
|
|
||||||
public:
|
|
||||||
MutableTransactionSignatureCreator(const CMutableTransaction* txToIn, unsigned int nInIn, const CAmount& amountIn, int nHashTypeIn) : TransactionSignatureCreator(&tx, nInIn, amountIn, nHashTypeIn), tx(*txToIn) {}
|
|
||||||
};
|
|
||||||
|
|
||||||
/** A signature creator that just produces 72-byte empty signatures. */
|
/** A signature creator that just produces 72-byte empty signatures. */
|
||||||
extern const BaseSignatureCreator& DUMMY_SIGNATURE_CREATOR;
|
extern const BaseSignatureCreator& DUMMY_SIGNATURE_CREATOR;
|
||||||
|
|
||||||
|
|
|
@ -2608,7 +2608,6 @@ bool CWallet::SignTransaction(CMutableTransaction &tx)
|
||||||
AssertLockHeld(cs_wallet); // mapWallet
|
AssertLockHeld(cs_wallet); // mapWallet
|
||||||
|
|
||||||
// sign the new tx
|
// sign the new tx
|
||||||
CTransaction txNewConst(tx);
|
|
||||||
int nIn = 0;
|
int nIn = 0;
|
||||||
for (const auto& input : tx.vin) {
|
for (const auto& input : tx.vin) {
|
||||||
std::map<uint256, CWalletTx>::const_iterator mi = mapWallet.find(input.prevout.hash);
|
std::map<uint256, CWalletTx>::const_iterator mi = mapWallet.find(input.prevout.hash);
|
||||||
|
@ -2618,7 +2617,7 @@ bool CWallet::SignTransaction(CMutableTransaction &tx)
|
||||||
const CScript& scriptPubKey = mi->second.tx->vout[input.prevout.n].scriptPubKey;
|
const CScript& scriptPubKey = mi->second.tx->vout[input.prevout.n].scriptPubKey;
|
||||||
const CAmount& amount = mi->second.tx->vout[input.prevout.n].nValue;
|
const CAmount& amount = mi->second.tx->vout[input.prevout.n].nValue;
|
||||||
SignatureData sigdata;
|
SignatureData sigdata;
|
||||||
if (!ProduceSignature(*this, TransactionSignatureCreator(&txNewConst, nIn, amount, SIGHASH_ALL), scriptPubKey, sigdata)) {
|
if (!ProduceSignature(*this, MutableTransactionSignatureCreator(&tx, nIn, amount, SIGHASH_ALL), scriptPubKey, sigdata)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
UpdateTransaction(tx, nIn, sigdata);
|
UpdateTransaction(tx, nIn, sigdata);
|
||||||
|
@ -3040,14 +3039,13 @@ bool CWallet::CreateTransaction(const std::vector<CRecipient>& vecSend, CTransac
|
||||||
|
|
||||||
if (sign)
|
if (sign)
|
||||||
{
|
{
|
||||||
CTransaction txNewConst(txNew);
|
|
||||||
int nIn = 0;
|
int nIn = 0;
|
||||||
for (const auto& coin : selected_coins)
|
for (const auto& coin : selected_coins)
|
||||||
{
|
{
|
||||||
const CScript& scriptPubKey = coin.txout.scriptPubKey;
|
const CScript& scriptPubKey = coin.txout.scriptPubKey;
|
||||||
SignatureData sigdata;
|
SignatureData sigdata;
|
||||||
|
|
||||||
if (!ProduceSignature(*this, TransactionSignatureCreator(&txNewConst, nIn, coin.txout.nValue, SIGHASH_ALL), scriptPubKey, sigdata))
|
if (!ProduceSignature(*this, MutableTransactionSignatureCreator(&txNew, nIn, coin.txout.nValue, SIGHASH_ALL), scriptPubKey, sigdata))
|
||||||
{
|
{
|
||||||
strFailReason = _("Signing transaction failed");
|
strFailReason = _("Signing transaction failed");
|
||||||
return false;
|
return false;
|
||||||
|
|
Loading…
Reference in a new issue