Flags ordered by their impact on performance

Additional error handling
This commit is contained in:
tevador 2019-04-21 23:23:13 +02:00
parent d30eef75af
commit bc78b628ea
8 changed files with 54 additions and 36 deletions

View file

@ -17,6 +17,7 @@ You should have received a copy of the GNU General Public License
along with RandomX. If not, see<http://www.gnu.org/licenses/>. along with RandomX. If not, see<http://www.gnu.org/licenses/>.
*/ */
#include <new>
#include "allocator.hpp" #include "allocator.hpp"
#include "intrin_portable.h" #include "intrin_portable.h"
#include "virtual_memory.hpp" #include "virtual_memory.hpp"
@ -26,7 +27,10 @@ namespace randomx {
template<size_t alignment> template<size_t alignment>
void* AlignedAllocator<alignment>::allocMemory(size_t count) { void* AlignedAllocator<alignment>::allocMemory(size_t count) {
return _mm_malloc(count, alignment); void *mem = _mm_malloc(count, alignment);
if (mem == nullptr)
throw std::bad_alloc();
return mem;
} }
template<size_t alignment> template<size_t alignment>

View file

@ -28,7 +28,7 @@ along with RandomX. If not, see<http://www.gnu.org/licenses/>.
extern "C" { extern "C" {
randomx_cache *randomx_alloc_cache(randomx_flags flags) { randomx_cache *randomx_alloc_cache(randomx_flags flags) {
randomx_cache *cache; randomx_cache *cache = nullptr;
try { try {
switch (flags & (RANDOMX_FLAG_JIT | RANDOMX_FLAG_LARGE_PAGES)) { switch (flags & (RANDOMX_FLAG_JIT | RANDOMX_FLAG_LARGE_PAGES)) {
@ -99,72 +99,72 @@ extern "C" {
delete dataset; delete dataset;
} }
randomx_vm *randomx_create_vm(randomx_flags flags) { randomx_vm *randomx_create_vm(randomx_flags flags, randomx_cache *cache, randomx_dataset *dataset) {
randomx_vm *vm = nullptr; randomx_vm *vm = nullptr;
try { try {
switch (flags & (RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_JIT | RANDOMX_FLAG_HARD_AES | RANDOMX_FLAG_LARGE_PAGES)) { switch (flags & (RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_JIT | RANDOMX_FLAG_HARD_AES | RANDOMX_FLAG_LARGE_PAGES)) {
case RANDOMX_FLAG_DEFAULT: //0 case RANDOMX_FLAG_DEFAULT:
vm = new randomx::InterpretedLightVmDefault(); vm = new randomx::InterpretedLightVmDefault();
break; break;
case RANDOMX_FLAG_FULL_MEM: //1 case RANDOMX_FLAG_FULL_MEM:
vm = new randomx::InterpretedVmDefault(); vm = new randomx::InterpretedVmDefault();
break; break;
case RANDOMX_FLAG_JIT: //2 case RANDOMX_FLAG_JIT:
vm = new randomx::CompiledLightVmDefault(); vm = new randomx::CompiledLightVmDefault();
break; break;
case RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_JIT: //3 case RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_JIT:
vm = new randomx::CompiledVmDefault(); vm = new randomx::CompiledVmDefault();
break; break;
case RANDOMX_FLAG_HARD_AES: //4 case RANDOMX_FLAG_HARD_AES:
vm = new randomx::InterpretedLightVmHardAes(); vm = new randomx::InterpretedLightVmHardAes();
break; break;
case RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_HARD_AES: //5 case RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_HARD_AES:
vm = new randomx::InterpretedVmHardAes(); vm = new randomx::InterpretedVmHardAes();
break; break;
case RANDOMX_FLAG_JIT | RANDOMX_FLAG_HARD_AES: //6 case RANDOMX_FLAG_JIT | RANDOMX_FLAG_HARD_AES:
vm = new randomx::CompiledLightVmHardAes(); vm = new randomx::CompiledLightVmHardAes();
break; break;
case RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_JIT | RANDOMX_FLAG_HARD_AES: //7 case RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_JIT | RANDOMX_FLAG_HARD_AES:
vm = new randomx::CompiledVmHardAes(); vm = new randomx::CompiledVmHardAes();
break; break;
case RANDOMX_FLAG_LARGE_PAGES: //8 case RANDOMX_FLAG_LARGE_PAGES:
vm = new randomx::InterpretedLightVmLargePage(); vm = new randomx::InterpretedLightVmLargePage();
break; break;
case RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_LARGE_PAGES: //9 case RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_LARGE_PAGES:
vm = new randomx::InterpretedVmLargePage(); vm = new randomx::InterpretedVmLargePage();
break; break;
case RANDOMX_FLAG_JIT | RANDOMX_FLAG_LARGE_PAGES: //10 case RANDOMX_FLAG_JIT | RANDOMX_FLAG_LARGE_PAGES:
vm = new randomx::CompiledLightVmLargePage(); vm = new randomx::CompiledLightVmLargePage();
break; break;
case RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_JIT | RANDOMX_FLAG_LARGE_PAGES: //11 case RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_JIT | RANDOMX_FLAG_LARGE_PAGES:
vm = new randomx::CompiledVmLargePage(); vm = new randomx::CompiledVmLargePage();
break; break;
case RANDOMX_FLAG_HARD_AES | RANDOMX_FLAG_LARGE_PAGES: //12 case RANDOMX_FLAG_HARD_AES | RANDOMX_FLAG_LARGE_PAGES:
vm = new randomx::InterpretedLightVmLargePageHardAes(); vm = new randomx::InterpretedLightVmLargePageHardAes();
break; break;
case RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_HARD_AES | RANDOMX_FLAG_LARGE_PAGES: //13 case RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_HARD_AES | RANDOMX_FLAG_LARGE_PAGES:
vm = new randomx::InterpretedVmLargePageHardAes(); vm = new randomx::InterpretedVmLargePageHardAes();
break; break;
case RANDOMX_FLAG_JIT | RANDOMX_FLAG_HARD_AES | RANDOMX_FLAG_LARGE_PAGES: //14 case RANDOMX_FLAG_JIT | RANDOMX_FLAG_HARD_AES | RANDOMX_FLAG_LARGE_PAGES:
vm = new randomx::CompiledLightVmLargePageHardAes(); vm = new randomx::CompiledLightVmLargePageHardAes();
break; break;
case RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_JIT | RANDOMX_FLAG_HARD_AES | RANDOMX_FLAG_LARGE_PAGES: //15 case RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_JIT | RANDOMX_FLAG_HARD_AES | RANDOMX_FLAG_LARGE_PAGES:
vm = new randomx::CompiledVmLargePageHardAes(); vm = new randomx::CompiledVmLargePageHardAes();
break; break;
@ -172,6 +172,12 @@ extern "C" {
UNREACHABLE; UNREACHABLE;
} }
if(cache != nullptr)
vm->setCache(cache);
if(dataset != nullptr)
vm->setDataset(dataset);
vm->allocate(); vm->allocate();
} }
catch (std::exception &ex) { catch (std::exception &ex) {

View file

@ -27,10 +27,10 @@ along with RandomX. If not, see<http://www.gnu.org/licenses/>.
typedef enum { typedef enum {
RANDOMX_FLAG_DEFAULT = 0, RANDOMX_FLAG_DEFAULT = 0,
RANDOMX_FLAG_FULL_MEM = 1, RANDOMX_FLAG_LARGE_PAGES = 1,
RANDOMX_FLAG_JIT = 2, RANDOMX_FLAG_HARD_AES = 2,
RANDOMX_FLAG_HARD_AES = 4, RANDOMX_FLAG_FULL_MEM = 4,
RANDOMX_FLAG_LARGE_PAGES = 8, RANDOMX_FLAG_JIT = 8,
} randomx_flags; } randomx_flags;
typedef struct randomx_dataset randomx_dataset; typedef struct randomx_dataset randomx_dataset;
@ -49,7 +49,7 @@ randomx_dataset *randomx_alloc_dataset(randomx_flags flags);
void randomx_init_dataset(randomx_dataset *dataset, randomx_cache *cache, unsigned long startBlock, unsigned long blockCount); void randomx_init_dataset(randomx_dataset *dataset, randomx_cache *cache, unsigned long startBlock, unsigned long blockCount);
void randomx_release_dataset(randomx_dataset *dataset); void randomx_release_dataset(randomx_dataset *dataset);
randomx_vm *randomx_create_vm(randomx_flags flags); randomx_vm *randomx_create_vm(randomx_flags flags, randomx_cache *cache, randomx_dataset *dataset);
void randomx_vm_set_cache(randomx_vm *machine, randomx_cache* cache); void randomx_vm_set_cache(randomx_vm *machine, randomx_cache* cache);
void randomx_vm_set_dataset(randomx_vm *machine, randomx_dataset *dataset); void randomx_vm_set_dataset(randomx_vm *machine, randomx_dataset *dataset);
void randomx_destroy_vm(randomx_vm *machine); void randomx_destroy_vm(randomx_vm *machine);

View file

@ -8,8 +8,7 @@ int main() {
randomx_cache *myCache = randomx_alloc_cache(RANDOMX_FLAG_DEFAULT); randomx_cache *myCache = randomx_alloc_cache(RANDOMX_FLAG_DEFAULT);
randomx_init_cache(myCache, mySeed, sizeof mySeed); randomx_init_cache(myCache, mySeed, sizeof mySeed);
randomx_vm *myMachine = randomx_create_vm(RANDOMX_FLAG_DEFAULT); randomx_vm *myMachine = randomx_create_vm(RANDOMX_FLAG_DEFAULT, myCache, NULL);
randomx_vm_set_cache(myMachine, myCache);
randomx_calculate_hash(myMachine, &myInput, sizeof myInput, hash); randomx_calculate_hash(myMachine, &myInput, sizeof myInput, hash);

View file

@ -26,8 +26,7 @@ int main() {
t2.join(); t2.join();
randomx_release_cache(myCache); randomx_release_cache(myCache);
randomx_vm *myMachine = randomx_create_vm((randomx_flags)(RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_JIT | RANDOMX_FLAG_HARD_AES | RANDOMX_FLAG_LARGE_PAGES)); randomx_vm *myMachine = randomx_create_vm((randomx_flags)(RANDOMX_FLAG_FULL_MEM | RANDOMX_FLAG_JIT | RANDOMX_FLAG_HARD_AES | RANDOMX_FLAG_LARGE_PAGES), nullptr, myDataset);
randomx_vm_set_dataset(myMachine, myDataset);
randomx_calculate_hash(myMachine, &myInput, sizeof myInput, hash); randomx_calculate_hash(myMachine, &myInput, sizeof myInput, hash);

View file

@ -201,11 +201,11 @@ int main(int argc, char** argv) {
std::cout << "Memory initialized in " << sw.getElapsed() << " s" << std::endl; std::cout << "Memory initialized in " << sw.getElapsed() << " s" << std::endl;
std::cout << "Initializing " << threadCount << " virtual machine(s) ..." << std::endl; std::cout << "Initializing " << threadCount << " virtual machine(s) ..." << std::endl;
for (int i = 0; i < threadCount; ++i) { for (int i = 0; i < threadCount; ++i) {
randomx_vm *vm = randomx_create_vm(flags); randomx_vm *vm = randomx_create_vm(flags, cache, dataset);
if (miningMode) if (vm == nullptr) {
randomx_vm_set_dataset(vm, dataset); std::cout << "ERROR: Unsupported virtual machine options" << std::endl;
else return 1;
randomx_vm_set_cache(vm, cache); }
vms.push_back(vm); vms.push_back(vm);
} }
std::cout << "Running benchmark (" << noncesCount << " nonces) ..." << std::endl; std::cout << "Running benchmark (" << noncesCount << " nonces) ..." << std::endl;

View file

@ -17,12 +17,13 @@ You should have received a copy of the GNU General Public License
along with RandomX. If not, see<http://www.gnu.org/licenses/>. along with RandomX. If not, see<http://www.gnu.org/licenses/>.
*/ */
#include <cstring>
#include <iomanip>
#include <stdexcept>
#include "virtual_machine.hpp" #include "virtual_machine.hpp"
#include "common.hpp" #include "common.hpp"
#include "aes_hash.hpp" #include "aes_hash.hpp"
#include "blake2/blake2.h" #include "blake2/blake2.h"
#include <cstring>
#include <iomanip>
#include "intrin_portable.h" #include "intrin_portable.h"
#include "allocator.hpp" #include "allocator.hpp"
@ -93,6 +94,8 @@ std::ostream& operator<<(std::ostream& os, const randomx::RegisterFile& rf) {
namespace randomx { namespace randomx {
alignas(16) volatile static __m128i aesDummy;
template<class Allocator, bool softAes> template<class Allocator, bool softAes>
VmBase<Allocator, softAes>::~VmBase() { VmBase<Allocator, softAes>::~VmBase() {
Allocator::freeMemory(scratchpad, ScratchpadSize); Allocator::freeMemory(scratchpad, ScratchpadSize);
@ -100,6 +103,13 @@ namespace randomx {
template<class Allocator, bool softAes> template<class Allocator, bool softAes>
void VmBase<Allocator, softAes>::allocate() { void VmBase<Allocator, softAes>::allocate() {
if (mem.memory == nullptr)
throw std::invalid_argument("Cache/Dataset not set");
if (!softAes) { //if hardware AES is not supported, it's better to fail now than to return a ticking bomb
__m128i tmp = _mm_load_si128((const __m128i*)&aesDummy);
tmp = _mm_aesenc_si128(tmp, tmp);
_mm_store_si128((__m128i*)&aesDummy, tmp);
}
scratchpad = (uint8_t*)Allocator::allocMemory(ScratchpadSize); scratchpad = (uint8_t*)Allocator::allocMemory(ScratchpadSize);
} }

View file

@ -37,7 +37,7 @@ namespace randomx {
AlignedAllocator<CacheLineSize>::freeMemory(ptr, sizeof(CompiledLightVm)); AlignedAllocator<CacheLineSize>::freeMemory(ptr, sizeof(CompiledLightVm));
} }
void setCache(randomx_cache* cache) override; void setCache(randomx_cache* cache) override;
void setDataset(randomx_dataset* dataset) override {} void setDataset(randomx_dataset* dataset) override { }
void run(void* seed) override; void run(void* seed) override;
using CompiledVm<Allocator, softAes>::mem; using CompiledVm<Allocator, softAes>::mem;