2 This file is part of GNUnet.
3 Copyright (C) 2014 GNUnet e.V.
5 GNUnet is free software: you can redistribute it and/or modify it
6 under the terms of the GNU Affero General Public License as published
7 by the Free Software Foundation, either version 3 of the License,
8 or (at your option) any later version.
10 GNUnet is distributed in the hope that it will be useful, but
11 WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Affero General Public License for more details.
15 You should have received a copy of the GNU Affero General Public License
16 along with this program. If not, see <http://www.gnu.org/licenses/>.
18 SPDX-License-Identifier: AGPL3.0-or-later
21 * @file util/gnunet-scrypt.c
22 * @brief tool to manipulate SCRYPT proofs of work.
26 #include "gnunet_util_lib.h"
30 * Amount of work required (W-bit collisions) for NSE proofs, in collision-bits.
32 static unsigned long long nse_work_required;
35 * Interval between proof find runs.
37 static struct GNUNET_TIME_Relative proof_find_delay;
39 static struct GNUNET_CRYPTO_EddsaPublicKey pub;
41 static uint64_t proof;
43 static struct GNUNET_SCHEDULER_Task *proof_task;
45 static const struct GNUNET_CONFIGURATION_Handle *cfg;
53 * Write our current proof to disk.
58 shutdown_task (void *cls)
61 GNUNET_DISK_fn_write (pwfn,
64 GNUNET_DISK_PERM_USER_READ |
65 GNUNET_DISK_PERM_USER_WRITE))
66 GNUNET_log_strerror_file (GNUNET_ERROR_TYPE_WARNING,
73 * Calculate the 'proof-of-work' hash (an expensive hash).
75 * @param buf data to hash
76 * @param buf_len number of bytes in @a buf
77 * @param result where to write the resulting hash
80 pow_hash (const void *buf,
82 struct GNUNET_HashCode *result)
85 gcry_kdf_derive (buf, buf_len,
88 "gnunet-proof-of-work", strlen ("gnunet-proof-of-work"),
89 2 /* iterations; keep cost of individual op small */,
90 sizeof (struct GNUNET_HashCode), result));
95 * Count the leading zeroes in hash.
97 * @param hash to count leading zeros in
98 * @return the number of leading zero bits.
101 count_leading_zeroes (const struct GNUNET_HashCode *hash)
103 unsigned int hash_count;
106 while (0 == GNUNET_CRYPTO_hash_get_bit (hash, hash_count))
113 * Find our proof of work.
115 * @param cls closure (unused)
116 * @param tc task context
119 find_proof (void *cls)
121 #define ROUND_SIZE 10
123 char buf[sizeof (struct GNUNET_CRYPTO_EddsaPublicKey) +
124 sizeof (uint64_t)] GNUNET_ALIGN;
125 struct GNUNET_HashCode result;
127 struct GNUNET_TIME_Absolute timestamp;
128 struct GNUNET_TIME_Relative elapsed;
130 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
131 "Got Proof of Work %llu\n",
132 (unsigned long long) proof);
134 GNUNET_memcpy (&buf[sizeof (uint64_t)], &pub,
135 sizeof (struct GNUNET_CRYPTO_EddsaPublicKey));
138 timestamp = GNUNET_TIME_absolute_get ();
139 while ((counter != UINT64_MAX) && (i < ROUND_SIZE))
141 GNUNET_memcpy (buf, &counter, sizeof (uint64_t));
142 pow_hash (buf, sizeof (buf), &result);
143 if (nse_work_required <= count_leading_zeroes (&result))
146 FPRINTF (stdout, "Proof of work found: %llu!\n",
147 (unsigned long long) proof);
148 GNUNET_SCHEDULER_shutdown ();
154 elapsed = GNUNET_TIME_absolute_get_duration (timestamp);
155 elapsed = GNUNET_TIME_relative_divide (elapsed, ROUND_SIZE);
156 GNUNET_log (GNUNET_ERROR_TYPE_INFO,
157 "Current: %llu [%s/proof]\n",
158 (unsigned long long) counter,
159 GNUNET_STRINGS_relative_time_to_string (elapsed, 0));
160 if (proof / (100 * ROUND_SIZE) < counter / (100 * ROUND_SIZE))
162 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Testing proofs currently at %llu\n",
163 (unsigned long long) counter);
164 /* remember progress every 100 rounds */
166 shutdown_task (NULL);
173 GNUNET_SCHEDULER_add_delayed_with_priority (proof_find_delay,
174 GNUNET_SCHEDULER_PRIORITY_IDLE,
180 * Main function that will be run by the scheduler.
183 * @param args remaining command-line arguments
184 * @param cfgfile name of the configuration file used (for saving, can be NULL!)
185 * @param cfg configuration
191 const struct GNUNET_CONFIGURATION_Handle *config)
193 struct GNUNET_CRYPTO_EddsaPrivateKey *pk;
197 /* load proof of work */
201 GNUNET_CONFIGURATION_get_value_filename (cfg,
206 GNUNET_log_config_missing (GNUNET_ERROR_TYPE_ERROR,
209 GNUNET_SCHEDULER_shutdown ();
213 GNUNET_log (GNUNET_ERROR_TYPE_INFO,
214 "Proof of Work file: %s\n",
216 if ( (GNUNET_YES != GNUNET_DISK_file_test (pwfn)) ||
218 GNUNET_DISK_fn_read (pwfn, &proof, sizeof (proof))))
221 /* load private key */
225 GNUNET_CONFIGURATION_get_value_filename (cfg,
230 GNUNET_log_config_missing (GNUNET_ERROR_TYPE_ERROR,
236 GNUNET_log (GNUNET_ERROR_TYPE_INFO,
237 "Private Key file: %s\n",
239 if (NULL == (pk = GNUNET_CRYPTO_eddsa_key_create_from_file (pkfn)))
242 _("Loading hostkey from `%s' failed.\n"),
248 GNUNET_CRYPTO_eddsa_key_get_public (pk, &pub);
250 pids = GNUNET_CRYPTO_eddsa_public_key_to_string (&pub);
251 GNUNET_log (GNUNET_ERROR_TYPE_INFO,
256 /* get target bit amount */
257 if (0 == nse_work_required)
260 GNUNET_CONFIGURATION_get_value_number (cfg,
265 GNUNET_log_config_missing (GNUNET_ERROR_TYPE_ERROR,
268 GNUNET_SCHEDULER_shutdown ();
271 if (nse_work_required >= sizeof (struct GNUNET_HashCode) * 8)
273 GNUNET_log_config_invalid (GNUNET_ERROR_TYPE_ERROR,
276 _("Value is too large.\n"));
277 GNUNET_SCHEDULER_shutdown ();
280 else if (0 == nse_work_required)
282 GNUNET_SCHEDULER_shutdown ();
286 GNUNET_log (GNUNET_ERROR_TYPE_INFO,
290 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
291 "Delay between tries: %s\n",
292 GNUNET_STRINGS_relative_time_to_string (proof_find_delay, 1));
294 GNUNET_SCHEDULER_add_with_priority (GNUNET_SCHEDULER_PRIORITY_IDLE,
297 GNUNET_SCHEDULER_add_shutdown (&shutdown_task,
303 * Program to manipulate ECC key files.
305 * @param argc number of arguments from the command line
306 * @param argv command line arguments
307 * @return 0 ok, 1 on error
313 struct GNUNET_GETOPT_CommandLineOption options[] = {
314 GNUNET_GETOPT_option_ulong ('b',
317 gettext_noop ("number of bits to require for the proof of work"),
319 GNUNET_GETOPT_option_filename ('k',
322 gettext_noop ("file with private key, otherwise default is used"),
324 GNUNET_GETOPT_option_filename ('o',
327 gettext_noop ("file with proof of work, otherwise default is used"),
329 GNUNET_GETOPT_option_relative_time ('t',
332 gettext_noop ("time to wait between calculations"),
334 GNUNET_GETOPT_OPTION_END
339 GNUNET_STRINGS_get_utf8_args (argc, argv, &argc, &argv))
343 GNUNET_PROGRAM_run (argc, argv,
344 "gnunet-scrypt [OPTIONS] prooffile",
345 gettext_noop ("Manipulate GNUnet proof of work files"),
349 GNUNET_free ((void*) argv);
350 GNUNET_free_non_null (pwfn);
354 /* end of gnunet-scrypt.c */