early-access version 1255
This commit is contained in:
222
externals/dynarmic/src/frontend/ir/value.cpp
vendored
Executable file
222
externals/dynarmic/src/frontend/ir/value.cpp
vendored
Executable file
@@ -0,0 +1,222 @@
|
||||
/* This file is part of the dynarmic project.
|
||||
* Copyright (c) 2016 MerryMage
|
||||
* SPDX-License-Identifier: 0BSD
|
||||
*/
|
||||
|
||||
#include "common/assert.h"
|
||||
#include "common/bit_util.h"
|
||||
#include "frontend/ir/microinstruction.h"
|
||||
#include "frontend/ir/opcodes.h"
|
||||
#include "frontend/ir/type.h"
|
||||
#include "frontend/ir/value.h"
|
||||
|
||||
namespace Dynarmic::IR {
|
||||
|
||||
Value::Value(Inst* value) : type(Type::Opaque) {
|
||||
inner.inst = value;
|
||||
}
|
||||
|
||||
Value::Value(A32::Reg value) : type(Type::A32Reg) {
|
||||
inner.imm_a32regref = value;
|
||||
}
|
||||
|
||||
Value::Value(A32::ExtReg value) : type(Type::A32ExtReg) {
|
||||
inner.imm_a32extregref = value;
|
||||
}
|
||||
|
||||
Value::Value(A64::Reg value) : type(Type::A64Reg) {
|
||||
inner.imm_a64regref = value;
|
||||
}
|
||||
|
||||
Value::Value(A64::Vec value) : type(Type::A64Vec) {
|
||||
inner.imm_a64vecref = value;
|
||||
}
|
||||
|
||||
Value::Value(bool value) : type(Type::U1) {
|
||||
inner.imm_u1 = value;
|
||||
}
|
||||
|
||||
Value::Value(u8 value) : type(Type::U8) {
|
||||
inner.imm_u8 = value;
|
||||
}
|
||||
|
||||
Value::Value(u16 value) : type(Type::U16) {
|
||||
inner.imm_u16 = value;
|
||||
}
|
||||
|
||||
Value::Value(u32 value) : type(Type::U32) {
|
||||
inner.imm_u32 = value;
|
||||
}
|
||||
|
||||
Value::Value(u64 value) : type(Type::U64) {
|
||||
inner.imm_u64 = value;
|
||||
}
|
||||
|
||||
Value::Value(CoprocessorInfo value) : type(Type::CoprocInfo) {
|
||||
inner.imm_coproc = value;
|
||||
}
|
||||
|
||||
Value::Value(Cond value) : type(Type::Cond) {
|
||||
inner.imm_cond = value;
|
||||
}
|
||||
|
||||
bool Value::IsIdentity() const {
|
||||
if (type == Type::Opaque)
|
||||
return inner.inst->GetOpcode() == Opcode::Identity;
|
||||
return false;
|
||||
}
|
||||
|
||||
bool Value::IsImmediate() const {
|
||||
if (IsIdentity())
|
||||
return inner.inst->GetArg(0).IsImmediate();
|
||||
return type != Type::Opaque;
|
||||
}
|
||||
|
||||
bool Value::IsEmpty() const {
|
||||
return type == Type::Void;
|
||||
}
|
||||
|
||||
Type Value::GetType() const {
|
||||
if (IsIdentity())
|
||||
return inner.inst->GetArg(0).GetType();
|
||||
if (type == Type::Opaque)
|
||||
return inner.inst->GetType();
|
||||
return type;
|
||||
}
|
||||
|
||||
A32::Reg Value::GetA32RegRef() const {
|
||||
ASSERT(type == Type::A32Reg);
|
||||
return inner.imm_a32regref;
|
||||
}
|
||||
|
||||
A32::ExtReg Value::GetA32ExtRegRef() const {
|
||||
ASSERT(type == Type::A32ExtReg);
|
||||
return inner.imm_a32extregref;
|
||||
}
|
||||
|
||||
A64::Reg Value::GetA64RegRef() const {
|
||||
ASSERT(type == Type::A64Reg);
|
||||
return inner.imm_a64regref;
|
||||
}
|
||||
|
||||
A64::Vec Value::GetA64VecRef() const {
|
||||
ASSERT(type == Type::A64Vec);
|
||||
return inner.imm_a64vecref;
|
||||
}
|
||||
|
||||
Inst* Value::GetInst() const {
|
||||
ASSERT(type == Type::Opaque);
|
||||
return inner.inst;
|
||||
}
|
||||
|
||||
Inst* Value::GetInstRecursive() const {
|
||||
ASSERT(type == Type::Opaque);
|
||||
if (IsIdentity())
|
||||
return inner.inst->GetArg(0).GetInstRecursive();
|
||||
return inner.inst;
|
||||
}
|
||||
|
||||
bool Value::GetU1() const {
|
||||
if (IsIdentity())
|
||||
return inner.inst->GetArg(0).GetU1();
|
||||
ASSERT(type == Type::U1);
|
||||
return inner.imm_u1;
|
||||
}
|
||||
|
||||
u8 Value::GetU8() const {
|
||||
if (IsIdentity())
|
||||
return inner.inst->GetArg(0).GetU8();
|
||||
ASSERT(type == Type::U8);
|
||||
return inner.imm_u8;
|
||||
}
|
||||
|
||||
u16 Value::GetU16() const {
|
||||
if (IsIdentity())
|
||||
return inner.inst->GetArg(0).GetU16();
|
||||
ASSERT(type == Type::U16);
|
||||
return inner.imm_u16;
|
||||
}
|
||||
|
||||
u32 Value::GetU32() const {
|
||||
if (IsIdentity())
|
||||
return inner.inst->GetArg(0).GetU32();
|
||||
ASSERT(type == Type::U32);
|
||||
return inner.imm_u32;
|
||||
}
|
||||
|
||||
u64 Value::GetU64() const {
|
||||
if (IsIdentity())
|
||||
return inner.inst->GetArg(0).GetU64();
|
||||
ASSERT(type == Type::U64);
|
||||
return inner.imm_u64;
|
||||
}
|
||||
|
||||
Value::CoprocessorInfo Value::GetCoprocInfo() const {
|
||||
if (IsIdentity())
|
||||
return inner.inst->GetArg(0).GetCoprocInfo();
|
||||
ASSERT(type == Type::CoprocInfo);
|
||||
return inner.imm_coproc;
|
||||
}
|
||||
|
||||
Cond Value::GetCond() const {
|
||||
if (IsIdentity())
|
||||
return inner.inst->GetArg(0).GetCond();
|
||||
ASSERT(type == Type::Cond);
|
||||
return inner.imm_cond;
|
||||
}
|
||||
|
||||
s64 Value::GetImmediateAsS64() const {
|
||||
ASSERT(IsImmediate());
|
||||
|
||||
switch (GetType()) {
|
||||
case IR::Type::U1:
|
||||
return s64(GetU1());
|
||||
case IR::Type::U8:
|
||||
return s64(Common::SignExtend<8, u64>(GetU8()));
|
||||
case IR::Type::U16:
|
||||
return s64(Common::SignExtend<16, u64>(GetU16()));
|
||||
case IR::Type::U32:
|
||||
return s64(Common::SignExtend<32, u64>(GetU32()));
|
||||
case IR::Type::U64:
|
||||
return s64(GetU64());
|
||||
default:
|
||||
ASSERT_FALSE("GetImmediateAsS64 called on an incompatible Value type.");
|
||||
}
|
||||
}
|
||||
|
||||
u64 Value::GetImmediateAsU64() const {
|
||||
ASSERT(IsImmediate());
|
||||
|
||||
switch (GetType()) {
|
||||
case IR::Type::U1:
|
||||
return u64(GetU1());
|
||||
case IR::Type::U8:
|
||||
return u64(GetU8());
|
||||
case IR::Type::U16:
|
||||
return u64(GetU16());
|
||||
case IR::Type::U32:
|
||||
return u64(GetU32());
|
||||
case IR::Type::U64:
|
||||
return u64(GetU64());
|
||||
default:
|
||||
ASSERT_FALSE("GetImmediateAsU64 called on an incompatible Value type.");
|
||||
}
|
||||
}
|
||||
|
||||
bool Value::IsSignedImmediate(s64 value) const {
|
||||
return IsImmediate() && GetImmediateAsS64() == value;
|
||||
}
|
||||
|
||||
bool Value::IsUnsignedImmediate(u64 value) const {
|
||||
return IsImmediate() && GetImmediateAsU64() == value;
|
||||
}
|
||||
|
||||
bool Value::HasAllBitsSet() const {
|
||||
return IsSignedImmediate(-1);
|
||||
}
|
||||
|
||||
bool Value::IsZero() const {
|
||||
return IsUnsignedImmediate(0);
|
||||
}
|
||||
|
||||
} // namespace Dynarmic::IR
|
||||
Reference in New Issue
Block a user