diff --git a/contrib/signet/issuer/issuer.sh b/contrib/signet/issuer/issuer.sh new file mode 100755 index 00000000000000..dbcacf1b4a71c5 --- /dev/null +++ b/contrib/signet/issuer/issuer.sh @@ -0,0 +1,76 @@ +#!/usr/bin/env bash +# Copyright (c) 2018 The Bitcoin Core developers +# Distributed under the MIT software license, see the accompanying +# file COPYING or http://www.opensource.org/licenses/mit-license.php. + +export LC_ALL=C + +# +# Issue blocks using a local node at a given interval. +# + +if [ $# -lt 3 ]; then + echo "syntax: $0 []" ; exit 1 +fi + +function log() +{ + echo "- $(date +%H:%M:%S): $*" +} + +min_time=$1 +shift +max_time=$1 +shift +bcli=$1 +shift + +# https://stackoverflow.com/questions/806906/how-do-i-test-if-a-variable-is-a-number-in-bash +re='^[0-9]+$' +if ! [[ $min_time =~ $re ]] ; then + echo "error: min_time $min_time is not a number" ; exit 1 +fi +if ! [[ $max_time =~ $re ]] ; then + echo "error: max_time $max_time is not a number" ; exit 1 +fi + +(( randinterval=max_time-min_time )) +if [ $randinterval -lt 1 ]; then + echo "error: interval min..max must be positive and greater than 0" ; exit 1 +fi + +if ! [ -e "$bcli" ]; then + which "$bcli" &> /dev/null + if [ $? -ne 0 ]; then + echo "error: unable to find bitcoin binary: $bcli" ; exit 1 + fi +fi + +echo "- checking node status" +conns=$($bcli "$@" getconnectioncount) + +if [ $? -ne 0 ]; then + echo "node error" ; exit 1 +fi + +if [ $conns -lt 1 ]; then + echo "warning: node is not connected to any other node" +fi + +log "node OK with $conns connection(s)" +log "mining in random intervals between $min_time .. $max_time seconds" +log "hit ^C to stop" + +while true; do + (( rv=$RANDOM%$randinterval )) + (( rv=rv+min_time )) + echo -n -e "- $(date +%H:%M:%S): next block in $rv seconds..." + sleep $rv + echo -n -e " [submit]" + blockhash=$($bcli "$@" getnewblockhex true) + if [ $? -ne 0 ]; then + echo "node error; aborting" ; exit 1 + fi + echo "" + log "broadcasting block $($bcli "$@" getblockcount) $blockhash to $($bcli "$@" getconnectioncount) peer(s)" +done diff --git a/src/chain.h b/src/chain.h index 2b6d2d082cb237..bd0e85c52fb1e6 100644 --- a/src/chain.h +++ b/src/chain.h @@ -383,6 +383,16 @@ class CDiskBlockIndex : public CBlockIndex READWRITE(nTime); READWRITE(nBits); READWRITE(nNonce); + if (g_solution_blocks && !(s.GetType() & SER_GETHASH)) { + uint256 hash = GetBlockHash(); + READWRITE(g_blockheader_payload_map[hash]); + size_t len = GetSizeOfCompactSize(g_blockheader_payload_map[hash].size()) + g_blockheader_payload_map[hash].size(); + while (len < g_solution_block_len) { + uint8_t padding = 0; + READWRITE(padding); + len++; + } + } } uint256 GetBlockHash() const diff --git a/src/chainparams.cpp b/src/chainparams.cpp index b8e0ea23dd5a45..40c5a6cbf4440e 100644 --- a/src/chainparams.cpp +++ b/src/chainparams.cpp @@ -17,13 +17,15 @@ #include #include -static CBlock CreateGenesisBlock(const char* pszTimestamp, const CScript& genesisOutputScript, uint32_t nTime, uint32_t nNonce, uint32_t nBits, int32_t nVersion, const CAmount& genesisReward) +#include + +static CBlock CreateGenesisBlock(const CScript& coinbase_sig, const CScript& genesisOutputScript, uint32_t nTime, uint32_t nNonce, uint32_t nBits, int32_t nVersion, const CAmount& genesisReward) { CMutableTransaction txNew; txNew.nVersion = 1; txNew.vin.resize(1); txNew.vout.resize(1); - txNew.vin[0].scriptSig = CScript() << 486604799 << CScriptNum(4) << std::vector((const unsigned char*)pszTimestamp, (const unsigned char*)pszTimestamp + strlen(pszTimestamp)); + txNew.vin[0].scriptSig = coinbase_sig; txNew.vout[0].nValue = genesisReward; txNew.vout[0].scriptPubKey = genesisOutputScript; @@ -38,6 +40,12 @@ static CBlock CreateGenesisBlock(const char* pszTimestamp, const CScript& genesi return genesis; } +static CBlock CreateGenesisBlock(const char* pszTimestamp, const CScript& genesisOutputScript, uint32_t nTime, uint32_t nNonce, uint32_t nBits, int32_t nVersion, const CAmount& genesisReward) +{ + CScript coinbase_sig = CScript() << 486604799 << CScriptNum(4) << std::vector((const unsigned char*)pszTimestamp, (const unsigned char*)pszTimestamp + strlen(pszTimestamp)); + return CreateGenesisBlock(coinbase_sig, genesisOutputScript, nTime, nNonce, nBits, nVersion, genesisReward); +} + /** * Build the genesis block. Note that the output of its generation * transaction cannot be spent since it did not originally exist in the @@ -268,6 +276,88 @@ class CTestNetParams : public CChainParams { } }; +/** + * SigNet version 2018-07-20 + */ +class SigNetParams : public CChainParams { +public: + SigNetParams(const ArgsManager& args) { + if (!args.IsArgSet("-signet_blockscript") || !args.IsArgSet("-signet_siglen")) { + throw std::runtime_error(strprintf("%s: -signet_blockscript and -signet_siglen are both mandatory for signet networks", __func__)); + } + if (args.GetArgs("-signet_blockscript").size() != 1) { + throw std::runtime_error(strprintf("%s: -signet_blockscript cannot be multiple values.", __func__)); + } + if (args.GetArgs("-signet_siglen").size() != 1) { + throw std::runtime_error(strprintf("%s: -signet_siglen cannot be multiple values.", __func__)); + } + + LogPrintf("SigNet with block script %s\n", gArgs.GetArgs("-signet_blockscript")[0]); + + strNetworkID = "signet"; + consensus.signature_pow = true; + consensus.blockscript = ParseHex(args.GetArgs("-signet_blockscript")[0]); + consensus.siglen = g_solution_block_len = atoi64(args.GetArgs("-signet_siglen")[0]); + consensus.nSubsidyHalvingInterval = 210000; + consensus.BIP34Height = 1; + consensus.BIP65Height = 1; + consensus.BIP66Height = 1; + consensus.nPowTargetTimespan = 14 * 24 * 60 * 60; // two weeks + consensus.nPowTargetSpacing = 10 * 60; + consensus.fPowAllowMinDifficultyBlocks = false; + consensus.fPowNoRetargeting = false; + consensus.nRuleChangeActivationThreshold = 1916; + consensus.nMinerConfirmationWindow = 2016; + consensus.powLimit = uint256S("00000000ffffffffffffffffffffffffffffffffffffffffffffffffffffffff"); + consensus.vDeployments[Consensus::DEPLOYMENT_TESTDUMMY].bit = 28; + consensus.vDeployments[Consensus::DEPLOYMENT_TESTDUMMY].nStartTime = 1539478800; + consensus.vDeployments[Consensus::DEPLOYMENT_TESTDUMMY].nTimeout = Consensus::BIP9Deployment::NO_TIMEOUT; + + // Deployment of BIP68, BIP112, and BIP113. + consensus.vDeployments[Consensus::DEPLOYMENT_CSV].bit = 0; + consensus.vDeployments[Consensus::DEPLOYMENT_CSV].nStartTime = 1539478800; + consensus.vDeployments[Consensus::DEPLOYMENT_CSV].nTimeout = Consensus::BIP9Deployment::NO_TIMEOUT; + + // Deployment of SegWit (BIP141, BIP143, and BIP147) + consensus.vDeployments[Consensus::DEPLOYMENT_SEGWIT].bit = 1; + consensus.vDeployments[Consensus::DEPLOYMENT_SEGWIT].nStartTime = 1539478800; + consensus.vDeployments[Consensus::DEPLOYMENT_SEGWIT].nTimeout = Consensus::BIP9Deployment::NO_TIMEOUT; + + pchMessageStart[0] = 0xf0; + pchMessageStart[1] = 0xc7; + pchMessageStart[2] = 0x70; + pchMessageStart[3] = 0x6a; + nDefaultPort = 38333; + nPruneAfterHeight = 1000; + + CHashWriter h(SER_DISK, 0); + h << consensus.blockscript << consensus.siglen; + uint256 hash = h.GetHash(); + CScript coinbase_sig = CScript() << std::vector(hash.begin(), hash.end()); + CScript genesis_out = CScript() << OP_RETURN; + genesis = CreateGenesisBlock(coinbase_sig, genesis_out, 1534313275, 0, 0x1d00ffff, 1, 50 * COIN); + consensus.hashGenesisBlock = genesis.GetHash(); + + vFixedSeeds.clear(); + vSeeds.clear(); + if (args.IsArgSet("-signet_seednode")) { + vSeeds = gArgs.GetArgs("-signet_seednode"); + } + + base58Prefixes[PUBKEY_ADDRESS] = std::vector{125}; + base58Prefixes[SCRIPT_ADDRESS] = std::vector{87}; + base58Prefixes[SECRET_KEY] = std::vector{217}; + base58Prefixes[EXT_PUBLIC_KEY] = {0x04, 0x35, 0x87, 0xCF}; + base58Prefixes[EXT_SECRET_KEY] = {0x04, 0x35, 0x83, 0x94}; + + bech32_hrp = "sb"; + + fDefaultConsistencyChecks = false; + fRequireStandard = false; + fMineBlocksOnDemand = false; + } +}; + /** * Regression test */ @@ -409,6 +499,10 @@ std::unique_ptr CreateChainParams(const std::string& chain) return std::unique_ptr(new CTestNetParams()); else if (chain == CBaseChainParams::REGTEST) return std::unique_ptr(new CRegTestParams(gArgs)); + else if (chain == CBaseChainParams::SIGNET) { + g_solution_blocks = true; + return std::unique_ptr(new SigNetParams(gArgs)); + } throw std::runtime_error(strprintf("%s: Unknown chain %s.", __func__, chain)); } diff --git a/src/chainparamsbase.cpp b/src/chainparamsbase.cpp index f0559a319acb7c..5ee50ad19bcf5b 100644 --- a/src/chainparamsbase.cpp +++ b/src/chainparamsbase.cpp @@ -13,6 +13,7 @@ const std::string CBaseChainParams::MAIN = "main"; const std::string CBaseChainParams::TESTNET = "test"; +const std::string CBaseChainParams::SIGNET = "signet"; const std::string CBaseChainParams::REGTEST = "regtest"; void SetupChainParamsBaseOptions() @@ -21,6 +22,10 @@ void SetupChainParamsBaseOptions() "This is intended for regression testing tools and app development.", true, OptionsCategory::CHAINPARAMS); gArgs.AddArg("-testnet", "Use the test chain", false, OptionsCategory::CHAINPARAMS); gArgs.AddArg("-vbparams=deployment:start:end", "Use given start/end times for specified version bits deployment (regtest-only)", true, OptionsCategory::CHAINPARAMS); + gArgs.AddArg("-signet", "Use the signet chain. Note that the network is defined by the signet_blockscript and signet_siglen parameters", false, OptionsCategory::CHAINPARAMS); + gArgs.AddArg("-signet_blockscript", "Blocks must satisfy the given script to be considered valid (only for -signet networks)", false, OptionsCategory::CHAINPARAMS); + gArgs.AddArg("-signet_siglen", "The length of the signature must be exactly this long (padded to this length, if shorter). All block headers in this network are of length 80 + this value", false, OptionsCategory::CHAINPARAMS); + gArgs.AddArg("-signet_seednode", "Specify a seed node for the signet network (may be used multiple times to specify multiple seed nodes)", false, OptionsCategory::CHAINPARAMS); } static std::unique_ptr globalChainBaseParams; @@ -39,8 +44,9 @@ std::unique_ptr CreateBaseChainParams(const std::string& chain return MakeUnique("testnet3", 18332); else if (chain == CBaseChainParams::REGTEST) return MakeUnique("regtest", 18443); - else - throw std::runtime_error(strprintf("%s: Unknown chain %s.", __func__, chain)); + else if (chain == CBaseChainParams::SIGNET) + return MakeUnique("signet", 38332); + throw std::runtime_error(strprintf("%s: Unknown chain %s.", __func__, chain)); } void SelectBaseParams(const std::string& chain) diff --git a/src/chainparamsbase.h b/src/chainparamsbase.h index 355df043d39e45..a960defbcf430f 100644 --- a/src/chainparamsbase.h +++ b/src/chainparamsbase.h @@ -16,9 +16,10 @@ class CBaseChainParams { public: - /** BIP70 chain name strings (main, test or regtest) */ + /** BIP70 chain name strings (main, test, sigtest or regtest) */ static const std::string MAIN; static const std::string TESTNET; + static const std::string SIGNET; static const std::string REGTEST; const std::string& DataDir() const { return strDataDir; } diff --git a/src/consensus/params.h b/src/consensus/params.h index 6c3a201f4f5fb7..560317d4c34908 100644 --- a/src/consensus/params.h +++ b/src/consensus/params.h @@ -75,7 +75,12 @@ struct Params { int64_t DifficultyAdjustmentInterval() const { return nPowTargetTimespan / nPowTargetSpacing; } uint256 nMinimumChainWork; uint256 defaultAssumeValid; + + bool signature_pow{false}; + std::vector blockscript; + uint32_t siglen; }; + } // namespace Consensus #endif // BITCOIN_CONSENSUS_PARAMS_H diff --git a/src/miner.cpp b/src/miner.cpp index 80a2f8f018073a..9e04af22d78b26 100644 --- a/src/miner.cpp +++ b/src/miner.cpp @@ -47,6 +47,10 @@ int64_t UpdateTime(CBlockHeader* pblock, const Consensus::Params& consensusParam BlockAssembler::Options::Options() { blockMinFeeRate = CFeeRate(DEFAULT_BLOCK_MIN_TX_FEE); nBlockMaxWeight = DEFAULT_BLOCK_MAX_WEIGHT; + + // Make room for the signature in the block header, if this is a signet + // block + if (g_solution_blocks) nBlockMaxWeight -= g_solution_block_len; } BlockAssembler::BlockAssembler(const CChainParams& params, const Options& options) : chainparams(params) diff --git a/src/policy/policy.cpp b/src/policy/policy.cpp index d4cc538492b322..49b4baa9f8aa2a 100644 --- a/src/policy/policy.cpp +++ b/src/policy/policy.cpp @@ -14,6 +14,7 @@ #include #include +unsigned int GetStandardScriptVerifyFlags() { return STANDARD_SCRIPT_VERIFY_FLAGS; } CAmount GetDustThreshold(const CTxOut& txout, const CFeeRate& dustRelayFeeIn) { diff --git a/src/policy/policy.h b/src/policy/policy.h index 3d47ac12676257..28e9eb1877b2ad 100644 --- a/src/policy/policy.h +++ b/src/policy/policy.h @@ -68,6 +68,8 @@ static constexpr unsigned int STANDARD_SCRIPT_VERIFY_FLAGS = MANDATORY_SCRIPT_VE SCRIPT_VERIFY_WITNESS_PUBKEYTYPE | SCRIPT_VERIFY_CONST_SCRIPTCODE; +unsigned int GetStandardScriptVerifyFlags(); + /** For convenience, standard but not mandatory verify flags. */ static constexpr unsigned int STANDARD_NOT_MANDATORY_VERIFY_FLAGS = STANDARD_SCRIPT_VERIFY_FLAGS & ~MANDATORY_SCRIPT_VERIFY_FLAGS; diff --git a/src/pow.cpp b/src/pow.cpp index 1414d375649254..ef7dfba9e0ccb8 100644 --- a/src/pow.cpp +++ b/src/pow.cpp @@ -9,6 +9,9 @@ #include #include #include +#include