2018-12-11 21:00:30 +01:00
|
|
|
/*
|
|
|
|
Copyright (c) 2018 tevador
|
|
|
|
|
|
|
|
This file is part of RandomX.
|
|
|
|
|
|
|
|
RandomX is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
RandomX is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with RandomX. If not, see<http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
//#define TRACE
|
|
|
|
#include "InterpretedVirtualMachine.hpp"
|
2018-12-13 23:11:55 +01:00
|
|
|
#include "CompiledVirtualMachine.hpp"
|
|
|
|
#include "AssemblyGeneratorX86.hpp"
|
2018-12-11 21:00:30 +01:00
|
|
|
#include "Stopwatch.hpp"
|
|
|
|
#include "blake2/blake2.h"
|
|
|
|
#include <fstream>
|
|
|
|
#include <iostream>
|
|
|
|
#include <iomanip>
|
|
|
|
#include <exception>
|
|
|
|
#include <cstring>
|
|
|
|
#include "Program.hpp"
|
|
|
|
#include <string>
|
2018-12-19 21:54:44 +01:00
|
|
|
#include <thread>
|
|
|
|
#include <atomic>
|
|
|
|
#include "dataset.hpp"
|
|
|
|
#include "Cache.hpp"
|
2019-02-04 17:07:00 +01:00
|
|
|
#include "hashAes1Rx4.hpp"
|
2018-12-11 21:00:30 +01:00
|
|
|
|
|
|
|
const uint8_t seed[32] = { 191, 182, 222, 175, 249, 89, 134, 104, 241, 68, 191, 62, 162, 166, 61, 64, 123, 191, 227, 193, 118, 60, 188, 53, 223, 133, 175, 24, 123, 230, 55, 74 };
|
|
|
|
|
2019-02-11 18:57:42 +01:00
|
|
|
const uint8_t blockTemplate__[] = {
|
|
|
|
0x07, 0x07, 0xf7, 0xa4, 0xf0, 0xd6, 0x05, 0xb3, 0x03, 0x26, 0x08, 0x16, 0xba, 0x3f, 0x10, 0x90, 0x2e, 0x1a, 0x14,
|
|
|
|
0x5a, 0xc5, 0xfa, 0xd3, 0xaa, 0x3a, 0xf6, 0xea, 0x44, 0xc1, 0x18, 0x69, 0xdc, 0x4f, 0x85, 0x3f, 0x00, 0x2b, 0x2e,
|
|
|
|
0xea, 0x00, 0x00, 0x00, 0x00, 0x77, 0xb2, 0x06, 0xa0, 0x2c, 0xa5, 0xb1, 0xd4, 0xce, 0x6b, 0xbf, 0xdf, 0x0a, 0xca,
|
|
|
|
0xc3, 0x8b, 0xde, 0xd3, 0x4d, 0x2d, 0xcd, 0xee, 0xf9, 0x5c, 0xd2, 0x0c, 0xef, 0xc1, 0x2f, 0x61, 0xd5, 0x61, 0x09
|
|
|
|
};
|
|
|
|
|
2018-12-11 21:00:30 +01:00
|
|
|
void dump(const char* buffer, uint64_t count, const char* name) {
|
|
|
|
std::ofstream fout(name, std::ios::out | std::ios::binary);
|
|
|
|
fout.write(buffer, count);
|
|
|
|
fout.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
constexpr char hexmap[] = "0123456789abcdef";
|
|
|
|
void outputHex(std::ostream& os, const char* data, int length) {
|
|
|
|
for (int i = 0; i < length; ++i) {
|
|
|
|
os << hexmap[(data[i] & 0xF0) >> 4];
|
|
|
|
os << hexmap[data[i] & 0x0F];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void readOption(const char* option, int argc, char** argv, bool& out) {
|
|
|
|
for (int i = 0; i < argc; ++i) {
|
|
|
|
if (strcmp(argv[i], option) == 0) {
|
|
|
|
out = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
out = false;
|
|
|
|
}
|
|
|
|
|
2018-12-19 21:54:44 +01:00
|
|
|
void readIntOption(const char* option, int argc, char** argv, int& out, int defaultValue) {
|
|
|
|
for (int i = 0; i < argc - 1; ++i) {
|
|
|
|
if (strcmp(argv[i], option) == 0 && (out = atoi(argv[i + 1])) > 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
out = defaultValue;
|
|
|
|
}
|
|
|
|
|
2018-12-11 21:00:30 +01:00
|
|
|
void readInt(int argc, char** argv, int& out, int defaultValue) {
|
|
|
|
for (int i = 0; i < argc; ++i) {
|
|
|
|
if (*argv[i] != '-' && (out = atoi(argv[i])) > 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
out = defaultValue;
|
|
|
|
}
|
|
|
|
|
2018-12-19 21:54:44 +01:00
|
|
|
class AtomicHash {
|
|
|
|
public:
|
|
|
|
AtomicHash() {
|
|
|
|
for (int i = 0; i < 4; ++i)
|
|
|
|
hash[i].store(0);
|
|
|
|
}
|
|
|
|
void xorWith(uint64_t update[4]) {
|
|
|
|
for (int i = 0; i < 4; ++i)
|
|
|
|
hash[i].fetch_xor(update[i]);
|
|
|
|
}
|
|
|
|
void print(std::ostream& os) {
|
|
|
|
for (int i = 0; i < 4; ++i)
|
|
|
|
print(hash[i], os);
|
|
|
|
os << std::endl;
|
|
|
|
}
|
|
|
|
private:
|
|
|
|
void print(std::atomic<uint64_t>& hash, std::ostream& os) {
|
|
|
|
auto h = hash.load();
|
|
|
|
outputHex(std::cout, (char*)&h, sizeof(h));
|
|
|
|
}
|
|
|
|
std::atomic<uint64_t> hash[4];
|
|
|
|
};
|
2018-12-11 21:00:30 +01:00
|
|
|
|
2018-12-23 18:02:17 +01:00
|
|
|
void printUsage(const char* executable) {
|
|
|
|
std::cout << "Usage: " << executable << " [OPTIONS]" << std::endl;
|
|
|
|
std::cout << "Supported options:" << std::endl;
|
2019-02-11 18:57:42 +01:00
|
|
|
std::cout << " --help shows this message" << std::endl;
|
|
|
|
std::cout << " --mine mining mode: 4 GiB dataset, x86-64 compiled VM" << std::endl;
|
|
|
|
std::cout << " (default: portable verification mode)" << std::endl;
|
|
|
|
std::cout << " --largePages use large pages" << std::endl;
|
|
|
|
std::cout << " --softAes use software AES (default: x86 AES-NI)" << std::endl;
|
|
|
|
std::cout << " --threads T use T threads (default: 1)" << std::endl;
|
|
|
|
std::cout << " --nonces N run N nonces (default: 1000)" << std::endl;
|
|
|
|
std::cout << " --genAsm generate x86-64 asm code for nonce N" << std::endl;
|
|
|
|
std::cout << " --genNative generate RandomX code for nonce N" << std::endl;
|
2018-12-23 18:02:17 +01:00
|
|
|
}
|
|
|
|
|
2018-12-21 21:09:55 +01:00
|
|
|
void generateAsm(int nonce) {
|
2019-02-09 15:45:26 +01:00
|
|
|
uint64_t hash[8];
|
2019-02-11 18:57:42 +01:00
|
|
|
uint8_t blockTemplate[sizeof(blockTemplate__)];
|
|
|
|
memcpy(blockTemplate, blockTemplate__, sizeof(blockTemplate));
|
2018-12-21 21:09:55 +01:00
|
|
|
int* noncePtr = (int*)(blockTemplate + 39);
|
|
|
|
*noncePtr = nonce;
|
|
|
|
blake2b(hash, sizeof(hash), blockTemplate, sizeof(blockTemplate), nullptr, 0);
|
|
|
|
RandomX::AssemblyGeneratorX86 asmX86;
|
2019-02-09 15:45:26 +01:00
|
|
|
RandomX::Program p;
|
|
|
|
fillAes1Rx4<false>(hash, sizeof(p), &p);
|
|
|
|
asmX86.generateProgram(p);
|
2018-12-21 21:09:55 +01:00
|
|
|
asmX86.printCode(std::cout);
|
|
|
|
}
|
|
|
|
|
2019-01-24 19:29:59 +01:00
|
|
|
void generateNative(int nonce) {
|
|
|
|
uint64_t hash[4];
|
2019-02-11 18:57:42 +01:00
|
|
|
uint8_t blockTemplate[sizeof(blockTemplate__)];
|
|
|
|
memcpy(blockTemplate, blockTemplate__, sizeof(blockTemplate));
|
2019-01-24 19:29:59 +01:00
|
|
|
int* noncePtr = (int*)(blockTemplate + 39);
|
|
|
|
*noncePtr = nonce;
|
|
|
|
blake2b(hash, sizeof(hash), blockTemplate, sizeof(blockTemplate), nullptr, 0);
|
2019-02-09 15:45:26 +01:00
|
|
|
alignas(16) RandomX::Program prog;
|
|
|
|
fillAes1Rx4<false>((void*)hash, sizeof(prog), &prog);
|
2019-01-24 19:29:59 +01:00
|
|
|
for (int i = 0; i < RandomX::ProgramLength; ++i) {
|
|
|
|
prog(i).dst %= 8;
|
|
|
|
prog(i).src %= 8;
|
|
|
|
}
|
|
|
|
std::cout << prog << std::endl;
|
|
|
|
}
|
|
|
|
|
2019-02-13 22:46:32 +01:00
|
|
|
template<bool softAes>
|
2019-01-20 00:44:01 +01:00
|
|
|
void mine(RandomX::VirtualMachine* vm, std::atomic<int>& atomicNonce, AtomicHash& result, int noncesCount, int thread, uint8_t* scratchpad) {
|
2019-02-04 17:07:00 +01:00
|
|
|
alignas(16) uint64_t hash[8];
|
2019-02-11 18:57:42 +01:00
|
|
|
uint8_t blockTemplate[sizeof(blockTemplate__)];
|
|
|
|
memcpy(blockTemplate, blockTemplate__, sizeof(blockTemplate));
|
2018-12-19 21:54:44 +01:00
|
|
|
int* noncePtr = (int*)(blockTemplate + 39);
|
|
|
|
int nonce = atomicNonce.fetch_add(1);
|
2018-12-13 23:11:55 +01:00
|
|
|
|
2018-12-19 21:54:44 +01:00
|
|
|
while (nonce < noncesCount) {
|
|
|
|
//std::cout << "Thread " << thread << " nonce " << nonce << std::endl;
|
|
|
|
*noncePtr = nonce;
|
2018-12-13 23:11:55 +01:00
|
|
|
blake2b(hash, sizeof(hash), blockTemplate, sizeof(blockTemplate), nullptr, 0);
|
2019-02-13 22:46:32 +01:00
|
|
|
fillAes1Rx4<softAes>((void*)hash, RandomX::ScratchpadSize, scratchpad);
|
2019-01-27 18:19:49 +01:00
|
|
|
vm->setScratchpad(scratchpad);
|
2018-12-21 21:09:55 +01:00
|
|
|
//dump((char*)((RandomX::CompiledVirtualMachine*)vm)->getProgram(), RandomX::CodeSize, "code-1337-jmp.txt");
|
2019-02-11 18:57:42 +01:00
|
|
|
for (int chain = 0; chain < RandomX::ChainLength - 1; ++chain) {
|
2019-02-13 22:46:32 +01:00
|
|
|
fillAes1Rx4<softAes>((void*)hash, sizeof(RandomX::Program), vm->getProgramBuffer());
|
2019-02-09 15:45:26 +01:00
|
|
|
vm->initialize();
|
2019-01-24 19:29:59 +01:00
|
|
|
vm->execute();
|
2019-02-09 15:45:26 +01:00
|
|
|
vm->getResult<false>(nullptr, 0, hash);
|
2019-01-24 19:29:59 +01:00
|
|
|
}
|
2019-02-13 22:46:32 +01:00
|
|
|
fillAes1Rx4<softAes>((void*)hash, sizeof(RandomX::Program), vm->getProgramBuffer());
|
2019-02-11 18:57:42 +01:00
|
|
|
vm->initialize();
|
|
|
|
vm->execute();
|
2019-02-13 22:46:32 +01:00
|
|
|
vm->getResult<softAes>(scratchpad, RandomX::ScratchpadSize, hash);
|
2018-12-19 21:54:44 +01:00
|
|
|
result.xorWith(hash);
|
|
|
|
if (RandomX::trace) {
|
|
|
|
std::cout << "Nonce: " << nonce << " ";
|
|
|
|
outputHex(std::cout, (char*)hash, sizeof(hash));
|
|
|
|
std::cout << std::endl;
|
|
|
|
}
|
|
|
|
nonce = atomicNonce.fetch_add(1);
|
2018-12-13 23:11:55 +01:00
|
|
|
}
|
2018-12-19 21:54:44 +01:00
|
|
|
}
|
2018-12-13 23:11:55 +01:00
|
|
|
|
2018-12-19 21:54:44 +01:00
|
|
|
int main(int argc, char** argv) {
|
2019-02-11 18:57:42 +01:00
|
|
|
bool softAes, genAsm, miningMode, help, largePages, async, genNative;
|
2018-12-19 21:54:44 +01:00
|
|
|
int programCount, threadCount;
|
2018-12-23 18:02:17 +01:00
|
|
|
readOption("--help", argc, argv, help);
|
|
|
|
|
|
|
|
if (help) {
|
|
|
|
printUsage(argv[0]);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-12-19 21:54:44 +01:00
|
|
|
readOption("--softAes", argc, argv, softAes);
|
|
|
|
readOption("--genAsm", argc, argv, genAsm);
|
2019-02-11 18:57:42 +01:00
|
|
|
readOption("--mine", argc, argv, miningMode);
|
2018-12-19 21:54:44 +01:00
|
|
|
readIntOption("--threads", argc, argv, threadCount, 1);
|
|
|
|
readIntOption("--nonces", argc, argv, programCount, 1000);
|
2019-01-04 19:44:15 +01:00
|
|
|
readOption("--largePages", argc, argv, largePages);
|
2019-01-15 00:01:11 +01:00
|
|
|
readOption("--async", argc, argv, async);
|
2019-01-24 19:29:59 +01:00
|
|
|
readOption("--genNative", argc, argv, genNative);
|
2018-12-13 23:11:55 +01:00
|
|
|
|
2018-12-21 21:09:55 +01:00
|
|
|
if (genAsm) {
|
|
|
|
generateAsm(programCount);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-01-24 19:29:59 +01:00
|
|
|
if (genNative) {
|
|
|
|
generateNative(programCount);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-01-18 17:57:47 +01:00
|
|
|
if (softAes)
|
|
|
|
std::cout << "Using software AES." << std::endl;
|
|
|
|
|
2018-12-19 21:54:44 +01:00
|
|
|
std::atomic<int> atomicNonce(0);
|
|
|
|
AtomicHash result;
|
|
|
|
std::vector<RandomX::VirtualMachine*> vms;
|
|
|
|
std::vector<std::thread> threads;
|
|
|
|
RandomX::dataset_t dataset;
|
2018-12-13 23:11:55 +01:00
|
|
|
|
2019-02-11 18:57:42 +01:00
|
|
|
std::cout << "RandomX - " << (miningMode ? "mining" : "verification") << " mode" << std::endl;
|
|
|
|
|
2018-12-19 21:54:44 +01:00
|
|
|
std::cout << "Initializing..." << std::endl;
|
2018-12-11 21:00:30 +01:00
|
|
|
try {
|
2018-12-19 21:54:44 +01:00
|
|
|
Stopwatch sw(true);
|
|
|
|
if (softAes) {
|
2019-01-18 17:57:47 +01:00
|
|
|
RandomX::datasetInitCache<true>(seed, dataset, largePages);
|
2018-12-18 22:00:58 +01:00
|
|
|
}
|
|
|
|
else {
|
2019-01-18 17:57:47 +01:00
|
|
|
RandomX::datasetInitCache<false>(seed, dataset, largePages);
|
2018-12-18 22:00:58 +01:00
|
|
|
}
|
2018-12-19 21:54:44 +01:00
|
|
|
if (RandomX::trace) {
|
|
|
|
std::cout << "Keys: " << std::endl;
|
2019-02-09 19:32:53 +01:00
|
|
|
for (unsigned i = 0; i < dataset.cache->getKeys().size(); ++i) {
|
2018-12-19 21:54:44 +01:00
|
|
|
outputHex(std::cout, (char*)&dataset.cache->getKeys()[i], sizeof(__m128i));
|
|
|
|
}
|
|
|
|
std::cout << std::endl;
|
|
|
|
std::cout << "Cache: " << std::endl;
|
|
|
|
outputHex(std::cout, (char*)dataset.cache->getCache(), sizeof(__m128i));
|
|
|
|
std::cout << std::endl;
|
|
|
|
}
|
2019-02-11 18:57:42 +01:00
|
|
|
if (!miningMode) {
|
2019-01-18 19:06:46 +01:00
|
|
|
std::cout << "Cache (256 MiB) initialized in " << sw.getElapsed() << " s" << std::endl;
|
2018-12-19 21:54:44 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
RandomX::Cache* cache = dataset.cache;
|
2019-01-04 19:44:15 +01:00
|
|
|
RandomX::datasetAlloc(dataset, largePages);
|
2018-12-23 15:12:54 +01:00
|
|
|
if (threadCount > 1) {
|
|
|
|
auto perThread = RandomX::DatasetBlockCount / threadCount;
|
|
|
|
auto remainder = RandomX::DatasetBlockCount % threadCount;
|
|
|
|
for (int i = 0; i < threadCount; ++i) {
|
|
|
|
auto count = perThread + (i == threadCount - 1 ? remainder : 0);
|
2019-02-13 22:46:32 +01:00
|
|
|
threads.push_back(std::thread(&RandomX::datasetInit, cache, dataset, i * perThread, count));
|
2018-12-23 15:12:54 +01:00
|
|
|
}
|
2019-02-09 19:32:53 +01:00
|
|
|
for (unsigned i = 0; i < threads.size(); ++i) {
|
2018-12-23 15:12:54 +01:00
|
|
|
threads[i].join();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2019-02-13 22:46:32 +01:00
|
|
|
RandomX::datasetInit(cache, dataset, 0, RandomX::DatasetBlockCount);
|
2018-12-19 21:54:44 +01:00
|
|
|
}
|
2019-01-18 17:57:47 +01:00
|
|
|
RandomX::Cache::dealloc(cache, largePages);
|
2018-12-19 21:54:44 +01:00
|
|
|
threads.clear();
|
2018-12-11 21:00:30 +01:00
|
|
|
std::cout << "Dataset (4 GiB) initialized in " << sw.getElapsed() << " s" << std::endl;
|
2018-12-19 21:54:44 +01:00
|
|
|
}
|
|
|
|
std::cout << "Initializing " << threadCount << " virtual machine(s)..." << std::endl;
|
|
|
|
for (int i = 0; i < threadCount; ++i) {
|
|
|
|
RandomX::VirtualMachine* vm;
|
2019-02-11 18:57:42 +01:00
|
|
|
if (miningMode) {
|
2019-01-15 00:01:11 +01:00
|
|
|
vm = new RandomX::CompiledVirtualMachine();
|
2018-12-19 21:54:44 +01:00
|
|
|
}
|
|
|
|
else {
|
2019-01-15 00:01:11 +01:00
|
|
|
vm = new RandomX::InterpretedVirtualMachine(softAes, async);
|
2018-12-19 21:54:44 +01:00
|
|
|
}
|
2019-01-15 00:01:11 +01:00
|
|
|
vm->setDataset(dataset);
|
2018-12-19 21:54:44 +01:00
|
|
|
vms.push_back(vm);
|
|
|
|
}
|
2019-01-20 00:44:01 +01:00
|
|
|
uint8_t* scratchpadMem;
|
|
|
|
if (largePages) {
|
2019-02-04 17:07:00 +01:00
|
|
|
scratchpadMem = (uint8_t*)allocLargePagesMemory(threadCount * RandomX::ScratchpadSize);
|
2019-01-20 00:44:01 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
scratchpadMem = (uint8_t*)_mm_malloc(threadCount * RandomX::ScratchpadSize, RandomX::CacheLineSize);
|
|
|
|
}
|
2019-02-11 18:57:42 +01:00
|
|
|
std::cout << "Running benchmark (" << programCount << " nonces) ..." << std::endl;
|
2018-12-11 21:00:30 +01:00
|
|
|
sw.restart();
|
2018-12-23 15:12:54 +01:00
|
|
|
if (threadCount > 1) {
|
2019-02-09 19:32:53 +01:00
|
|
|
for (unsigned i = 0; i < vms.size(); ++i) {
|
2019-02-13 22:46:32 +01:00
|
|
|
if (softAes)
|
|
|
|
threads.push_back(std::thread(&mine<true>, vms[i], std::ref(atomicNonce), std::ref(result), programCount, i, scratchpadMem + RandomX::ScratchpadSize * i));
|
|
|
|
else
|
|
|
|
threads.push_back(std::thread(&mine<false>, vms[i], std::ref(atomicNonce), std::ref(result), programCount, i, scratchpadMem + RandomX::ScratchpadSize * i));
|
2018-12-23 15:12:54 +01:00
|
|
|
}
|
2019-02-09 19:32:53 +01:00
|
|
|
for (unsigned i = 0; i < threads.size(); ++i) {
|
2018-12-23 15:12:54 +01:00
|
|
|
threads[i].join();
|
|
|
|
}
|
2018-12-19 21:54:44 +01:00
|
|
|
}
|
2018-12-23 15:12:54 +01:00
|
|
|
else {
|
2019-02-13 22:46:32 +01:00
|
|
|
if(softAes)
|
|
|
|
mine<true>(vms[0], std::ref(atomicNonce), std::ref(result), programCount, 0, scratchpadMem);
|
|
|
|
else
|
|
|
|
mine<false>(vms[0], std::ref(atomicNonce), std::ref(result), programCount, 0, scratchpadMem);
|
2019-02-11 18:57:42 +01:00
|
|
|
if (miningMode)
|
|
|
|
std::cout << "Average program size: " << ((RandomX::CompiledVirtualMachine*)vms[0])->getTotalSize() / programCount / RandomX::ChainLength << std::endl;
|
2018-12-11 21:00:30 +01:00
|
|
|
}
|
2018-12-13 23:11:55 +01:00
|
|
|
double elapsed = sw.getElapsed();
|
2018-12-19 12:38:10 +01:00
|
|
|
std::cout << "Calculated result: ";
|
2018-12-19 21:54:44 +01:00
|
|
|
result.print(std::cout);
|
2019-02-11 18:57:42 +01:00
|
|
|
/*if(programCount == 1000)
|
|
|
|
std::cout << "Reference result: 3e1c5f9b9d0bf8ffa250f860bf5f7ab76ac823b206ddee6a592660119a3640c6" << std::endl;*/
|
|
|
|
if (!miningMode) {
|
2019-02-09 19:32:53 +01:00
|
|
|
std::cout << "Performance: " << 1000 * elapsed / programCount << " ms per hash" << std::endl;
|
2019-01-15 00:01:11 +01:00
|
|
|
}
|
|
|
|
else {
|
2019-02-09 19:32:53 +01:00
|
|
|
std::cout << "Performance: " << programCount / elapsed << " hashes per second" << std::endl;
|
2019-01-15 00:01:11 +01:00
|
|
|
}
|
2018-12-11 21:00:30 +01:00
|
|
|
}
|
|
|
|
catch (std::exception& e) {
|
|
|
|
std::cout << "ERROR: " << e.what() << std::endl;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|