2018-12-13 22:11:55 +00: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/>.
|
|
|
|
*/
|
2018-12-15 22:13:17 +00:00
|
|
|
//#define TRACE
|
2018-12-13 22:11:55 +00:00
|
|
|
#include "AssemblyGeneratorX86.hpp"
|
|
|
|
#include "Pcg32.hpp"
|
|
|
|
#include "common.hpp"
|
|
|
|
#include "instructions.hpp"
|
|
|
|
|
|
|
|
namespace RandomX {
|
|
|
|
|
2018-12-16 12:43:18 +00:00
|
|
|
static const char* regR[8] = { "r8", "r9", "r10", "r11", "r12", "r13", "r14", "r15" };
|
|
|
|
static const char* regR32[8] = { "r8d", "r9d", "r10d", "r11d", "r12d", "r13d", "r14d", "r15d" };
|
|
|
|
static const char* regF[8] = { "xmm8", "xmm9", "xmm2", "xmm3", "xmm4", "xmm5", "xmm6", "xmm7" };
|
2018-12-13 22:11:55 +00:00
|
|
|
|
|
|
|
void AssemblyGeneratorX86::generateProgram(const void* seed) {
|
|
|
|
asmCode.str(std::string()); //clear
|
|
|
|
Pcg32 gen(seed);
|
|
|
|
for (unsigned i = 0; i < sizeof(RegisterFile) / sizeof(Pcg32::result_type); ++i) {
|
|
|
|
gen();
|
|
|
|
}
|
|
|
|
Instruction instr;
|
|
|
|
for (unsigned i = 0; i < ProgramLength; ++i) {
|
|
|
|
for (unsigned j = 0; j < sizeof(instr) / sizeof(Pcg32::result_type); ++j) {
|
|
|
|
*(((uint32_t*)&instr) + j) = gen();
|
|
|
|
}
|
|
|
|
generateCode(instr, i);
|
|
|
|
asmCode << std::endl;
|
|
|
|
}
|
|
|
|
if(ProgramLength > 0)
|
|
|
|
asmCode << "\tjmp rx_i_0" << std::endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::generateCode(Instruction& instr, int i) {
|
|
|
|
asmCode << "rx_i_" << i << ": ;" << instr.getName() << std::endl;
|
|
|
|
asmCode << "\tdec edi" << std::endl;
|
|
|
|
asmCode << "\tjs rx_finish" << std::endl;
|
|
|
|
auto generator = engine[instr.opcode];
|
|
|
|
(this->*generator)(instr, i);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::gena(Instruction& instr) {
|
2018-12-20 17:36:09 +00:00
|
|
|
asmCode << "\txor " << regR[instr.rega % RegistersCount] << ", 0" << std::hex << instr.addra << "h" << std::dec << std::endl;
|
2018-12-13 22:11:55 +00:00
|
|
|
switch (instr.loca & 7)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
case 1:
|
|
|
|
case 2:
|
|
|
|
case 3:
|
|
|
|
asmCode << "\tmov ecx, " << regR32[instr.rega % RegistersCount] << std::endl;
|
|
|
|
asmCode << "\tcall rx_read_dataset" << std::endl;
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 4:
|
|
|
|
asmCode << "\tmov eax, " << regR32[instr.rega % RegistersCount] << std::endl;
|
|
|
|
asmCode << "\tand eax, " << (ScratchpadL2 - 1) << std::endl;
|
|
|
|
asmCode << "\tmov rax, qword ptr [rsi + rax * 8]" << std::endl;
|
|
|
|
return;
|
|
|
|
|
|
|
|
default:
|
|
|
|
asmCode << "\tmov eax, " << regR32[instr.rega % RegistersCount] << std::endl;
|
|
|
|
asmCode << "\tand eax, " << (ScratchpadL1 - 1) << std::endl;
|
|
|
|
asmCode << "\tmov rax, qword ptr [rsi + rax * 8]" << std::endl;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::genbr0(Instruction& instr, const char* instrx86) {
|
|
|
|
switch (instr.locb & 7)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
case 1:
|
|
|
|
case 2:
|
|
|
|
case 3:
|
|
|
|
case 4:
|
|
|
|
case 5:
|
|
|
|
asmCode << "\tmov rcx, " << regR[instr.regb % RegistersCount] << std::endl;
|
|
|
|
asmCode << "\t" << instrx86 << " rax, cl" << std::endl;
|
|
|
|
return;
|
|
|
|
default:
|
2018-12-20 17:36:09 +00:00
|
|
|
asmCode << "\t" << instrx86 << " rax, " << (instr.imm8 & 63) << std::endl;;
|
2018-12-13 22:11:55 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::genbr1(Instruction& instr) {
|
|
|
|
switch (instr.locb & 7)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
case 1:
|
|
|
|
case 2:
|
|
|
|
case 3:
|
|
|
|
case 4:
|
|
|
|
case 5:
|
|
|
|
asmCode << regR[instr.regb % RegistersCount] << std::endl;
|
|
|
|
return;
|
|
|
|
default:
|
2018-12-20 17:36:09 +00:00
|
|
|
asmCode << instr.imm32 << std::endl;;
|
2018-12-13 22:11:55 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::genbr132(Instruction& instr) {
|
|
|
|
switch (instr.locb & 7)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
case 1:
|
|
|
|
case 2:
|
|
|
|
case 3:
|
|
|
|
case 4:
|
|
|
|
case 5:
|
|
|
|
asmCode << regR32[instr.regb % RegistersCount] << std::endl;
|
|
|
|
return;
|
|
|
|
default:
|
2018-12-20 17:36:09 +00:00
|
|
|
asmCode << instr.imm32 << std::endl;;
|
2018-12-13 22:11:55 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::genbf(Instruction& instr, const char* instrx86) {
|
|
|
|
asmCode << "\tand rax, -2048" << std::endl;
|
|
|
|
asmCode << "\tcvtsi2sd xmm0, rax" << std::endl;
|
|
|
|
switch (instr.locb & 7)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
case 1:
|
|
|
|
case 2:
|
|
|
|
case 3:
|
|
|
|
case 4:
|
|
|
|
case 5:
|
|
|
|
asmCode << "\t" << instrx86 << " xmm0, " << regF[instr.regb % RegistersCount] << std::endl;
|
|
|
|
return;
|
|
|
|
default:
|
|
|
|
convertible_t bimm;
|
2018-12-20 17:36:09 +00:00
|
|
|
bimm.f64 = (double)instr.imm32;
|
2018-12-13 22:11:55 +00:00
|
|
|
asmCode << "\tmov rax, " << bimm.i64 << std::endl;
|
2018-12-16 12:43:18 +00:00
|
|
|
asmCode << "\tmovd xmm1, rax" << std::endl;
|
|
|
|
asmCode << "\t" << instrx86 << " xmm0, xmm1" << std::endl;
|
2018-12-13 22:11:55 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::gencr(Instruction& instr) {
|
|
|
|
switch (instr.locc & 7)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
asmCode << "\tmov rcx, rax" << std::endl;
|
|
|
|
asmCode << "\tmov eax, " << regR32[instr.regc % RegistersCount] << std::endl;
|
2018-12-20 17:36:09 +00:00
|
|
|
asmCode << "\txor eax, 0" << std::hex << instr.addrc << "h" << std::dec << std::endl;
|
2018-12-13 22:11:55 +00:00
|
|
|
asmCode << "\tand eax, " << (ScratchpadL2 - 1) << std::endl;
|
|
|
|
asmCode << "\tmov qword ptr [rsi + rax * 8], rcx" << std::endl;
|
2018-12-15 22:13:17 +00:00
|
|
|
if (trace) {
|
|
|
|
asmCode << "\tmov qword ptr [rsi + rdi * 8 + 262144], rcx" << std::endl;
|
|
|
|
}
|
2018-12-13 22:11:55 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
case 2:
|
|
|
|
case 3:
|
|
|
|
asmCode << "\tmov rcx, rax" << std::endl;
|
|
|
|
asmCode << "\tmov eax, " << regR32[instr.regc % RegistersCount] << std::endl;
|
2018-12-20 17:36:09 +00:00
|
|
|
asmCode << "\txor eax, 0" << std::hex << instr.addrc << "h" << std::dec << std::endl;
|
2018-12-13 22:11:55 +00:00
|
|
|
asmCode << "\tand eax, " << (ScratchpadL1 - 1) << std::endl;
|
|
|
|
asmCode << "\tmov qword ptr [rsi + rax * 8], rcx" << std::endl;
|
2018-12-15 22:13:17 +00:00
|
|
|
if (trace) {
|
|
|
|
asmCode << "\tmov qword ptr [rsi + rdi * 8 + 262144], rcx" << std::endl;
|
|
|
|
}
|
2018-12-13 22:11:55 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
default:
|
|
|
|
asmCode << "\tmov " << regR[instr.regc % RegistersCount] << ", rax" << std::endl;
|
2018-12-15 22:13:17 +00:00
|
|
|
if (trace) {
|
|
|
|
asmCode << "\tmov qword ptr [rsi + rdi * 8 + 262144], rax" << std::endl;
|
|
|
|
}
|
2018-12-13 22:11:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::gencf(Instruction& instr) {
|
|
|
|
switch (instr.locc & 7)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
asmCode << "\tmov eax, " << regR32[instr.regc % RegistersCount] << std::endl;
|
2018-12-20 17:36:09 +00:00
|
|
|
asmCode << "\txor eax, 0" << std::hex << instr.addrc << "h" << std::dec << std::endl;
|
2018-12-13 22:11:55 +00:00
|
|
|
asmCode << "\tand eax, " << (ScratchpadL2 - 1) << std::endl;
|
|
|
|
asmCode << "\tmovd qword ptr [rsi + rax * 8], xmm0" << std::endl;
|
2018-12-15 22:13:17 +00:00
|
|
|
break;
|
2018-12-13 22:11:55 +00:00
|
|
|
|
|
|
|
case 1:
|
|
|
|
case 2:
|
|
|
|
case 3:
|
|
|
|
asmCode << "\tmov eax, " << regR32[instr.regc % RegistersCount] << std::endl;
|
2018-12-20 17:36:09 +00:00
|
|
|
asmCode << "\txor eax, 0" << std::hex << instr.addrc << "h" << std::dec << std::endl;
|
2018-12-13 22:11:55 +00:00
|
|
|
asmCode << "\tand eax, " << (ScratchpadL1 - 1) << std::endl;
|
|
|
|
asmCode << "\tmovd qword ptr [rsi + rax * 8], xmm0" << std::endl;
|
2018-12-15 22:13:17 +00:00
|
|
|
break;
|
2018-12-13 22:11:55 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
asmCode << "\tmovsd " << regF[instr.regc % RegistersCount] << ", xmm0" << std::endl;
|
2018-12-15 22:13:17 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (trace) {
|
|
|
|
asmCode << "\tmovd qword ptr [rsi + rdi * 8 + 262144], xmm0" << std::endl;
|
2018-12-13 22:11:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_ADD_64(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tadd rax, ";
|
|
|
|
genbr1(instr);
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_ADD_32(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tadd eax, ";
|
|
|
|
genbr132(instr);
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_SUB_64(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tsub rax, ";
|
|
|
|
genbr1(instr);
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_SUB_32(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tsub eax, ";
|
|
|
|
genbr132(instr);
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_MUL_64(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\timul rax, ";
|
|
|
|
if ((instr.locb & 7) >= 6) {
|
|
|
|
asmCode << "rax, ";
|
|
|
|
}
|
|
|
|
genbr1(instr);
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_MULH_64(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tmov rcx, ";
|
|
|
|
genbr1(instr);
|
|
|
|
asmCode << "\tmul rcx" << std::endl;
|
|
|
|
asmCode << "\tmov rax, rdx" << std::endl;
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_MUL_32(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tmov ecx, eax" << std::endl;
|
|
|
|
asmCode << "\tmov eax, ";
|
|
|
|
genbr132(instr);
|
|
|
|
asmCode << "\timul rax, rcx" << std::endl;
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_IMUL_32(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tmovsxd rcx, eax" << std::endl;
|
|
|
|
if ((instr.locb & 7) >= 6) {
|
2018-12-20 17:36:09 +00:00
|
|
|
asmCode << "\tmov rax, " << instr.imm32 << std::endl;
|
2018-12-13 22:11:55 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
asmCode << "\tmovsxd rax, " << regR32[instr.regb % RegistersCount] << std::endl;
|
|
|
|
}
|
|
|
|
asmCode << "\timul rax, rcx" << std::endl;
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_IMULH_64(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tmov rcx, ";
|
|
|
|
genbr1(instr);
|
|
|
|
asmCode << "\timul rcx" << std::endl;
|
|
|
|
asmCode << "\tmov rax, rdx" << std::endl;
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_DIV_64(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
if ((instr.locb & 7) >= 6) {
|
2018-12-20 17:36:09 +00:00
|
|
|
if (instr.imm32 == 0) {
|
2018-12-13 22:11:55 +00:00
|
|
|
asmCode << "\tmov ecx, 1" << std::endl;
|
|
|
|
}
|
|
|
|
else {
|
2018-12-20 17:36:09 +00:00
|
|
|
asmCode << "\tmov ecx, " << instr.imm32 << std::endl;
|
2018-12-13 22:11:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2018-12-21 21:41:35 +00:00
|
|
|
asmCode << "\tmov ecx, 1" << std::endl;
|
2018-12-13 22:11:55 +00:00
|
|
|
asmCode << "\tmov edx, " << regR32[instr.regb % RegistersCount] << std::endl;
|
|
|
|
asmCode << "\ttest edx, edx" << std::endl;
|
|
|
|
asmCode << "\tcmovne ecx, edx" << std::endl;
|
|
|
|
}
|
|
|
|
asmCode << "\txor edx, edx" << std::endl;
|
|
|
|
asmCode << "\tdiv rcx" << std::endl;
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_IDIV_64(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tmov edx, ";
|
|
|
|
genbr132(instr);
|
|
|
|
asmCode << "\tcmp edx, -1" << std::endl;
|
|
|
|
asmCode << "\tjne short safe_idiv_" << i << std::endl;
|
|
|
|
asmCode << "\tmov rcx, rax" << std::endl;
|
|
|
|
asmCode << "\trol rcx, 1" << std::endl;
|
|
|
|
asmCode << "\tdec rcx" << std::endl;
|
|
|
|
asmCode << "\tjz short result_idiv_" << i << std::endl;
|
|
|
|
asmCode << "safe_idiv_" << i << ":" << std::endl;
|
|
|
|
asmCode << "\tmov ecx, 1" << std::endl;
|
|
|
|
asmCode << "\ttest edx, edx" << std::endl;
|
|
|
|
asmCode << "\tcmovne ecx, edx" << std::endl;
|
|
|
|
asmCode << "\tmovsxd rcx, ecx" << std::endl;
|
|
|
|
asmCode << "\tcqo" << std::endl;
|
|
|
|
asmCode << "\tidiv rcx" << std::endl;
|
|
|
|
asmCode << "result_idiv_" << i << ":" << std::endl;
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_AND_64(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tand rax, ";
|
|
|
|
genbr1(instr);
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_AND_32(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tand eax, ";
|
|
|
|
genbr132(instr);
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_OR_64(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tor rax, ";
|
|
|
|
genbr1(instr);
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_OR_32(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tor eax, ";
|
|
|
|
genbr132(instr);
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_XOR_64(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\txor rax, ";
|
|
|
|
genbr1(instr);
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_XOR_32(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\txor eax, ";
|
|
|
|
genbr132(instr);
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_SHL_64(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
genbr0(instr, "shl");
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_SHR_64(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
genbr0(instr, "shr");
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_SAR_64(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
genbr0(instr, "sar");
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_ROL_64(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
genbr0(instr, "rol");
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_ROR_64(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
genbr0(instr, "ror");
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_FPADD(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
genbf(instr, "addsd");
|
|
|
|
gencf(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_FPSUB(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
genbf(instr, "subsd");
|
|
|
|
gencf(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_FPMUL(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tor rax, 2048" << std::endl;
|
|
|
|
genbf(instr, "mulsd");
|
|
|
|
gencf(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_FPDIV(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tor rax, 2048" << std::endl;
|
|
|
|
genbf(instr, "divsd");
|
|
|
|
gencf(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_FPSQRT(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tmov rcx, 9223372036854773760" << std::endl;
|
|
|
|
asmCode << "\tand rax, rcx" << std::endl;
|
|
|
|
asmCode << "\tcvtsi2sd xmm0, rax" << std::endl;
|
|
|
|
asmCode << "\tsqrtsd xmm0, xmm0" << std::endl;
|
|
|
|
gencf(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_FPROUND(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tmov rcx, rax" << std::endl;
|
|
|
|
asmCode << "\tshl eax, 13" << std::endl;
|
|
|
|
asmCode << "\tand rcx, -2048" << std::endl;
|
|
|
|
asmCode << "\tand eax, 24576" << std::endl;
|
|
|
|
asmCode << "\tcvtsi2sd xmm0, rcx" << std::endl;
|
|
|
|
asmCode << "\tor eax, 40896" << std::endl;
|
|
|
|
asmCode << "\tmov dword ptr [rsp - 8], eax" << std::endl;
|
|
|
|
asmCode << "\tldmxcsr dword ptr [rsp - 8]" << std::endl;
|
|
|
|
gencf(instr);
|
|
|
|
}
|
|
|
|
|
2018-12-21 21:41:35 +00:00
|
|
|
static inline const char* jumpCondition(Instruction& instr, bool invert = false) {
|
|
|
|
switch ((instr.locb & 7) ^ invert)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
return "jbe";
|
|
|
|
case 1:
|
|
|
|
return "ja";
|
|
|
|
case 2:
|
|
|
|
return "js";
|
|
|
|
case 3:
|
|
|
|
return "jns";
|
|
|
|
case 4:
|
|
|
|
return "jo";
|
|
|
|
case 5:
|
|
|
|
return "jno";
|
|
|
|
case 6:
|
|
|
|
return "jl";
|
|
|
|
case 7:
|
|
|
|
return "jge";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-13 22:11:55 +00:00
|
|
|
void AssemblyGeneratorX86::h_CALL(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
2018-12-21 21:41:35 +00:00
|
|
|
asmCode << "\tcmp " << regR32[instr.regb % RegistersCount] << ", " << instr.imm32 << std::endl;
|
|
|
|
asmCode << "\t" << jumpCondition(instr);
|
|
|
|
asmCode << " short taken_call_" << i << std::endl;
|
|
|
|
gencr(instr);
|
|
|
|
asmCode << "\tjmp rx_i_" << wrapInstr(i + 1) << std::endl;
|
|
|
|
asmCode << "taken_call_" << i << ":" << std::endl;
|
2018-12-15 22:13:17 +00:00
|
|
|
if (trace) {
|
|
|
|
asmCode << "\tmov qword ptr [rsi + rdi * 8 + 262144], rax" << std::endl;
|
|
|
|
}
|
2018-12-13 22:11:55 +00:00
|
|
|
asmCode << "\tpush rax" << std::endl;
|
2018-12-20 17:36:09 +00:00
|
|
|
asmCode << "\tcall rx_i_" << wrapInstr(i + (instr.imm8 & 127) + 2) << std::endl;
|
2018-12-13 22:11:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void AssemblyGeneratorX86::h_RET(Instruction& instr, int i) {
|
|
|
|
gena(instr);
|
|
|
|
asmCode << "\tcmp rsp, rbp" << std::endl;
|
|
|
|
asmCode << "\tje short not_taken_ret_" << i << std::endl;
|
|
|
|
asmCode << "\txor rax, qword ptr [rsp + 8]" << std::endl;
|
|
|
|
gencr(instr);
|
|
|
|
asmCode << "\tret 8" << std::endl;
|
|
|
|
asmCode << "not_taken_ret_" << i << ":" << std::endl;
|
|
|
|
gencr(instr);
|
|
|
|
}
|
|
|
|
|
|
|
|
#include "instructionWeights.hpp"
|
|
|
|
#define INST_HANDLE(x) REPN(&AssemblyGeneratorX86::h_##x, WT(x))
|
|
|
|
|
|
|
|
InstructionGenerator AssemblyGeneratorX86::engine[256] = {
|
|
|
|
INST_HANDLE(ADD_64)
|
|
|
|
INST_HANDLE(ADD_32)
|
|
|
|
INST_HANDLE(SUB_64)
|
|
|
|
INST_HANDLE(SUB_32)
|
|
|
|
INST_HANDLE(MUL_64)
|
|
|
|
INST_HANDLE(MULH_64)
|
|
|
|
INST_HANDLE(MUL_32)
|
|
|
|
INST_HANDLE(IMUL_32)
|
|
|
|
INST_HANDLE(IMULH_64)
|
|
|
|
INST_HANDLE(DIV_64)
|
|
|
|
INST_HANDLE(IDIV_64)
|
|
|
|
INST_HANDLE(AND_64)
|
|
|
|
INST_HANDLE(AND_32)
|
|
|
|
INST_HANDLE(OR_64)
|
|
|
|
INST_HANDLE(OR_32)
|
|
|
|
INST_HANDLE(XOR_64)
|
|
|
|
INST_HANDLE(XOR_32)
|
|
|
|
INST_HANDLE(SHL_64)
|
|
|
|
INST_HANDLE(SHR_64)
|
|
|
|
INST_HANDLE(SAR_64)
|
|
|
|
INST_HANDLE(ROL_64)
|
|
|
|
INST_HANDLE(ROR_64)
|
|
|
|
INST_HANDLE(FPADD)
|
|
|
|
INST_HANDLE(FPSUB)
|
|
|
|
INST_HANDLE(FPMUL)
|
|
|
|
INST_HANDLE(FPDIV)
|
|
|
|
INST_HANDLE(FPSQRT)
|
|
|
|
INST_HANDLE(FPROUND)
|
|
|
|
INST_HANDLE(CALL)
|
|
|
|
INST_HANDLE(RET)
|
|
|
|
};
|
|
|
|
}
|