2014-08-27 17:22:33 +02:00
// Copyright (c) 2009-2010 Satoshi Nakamoto
2018-01-03 02:12:05 +09:00
// Copyright (c) 2009-2017 The Bitcoin Core developers
2014-09-09 10:00:42 +02:00
// Distributed under the MIT software license, see the accompanying
2014-08-27 17:22:33 +02:00
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2017-11-10 13:57:53 +13:00
# include <script/sign.h>
2014-08-27 17:22:33 +02:00
2017-11-10 13:57:53 +13:00
# include <key.h>
# include <policy/policy.h>
# include <primitives/transaction.h>
# include <script/standard.h>
# include <uint256.h>
2014-08-27 17:22:33 +02:00
2015-10-29 07:11:24 +01:00
typedef std : : vector < unsigned char > valtype ;
2014-08-27 17:22:33 +02:00
2018-03-27 13:15:10 -07:00
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 ) { }
2014-11-04 10:06:20 -08:00
2018-03-27 13:15:10 -07:00
bool TransactionSignatureCreator : : CreateSig ( const SigningProvider & provider , std : : vector < unsigned char > & vchSig , const CKeyID & address , const CScript & scriptCode , SigVersion sigversion ) const
2014-08-27 17:22:33 +02:00
{
CKey key ;
2018-03-27 13:15:10 -07:00
if ( ! provider . GetKey ( address , key ) )
2014-08-27 17:22:33 +02:00
return false ;
2016-10-02 01:19:33 +08:00
// Signing with uncompressed keys is disabled in witness scripts
2018-03-09 15:03:40 +01:00
if ( sigversion = = SigVersion : : WITNESS_V0 & & ! key . IsCompressed ( ) )
2016-10-02 01:19:33 +08:00
return false ;
2016-03-31 14:54:58 +02:00
uint256 hash = SignatureHash ( scriptCode , * txTo , nIn , nHashType , amount , sigversion ) ;
2014-08-27 17:22:33 +02:00
if ( ! key . Sign ( hash , vchSig ) )
return false ;
vchSig . push_back ( ( unsigned char ) nHashType ) ;
2014-11-04 10:06:20 -08:00
return true ;
}
2014-08-27 17:22:33 +02:00
2018-03-27 13:15:10 -07:00
static bool Sign1 ( const SigningProvider & provider , const CKeyID & address , const BaseSignatureCreator & creator , const CScript & scriptCode , std : : vector < valtype > & ret , SigVersion sigversion )
2014-11-04 10:06:20 -08:00
{
2017-01-04 14:42:14 +09:00
std : : vector < unsigned char > vchSig ;
2018-03-27 13:15:10 -07:00
if ( ! creator . CreateSig ( provider , vchSig , address , scriptCode , sigversion ) )
2014-11-04 10:06:20 -08:00
return false ;
2016-03-31 14:54:58 +02:00
ret . push_back ( vchSig ) ;
2014-08-27 17:22:33 +02:00
return true ;
}
2018-03-27 13:15:10 -07:00
static bool SignN ( const SigningProvider & provider , const std : : vector < valtype > & multisigdata , const BaseSignatureCreator & creator , const CScript & scriptCode , std : : vector < valtype > & ret , SigVersion sigversion )
2014-08-27 17:22:33 +02:00
{
int nSigned = 0 ;
int nRequired = multisigdata . front ( ) [ 0 ] ;
for ( unsigned int i = 1 ; i < multisigdata . size ( ) - 1 & & nSigned < nRequired ; i + + )
{
const valtype & pubkey = multisigdata [ i ] ;
CKeyID keyID = CPubKey ( pubkey ) . GetID ( ) ;
2018-03-27 13:15:10 -07:00
if ( Sign1 ( provider , keyID , creator , scriptCode , ret , sigversion ) )
2014-08-27 17:22:33 +02:00
+ + nSigned ;
}
return nSigned = = nRequired ;
}
2014-11-10 14:40:01 +08:00
/**
2014-11-04 10:06:20 -08:00
* Sign scriptPubKey using signature made with creator .
2014-11-10 14:40:01 +08:00
* Signatures are returned in scriptSigRet ( or returns false if scriptPubKey can ' t be signed ) ,
* unless whichTypeRet is TX_SCRIPTHASH , in which case scriptSigRet is the redemption script .
* Returns false if scriptPubKey could not be completely satisfied .
*/
2018-03-27 13:15:10 -07:00
static bool SignStep ( const SigningProvider & provider , const BaseSignatureCreator & creator , const CScript & scriptPubKey ,
2016-03-31 14:54:58 +02:00
std : : vector < valtype > & ret , txnouttype & whichTypeRet , SigVersion sigversion )
2014-08-27 17:22:33 +02:00
{
2016-03-31 14:54:58 +02:00
CScript scriptRet ;
uint160 h160 ;
ret . clear ( ) ;
2014-08-27 17:22:33 +02:00
2017-01-04 14:42:14 +09:00
std : : vector < valtype > vSolutions ;
2014-08-27 17:22:33 +02:00
if ( ! Solver ( scriptPubKey , whichTypeRet , vSolutions ) )
return false ;
CKeyID keyID ;
switch ( whichTypeRet )
{
case TX_NONSTANDARD :
case TX_NULL_DATA :
2017-08-25 19:55:52 -07:00
case TX_WITNESS_UNKNOWN :
2014-08-27 17:22:33 +02:00
return false ;
case TX_PUBKEY :
keyID = CPubKey ( vSolutions [ 0 ] ) . GetID ( ) ;
2018-03-27 13:15:10 -07:00
return Sign1 ( provider , keyID , creator , scriptPubKey , ret , sigversion ) ;
2014-08-27 17:22:33 +02:00
case TX_PUBKEYHASH :
keyID = CKeyID ( uint160 ( vSolutions [ 0 ] ) ) ;
2018-03-27 13:15:10 -07:00
if ( ! Sign1 ( provider , keyID , creator , scriptPubKey , ret , sigversion ) )
2014-08-27 17:22:33 +02:00
return false ;
else
{
CPubKey vch ;
2018-03-27 13:15:10 -07:00
provider . GetPubKey ( keyID , vch ) ;
2016-03-31 14:54:58 +02:00
ret . push_back ( ToByteVector ( vch ) ) ;
2014-08-27 17:22:33 +02:00
}
return true ;
case TX_SCRIPTHASH :
2018-03-27 13:15:10 -07:00
if ( provider . GetCScript ( uint160 ( vSolutions [ 0 ] ) , scriptRet ) ) {
2016-03-31 14:54:58 +02:00
ret . push_back ( std : : vector < unsigned char > ( scriptRet . begin ( ) , scriptRet . end ( ) ) ) ;
return true ;
}
return false ;
2014-08-27 17:22:33 +02:00
case TX_MULTISIG :
2016-03-31 14:54:58 +02:00
ret . push_back ( valtype ( ) ) ; // workaround CHECKMULTISIG bug
2018-03-27 13:15:10 -07:00
return ( SignN ( provider , vSolutions , creator , scriptPubKey , ret , sigversion ) ) ;
2016-03-31 14:54:58 +02:00
case TX_WITNESS_V0_KEYHASH :
ret . push_back ( vSolutions [ 0 ] ) ;
return true ;
case TX_WITNESS_V0_SCRIPTHASH :
CRIPEMD160 ( ) . Write ( & vSolutions [ 0 ] [ 0 ] , vSolutions [ 0 ] . size ( ) ) . Finalize ( h160 . begin ( ) ) ;
2018-03-27 13:15:10 -07:00
if ( provider . GetCScript ( h160 , scriptRet ) ) {
2016-03-31 14:54:58 +02:00
ret . push_back ( std : : vector < unsigned char > ( scriptRet . begin ( ) , scriptRet . end ( ) ) ) ;
return true ;
}
return false ;
default :
return false ;
2014-08-27 17:22:33 +02:00
}
}
2017-01-04 14:42:14 +09:00
static CScript PushAll ( const std : : vector < valtype > & values )
2014-08-27 17:22:33 +02:00
{
2016-03-31 14:54:58 +02:00
CScript result ;
2017-06-02 03:18:57 +02:00
for ( const valtype & v : values ) {
2016-03-31 14:54:58 +02:00
if ( v . size ( ) = = 0 ) {
result < < OP_0 ;
} else if ( v . size ( ) = = 1 & & v [ 0 ] > = 1 & & v [ 0 ] < = 16 ) {
result < < CScript : : EncodeOP_N ( v [ 0 ] ) ;
} else {
result < < v ;
}
}
return result ;
}
2018-03-27 13:15:10 -07:00
bool ProduceSignature ( const SigningProvider & provider , const BaseSignatureCreator & creator , const CScript & fromPubKey , SignatureData & sigdata )
2016-03-31 14:54:58 +02:00
{
std : : vector < valtype > result ;
2014-08-27 17:22:33 +02:00
txnouttype whichType ;
2018-03-27 13:15:10 -07:00
bool solved = SignStep ( provider , creator , fromPubKey , result , whichType , SigVersion : : BASE ) ;
2016-03-31 14:54:58 +02:00
bool P2SH = false ;
CScript subscript ;
sigdata . scriptWitness . stack . clear ( ) ;
2014-08-27 17:22:33 +02:00
2016-03-31 14:54:58 +02:00
if ( solved & & whichType = = TX_SCRIPTHASH )
2014-08-27 17:22:33 +02:00
{
2016-03-31 14:54:58 +02:00
// Solver returns the subscript that needs to be evaluated;
2014-08-27 17:22:33 +02:00
// the final scriptSig is the signatures from that
// and then the serialized subscript:
2017-12-13 15:16:23 -05:00
subscript = CScript ( result [ 0 ] . begin ( ) , result [ 0 ] . end ( ) ) ;
2018-03-27 13:15:10 -07:00
solved = solved & & SignStep ( provider , creator , subscript , result , whichType , SigVersion : : BASE ) & & whichType ! = TX_SCRIPTHASH ;
2016-03-31 14:54:58 +02:00
P2SH = true ;
}
2014-08-27 17:22:33 +02:00
2016-03-31 14:54:58 +02:00
if ( solved & & whichType = = TX_WITNESS_V0_KEYHASH )
{
CScript witnessscript ;
witnessscript < < OP_DUP < < OP_HASH160 < < ToByteVector ( result [ 0 ] ) < < OP_EQUALVERIFY < < OP_CHECKSIG ;
2014-08-27 17:22:33 +02:00
txnouttype subType ;
2018-03-27 13:15:10 -07:00
solved = solved & & SignStep ( provider , creator , witnessscript , result , subType , SigVersion : : WITNESS_V0 ) ;
2016-03-31 14:54:58 +02:00
sigdata . scriptWitness . stack = result ;
result . clear ( ) ;
}
else if ( solved & & whichType = = TX_WITNESS_V0_SCRIPTHASH )
{
CScript witnessscript ( result [ 0 ] . begin ( ) , result [ 0 ] . end ( ) ) ;
txnouttype subType ;
2018-03-27 13:15:10 -07:00
solved = solved & & SignStep ( provider , creator , witnessscript , result , subType , SigVersion : : WITNESS_V0 ) & & subType ! = TX_SCRIPTHASH & & subType ! = TX_WITNESS_V0_SCRIPTHASH & & subType ! = TX_WITNESS_V0_KEYHASH ;
2016-03-31 14:54:58 +02:00
result . push_back ( std : : vector < unsigned char > ( witnessscript . begin ( ) , witnessscript . end ( ) ) ) ;
sigdata . scriptWitness . stack = result ;
result . clear ( ) ;
2014-08-27 17:22:33 +02:00
}
2016-03-31 14:54:58 +02:00
if ( P2SH ) {
result . push_back ( std : : vector < unsigned char > ( subscript . begin ( ) , subscript . end ( ) ) ) ;
}
sigdata . scriptSig = PushAll ( result ) ;
2014-08-27 17:22:33 +02:00
// Test solution
2016-03-31 14:54:58 +02:00
return solved & & VerifyScript ( sigdata . scriptSig , fromPubKey , & sigdata . scriptWitness , STANDARD_SCRIPT_VERIFY_FLAGS , creator . Checker ( ) ) ;
}
SignatureData DataFromTransaction ( const CMutableTransaction & tx , unsigned int nIn )
{
SignatureData data ;
assert ( tx . vin . size ( ) > nIn ) ;
data . scriptSig = tx . vin [ nIn ] . scriptSig ;
2016-08-04 02:49:16 +02:00
data . scriptWitness = tx . vin [ nIn ] . scriptWitness ;
2016-03-31 14:54:58 +02:00
return data ;
}
2018-03-05 16:37:24 -05:00
void UpdateInput ( CTxIn & input , const SignatureData & data )
{
input . scriptSig = data . scriptSig ;
input . scriptWitness = data . scriptWitness ;
}
2016-03-31 14:54:58 +02:00
void UpdateTransaction ( CMutableTransaction & tx , unsigned int nIn , const SignatureData & data )
{
assert ( tx . vin . size ( ) > nIn ) ;
2018-03-05 16:37:24 -05:00
UpdateInput ( tx . vin [ nIn ] , data ) ;
2014-11-04 10:06:20 -08:00
}
2018-03-17 19:19:09 -07:00
bool SignSignature ( const SigningProvider & provider , const CScript & fromPubKey , CMutableTransaction & txTo , unsigned int nIn , const CAmount & amount , int nHashType )
2014-11-04 10:06:20 -08:00
{
assert ( nIn < txTo . vin . size ( ) ) ;
CTransaction txToConst ( txTo ) ;
2018-03-27 13:15:10 -07:00
TransactionSignatureCreator creator ( & txToConst , nIn , amount , nHashType ) ;
2014-11-04 10:06:20 -08:00
2016-03-31 14:54:58 +02:00
SignatureData sigdata ;
2018-03-27 13:15:10 -07:00
bool ret = ProduceSignature ( provider , creator , fromPubKey , sigdata ) ;
2016-03-31 14:54:58 +02:00
UpdateTransaction ( txTo , nIn , sigdata ) ;
return ret ;
2014-08-27 17:22:33 +02:00
}
2018-03-17 19:19:09 -07:00
bool SignSignature ( const SigningProvider & provider , const CTransaction & txFrom , CMutableTransaction & txTo , unsigned int nIn , int nHashType )
2014-08-27 17:22:33 +02:00
{
assert ( nIn < txTo . vin . size ( ) ) ;
CTxIn & txin = txTo . vin [ nIn ] ;
assert ( txin . prevout . n < txFrom . vout . size ( ) ) ;
const CTxOut & txout = txFrom . vout [ txin . prevout . n ] ;
2018-03-17 19:19:09 -07:00
return SignSignature ( provider , txout . scriptPubKey , txTo , nIn , txout . nValue , nHashType ) ;
2014-08-27 17:22:33 +02:00
}
2017-01-04 14:42:14 +09:00
static std : : vector < valtype > CombineMultisig ( const CScript & scriptPubKey , const BaseSignatureChecker & checker ,
const std : : vector < valtype > & vSolutions ,
const std : : vector < valtype > & sigs1 , const std : : vector < valtype > & sigs2 , SigVersion sigversion )
2014-08-27 17:22:33 +02:00
{
// Combine all the signatures we've got:
2017-01-04 14:42:14 +09:00
std : : set < valtype > allsigs ;
2017-06-02 03:18:57 +02:00
for ( const valtype & v : sigs1 )
2014-08-27 17:22:33 +02:00
{
if ( ! v . empty ( ) )
allsigs . insert ( v ) ;
}
2017-06-02 03:18:57 +02:00
for ( const valtype & v : sigs2 )
2014-08-27 17:22:33 +02:00
{
if ( ! v . empty ( ) )
allsigs . insert ( v ) ;
}
// Build a map of pubkey -> signature by matching sigs to pubkeys:
assert ( vSolutions . size ( ) > 1 ) ;
unsigned int nSigsRequired = vSolutions . front ( ) [ 0 ] ;
unsigned int nPubKeys = vSolutions . size ( ) - 2 ;
2017-01-04 14:42:14 +09:00
std : : map < valtype , valtype > sigs ;
2017-06-02 03:18:57 +02:00
for ( const valtype & sig : allsigs )
2014-08-27 17:22:33 +02:00
{
for ( unsigned int i = 0 ; i < nPubKeys ; i + + )
{
const valtype & pubkey = vSolutions [ i + 1 ] ;
if ( sigs . count ( pubkey ) )
continue ; // Already got a sig for this pubkey
2016-03-31 14:54:58 +02:00
if ( checker . CheckSig ( sig , pubkey , scriptPubKey , sigversion ) )
2014-08-27 17:22:33 +02:00
{
sigs [ pubkey ] = sig ;
break ;
}
}
}
// Now build a merged CScript:
unsigned int nSigsHave = 0 ;
2016-03-31 14:54:58 +02:00
std : : vector < valtype > result ; result . push_back ( valtype ( ) ) ; // pop-one-too-many workaround
2014-08-27 17:22:33 +02:00
for ( unsigned int i = 0 ; i < nPubKeys & & nSigsHave < nSigsRequired ; i + + )
{
if ( sigs . count ( vSolutions [ i + 1 ] ) )
{
2016-03-31 14:54:58 +02:00
result . push_back ( sigs [ vSolutions [ i + 1 ] ] ) ;
2014-08-27 17:22:33 +02:00
+ + nSigsHave ;
}
}
// Fill any missing with OP_0:
for ( unsigned int i = nSigsHave ; i < nSigsRequired ; i + + )
2016-03-31 14:54:58 +02:00
result . push_back ( valtype ( ) ) ;
2014-08-27 17:22:33 +02:00
return result ;
}
2016-03-31 14:54:58 +02:00
namespace
{
struct Stacks
{
std : : vector < valtype > script ;
std : : vector < valtype > witness ;
Stacks ( ) { }
explicit Stacks ( const std : : vector < valtype > & scriptSigStack_ ) : script ( scriptSigStack_ ) , witness ( ) { }
explicit Stacks ( const SignatureData & data ) : witness ( data . scriptWitness . stack ) {
2018-03-09 15:03:40 +01:00
EvalScript ( script , data . scriptSig , SCRIPT_VERIFY_STRICTENC , BaseSignatureChecker ( ) , SigVersion : : BASE ) ;
2016-03-31 14:54:58 +02:00
}
SignatureData Output ( ) const {
SignatureData result ;
result . scriptSig = PushAll ( script ) ;
result . scriptWitness . stack = witness ;
return result ;
}
} ;
}
static Stacks CombineSignatures ( const CScript & scriptPubKey , const BaseSignatureChecker & checker ,
2017-01-04 14:42:14 +09:00
const txnouttype txType , const std : : vector < valtype > & vSolutions ,
2016-03-31 14:54:58 +02:00
Stacks sigs1 , Stacks sigs2 , SigVersion sigversion )
2014-08-27 17:22:33 +02:00
{
switch ( txType )
{
case TX_NONSTANDARD :
case TX_NULL_DATA :
2017-08-25 19:55:52 -07:00
case TX_WITNESS_UNKNOWN :
2014-08-27 17:22:33 +02:00
// Don't know anything about this, assume bigger one is correct:
2016-03-31 14:54:58 +02:00
if ( sigs1 . script . size ( ) > = sigs2 . script . size ( ) )
return sigs1 ;
return sigs2 ;
2014-08-27 17:22:33 +02:00
case TX_PUBKEY :
case TX_PUBKEYHASH :
// Signatures are bigger than placeholders or empty scripts:
2016-03-31 14:54:58 +02:00
if ( sigs1 . script . empty ( ) | | sigs1 . script [ 0 ] . empty ( ) )
return sigs2 ;
return sigs1 ;
case TX_WITNESS_V0_KEYHASH :
// Signatures are bigger than placeholders or empty scripts:
if ( sigs1 . witness . empty ( ) | | sigs1 . witness [ 0 ] . empty ( ) )
return sigs2 ;
return sigs1 ;
2014-08-27 17:22:33 +02:00
case TX_SCRIPTHASH :
2016-03-31 14:54:58 +02:00
if ( sigs1 . script . empty ( ) | | sigs1 . script . back ( ) . empty ( ) )
return sigs2 ;
else if ( sigs2 . script . empty ( ) | | sigs2 . script . back ( ) . empty ( ) )
return sigs1 ;
2014-08-27 17:22:33 +02:00
else
{
// Recur to combine:
2016-03-31 14:54:58 +02:00
valtype spk = sigs1 . script . back ( ) ;
2014-08-27 17:22:33 +02:00
CScript pubKey2 ( spk . begin ( ) , spk . end ( ) ) ;
txnouttype txType2 ;
2017-01-04 14:42:14 +09:00
std : : vector < std : : vector < unsigned char > > vSolutions2 ;
2014-08-27 17:22:33 +02:00
Solver ( pubKey2 , txType2 , vSolutions2 ) ;
2016-03-31 14:54:58 +02:00
sigs1 . script . pop_back ( ) ;
sigs2 . script . pop_back ( ) ;
Stacks result = CombineSignatures ( pubKey2 , checker , txType2 , vSolutions2 , sigs1 , sigs2 , sigversion ) ;
result . script . push_back ( spk ) ;
2014-08-27 17:22:33 +02:00
return result ;
}
case TX_MULTISIG :
2016-03-31 14:54:58 +02:00
return Stacks ( CombineMultisig ( scriptPubKey , checker , vSolutions , sigs1 . script , sigs2 . script , sigversion ) ) ;
case TX_WITNESS_V0_SCRIPTHASH :
if ( sigs1 . witness . empty ( ) | | sigs1 . witness . back ( ) . empty ( ) )
return sigs2 ;
else if ( sigs2 . witness . empty ( ) | | sigs2 . witness . back ( ) . empty ( ) )
return sigs1 ;
else
{
// Recur to combine:
CScript pubKey2 ( sigs1 . witness . back ( ) . begin ( ) , sigs1 . witness . back ( ) . end ( ) ) ;
txnouttype txType2 ;
2017-01-04 14:42:14 +09:00
std : : vector < valtype > vSolutions2 ;
2016-03-31 14:54:58 +02:00
Solver ( pubKey2 , txType2 , vSolutions2 ) ;
sigs1 . witness . pop_back ( ) ;
sigs1 . script = sigs1 . witness ;
sigs1 . witness . clear ( ) ;
sigs2 . witness . pop_back ( ) ;
sigs2 . script = sigs2 . witness ;
sigs2 . witness . clear ( ) ;
2018-03-09 15:03:40 +01:00
Stacks result = CombineSignatures ( pubKey2 , checker , txType2 , vSolutions2 , sigs1 , sigs2 , SigVersion : : WITNESS_V0 ) ;
2016-03-31 14:54:58 +02:00
result . witness = result . script ;
result . script . clear ( ) ;
result . witness . push_back ( valtype ( pubKey2 . begin ( ) , pubKey2 . end ( ) ) ) ;
return result ;
}
default :
return Stacks ( ) ;
2014-08-27 17:22:33 +02:00
}
}
2016-03-31 14:54:58 +02:00
SignatureData CombineSignatures ( const CScript & scriptPubKey , const BaseSignatureChecker & checker ,
const SignatureData & scriptSig1 , const SignatureData & scriptSig2 )
2014-08-27 17:22:33 +02:00
{
txnouttype txType ;
2017-01-04 14:42:14 +09:00
std : : vector < std : : vector < unsigned char > > vSolutions ;
2014-08-27 17:22:33 +02:00
Solver ( scriptPubKey , txType , vSolutions ) ;
2018-03-09 15:03:40 +01:00
return CombineSignatures ( scriptPubKey , checker , txType , vSolutions , Stacks ( scriptSig1 ) , Stacks ( scriptSig2 ) , SigVersion : : BASE ) . Output ( ) ;
2014-08-27 17:22:33 +02:00
}
2015-05-01 15:21:06 +02:00
namespace {
/** Dummy signature checker which accepts all signatures. */
2018-03-27 13:34:39 -07:00
class DummySignatureChecker final : public BaseSignatureChecker
2015-05-01 15:21:06 +02:00
{
public :
DummySignatureChecker ( ) { }
2018-03-27 13:34:39 -07:00
bool CheckSig ( const std : : vector < unsigned char > & scriptSig , const std : : vector < unsigned char > & vchPubKey , const CScript & scriptCode , SigVersion sigversion ) const override { return true ; }
} ;
const DummySignatureChecker DUMMY_CHECKER ;
2015-05-01 15:21:06 +02:00
2018-03-27 13:34:39 -07:00
class DummySignatureCreator final : public BaseSignatureCreator {
public :
DummySignatureCreator ( ) { }
const BaseSignatureChecker & Checker ( ) const override { return DUMMY_CHECKER ; }
bool CreateSig ( const SigningProvider & provider , std : : vector < unsigned char > & vchSig , const CKeyID & keyid , const CScript & scriptCode , SigVersion sigversion ) const override
2015-05-01 15:21:06 +02:00
{
2018-03-27 13:34:39 -07:00
// Create a dummy signature that is a valid DER-encoding
vchSig . assign ( 72 , ' \000 ' ) ;
vchSig [ 0 ] = 0x30 ;
vchSig [ 1 ] = 69 ;
vchSig [ 2 ] = 0x02 ;
vchSig [ 3 ] = 33 ;
vchSig [ 4 ] = 0x01 ;
vchSig [ 4 + 33 ] = 0x02 ;
vchSig [ 5 + 33 ] = 32 ;
vchSig [ 6 + 33 ] = 0x01 ;
vchSig [ 6 + 33 + 32 ] = SIGHASH_ALL ;
2015-05-01 15:21:06 +02:00
return true ;
}
} ;
}
2018-03-27 13:34:39 -07:00
const BaseSignatureCreator & DUMMY_SIGNATURE_CREATOR = DummySignatureCreator ( ) ;
2017-11-30 16:48:31 -08:00
2018-03-17 19:19:09 -07:00
bool IsSolvable ( const SigningProvider & provider , const CScript & script )
2017-11-30 16:48:31 -08:00
{
// This check is to make sure that the script we created can actually be solved for and signed by us
// if we were to have the private keys. This is just to make sure that the script is valid and that,
// if found in a transaction, we would still accept and relay that transaction. In particular,
// it will reject witness outputs that require signing with an uncompressed public key.
SignatureData sigs ;
// Make sure that STANDARD_SCRIPT_VERIFY_FLAGS includes SCRIPT_VERIFY_WITNESS_PUBKEYTYPE, the most
// important property this function is designed to test for.
static_assert ( STANDARD_SCRIPT_VERIFY_FLAGS & SCRIPT_VERIFY_WITNESS_PUBKEYTYPE , " IsSolvable requires standard script flags to include WITNESS_PUBKEYTYPE " ) ;
2018-03-27 13:34:39 -07:00
if ( ProduceSignature ( provider , DUMMY_SIGNATURE_CREATOR , script , sigs ) ) {
2017-11-30 16:48:31 -08:00
// VerifyScript check is just defensive, and should never fail.
2018-03-27 13:34:39 -07:00
assert ( VerifyScript ( sigs . scriptSig , script , & sigs . scriptWitness , STANDARD_SCRIPT_VERIFY_FLAGS , DUMMY_CHECKER ) ) ;
2017-11-30 16:48:31 -08:00
return true ;
}
return false ;
}