2016-07-01 15:01:06 +02:00
|
|
|
/* This file is part of the dynarmic project.
|
|
|
|
* Copyright (c) 2016 MerryMage
|
|
|
|
* This software may be used and distributed according to the terms of the GNU
|
|
|
|
* General Public License version 2 or any later version.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2016-08-17 16:53:36 +02:00
|
|
|
#include <initializer_list>
|
|
|
|
|
|
|
|
#include "common/common_types.h"
|
2016-07-04 11:22:11 +02:00
|
|
|
#include "frontend/arm_types.h"
|
2016-08-17 16:53:36 +02:00
|
|
|
#include "frontend/ir/basic_block.h"
|
2016-07-04 11:22:11 +02:00
|
|
|
#include "frontend/ir/opcodes.h"
|
2016-08-17 16:53:36 +02:00
|
|
|
#include "frontend/ir/terminal.h"
|
|
|
|
#include "frontend/ir/value.h"
|
|
|
|
|
|
|
|
// ARM JIT Microinstruction Intermediate Representation
|
|
|
|
//
|
|
|
|
// This intermediate representation is an SSA IR. It is designed primarily for analysis,
|
|
|
|
// though it can be lowered into a reduced form for interpretation. Each IR node (Value)
|
|
|
|
// is a microinstruction of an idealised ARM CPU. The choice of microinstructions is made
|
|
|
|
// not based on any existing microarchitecture but on ease of implementation.
|
2016-07-01 15:01:06 +02:00
|
|
|
|
|
|
|
namespace Dynarmic {
|
|
|
|
namespace Arm {
|
|
|
|
|
2016-08-12 19:17:31 +02:00
|
|
|
/**
|
|
|
|
* Convenience class to construct a basic block of the intermediate representation.
|
|
|
|
* `block` is the resulting block.
|
|
|
|
* The user of this class updates `current_location` as appropriate.
|
|
|
|
*/
|
2016-07-01 15:01:06 +02:00
|
|
|
class IREmitter {
|
|
|
|
public:
|
2016-07-04 15:37:50 +02:00
|
|
|
explicit IREmitter(LocationDescriptor descriptor) : block(descriptor), current_location(descriptor) {}
|
|
|
|
|
|
|
|
IR::Block block;
|
|
|
|
LocationDescriptor current_location;
|
2016-07-01 15:01:06 +02:00
|
|
|
|
|
|
|
struct ResultAndCarry {
|
2016-07-23 00:55:00 +02:00
|
|
|
IR::Value result;
|
|
|
|
IR::Value carry;
|
2016-07-01 15:01:06 +02:00
|
|
|
};
|
|
|
|
|
2016-07-08 11:09:18 +02:00
|
|
|
struct ResultAndCarryAndOverflow {
|
2016-07-23 00:55:00 +02:00
|
|
|
IR::Value result;
|
|
|
|
IR::Value carry;
|
|
|
|
IR::Value overflow;
|
2016-07-08 11:09:18 +02:00
|
|
|
};
|
|
|
|
|
2016-07-01 15:01:06 +02:00
|
|
|
void Unimplemented();
|
2016-07-11 23:43:53 +02:00
|
|
|
u32 PC();
|
|
|
|
u32 AlignPC(size_t alignment);
|
2016-07-01 15:01:06 +02:00
|
|
|
|
2016-07-23 00:55:00 +02:00
|
|
|
IR::Value Imm1(bool value);
|
|
|
|
IR::Value Imm8(u8 value);
|
|
|
|
IR::Value Imm32(u32 value);
|
|
|
|
|
|
|
|
IR::Value GetRegister(Reg source_reg);
|
2016-08-05 19:54:19 +02:00
|
|
|
IR::Value GetExtendedRegister(ExtReg source_reg);
|
2016-07-23 00:55:00 +02:00
|
|
|
void SetRegister(const Reg dest_reg, const IR::Value& value);
|
2016-08-05 19:54:19 +02:00
|
|
|
void SetExtendedRegister(const ExtReg dest_reg, const IR::Value& value);
|
2016-07-23 00:55:00 +02:00
|
|
|
|
|
|
|
void ALUWritePC(const IR::Value& value);
|
|
|
|
void BranchWritePC(const IR::Value& value);
|
|
|
|
void BXWritePC(const IR::Value& value);
|
|
|
|
void LoadWritePC(const IR::Value& value);
|
|
|
|
void CallSupervisor(const IR::Value& value);
|
2016-08-13 01:10:23 +02:00
|
|
|
void PushRSB(const LocationDescriptor& return_location);
|
2016-07-23 00:55:00 +02:00
|
|
|
|
2016-08-14 20:39:16 +02:00
|
|
|
IR::Value GetCpsr();
|
|
|
|
void SetCpsr(const IR::Value& value);
|
2016-07-23 00:55:00 +02:00
|
|
|
IR::Value GetCFlag();
|
|
|
|
void SetNFlag(const IR::Value& value);
|
|
|
|
void SetZFlag(const IR::Value& value);
|
|
|
|
void SetCFlag(const IR::Value& value);
|
|
|
|
void SetVFlag(const IR::Value& value);
|
2016-08-06 23:04:52 +02:00
|
|
|
void OrQFlag(const IR::Value& value);
|
2016-07-23 00:55:00 +02:00
|
|
|
|
2016-08-04 23:04:42 +02:00
|
|
|
IR::Value Pack2x32To1x64(const IR::Value& lo, const IR::Value& hi);
|
|
|
|
IR::Value LeastSignificantWord(const IR::Value& value);
|
2016-08-06 22:03:57 +02:00
|
|
|
ResultAndCarry MostSignificantWord(const IR::Value& value);
|
2016-07-23 00:55:00 +02:00
|
|
|
IR::Value LeastSignificantHalf(const IR::Value& value);
|
|
|
|
IR::Value LeastSignificantByte(const IR::Value& value);
|
|
|
|
IR::Value MostSignificantBit(const IR::Value& value);
|
|
|
|
IR::Value IsZero(const IR::Value& value);
|
2016-08-04 23:04:42 +02:00
|
|
|
IR::Value IsZero64(const IR::Value& value);
|
2016-07-23 00:55:00 +02:00
|
|
|
|
|
|
|
ResultAndCarry LogicalShiftLeft(const IR::Value& value_in, const IR::Value& shift_amount, const IR::Value& carry_in);
|
|
|
|
ResultAndCarry LogicalShiftRight(const IR::Value& value_in, const IR::Value& shift_amount, const IR::Value& carry_in);
|
2016-08-07 15:23:33 +02:00
|
|
|
IR::Value LogicalShiftRight64(const IR::Value& value_in, const IR::Value& shift_amount);
|
2016-07-23 00:55:00 +02:00
|
|
|
ResultAndCarry ArithmeticShiftRight(const IR::Value& value_in, const IR::Value& shift_amount, const IR::Value& carry_in);
|
|
|
|
ResultAndCarry RotateRight(const IR::Value& value_in, const IR::Value& shift_amount, const IR::Value& carry_in);
|
2016-07-31 20:07:35 +02:00
|
|
|
ResultAndCarry RotateRightExtended(const IR::Value& value_in, const IR::Value& carry_in);
|
2016-07-23 00:55:00 +02:00
|
|
|
ResultAndCarryAndOverflow AddWithCarry(const IR::Value& a, const IR::Value& b, const IR::Value& carry_in);
|
|
|
|
IR::Value Add(const IR::Value& a, const IR::Value& b);
|
2016-08-04 23:04:42 +02:00
|
|
|
IR::Value Add64(const IR::Value& a, const IR::Value& b);
|
2016-07-23 00:55:00 +02:00
|
|
|
ResultAndCarryAndOverflow SubWithCarry(const IR::Value& a, const IR::Value& b, const IR::Value& carry_in);
|
|
|
|
IR::Value Sub(const IR::Value& a, const IR::Value& b);
|
2016-08-06 07:09:47 +02:00
|
|
|
IR::Value Sub64(const IR::Value& a, const IR::Value& b);
|
2016-08-04 23:04:42 +02:00
|
|
|
IR::Value Mul(const IR::Value& a, const IR::Value& b);
|
|
|
|
IR::Value Mul64(const IR::Value& a, const IR::Value& b);
|
2016-07-23 00:55:00 +02:00
|
|
|
IR::Value And(const IR::Value& a, const IR::Value& b);
|
|
|
|
IR::Value Eor(const IR::Value& a, const IR::Value& b);
|
|
|
|
IR::Value Or(const IR::Value& a, const IR::Value& b);
|
|
|
|
IR::Value Not(const IR::Value& a);
|
2016-08-04 23:04:42 +02:00
|
|
|
IR::Value SignExtendWordToLong(const IR::Value& a);
|
2016-07-23 00:55:00 +02:00
|
|
|
IR::Value SignExtendHalfToWord(const IR::Value& a);
|
|
|
|
IR::Value SignExtendByteToWord(const IR::Value& a);
|
2016-08-04 23:04:42 +02:00
|
|
|
IR::Value ZeroExtendWordToLong(const IR::Value& a);
|
2016-07-23 00:55:00 +02:00
|
|
|
IR::Value ZeroExtendHalfToWord(const IR::Value& a);
|
|
|
|
IR::Value ZeroExtendByteToWord(const IR::Value& a);
|
|
|
|
IR::Value ByteReverseWord(const IR::Value& a);
|
|
|
|
IR::Value ByteReverseHalf(const IR::Value& a);
|
|
|
|
IR::Value ByteReverseDual(const IR::Value& a);
|
2016-08-12 19:26:14 +02:00
|
|
|
IR::Value PackedSaturatedAddU8(const IR::Value& a, const IR::Value& b);
|
|
|
|
IR::Value PackedSaturatedAddS8(const IR::Value& a, const IR::Value& b);
|
2016-08-12 17:53:16 +02:00
|
|
|
IR::Value PackedSaturatedSubU8(const IR::Value& a, const IR::Value& b);
|
2016-08-12 19:18:38 +02:00
|
|
|
IR::Value PackedSaturatedSubS8(const IR::Value& a, const IR::Value& b);
|
2016-08-12 19:42:16 +02:00
|
|
|
IR::Value PackedSaturatedAddU16(const IR::Value& a, const IR::Value& b);
|
|
|
|
IR::Value PackedSaturatedAddS16(const IR::Value& a, const IR::Value& b);
|
|
|
|
IR::Value PackedSaturatedSubU16(const IR::Value& a, const IR::Value& b);
|
|
|
|
IR::Value PackedSaturatedSubS16(const IR::Value& a, const IR::Value& b);
|
2016-07-23 00:55:00 +02:00
|
|
|
|
2016-08-07 20:25:12 +02:00
|
|
|
IR::Value TransferToFP32(const IR::Value& a);
|
|
|
|
IR::Value TransferToFP64(const IR::Value& a);
|
|
|
|
IR::Value TransferFromFP32(const IR::Value& a);
|
|
|
|
IR::Value TransferFromFP64(const IR::Value& a);
|
2016-08-07 02:27:18 +02:00
|
|
|
IR::Value FPAbs32(const IR::Value& a);
|
|
|
|
IR::Value FPAbs64(const IR::Value& a);
|
2016-08-06 18:21:29 +02:00
|
|
|
IR::Value FPAdd32(const IR::Value& a, const IR::Value& b, bool fpscr_controlled);
|
|
|
|
IR::Value FPAdd64(const IR::Value& a, const IR::Value& b, bool fpscr_controlled);
|
2016-08-07 11:56:12 +02:00
|
|
|
IR::Value FPDiv32(const IR::Value& a, const IR::Value& b, bool fpscr_controlled);
|
|
|
|
IR::Value FPDiv64(const IR::Value& a, const IR::Value& b, bool fpscr_controlled);
|
2016-08-07 11:21:14 +02:00
|
|
|
IR::Value FPMul32(const IR::Value& a, const IR::Value& b, bool fpscr_controlled);
|
|
|
|
IR::Value FPMul64(const IR::Value& a, const IR::Value& b, bool fpscr_controlled);
|
2016-08-07 11:56:12 +02:00
|
|
|
IR::Value FPNeg32(const IR::Value& a);
|
|
|
|
IR::Value FPNeg64(const IR::Value& a);
|
2016-08-07 13:19:07 +02:00
|
|
|
IR::Value FPSqrt32(const IR::Value& a);
|
|
|
|
IR::Value FPSqrt64(const IR::Value& a);
|
2016-08-07 02:41:25 +02:00
|
|
|
IR::Value FPSub32(const IR::Value& a, const IR::Value& b, bool fpscr_controlled);
|
|
|
|
IR::Value FPSub64(const IR::Value& a, const IR::Value& b, bool fpscr_controlled);
|
2016-08-23 23:04:46 +02:00
|
|
|
IR::Value FPDoubleToSingle(const IR::Value& a, bool fpscr_controlled);
|
|
|
|
IR::Value FPSingleToDouble(const IR::Value& a, bool fpscr_controlled);
|
|
|
|
IR::Value FPSingleToS32(const IR::Value& a, bool round_towards_zero, bool fpscr_controlled);
|
|
|
|
IR::Value FPSingleToU32(const IR::Value& a, bool round_towards_zero, bool fpscr_controlled);
|
|
|
|
IR::Value FPDoubleToS32(const IR::Value& a, bool round_towards_zero, bool fpscr_controlled);
|
|
|
|
IR::Value FPDoubleToU32(const IR::Value& a, bool round_towards_zero, bool fpscr_controlled);
|
|
|
|
IR::Value FPS32ToSingle(const IR::Value& a, bool round_to_nearest, bool fpscr_controlled);
|
|
|
|
IR::Value FPU32ToSingle(const IR::Value& a, bool round_to_nearest, bool fpscr_controlled);
|
|
|
|
IR::Value FPS32ToDouble(const IR::Value& a, bool round_to_nearest, bool fpscr_controlled);
|
|
|
|
IR::Value FPU32ToDouble(const IR::Value& a, bool round_to_nearest, bool fpscr_controlled);
|
2016-08-06 18:21:29 +02:00
|
|
|
|
TranslateArm: Implement CLREX, LDREX, LDREXB, LDREXD, LDREXH, STREX, STREXB, STREXD, STREXH, SWP, SWPB
2016-08-09 23:48:20 +02:00
|
|
|
void ClearExlcusive();
|
|
|
|
void SetExclusive(const IR::Value& vaddr, size_t byte_size);
|
2016-07-23 00:55:00 +02:00
|
|
|
IR::Value ReadMemory8(const IR::Value& vaddr);
|
|
|
|
IR::Value ReadMemory16(const IR::Value& vaddr);
|
|
|
|
IR::Value ReadMemory32(const IR::Value& vaddr);
|
|
|
|
IR::Value ReadMemory64(const IR::Value& vaddr);
|
|
|
|
void WriteMemory8(const IR::Value& vaddr, const IR::Value& value);
|
|
|
|
void WriteMemory16(const IR::Value& vaddr, const IR::Value& value);
|
|
|
|
void WriteMemory32(const IR::Value& vaddr, const IR::Value& value);
|
|
|
|
void WriteMemory64(const IR::Value& vaddr, const IR::Value& value);
|
TranslateArm: Implement CLREX, LDREX, LDREXB, LDREXD, LDREXH, STREX, STREXB, STREXD, STREXH, SWP, SWPB
2016-08-09 23:48:20 +02:00
|
|
|
IR::Value ExclusiveWriteMemory8(const IR::Value& vaddr, const IR::Value& value);
|
|
|
|
IR::Value ExclusiveWriteMemory16(const IR::Value& vaddr, const IR::Value& value);
|
|
|
|
IR::Value ExclusiveWriteMemory32(const IR::Value& vaddr, const IR::Value& value);
|
|
|
|
IR::Value ExclusiveWriteMemory64(const IR::Value& vaddr, const IR::Value& value_lo, const IR::Value& value_hi);
|
2016-07-11 23:43:53 +02:00
|
|
|
|
2016-08-05 15:07:27 +02:00
|
|
|
void Breakpoint();
|
|
|
|
|
2016-07-07 11:53:09 +02:00
|
|
|
void SetTerm(const IR::Terminal& terminal);
|
|
|
|
|
2016-07-01 15:01:06 +02:00
|
|
|
private:
|
2016-07-23 00:55:00 +02:00
|
|
|
IR::Value Inst(IR::Opcode op, std::initializer_list<IR::Value> args);
|
2016-07-01 15:01:06 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace Arm
|
|
|
|
} // namespace Dynarmic
|