Branch data Line data Source code
1 : : // Copyright (c) 2009-2022 The Bitcoin Core developers
2 : : // Distributed under the MIT software license, see the accompanying
3 : : // file COPYING or http://www.opensource.org/licenses/mit-license.php.
4 : :
5 : : #include <bitcoin-build-config.h> // IWYU pragma: keep
6 : :
7 : : #include <arith_uint256.h>
8 : : #include <chain.h>
9 : : #include <chainparams.h>
10 : : #include <chainparamsbase.h>
11 : : #include <clientversion.h>
12 : : #include <common/args.h>
13 : : #include <common/system.h>
14 : : #include <compat/compat.h>
15 : : #include <core_io.h>
16 : : #include <streams.h>
17 : : #include <util/exception.h>
18 : : #include <util/strencodings.h>
19 : : #include <util/translation.h>
20 : :
21 : : #include <atomic>
22 : : #include <cstdio>
23 : : #include <functional>
24 : : #include <memory>
25 : : #include <thread>
26 : :
27 : : static const int CONTINUE_EXECUTION=-1;
28 : :
29 : : const TranslateFn G_TRANSLATION_FUN{nullptr};
30 : :
31 : 5 : static void SetupBitcoinUtilArgs(ArgsManager &argsman)
32 : : {
33 : 5 : SetupHelpOptions(argsman);
34 : :
35 [ + - + - ]: 10 : argsman.AddArg("-version", "Print version and exit", ArgsManager::ALLOW_ANY, OptionsCategory::OPTIONS);
36 : :
37 [ + - + - ]: 10 : argsman.AddCommand("grind", "Perform proof of work on hex header string");
38 : :
39 : 5 : SetupChainParamsBaseOptions(argsman);
40 : 5 : }
41 : :
42 : : // This function returns either one of EXIT_ codes when it's expected to stop the process or
43 : : // CONTINUE_EXECUTION when it's expected to continue further.
44 : 5 : static int AppInitUtil(ArgsManager& args, int argc, char* argv[])
45 : : {
46 : 5 : SetupBitcoinUtilArgs(args);
47 [ + - ]: 5 : std::string error;
48 [ + - + + ]: 5 : if (!args.ParseParameters(argc, argv, error)) {
49 [ + - ]: 2 : tfm::format(std::cerr, "Error parsing command line arguments: %s\n", error);
50 : : return EXIT_FAILURE;
51 : : }
52 : :
53 [ + - + - : 6 : if (HelpRequested(args) || args.GetBoolArg("-version", false)) {
+ - + - +
- - + ]
54 : : // First part of help message is specific to this utility
55 [ # # # # ]: 0 : std::string strUsage = CLIENT_NAME " bitcoin-util utility version " + FormatFullVersion() + "\n";
56 : :
57 [ # # # # : 0 : if (args.GetBoolArg("-version", false)) {
# # ]
58 [ # # # # ]: 0 : strUsage += FormatParagraph(LicenseInfo());
59 : : } else {
60 [ # # ]: 0 : strUsage += "\n"
61 : : "The bitcoin-util tool provides bitcoin related functionality that does not rely on the ability to access a running node. Available [commands] are listed below.\n"
62 : : "\n"
63 : : "Usage: bitcoin-util [options] [command]\n"
64 : 0 : "or: bitcoin-util [options] grind <hex-block-header>\n";
65 [ # # # # ]: 0 : strUsage += "\n" + args.GetHelpMessage();
66 : : }
67 : :
68 [ # # ]: 0 : tfm::format(std::cout, "%s", strUsage);
69 : :
70 [ # # ]: 0 : if (argc < 2) {
71 [ # # ]: 0 : tfm::format(std::cerr, "Error: too few parameters\n");
72 : : return EXIT_FAILURE;
73 : : }
74 : : return EXIT_SUCCESS;
75 : 0 : }
76 : :
77 : : // Check for chain settings (Params() calls are only valid after this clause)
78 : 3 : try {
79 [ + - + - ]: 3 : SelectParams(args.GetChainType());
80 [ - - ]: 0 : } catch (const std::exception& e) {
81 [ - - ]: 0 : tfm::format(std::cerr, "Error: %s\n", e.what());
82 : 0 : return EXIT_FAILURE;
83 : 0 : }
84 : :
85 : : return CONTINUE_EXECUTION;
86 : 5 : }
87 : :
88 : 0 : static void grind_task(uint32_t nBits, CBlockHeader header, uint32_t offset, uint32_t step, std::atomic<bool>& found, uint32_t& proposed_nonce)
89 : : {
90 : 0 : arith_uint256 target;
91 : 0 : bool neg, over;
92 : 0 : target.SetCompact(nBits, &neg, &over);
93 [ # # # # : 0 : if (target == 0 || neg || over) return;
# # ]
94 : 0 : header.nNonce = offset;
95 : :
96 : 0 : uint32_t finish = std::numeric_limits<uint32_t>::max() - step;
97 : 0 : finish = finish - (finish % step) + offset;
98 : :
99 [ # # # # ]: 0 : while (!found && header.nNonce < finish) {
100 [ # # ]: 0 : const uint32_t next = (finish - header.nNonce < 5000*step) ? finish : header.nNonce + 5000*step;
101 : 0 : do {
102 [ # # ]: 0 : if (UintToArith256(header.GetHash()) <= target) {
103 [ # # ]: 0 : if (!found.exchange(true)) {
104 : 0 : proposed_nonce = header.nNonce;
105 : : }
106 : 0 : return;
107 : : }
108 : 0 : header.nNonce += step;
109 [ # # ]: 0 : } while(header.nNonce != next);
110 : : }
111 : : }
112 : :
113 : 3 : static int Grind(const std::vector<std::string>& args, std::string& strPrint)
114 : : {
115 [ + + ]: 3 : if (args.size() != 1) {
116 : 2 : strPrint = "Must specify block header to grind";
117 : 2 : return EXIT_FAILURE;
118 : : }
119 : :
120 : 1 : CBlockHeader header;
121 [ + - ]: 1 : if (!DecodeHexBlockHeader(header, args[0])) {
122 : 1 : strPrint = "Could not decode block header";
123 : 1 : return EXIT_FAILURE;
124 : : }
125 : :
126 : 0 : uint32_t nBits = header.nBits;
127 : 0 : std::atomic<bool> found{false};
128 : 0 : uint32_t proposed_nonce{};
129 : :
130 : 0 : std::vector<std::thread> threads;
131 [ # # ]: 0 : int n_tasks = std::max(1u, std::thread::hardware_concurrency());
132 [ # # ]: 0 : threads.reserve(n_tasks);
133 [ # # ]: 0 : for (int i = 0; i < n_tasks; ++i) {
134 [ # # ]: 0 : threads.emplace_back(grind_task, nBits, header, i, n_tasks, std::ref(found), std::ref(proposed_nonce));
135 : : }
136 [ # # ]: 0 : for (auto& t : threads) {
137 [ # # ]: 0 : t.join();
138 : : }
139 [ # # ]: 0 : if (found) {
140 : 0 : header.nNonce = proposed_nonce;
141 : : } else {
142 [ # # ]: 0 : strPrint = "Could not satisfy difficulty target";
143 : : return EXIT_FAILURE;
144 : : }
145 : :
146 : 0 : DataStream ss{};
147 [ # # ]: 0 : ss << header;
148 [ # # ]: 0 : strPrint = HexStr(ss);
149 : 0 : return EXIT_SUCCESS;
150 : 0 : }
151 : :
152 : 5 : MAIN_FUNCTION
153 : : {
154 : 5 : ArgsManager& args = gArgs;
155 : 5 : SetupEnvironment();
156 : :
157 : 5 : try {
158 [ + - ]: 5 : int ret = AppInitUtil(args, argc, argv);
159 [ + + ]: 5 : if (ret != CONTINUE_EXECUTION) {
160 : : return ret;
161 : : }
162 [ - - ]: 0 : } catch (const std::exception& e) {
163 [ - - ]: 0 : PrintExceptionContinue(&e, "AppInitUtil()");
164 : 0 : return EXIT_FAILURE;
165 : 0 : } catch (...) {
166 [ - - ]: 0 : PrintExceptionContinue(nullptr, "AppInitUtil()");
167 : 0 : return EXIT_FAILURE;
168 : 0 : }
169 : :
170 : 3 : const auto cmd = args.GetCommand();
171 [ - + ]: 3 : if (!cmd) {
172 [ # # ]: 0 : tfm::format(std::cerr, "Error: must specify a command\n");
173 : : return EXIT_FAILURE;
174 : : }
175 : :
176 : 3 : int ret = EXIT_FAILURE;
177 [ + - ]: 3 : std::string strPrint;
178 : 3 : try {
179 [ + - ]: 3 : if (cmd->command == "grind") {
180 [ + - ]: 3 : ret = Grind(cmd->args, strPrint);
181 : : } else {
182 : 0 : assert(false); // unknown command should be caught earlier
183 : : }
184 [ - - ]: 0 : } catch (const std::exception& e) {
185 [ - - ]: 0 : strPrint = std::string("error: ") + e.what();
186 : 0 : } catch (...) {
187 [ - - ]: 0 : strPrint = "unknown error";
188 [ - - ]: 0 : }
189 : :
190 [ + - ]: 3 : if (strPrint != "") {
191 [ + - + - ]: 6 : tfm::format(ret == 0 ? std::cout : std::cerr, "%s\n", strPrint);
192 : : }
193 : :
194 : 3 : return ret;
195 : 6 : }
|