2016-07-01 21:01:06 +08:00
|
|
|
/* This file is part of the dynarmic project.
|
|
|
|
* Copyright (c) 2016 MerryMage
|
2020-04-23 15:25:11 +01:00
|
|
|
* SPDX-License-Identifier: 0BSD
|
2016-07-01 21:01:06 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2016-08-17 10:53:36 -04:00
|
|
|
#include <array>
|
2018-01-04 21:12:02 +00:00
|
|
|
#include <functional>
|
2018-10-15 00:17:56 +03:00
|
|
|
#include <optional>
|
2018-01-26 21:35:42 -05:00
|
|
|
#include <utility>
|
2016-08-17 10:53:36 -04:00
|
|
|
#include <vector>
|
|
|
|
|
2016-08-24 20:07:08 +01:00
|
|
|
#include <xbyak.h>
|
2016-07-01 21:01:06 +08:00
|
|
|
|
2018-08-14 13:13:47 -05:00
|
|
|
#include "backend/x64/block_of_code.h"
|
|
|
|
#include "backend/x64/hostloc.h"
|
|
|
|
#include "backend/x64/oparg.h"
|
2016-07-01 21:01:06 +08:00
|
|
|
#include "common/common_types.h"
|
2018-01-18 11:36:48 +00:00
|
|
|
#include "frontend/ir/cond.h"
|
2016-08-17 10:53:36 -04:00
|
|
|
#include "frontend/ir/microinstruction.h"
|
|
|
|
#include "frontend/ir/value.h"
|
2016-07-01 21:01:06 +08:00
|
|
|
|
2020-04-08 11:46:36 +01:00
|
|
|
namespace Dynarmic::Backend::X64 {
|
2016-07-01 21:01:06 +08:00
|
|
|
|
2017-02-24 21:09:12 +00:00
|
|
|
class RegAlloc;
|
|
|
|
|
2017-02-24 18:42:59 +00:00
|
|
|
struct HostLocInfo {
|
|
|
|
public:
|
2017-02-26 23:16:41 +00:00
|
|
|
bool IsLocked() const;
|
|
|
|
bool IsEmpty() const;
|
|
|
|
bool IsLastUse() const;
|
|
|
|
|
|
|
|
void ReadLock();
|
|
|
|
void WriteLock();
|
2017-11-27 19:38:22 +00:00
|
|
|
void AddArgReference();
|
2018-08-19 21:07:37 +01:00
|
|
|
void ReleaseOne();
|
|
|
|
void ReleaseAll();
|
2017-02-24 18:42:59 +00:00
|
|
|
|
2018-01-18 13:00:07 +00:00
|
|
|
bool ContainsValue(const IR::Inst* inst) const;
|
|
|
|
size_t GetMaxBitWidth() const;
|
|
|
|
|
|
|
|
void AddValue(IR::Inst* inst);
|
|
|
|
|
2017-02-24 18:42:59 +00:00
|
|
|
private:
|
2018-01-18 13:00:07 +00:00
|
|
|
// Current instruction state
|
2018-08-19 21:07:37 +01:00
|
|
|
size_t is_being_used_count = 0;
|
2017-02-24 20:19:50 +00:00
|
|
|
bool is_scratch = false;
|
2017-11-27 19:38:22 +00:00
|
|
|
|
2018-01-18 13:00:07 +00:00
|
|
|
// Block state
|
2017-11-27 19:38:22 +00:00
|
|
|
size_t current_references = 0;
|
|
|
|
size_t accumulated_uses = 0;
|
|
|
|
size_t total_uses = 0;
|
2018-01-18 13:00:07 +00:00
|
|
|
|
|
|
|
// Value state
|
|
|
|
std::vector<IR::Inst*> values;
|
|
|
|
size_t max_bit_width = 0;
|
2017-02-24 18:42:59 +00:00
|
|
|
};
|
|
|
|
|
2017-02-24 21:09:12 +00:00
|
|
|
struct Argument {
|
|
|
|
public:
|
2018-10-15 00:17:56 +03:00
|
|
|
using copyable_reference = std::reference_wrapper<Argument>;
|
|
|
|
|
2017-02-26 23:16:41 +00:00
|
|
|
IR::Type GetType() const;
|
|
|
|
bool IsImmediate() const;
|
2018-08-18 21:08:34 +01:00
|
|
|
bool IsVoid() const;
|
2017-02-24 21:09:12 +00:00
|
|
|
|
2018-01-07 11:41:17 +00:00
|
|
|
bool FitsInImmediateU32() const;
|
2018-01-13 17:59:50 +00:00
|
|
|
bool FitsInImmediateS32() const;
|
2018-01-07 11:41:17 +00:00
|
|
|
|
2017-02-24 21:25:31 +00:00
|
|
|
bool GetImmediateU1() const;
|
2017-02-24 21:09:12 +00:00
|
|
|
u8 GetImmediateU8() const;
|
|
|
|
u16 GetImmediateU16() const;
|
|
|
|
u32 GetImmediateU32() const;
|
2018-01-13 17:59:50 +00:00
|
|
|
u64 GetImmediateS32() const;
|
2017-02-24 21:09:12 +00:00
|
|
|
u64 GetImmediateU64() const;
|
2018-01-18 11:36:48 +00:00
|
|
|
IR::Cond GetImmediateCond() const;
|
2017-02-24 21:09:12 +00:00
|
|
|
|
|
|
|
/// Is this value currently in a GPR?
|
|
|
|
bool IsInGpr() const;
|
|
|
|
/// Is this value currently in a XMM?
|
|
|
|
bool IsInXmm() const;
|
|
|
|
/// Is this value currently in memory?
|
|
|
|
bool IsInMemory() const;
|
|
|
|
|
|
|
|
private:
|
|
|
|
friend class RegAlloc;
|
2018-08-18 21:08:34 +01:00
|
|
|
explicit Argument(RegAlloc& reg_alloc) : reg_alloc(reg_alloc) {}
|
2017-02-24 21:09:12 +00:00
|
|
|
|
|
|
|
bool allocated = false;
|
|
|
|
RegAlloc& reg_alloc;
|
|
|
|
IR::Value value;
|
|
|
|
};
|
|
|
|
|
2016-07-01 21:01:06 +08:00
|
|
|
class RegAlloc final {
|
|
|
|
public:
|
2018-08-18 21:08:34 +01:00
|
|
|
using ArgumentInfo = std::array<Argument, IR::max_arg_count>;
|
2018-08-13 11:10:34 -04:00
|
|
|
|
2020-04-08 13:19:26 +01:00
|
|
|
explicit RegAlloc(BlockOfCode& code, size_t num_spills, std::function<Xbyak::Address(HostLoc)> spill_to_addr, std::vector<HostLoc> gpr_order, std::vector<HostLoc> xmm_order);
|
2016-07-01 21:01:06 +08:00
|
|
|
|
2018-08-13 11:10:34 -04:00
|
|
|
ArgumentInfo GetArgumentInfo(IR::Inst* inst);
|
2017-02-24 21:09:12 +00:00
|
|
|
|
2017-02-26 22:28:32 +00:00
|
|
|
Xbyak::Reg64 UseGpr(Argument& arg);
|
|
|
|
Xbyak::Xmm UseXmm(Argument& arg);
|
|
|
|
OpArg UseOpArg(Argument& arg);
|
|
|
|
void Use(Argument& arg, HostLoc host_loc);
|
2017-02-24 21:09:12 +00:00
|
|
|
|
2017-02-26 22:28:32 +00:00
|
|
|
Xbyak::Reg64 UseScratchGpr(Argument& arg);
|
|
|
|
Xbyak::Xmm UseScratchXmm(Argument& arg);
|
|
|
|
void UseScratch(Argument& arg, HostLoc host_loc);
|
2017-02-24 21:09:12 +00:00
|
|
|
|
2017-02-26 22:28:32 +00:00
|
|
|
void DefineValue(IR::Inst* inst, const Xbyak::Reg& reg);
|
|
|
|
void DefineValue(IR::Inst* inst, Argument& arg);
|
2017-02-24 21:09:12 +00:00
|
|
|
|
2018-08-19 21:07:37 +01:00
|
|
|
void Release(const Xbyak::Reg& reg);
|
|
|
|
|
2020-04-08 13:19:26 +01:00
|
|
|
Xbyak::Reg64 ScratchGpr();
|
|
|
|
Xbyak::Reg64 ScratchGpr(HostLoc desired_location);
|
|
|
|
Xbyak::Xmm ScratchXmm();
|
|
|
|
Xbyak::Xmm ScratchXmm(HostLoc desired_location);
|
2016-07-01 21:01:06 +08:00
|
|
|
|
2018-10-15 00:17:56 +03:00
|
|
|
void HostCall(IR::Inst* result_def = nullptr,
|
|
|
|
std::optional<Argument::copyable_reference> arg0 = {},
|
|
|
|
std::optional<Argument::copyable_reference> arg1 = {},
|
|
|
|
std::optional<Argument::copyable_reference> arg2 = {},
|
|
|
|
std::optional<Argument::copyable_reference> arg3 = {});
|
2016-07-11 15:28:10 +01:00
|
|
|
|
2016-07-01 21:01:06 +08:00
|
|
|
// TODO: Values in host flags
|
|
|
|
|
|
|
|
void EndOfAllocScope();
|
|
|
|
|
2016-07-11 22:43:53 +01:00
|
|
|
void AssertNoMoreUses();
|
|
|
|
|
2016-07-01 21:01:06 +08:00
|
|
|
private:
|
2017-02-24 21:09:12 +00:00
|
|
|
friend struct Argument;
|
|
|
|
|
2020-04-08 13:19:26 +01:00
|
|
|
std::vector<HostLoc> gpr_order;
|
|
|
|
std::vector<HostLoc> xmm_order;
|
|
|
|
|
|
|
|
HostLoc SelectARegister(const std::vector<HostLoc>& desired_locations) const;
|
2018-10-15 00:17:56 +03:00
|
|
|
std::optional<HostLoc> ValueLocation(const IR::Inst* value) const;
|
2016-08-05 14:10:39 +01:00
|
|
|
|
2020-04-08 13:19:26 +01:00
|
|
|
HostLoc UseImpl(IR::Value use_value, const std::vector<HostLoc>& desired_locations);
|
|
|
|
HostLoc UseScratchImpl(IR::Value use_value, const std::vector<HostLoc>& desired_locations);
|
|
|
|
HostLoc ScratchImpl(const std::vector<HostLoc>& desired_locations);
|
2017-02-26 22:28:32 +00:00
|
|
|
void DefineValueImpl(IR::Inst* def_inst, HostLoc host_loc);
|
|
|
|
void DefineValueImpl(IR::Inst* def_inst, const IR::Value& use_inst);
|
2017-02-24 19:08:58 +00:00
|
|
|
|
2019-05-24 01:59:04 -04:00
|
|
|
HostLoc LoadImmediate(IR::Value imm, HostLoc host_loc);
|
2017-02-24 19:42:36 +00:00
|
|
|
void Move(HostLoc to, HostLoc from);
|
2018-07-31 21:55:14 +01:00
|
|
|
void CopyToScratch(size_t bit_width, HostLoc to, HostLoc from);
|
2017-02-24 19:42:36 +00:00
|
|
|
void Exchange(HostLoc a, HostLoc b);
|
|
|
|
void MoveOutOfTheWay(HostLoc reg);
|
|
|
|
|
2016-07-01 21:01:06 +08:00
|
|
|
void SpillRegister(HostLoc loc);
|
|
|
|
HostLoc FindFreeSpill() const;
|
|
|
|
|
2018-01-04 21:12:02 +00:00
|
|
|
std::vector<HostLocInfo> hostloc_info;
|
2017-02-26 22:28:32 +00:00
|
|
|
HostLocInfo& LocInfo(HostLoc loc);
|
|
|
|
const HostLocInfo& LocInfo(HostLoc loc) const;
|
2018-01-04 21:12:02 +00:00
|
|
|
|
2018-02-03 14:28:57 +00:00
|
|
|
BlockOfCode& code;
|
2018-01-04 21:12:02 +00:00
|
|
|
std::function<Xbyak::Address(HostLoc)> spill_to_addr;
|
2018-07-31 21:55:14 +01:00
|
|
|
void EmitMove(size_t bit_width, HostLoc to, HostLoc from);
|
2018-01-04 21:12:02 +00:00
|
|
|
void EmitExchange(HostLoc a, HostLoc b);
|
2016-07-01 21:01:06 +08:00
|
|
|
};
|
|
|
|
|
2020-04-08 11:46:36 +01:00
|
|
|
} // namespace Dynarmic::Backend::X64
|