dynarmic/src/frontend/A64/location_descriptor.h

94 lines
2.8 KiB
C
Raw Normal View History

2018-01-04 23:05:27 +00: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
#include <functional>
#include <iosfwd>
#include <tuple>
#include "common/bit_util.h"
#include "common/common_types.h"
2018-07-17 19:53:21 +01:00
#include "common/fp/fpcr.h"
2018-01-04 23:05:27 +00:00
#include "frontend/ir/location_descriptor.h"
namespace Dynarmic::A64 {
2018-01-04 23:05:27 +00:00
/**
* LocationDescriptor describes the location of a basic block.
* The location is not solely based on the PC because other flags influence the way
* instructions should be translated.
*/
class LocationDescriptor {
public:
static constexpr u64 PC_MASK = 0x00FF'FFFF'FFFF'FFFFull;
static constexpr u32 FPCR_MASK = 0x07C0'0000;
2018-07-17 19:53:21 +01:00
LocationDescriptor(u64 pc, FP::FPCR fpcr) : pc(pc & PC_MASK), fpcr(fpcr.Value() & FPCR_MASK) {}
2018-01-04 23:05:27 +00:00
explicit LocationDescriptor(const IR::LocationDescriptor& o)
: pc(o.Value() & PC_MASK), fpcr((o.Value() >> 37) & FPCR_MASK) {}
u64 PC() const { return Common::SignExtend<56>(pc); }
2018-07-17 19:53:21 +01:00
FP::FPCR FPCR() const { return fpcr; }
2018-01-04 23:05:27 +00:00
bool operator == (const LocationDescriptor& o) const {
return std::tie(pc, fpcr) == std::tie(o.pc, o.fpcr);
}
bool operator != (const LocationDescriptor& o) const {
return !operator==(o);
}
2018-01-07 13:42:11 +00:00
LocationDescriptor SetPC(u64 new_pc) const {
2018-01-04 23:05:27 +00:00
return LocationDescriptor(new_pc, fpcr);
}
LocationDescriptor AdvancePC(int amount) const {
return LocationDescriptor(static_cast<u64>(pc + amount), fpcr);
}
u64 UniqueHash() const {
// This value MUST BE UNIQUE.
// This calculation has to match up with EmitTerminalPopRSBHint
u64 fpcr_u64 = static_cast<u64>(fpcr.Value()) << 37;
return pc | fpcr_u64;
}
operator IR::LocationDescriptor() const {
return IR::LocationDescriptor{UniqueHash()};
}
private:
2018-07-17 19:53:21 +01:00
u64 pc; ///< Current program counter value.
FP::FPCR fpcr; ///< Floating point control register.
2018-01-04 23:05:27 +00:00
};
/**
* Provides a string representation of a LocationDescriptor.
*
* @param o Output stream
* @param descriptor The descriptor to get a string representation of
*/
std::ostream& operator<<(std::ostream& o, const LocationDescriptor& descriptor);
} // namespace Dynarmic::A64
2018-01-04 23:05:27 +00:00
namespace std {
template <>
struct less<Dynarmic::A64::LocationDescriptor> {
bool operator()(const Dynarmic::A64::LocationDescriptor& x, const Dynarmic::A64::LocationDescriptor& y) const {
return x.UniqueHash() < y.UniqueHash();
}
};
template <>
struct hash<Dynarmic::A64::LocationDescriptor> {
size_t operator()(const Dynarmic::A64::LocationDescriptor& x) const {
return std::hash<u64>()(x.UniqueHash());
}
};
} // namespace std