dynarmic/src/frontend/imm.h
2020-04-23 15:45:57 +01:00

160 lines
3.9 KiB
C++

/* This file is part of the dynarmic project.
* Copyright (c) 2018 MerryMage
* SPDX-License-Identifier: 0BSD
*/
#pragma once
#include "common/assert.h"
#include "common/bit_util.h"
#include "common/common_types.h"
#include "common/math_util.h"
namespace Dynarmic {
/**
* Imm represents an immediate value in an AArch32/AArch64 instruction.
* Imm is used during translation as a typesafe way of passing around immediates of fixed sizes.
*/
template <size_t bit_size_>
class Imm {
public:
static constexpr size_t bit_size = bit_size_;
explicit Imm(u32 value) : value(value) {
ASSERT_MSG((Common::Bits<0, bit_size-1>(value) == value), "More bits in value than expected");
}
template <typename T = u32>
T ZeroExtend() const {
static_assert(Common::BitSize<T>() >= bit_size);
return static_cast<T>(value);
}
template <typename T = s32>
T SignExtend() const {
static_assert(Common::BitSize<T>() >= bit_size);
return Common::SignExtend<bit_size, T>(value);
}
template <size_t bit>
bool Bit() const {
static_assert(bit < bit_size);
return Common::Bit<bit>(value);
}
template <size_t begin_bit, size_t end_bit, typename T = u32>
T Bits() const {
static_assert(begin_bit <= end_bit && end_bit < bit_size);
static_assert(Common::BitSize<T>() >= end_bit - begin_bit + 1);
return static_cast<T>(Common::Bits<begin_bit, end_bit>(value));
}
bool operator==(Imm other) const {
return value == other.value;
}
bool operator!=(Imm other) const {
return !operator==(other);
}
bool operator<(Imm other) const {
return value < other.value;
}
bool operator<=(Imm other) const {
return value <= other.value;
}
bool operator>(Imm other) const {
return value > other.value;
}
bool operator>=(Imm other) const {
return value >= other.value;
}
private:
static_assert(bit_size != 0, "Cannot have a zero-sized immediate");
static_assert(bit_size <= 32, "Cannot have an immediate larger than the instruction size");
u32 value;
};
template <size_t bit_size>
bool operator==(u32 a, Imm<bit_size> b) {
return Imm<bit_size>{a} == b;
}
template <size_t bit_size>
bool operator==(Imm<bit_size> a, u32 b) {
return Imm<bit_size>{b} == a;
}
template <size_t bit_size>
bool operator!=(u32 a, Imm<bit_size> b) {
return !operator==(a, b);
}
template <size_t bit_size>
bool operator!=(Imm<bit_size> a, u32 b) {
return !operator==(a, b);
}
template <size_t bit_size>
bool operator<(u32 a, Imm<bit_size> b) {
return Imm<bit_size>{a} < b;
}
template <size_t bit_size>
bool operator<(Imm<bit_size> a, u32 b) {
return a < Imm<bit_size>{b};
}
template <size_t bit_size>
bool operator<=(u32 a, Imm<bit_size> b) {
return !operator<(b, a);
}
template <size_t bit_size>
bool operator<=(Imm<bit_size> a, u32 b) {
return !operator<(b, a);
}
template <size_t bit_size>
bool operator>(u32 a, Imm<bit_size> b) {
return operator<(b, a);
}
template <size_t bit_size>
bool operator>(Imm<bit_size> a, u32 b) {
return operator<(b, a);
}
template <size_t bit_size>
bool operator>=(u32 a, Imm<bit_size> b) {
return !operator<(a, b);
}
template <size_t bit_size>
bool operator>=(Imm<bit_size> a, u32 b) {
return !operator<(a, b);
}
/**
* Concatenate immediates together.
* Left to right corresponds to most significant imm to least significant imm.
* This is equivalent to a:b:...:z in ASL.
*/
template <size_t first_bit_size, size_t ...rest_bit_sizes>
auto concatenate(Imm<first_bit_size> first, Imm<rest_bit_sizes> ...rest) {
if constexpr (sizeof...(rest) == 0) {
return first;
} else {
const auto concat_rest = concatenate(rest...);
const u32 value = (first.ZeroExtend() << concat_rest.bit_size) | concat_rest.ZeroExtend();
return Imm<Common::Sum(first_bit_size, rest_bit_sizes...)>{value};
}
}
} // namespace Dynarmic