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/>.
|
|
|
|
*/
|
2019-04-20 16:53:06 +02:00
|
|
|
|
2018-12-11 21:00:30 +01:00
|
|
|
#include <iostream>
|
|
|
|
#include <iomanip>
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <sstream>
|
|
|
|
#include <cmath>
|
2019-02-09 19:32:53 +01:00
|
|
|
#include <cfloat>
|
2019-02-24 14:48:07 +01:00
|
|
|
#include <climits>
|
2019-04-20 16:53:06 +02:00
|
|
|
#include "vm_interpreted.hpp"
|
|
|
|
#include "dataset.hpp"
|
|
|
|
#include "intrin_portable.h"
|
2019-02-22 17:48:26 +01:00
|
|
|
#include "reciprocal.h"
|
2018-12-11 21:00:30 +01:00
|
|
|
|
2019-04-20 11:08:01 +02:00
|
|
|
namespace randomx {
|
2018-12-11 21:00:30 +01:00
|
|
|
|
2019-04-16 18:58:44 +02:00
|
|
|
static int_reg_t Zero = 0;
|
|
|
|
|
2019-04-20 11:08:01 +02:00
|
|
|
template<class Allocator, bool softAes>
|
|
|
|
void InterpretedVm<Allocator, softAes>::setDataset(randomx_dataset* dataset) {
|
2019-04-28 12:44:28 +02:00
|
|
|
datasetPtr = dataset;
|
2019-04-20 11:08:01 +02:00
|
|
|
mem.memory = dataset->memory;
|
2019-01-15 00:01:11 +01:00
|
|
|
}
|
|
|
|
|
2019-04-20 11:08:01 +02:00
|
|
|
template<class Allocator, bool softAes>
|
2019-04-20 12:49:24 +02:00
|
|
|
void InterpretedVm<Allocator, softAes>::run(void* seed) {
|
|
|
|
VmBase<Allocator, softAes>::generateProgram(seed);
|
2019-04-20 11:08:01 +02:00
|
|
|
randomx_vm::initialize();
|
2019-04-20 12:49:24 +02:00
|
|
|
execute();
|
2019-02-09 15:45:26 +01:00
|
|
|
}
|
|
|
|
|
2019-04-20 11:08:01 +02:00
|
|
|
template<class Allocator, bool softAes>
|
2019-04-28 16:42:45 +02:00
|
|
|
void InterpretedVm<Allocator, softAes>::executeBytecode(int_reg_t(&r)[RegistersCount], __m128d (&f)[RegisterCountFlt], __m128d (&e)[RegisterCountFlt], __m128d (&a)[RegisterCountFlt]) {
|
|
|
|
for (int pc = 0; pc < RANDOMX_PROGRAM_SIZE; ++pc) {
|
|
|
|
executeBytecode(pc, r, f, e, a);
|
2019-03-20 23:38:37 +01:00
|
|
|
}
|
2019-02-09 15:45:26 +01:00
|
|
|
}
|
|
|
|
|
2019-04-20 11:08:01 +02:00
|
|
|
template<class Allocator, bool softAes>
|
|
|
|
FORCE_INLINE void* InterpretedVm<Allocator, softAes>::getScratchpadAddress(InstructionByteCode& ibc) {
|
2019-04-16 18:58:44 +02:00
|
|
|
uint32_t addr = (*ibc.isrc + ibc.imm) & ibc.memMask;
|
|
|
|
return scratchpad + addr;
|
|
|
|
}
|
|
|
|
|
2019-04-20 11:08:01 +02:00
|
|
|
template<class Allocator, bool softAes>
|
|
|
|
FORCE_INLINE __m128d InterpretedVm<Allocator, softAes>::maskRegisterExponentMantissa(__m128d x) {
|
2019-04-30 21:14:50 +02:00
|
|
|
const __m128d xmantissaMask = _mm_castsi128_pd(_mm_set_epi64x(dynamicMantissaMask, dynamicMantissaMask));
|
|
|
|
const __m128d xexponentMask = _mm_load_pd((const double*)&config.eMask);
|
|
|
|
x = _mm_and_pd(x, xmantissaMask);
|
|
|
|
x = _mm_or_pd(x, xexponentMask);
|
2019-04-17 16:18:02 +02:00
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
2019-04-20 11:08:01 +02:00
|
|
|
template<class Allocator, bool softAes>
|
2019-04-28 16:42:45 +02:00
|
|
|
void InterpretedVm<Allocator, softAes>::executeBytecode(int& pc, int_reg_t(&r)[RegistersCount], __m128d (&f)[RegisterCountFlt], __m128d (&e)[RegisterCountFlt], __m128d (&a)[RegisterCountFlt]) {
|
|
|
|
auto& ibc = byteCode[pc];
|
2019-02-09 15:45:26 +01:00
|
|
|
switch (ibc.type)
|
|
|
|
{
|
2019-04-11 00:01:22 +02:00
|
|
|
case InstructionType::IADD_RS: {
|
|
|
|
*ibc.idst += (*ibc.isrc << ibc.shift) + ibc.imm;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::IADD_M: {
|
2019-04-16 18:58:44 +02:00
|
|
|
*ibc.idst += load64(getScratchpadAddress(ibc));
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::ISUB_R: {
|
|
|
|
*ibc.idst -= *ibc.isrc;
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::ISUB_M: {
|
2019-04-16 18:58:44 +02:00
|
|
|
*ibc.idst -= load64(getScratchpadAddress(ibc));
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
2019-02-22 17:48:26 +01:00
|
|
|
case InstructionType::IMUL_R: { //also handles IMUL_RCP
|
2019-02-09 15:45:26 +01:00
|
|
|
*ibc.idst *= *ibc.isrc;
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::IMUL_M: {
|
2019-04-16 18:58:44 +02:00
|
|
|
*ibc.idst *= load64(getScratchpadAddress(ibc));
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::IMULH_R: {
|
|
|
|
*ibc.idst = mulh(*ibc.idst, *ibc.isrc);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::IMULH_M: {
|
2019-04-16 18:58:44 +02:00
|
|
|
*ibc.idst = mulh(*ibc.idst, load64(getScratchpadAddress(ibc)));
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::ISMULH_R: {
|
|
|
|
*ibc.idst = smulh(unsigned64ToSigned2sCompl(*ibc.idst), unsigned64ToSigned2sCompl(*ibc.isrc));
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::ISMULH_M: {
|
2019-04-16 18:58:44 +02:00
|
|
|
*ibc.idst = smulh(unsigned64ToSigned2sCompl(*ibc.idst), unsigned64ToSigned2sCompl(load64(getScratchpadAddress(ibc))));
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::INEG_R: {
|
|
|
|
*ibc.idst = ~(*ibc.idst) + 1; //two's complement negative
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::IXOR_R: {
|
|
|
|
*ibc.idst ^= *ibc.isrc;
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::IXOR_M: {
|
2019-04-16 18:58:44 +02:00
|
|
|
*ibc.idst ^= load64(getScratchpadAddress(ibc));
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::IROR_R: {
|
|
|
|
*ibc.idst = rotr(*ibc.idst, *ibc.isrc & 63);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::IROL_R: {
|
|
|
|
*ibc.idst = rotl(*ibc.idst, *ibc.isrc & 63);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::ISWAP_R: {
|
|
|
|
int_reg_t temp = *ibc.isrc;
|
|
|
|
*ibc.isrc = *ibc.idst;
|
|
|
|
*ibc.idst = temp;
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::FSWAP_R: {
|
|
|
|
*ibc.fdst = _mm_shuffle_pd(*ibc.fdst, *ibc.fdst, 1);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::FADD_R: {
|
|
|
|
*ibc.fdst = _mm_add_pd(*ibc.fdst, *ibc.fsrc);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::FADD_M: {
|
2019-04-16 18:58:44 +02:00
|
|
|
__m128d fsrc = load_cvt_i32x2(getScratchpadAddress(ibc));
|
2019-02-09 15:45:26 +01:00
|
|
|
*ibc.fdst = _mm_add_pd(*ibc.fdst, fsrc);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::FSUB_R: {
|
|
|
|
*ibc.fdst = _mm_sub_pd(*ibc.fdst, *ibc.fsrc);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::FSUB_M: {
|
2019-04-16 18:58:44 +02:00
|
|
|
__m128d fsrc = load_cvt_i32x2(getScratchpadAddress(ibc));
|
2019-02-09 15:45:26 +01:00
|
|
|
*ibc.fdst = _mm_sub_pd(*ibc.fdst, fsrc);
|
|
|
|
} break;
|
|
|
|
|
2019-02-13 00:01:34 +01:00
|
|
|
case InstructionType::FSCAL_R: {
|
2019-02-15 10:41:02 +01:00
|
|
|
const __m128d mask = _mm_castsi128_pd(_mm_set1_epi64x(0x81F0000000000000));
|
|
|
|
*ibc.fdst = _mm_xor_pd(*ibc.fdst, mask);
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::FMUL_R: {
|
|
|
|
*ibc.fdst = _mm_mul_pd(*ibc.fdst, *ibc.fsrc);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::FDIV_M: {
|
2019-04-17 16:18:02 +02:00
|
|
|
__m128d fsrc = maskRegisterExponentMantissa(load_cvt_i32x2(getScratchpadAddress(ibc)));
|
2019-02-24 14:48:07 +01:00
|
|
|
*ibc.fdst = _mm_div_pd(*ibc.fdst, fsrc);
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::FSQRT_R: {
|
|
|
|
*ibc.fdst = _mm_sqrt_pd(*ibc.fdst);
|
|
|
|
} break;
|
|
|
|
|
2019-04-29 23:38:23 +02:00
|
|
|
case InstructionType::CBRANCH: {
|
|
|
|
*ibc.isrc += ibc.imm;
|
|
|
|
if ((*ibc.isrc & ibc.memMask) == 0) {
|
2019-04-28 16:42:45 +02:00
|
|
|
pc = ibc.target;
|
2019-03-20 23:38:37 +01:00
|
|
|
}
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::CFROUND: {
|
|
|
|
setRoundMode(rotr(*ibc.isrc, ibc.imm) % 4);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::ISTORE: {
|
2019-04-16 18:58:44 +02:00
|
|
|
store64(scratchpad + ((*ibc.idst + ibc.imm) & ibc.memMask), *ibc.isrc);
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
case InstructionType::NOP: {
|
|
|
|
//nothing
|
|
|
|
} break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
UNREACHABLE;
|
2018-12-11 21:00:30 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-20 11:08:01 +02:00
|
|
|
template<class Allocator, bool softAes>
|
|
|
|
void InterpretedVm<Allocator, softAes>::execute() {
|
2019-04-28 16:42:45 +02:00
|
|
|
int_reg_t r[RegistersCount] = { 0 };
|
|
|
|
__m128d f[RegisterCountFlt];
|
|
|
|
__m128d e[RegisterCountFlt];
|
|
|
|
__m128d a[RegisterCountFlt];
|
2019-02-09 15:45:26 +01:00
|
|
|
|
2019-04-28 16:42:45 +02:00
|
|
|
for(unsigned i = 0; i < RegisterCountFlt; ++i)
|
|
|
|
a[i] = _mm_load_pd(®.a[i].lo);
|
2019-02-09 15:45:26 +01:00
|
|
|
|
|
|
|
precompileProgram(r, f, e, a);
|
|
|
|
|
|
|
|
uint32_t spAddr0 = mem.mx;
|
|
|
|
uint32_t spAddr1 = mem.ma;
|
|
|
|
|
2019-03-08 15:34:34 +01:00
|
|
|
for(unsigned ic = 0; ic < RANDOMX_PROGRAM_ITERATIONS; ++ic) {
|
2019-04-17 16:18:02 +02:00
|
|
|
uint64_t spMix = r[config.readReg0] ^ r[config.readReg1];
|
2019-02-16 23:18:45 +01:00
|
|
|
spAddr0 ^= spMix;
|
2019-02-09 15:45:26 +01:00
|
|
|
spAddr0 &= ScratchpadL3Mask64;
|
2019-02-16 23:18:45 +01:00
|
|
|
spAddr1 ^= spMix >> 32;
|
|
|
|
spAddr1 &= ScratchpadL3Mask64;
|
2019-02-09 15:45:26 +01:00
|
|
|
|
2019-04-28 16:42:45 +02:00
|
|
|
for (unsigned i = 0; i < RegistersCount; ++i)
|
|
|
|
r[i] ^= load64(scratchpad + spAddr0 + 8 * i);
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < RegisterCountFlt; ++i)
|
|
|
|
f[i] = load_cvt_i32x2(scratchpad + spAddr1 + 8 * i);
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < RegisterCountFlt; ++i)
|
|
|
|
e[i] = maskRegisterExponentMantissa(load_cvt_i32x2(scratchpad + spAddr1 + 8 * (RegisterCountFlt + i)));
|
2019-02-16 23:18:45 +01:00
|
|
|
|
2019-03-20 23:38:37 +01:00
|
|
|
executeBytecode(r, f, e, a);
|
|
|
|
|
2019-04-17 16:18:02 +02:00
|
|
|
mem.mx ^= r[config.readReg2] ^ r[config.readReg3];
|
2019-03-22 12:53:16 +01:00
|
|
|
mem.mx &= CacheLineAlignMask;
|
2019-04-26 16:05:30 +02:00
|
|
|
datasetRead(datasetOffset + mem.ma, r);
|
2019-03-20 23:38:37 +01:00
|
|
|
std::swap(mem.mx, mem.ma);
|
2019-02-09 15:45:26 +01:00
|
|
|
|
2019-04-28 16:42:45 +02:00
|
|
|
for (unsigned i = 0; i < RegistersCount; ++i)
|
|
|
|
store64(scratchpad + spAddr1 + 8 * i, r[i]);
|
2019-02-16 23:18:45 +01:00
|
|
|
|
2019-04-28 16:42:45 +02:00
|
|
|
for (unsigned i = 0; i < RegisterCountFlt; ++i)
|
|
|
|
f[i] = _mm_xor_pd(f[i], e[i]);
|
2019-02-16 23:18:45 +01:00
|
|
|
|
2019-04-28 16:42:45 +02:00
|
|
|
for (unsigned i = 0; i < RegisterCountFlt; ++i)
|
|
|
|
_mm_store_pd((double*)(scratchpad + spAddr0 + 16 * i), f[i]);
|
2019-02-09 15:45:26 +01:00
|
|
|
|
|
|
|
spAddr0 = 0;
|
|
|
|
spAddr1 = 0;
|
2018-12-11 21:00:30 +01:00
|
|
|
}
|
2019-02-04 17:07:00 +01:00
|
|
|
|
2019-04-28 16:42:45 +02:00
|
|
|
for (unsigned i = 0; i < RegistersCount; ++i)
|
|
|
|
store64(®.r[i], r[i]);
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < RegisterCountFlt; ++i)
|
|
|
|
_mm_store_pd(®.f[i].lo, f[i]);
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < RegisterCountFlt; ++i)
|
|
|
|
_mm_store_pd(®.e[i].lo, e[i]);
|
2018-12-11 21:00:30 +01:00
|
|
|
}
|
|
|
|
|
2019-04-20 11:08:01 +02:00
|
|
|
template<class Allocator, bool softAes>
|
2019-04-28 16:42:45 +02:00
|
|
|
void InterpretedVm<Allocator, softAes>::datasetRead(uint32_t address, int_reg_t(&r)[RegistersCount]) {
|
2019-04-20 11:08:01 +02:00
|
|
|
uint64_t* datasetLine = (uint64_t*)(mem.memory + address);
|
|
|
|
for (int i = 0; i < RegistersCount; ++i)
|
|
|
|
r[i] ^= datasetLine[i];
|
2019-04-11 00:01:22 +02:00
|
|
|
}
|
|
|
|
|
2019-04-20 16:53:06 +02:00
|
|
|
#include "instruction_weights.hpp"
|
2019-02-04 17:07:00 +01:00
|
|
|
|
2019-04-20 11:08:01 +02:00
|
|
|
template<class Allocator, bool softAes>
|
2019-04-28 16:42:45 +02:00
|
|
|
void InterpretedVm<Allocator, softAes>::precompileProgram(int_reg_t(&r)[RegistersCount], __m128d (&f)[RegisterCountFlt], __m128d (&e)[RegisterCountFlt], __m128d (&a)[RegisterCountFlt]) {
|
2019-04-30 10:20:28 +02:00
|
|
|
RegisterUsage registerUsage[RegistersCount];
|
2019-04-28 16:42:45 +02:00
|
|
|
for (unsigned i = 0; i < RegistersCount; ++i) {
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[i].lastUsed = -1;
|
|
|
|
registerUsage[i].count = 0;
|
2019-03-20 23:38:37 +01:00
|
|
|
}
|
2019-03-08 15:34:34 +01:00
|
|
|
for (unsigned i = 0; i < RANDOMX_PROGRAM_SIZE; ++i) {
|
2019-02-09 15:45:26 +01:00
|
|
|
auto& instr = program(i);
|
|
|
|
auto& ibc = byteCode[i];
|
|
|
|
switch (instr.opcode) {
|
2019-04-06 12:00:56 +02:00
|
|
|
CASE_REP(IADD_RS) {
|
2019-02-09 15:45:26 +01:00
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
auto src = instr.src % RegistersCount;
|
2019-04-11 00:01:22 +02:00
|
|
|
ibc.type = InstructionType::IADD_RS;
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.idst = &r[dst];
|
2019-04-16 18:58:44 +02:00
|
|
|
if (dst != RegisterNeedsDisplacement) {
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.isrc = &r[src];
|
2019-04-29 23:38:23 +02:00
|
|
|
ibc.shift = instr.getModShift();
|
2019-04-11 00:01:22 +02:00
|
|
|
ibc.imm = 0;
|
2019-02-09 15:45:26 +01:00
|
|
|
}
|
|
|
|
else {
|
2019-04-11 00:01:22 +02:00
|
|
|
ibc.isrc = &r[src];
|
2019-04-29 23:38:23 +02:00
|
|
|
ibc.shift = instr.getModShift();
|
2019-03-11 23:04:34 +01:00
|
|
|
ibc.imm = signExtend2sCompl(instr.getImm32());
|
2019-02-09 15:45:26 +01:00
|
|
|
}
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[dst].lastUsed = i;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
2019-02-04 17:07:00 +01:00
|
|
|
|
2019-02-09 15:45:26 +01:00
|
|
|
CASE_REP(IADD_M) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
auto src = instr.src % RegistersCount;
|
|
|
|
ibc.type = InstructionType::IADD_M;
|
|
|
|
ibc.idst = &r[dst];
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.imm = signExtend2sCompl(instr.getImm32());
|
2019-04-28 16:42:45 +02:00
|
|
|
if (src != dst) {
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.isrc = &r[src];
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.memMask = (instr.getModMem() ? ScratchpadL1Mask : ScratchpadL2Mask);
|
2019-02-09 15:45:26 +01:00
|
|
|
}
|
|
|
|
else {
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.isrc = &Zero;
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.memMask = ScratchpadL3Mask;
|
|
|
|
}
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[dst].lastUsed = i;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(ISUB_R) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
auto src = instr.src % RegistersCount;
|
|
|
|
ibc.type = InstructionType::ISUB_R;
|
|
|
|
ibc.idst = &r[dst];
|
|
|
|
if (src != dst) {
|
|
|
|
ibc.isrc = &r[src];
|
|
|
|
}
|
|
|
|
else {
|
2019-03-11 23:04:34 +01:00
|
|
|
ibc.imm = signExtend2sCompl(instr.getImm32());
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.isrc = &ibc.imm;
|
|
|
|
}
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[dst].lastUsed = i;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(ISUB_M) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
auto src = instr.src % RegistersCount;
|
|
|
|
ibc.type = InstructionType::ISUB_M;
|
|
|
|
ibc.idst = &r[dst];
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.imm = signExtend2sCompl(instr.getImm32());
|
2019-04-28 16:42:45 +02:00
|
|
|
if (src != dst) {
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.isrc = &r[src];
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.memMask = (instr.getModMem() ? ScratchpadL1Mask : ScratchpadL2Mask);
|
2019-02-09 15:45:26 +01:00
|
|
|
}
|
|
|
|
else {
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.isrc = &Zero;
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.memMask = ScratchpadL3Mask;
|
|
|
|
}
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[dst].lastUsed = i;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(IMUL_R) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
auto src = instr.src % RegistersCount;
|
|
|
|
ibc.type = InstructionType::IMUL_R;
|
|
|
|
ibc.idst = &r[dst];
|
|
|
|
if (src != dst) {
|
|
|
|
ibc.isrc = &r[src];
|
|
|
|
}
|
|
|
|
else {
|
2019-03-11 23:04:34 +01:00
|
|
|
ibc.imm = signExtend2sCompl(instr.getImm32());
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.isrc = &ibc.imm;
|
|
|
|
}
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[dst].lastUsed = i;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(IMUL_M) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
auto src = instr.src % RegistersCount;
|
|
|
|
ibc.type = InstructionType::IMUL_M;
|
|
|
|
ibc.idst = &r[dst];
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.imm = signExtend2sCompl(instr.getImm32());
|
2019-04-28 16:42:45 +02:00
|
|
|
if (src != dst) {
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.isrc = &r[src];
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.memMask = (instr.getModMem() ? ScratchpadL1Mask : ScratchpadL2Mask);
|
2019-02-09 15:45:26 +01:00
|
|
|
}
|
|
|
|
else {
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.isrc = &Zero;
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.memMask = ScratchpadL3Mask;
|
|
|
|
}
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[dst].lastUsed = i;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(IMULH_R) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
auto src = instr.src % RegistersCount;
|
|
|
|
ibc.type = InstructionType::IMULH_R;
|
|
|
|
ibc.idst = &r[dst];
|
|
|
|
ibc.isrc = &r[src];
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[dst].lastUsed = i;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(IMULH_M) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
auto src = instr.src % RegistersCount;
|
|
|
|
ibc.type = InstructionType::IMULH_M;
|
|
|
|
ibc.idst = &r[dst];
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.imm = signExtend2sCompl(instr.getImm32());
|
2019-04-28 16:42:45 +02:00
|
|
|
if (src != dst) {
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.isrc = &r[src];
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.memMask = (instr.getModMem() ? ScratchpadL1Mask : ScratchpadL2Mask);
|
2019-02-09 15:45:26 +01:00
|
|
|
}
|
|
|
|
else {
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.isrc = &Zero;
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.memMask = ScratchpadL3Mask;
|
|
|
|
}
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[dst].lastUsed = i;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(ISMULH_R) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
auto src = instr.src % RegistersCount;
|
|
|
|
ibc.type = InstructionType::ISMULH_R;
|
|
|
|
ibc.idst = &r[dst];
|
|
|
|
ibc.isrc = &r[src];
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[dst].lastUsed = i;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(ISMULH_M) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
auto src = instr.src % RegistersCount;
|
|
|
|
ibc.type = InstructionType::ISMULH_M;
|
|
|
|
ibc.idst = &r[dst];
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.imm = signExtend2sCompl(instr.getImm32());
|
2019-04-28 16:42:45 +02:00
|
|
|
if (src != dst) {
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.isrc = &r[src];
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.memMask = (instr.getModMem() ? ScratchpadL1Mask : ScratchpadL2Mask);
|
2019-02-09 15:45:26 +01:00
|
|
|
}
|
|
|
|
else {
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.isrc = &Zero;
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.memMask = ScratchpadL3Mask;
|
|
|
|
}
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[dst].lastUsed = i;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
2019-02-22 17:48:26 +01:00
|
|
|
CASE_REP(IMUL_RCP) {
|
2019-03-11 23:04:34 +01:00
|
|
|
uint32_t divisor = instr.getImm32();
|
2019-02-09 15:45:26 +01:00
|
|
|
if (divisor != 0) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
2019-02-22 17:48:26 +01:00
|
|
|
ibc.type = InstructionType::IMUL_R;
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.idst = &r[dst];
|
2019-04-20 16:53:06 +02:00
|
|
|
ibc.imm = randomx_reciprocal(divisor);
|
2019-02-22 17:48:26 +01:00
|
|
|
ibc.isrc = &ibc.imm;
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[dst].lastUsed = i;
|
2019-02-09 15:45:26 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
ibc.type = InstructionType::NOP;
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(INEG_R) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
ibc.type = InstructionType::INEG_R;
|
|
|
|
ibc.idst = &r[dst];
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[dst].lastUsed = i;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(IXOR_R) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
auto src = instr.src % RegistersCount;
|
|
|
|
ibc.type = InstructionType::IXOR_R;
|
|
|
|
ibc.idst = &r[dst];
|
|
|
|
if (src != dst) {
|
|
|
|
ibc.isrc = &r[src];
|
|
|
|
}
|
|
|
|
else {
|
2019-03-11 23:04:34 +01:00
|
|
|
ibc.imm = signExtend2sCompl(instr.getImm32());
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.isrc = &ibc.imm;
|
|
|
|
}
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[dst].lastUsed = i;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(IXOR_M) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
auto src = instr.src % RegistersCount;
|
|
|
|
ibc.type = InstructionType::IXOR_M;
|
|
|
|
ibc.idst = &r[dst];
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.imm = signExtend2sCompl(instr.getImm32());
|
2019-04-28 16:42:45 +02:00
|
|
|
if (src != dst) {
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.isrc = &r[src];
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.memMask = (instr.getModMem() ? ScratchpadL1Mask : ScratchpadL2Mask);
|
2019-02-09 15:45:26 +01:00
|
|
|
}
|
|
|
|
else {
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.isrc = &Zero;
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.memMask = ScratchpadL3Mask;
|
|
|
|
}
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[dst].lastUsed = i;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(IROR_R) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
auto src = instr.src % RegistersCount;
|
|
|
|
ibc.type = InstructionType::IROR_R;
|
|
|
|
ibc.idst = &r[dst];
|
|
|
|
if (src != dst) {
|
|
|
|
ibc.isrc = &r[src];
|
|
|
|
}
|
|
|
|
else {
|
2019-03-11 23:04:34 +01:00
|
|
|
ibc.imm = instr.getImm32();
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.isrc = &ibc.imm;
|
|
|
|
}
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[dst].lastUsed = i;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(IROL_R) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
auto src = instr.src % RegistersCount;
|
|
|
|
ibc.type = InstructionType::IROL_R;
|
|
|
|
ibc.idst = &r[dst];
|
|
|
|
if (src != dst) {
|
|
|
|
ibc.isrc = &r[src];
|
|
|
|
}
|
|
|
|
else {
|
2019-03-11 23:04:34 +01:00
|
|
|
ibc.imm = instr.getImm32();
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.isrc = &ibc.imm;
|
|
|
|
}
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[dst].lastUsed = i;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(ISWAP_R) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
auto src = instr.src % RegistersCount;
|
|
|
|
if (src != dst) {
|
|
|
|
ibc.idst = &r[dst];
|
|
|
|
ibc.isrc = &r[src];
|
|
|
|
ibc.type = InstructionType::ISWAP_R;
|
2019-04-30 10:20:28 +02:00
|
|
|
registerUsage[dst].lastUsed = i;
|
|
|
|
registerUsage[src].lastUsed = i;
|
2019-02-09 15:45:26 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
ibc.type = InstructionType::NOP;
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(FSWAP_R) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
ibc.type = InstructionType::FSWAP_R;
|
2019-04-28 16:42:45 +02:00
|
|
|
if (dst < RegisterCountFlt)
|
2019-02-18 22:09:20 +01:00
|
|
|
ibc.fdst = &f[dst];
|
|
|
|
else
|
2019-04-28 16:42:45 +02:00
|
|
|
ibc.fdst = &e[dst - RegisterCountFlt];
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(FADD_R) {
|
2019-04-28 16:42:45 +02:00
|
|
|
auto dst = instr.dst % RegisterCountFlt;
|
|
|
|
auto src = instr.src % RegisterCountFlt;
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.type = InstructionType::FADD_R;
|
|
|
|
ibc.fdst = &f[dst];
|
|
|
|
ibc.fsrc = &a[src];
|
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(FADD_M) {
|
2019-04-28 16:42:45 +02:00
|
|
|
auto dst = instr.dst % RegisterCountFlt;
|
|
|
|
auto src = instr.src % RegistersCount;
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.type = InstructionType::FADD_M;
|
|
|
|
ibc.fdst = &f[dst];
|
|
|
|
ibc.isrc = &r[src];
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.memMask = (instr.getModMem() ? ScratchpadL1Mask : ScratchpadL2Mask);
|
|
|
|
ibc.imm = signExtend2sCompl(instr.getImm32());
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(FSUB_R) {
|
2019-04-28 16:42:45 +02:00
|
|
|
auto dst = instr.dst % RegisterCountFlt;
|
|
|
|
auto src = instr.src % RegisterCountFlt;
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.type = InstructionType::FSUB_R;
|
|
|
|
ibc.fdst = &f[dst];
|
|
|
|
ibc.fsrc = &a[src];
|
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(FSUB_M) {
|
2019-04-28 16:42:45 +02:00
|
|
|
auto dst = instr.dst % RegisterCountFlt;
|
|
|
|
auto src = instr.src % RegistersCount;
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.type = InstructionType::FSUB_M;
|
|
|
|
ibc.fdst = &f[dst];
|
|
|
|
ibc.isrc = &r[src];
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.memMask = (instr.getModMem() ? ScratchpadL1Mask : ScratchpadL2Mask);
|
|
|
|
ibc.imm = signExtend2sCompl(instr.getImm32());
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
2018-12-11 21:00:30 +01:00
|
|
|
|
2019-02-13 00:01:34 +01:00
|
|
|
CASE_REP(FSCAL_R) {
|
2019-04-28 16:42:45 +02:00
|
|
|
auto dst = instr.dst % RegisterCountFlt;
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.fdst = &f[dst];
|
2019-02-13 00:01:34 +01:00
|
|
|
ibc.type = InstructionType::FSCAL_R;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
2019-01-24 19:29:59 +01:00
|
|
|
|
2019-02-09 15:45:26 +01:00
|
|
|
CASE_REP(FMUL_R) {
|
2019-04-28 16:42:45 +02:00
|
|
|
auto dst = instr.dst % RegisterCountFlt;
|
|
|
|
auto src = instr.src % RegisterCountFlt;
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.type = InstructionType::FMUL_R;
|
|
|
|
ibc.fdst = &e[dst];
|
|
|
|
ibc.fsrc = &a[src];
|
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(FDIV_M) {
|
2019-04-28 16:42:45 +02:00
|
|
|
auto dst = instr.dst % RegisterCountFlt;
|
|
|
|
auto src = instr.src % RegistersCount;
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.type = InstructionType::FDIV_M;
|
|
|
|
ibc.fdst = &e[dst];
|
|
|
|
ibc.isrc = &r[src];
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.memMask = (instr.getModMem() ? ScratchpadL1Mask : ScratchpadL2Mask);
|
|
|
|
ibc.imm = signExtend2sCompl(instr.getImm32());
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(FSQRT_R) {
|
2019-04-28 16:42:45 +02:00
|
|
|
auto dst = instr.dst % RegisterCountFlt;
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.type = InstructionType::FSQRT_R;
|
|
|
|
ibc.fdst = &e[dst];
|
|
|
|
} break;
|
|
|
|
|
2019-04-29 23:38:23 +02:00
|
|
|
CASE_REP(CBRANCH) {
|
|
|
|
ibc.type = InstructionType::CBRANCH;
|
2019-03-20 23:38:37 +01:00
|
|
|
//jump condition
|
|
|
|
int reg = getConditionRegister(registerUsage);
|
2019-04-29 23:38:23 +02:00
|
|
|
ibc.isrc = &r[reg];
|
2019-04-30 10:20:28 +02:00
|
|
|
ibc.target = registerUsage[reg].lastUsed;
|
|
|
|
registerUsage[reg].count++;
|
2019-04-29 23:38:23 +02:00
|
|
|
int shift = instr.getModCond();
|
|
|
|
const uint64_t conditionMask = ConditionMask << instr.getModCond();
|
|
|
|
ibc.imm = signExtend2sCompl(instr.getImm32()) | (1ULL << shift);
|
|
|
|
ibc.memMask = ConditionMask << shift;
|
2019-04-30 10:20:28 +02:00
|
|
|
//mark all registers as used
|
|
|
|
for (unsigned j = 0; j < RegistersCount; ++j) {
|
|
|
|
registerUsage[j].lastUsed = i;
|
2019-03-20 23:38:37 +01:00
|
|
|
}
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(CFROUND) {
|
2019-04-28 16:42:45 +02:00
|
|
|
auto src = instr.src % RegistersCount;
|
2019-02-09 15:45:26 +01:00
|
|
|
ibc.isrc = &r[src];
|
|
|
|
ibc.type = InstructionType::CFROUND;
|
2019-03-11 23:04:34 +01:00
|
|
|
ibc.imm = instr.getImm32() & 63;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(ISTORE) {
|
|
|
|
auto dst = instr.dst % RegistersCount;
|
|
|
|
auto src = instr.src % RegistersCount;
|
|
|
|
ibc.type = InstructionType::ISTORE;
|
|
|
|
ibc.idst = &r[dst];
|
|
|
|
ibc.isrc = &r[src];
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.imm = signExtend2sCompl(instr.getImm32());
|
2019-04-29 23:38:23 +02:00
|
|
|
if (instr.getModCond() < StoreL3Condition)
|
2019-04-16 18:58:44 +02:00
|
|
|
ibc.memMask = (instr.getModMem() ? ScratchpadL1Mask : ScratchpadL2Mask);
|
|
|
|
else
|
|
|
|
ibc.memMask = ScratchpadL3Mask;
|
2019-02-09 15:45:26 +01:00
|
|
|
} break;
|
|
|
|
|
|
|
|
CASE_REP(NOP) {
|
|
|
|
ibc.type = InstructionType::NOP;
|
|
|
|
} break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
UNREACHABLE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-04-20 11:08:01 +02:00
|
|
|
|
|
|
|
template class InterpretedVm<AlignedAllocator<CacheLineSize>, false>;
|
|
|
|
template class InterpretedVm<AlignedAllocator<CacheLineSize>, true>;
|
|
|
|
template class InterpretedVm<LargePageAllocator, false>;
|
|
|
|
template class InterpretedVm<LargePageAllocator, true>;
|
2018-12-11 21:00:30 +01:00
|
|
|
}
|