aarch64: Initial work to transition backend to ISLE (#3541)
* aarch64: Initial work to transition backend to ISLE This commit is what is hoped to be the initial commit towards migrating the aarch64 backend to ISLE. There's seemingly a lot of changes here but it's intended to largely be code motion. The current thinking is to closely follow the x64 backend for how all this is handled and organized. Major changes in this PR are: * The `Inst` enum is now defined in ISLE. This avoids having to define it in two places (once in Rust and once in ISLE). I've preserved all the comments in the ISLE and otherwise this isn't actually a functional change from the Rust perspective, it's still the same enum according to Rust. * Lots of little enums and things were moved to ISLE as well. As with `Inst` their definitions didn't change, only where they're defined. This will give future ISLE PRs access to all these operations. * Initial code for lowering `iconst`, `null`, and `bconst` are implemented. Ironically none of this is actually used right now because constant lowering is handled in `put_input_in_regs` which specially handles constants. Nonetheless I wanted to get at least something simple working which shows off how to special case various things that are specific to AArch64. In a future PR I plan to hook up const-lowering in ISLE to this path so even though `iconst`-the-clif-instruction is never lowered this should use the const lowering defined in ISLE rather than elsewhere in the backend (eventually leading to the deletion of the non-ISLE lowering). * The `IsleContext` skeleton is created and set up for future additions. * Some code for ISLE that's shared across all backends now lives in `isle_prelude_methods!()` and is deduplicated between the AArch64 backend and the x64 backend. * Register mapping is tweaked to do the same thing for AArch64 that it does for x64. Namely mapping virtual registers is supported instead of just virtual to machine registers. My main goal with this PR was to get AArch64 into a place where new instructions can be added with relative ease. Additionally I'm hoping to figure out as part of this change how much to share for ISLE between AArch64 and x64 (and other backends). * Don't use priorities with rules * Update .gitattributes with concise syntax * Deduplicate some type definitions * Rebuild ISLE * Move isa::isle to machinst::isle
This commit is contained in:
@@ -1,7 +1,7 @@
|
||||
//! Instruction operand sub-components (aka "parts"): definitions and printing.
|
||||
|
||||
use super::regs::{self, show_ireg_sized};
|
||||
use super::{EmitState, RegMapper};
|
||||
use super::EmitState;
|
||||
use crate::ir::condcodes::{FloatCC, IntCC};
|
||||
use crate::ir::{MemFlags, Type};
|
||||
use crate::isa::x64::inst::Inst;
|
||||
|
||||
@@ -2565,64 +2565,17 @@ fn x64_get_regs(inst: &Inst, collector: &mut RegUsageCollector) {
|
||||
//=============================================================================
|
||||
// Instructions and subcomponents: map_regs
|
||||
|
||||
// Define our own register-mapping trait so we can do arbitrary register
|
||||
// renaming that are more free form than what `regalloc` constrains us to with
|
||||
// its `RegUsageMapper` trait definition.
|
||||
pub trait RegMapper {
|
||||
fn get_use(&self, reg: Reg) -> Option<Reg>;
|
||||
fn get_def(&self, reg: Reg) -> Option<Reg>;
|
||||
fn get_mod(&self, reg: Reg) -> Option<Reg>;
|
||||
}
|
||||
|
||||
impl<T> RegMapper for T
|
||||
where
|
||||
T: regalloc::RegUsageMapper,
|
||||
{
|
||||
fn get_use(&self, reg: Reg) -> Option<Reg> {
|
||||
let v = reg.as_virtual_reg()?;
|
||||
self.get_use(v).map(|r| r.to_reg())
|
||||
}
|
||||
|
||||
fn get_def(&self, reg: Reg) -> Option<Reg> {
|
||||
let v = reg.as_virtual_reg()?;
|
||||
self.get_def(v).map(|r| r.to_reg())
|
||||
}
|
||||
|
||||
fn get_mod(&self, reg: Reg) -> Option<Reg> {
|
||||
let v = reg.as_virtual_reg()?;
|
||||
self.get_mod(v).map(|r| r.to_reg())
|
||||
}
|
||||
}
|
||||
|
||||
fn map_use<RM: RegMapper>(m: &RM, r: &mut Reg) {
|
||||
if let Some(new) = m.get_use(*r) {
|
||||
*r = new;
|
||||
}
|
||||
}
|
||||
|
||||
fn map_def<RM: RegMapper>(m: &RM, r: &mut Writable<Reg>) {
|
||||
if let Some(new) = m.get_def(r.to_reg()) {
|
||||
*r = Writable::from_reg(new);
|
||||
}
|
||||
}
|
||||
|
||||
fn map_mod<RM: RegMapper>(m: &RM, r: &mut Writable<Reg>) {
|
||||
if let Some(new) = m.get_mod(r.to_reg()) {
|
||||
*r = Writable::from_reg(new);
|
||||
}
|
||||
}
|
||||
|
||||
impl Amode {
|
||||
fn map_uses<RM: RegMapper>(&mut self, map: &RM) {
|
||||
match self {
|
||||
Amode::ImmReg { ref mut base, .. } => map_use(map, base),
|
||||
Amode::ImmReg { ref mut base, .. } => map.map_use(base),
|
||||
Amode::ImmRegRegShift {
|
||||
ref mut base,
|
||||
ref mut index,
|
||||
..
|
||||
} => {
|
||||
map_use(map, base);
|
||||
map_use(map, index);
|
||||
map.map_use(base);
|
||||
map.map_use(index);
|
||||
}
|
||||
Amode::RipRelative { .. } => {
|
||||
// RIP isn't involved in regalloc.
|
||||
@@ -2645,7 +2598,7 @@ impl Amode {
|
||||
impl RegMemImm {
|
||||
fn map_uses<RM: RegMapper>(&mut self, map: &RM) {
|
||||
match self {
|
||||
RegMemImm::Reg { ref mut reg } => map_use(map, reg),
|
||||
RegMemImm::Reg { ref mut reg } => map.map_use(reg),
|
||||
RegMemImm::Mem { ref mut addr } => addr.map_uses(map),
|
||||
RegMemImm::Imm { .. } => {}
|
||||
}
|
||||
@@ -2655,7 +2608,7 @@ impl RegMemImm {
|
||||
match self {
|
||||
Self::Reg { reg } => {
|
||||
let mut writable_src = Writable::from_reg(*reg);
|
||||
map_def(mapper, &mut writable_src);
|
||||
mapper.map_def(&mut writable_src);
|
||||
*self = Self::reg(writable_src.to_reg());
|
||||
}
|
||||
_ => panic!("unexpected RegMemImm kind in map_src_reg_as_def"),
|
||||
@@ -2666,7 +2619,7 @@ impl RegMemImm {
|
||||
impl RegMem {
|
||||
fn map_uses<RM: RegMapper>(&mut self, map: &RM) {
|
||||
match self {
|
||||
RegMem::Reg { ref mut reg } => map_use(map, reg),
|
||||
RegMem::Reg { ref mut reg } => map.map_use(reg),
|
||||
RegMem::Mem { ref mut addr, .. } => addr.map_uses(map),
|
||||
}
|
||||
}
|
||||
@@ -2675,7 +2628,7 @@ impl RegMem {
|
||||
match self {
|
||||
Self::Reg { reg } => {
|
||||
let mut writable_src = Writable::from_reg(*reg);
|
||||
map_def(mapper, &mut writable_src);
|
||||
mapper.map_def(&mut writable_src);
|
||||
*self = Self::reg(writable_src.to_reg());
|
||||
}
|
||||
_ => panic!("unexpected RegMem kind in map_src_reg_as_def"),
|
||||
@@ -2698,25 +2651,25 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
debug_assert_eq!(*src1, dst.to_reg());
|
||||
if produces_const {
|
||||
src2.map_as_def(mapper);
|
||||
map_def(mapper, dst);
|
||||
mapper.map_def(dst);
|
||||
*src1 = dst.to_reg();
|
||||
} else {
|
||||
src2.map_uses(mapper);
|
||||
map_mod(mapper, dst);
|
||||
mapper.map_mod(dst);
|
||||
*src1 = dst.to_reg();
|
||||
}
|
||||
}
|
||||
Inst::Not { src, dst, .. } | Inst::Neg { src, dst, .. } => {
|
||||
debug_assert_eq!(*src, dst.to_reg());
|
||||
map_mod(mapper, dst);
|
||||
mapper.map_mod(dst);
|
||||
*src = dst.to_reg();
|
||||
}
|
||||
Inst::Div { divisor, .. } => divisor.map_uses(mapper),
|
||||
Inst::MulHi { src2, .. } => src2.map_uses(mapper),
|
||||
Inst::CheckedDivOrRemSeq { divisor, tmp, .. } => {
|
||||
map_mod(mapper, divisor);
|
||||
mapper.map_mod(divisor);
|
||||
if let Some(tmp) = tmp {
|
||||
map_def(mapper, tmp)
|
||||
mapper.map_def(tmp)
|
||||
}
|
||||
}
|
||||
Inst::SignExtendData { .. } => {}
|
||||
@@ -2736,7 +2689,7 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
..
|
||||
} => {
|
||||
src.map_uses(mapper);
|
||||
map_def(mapper, dst);
|
||||
mapper.map_def(dst);
|
||||
}
|
||||
Inst::XmmRmRImm {
|
||||
ref op,
|
||||
@@ -2748,7 +2701,7 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
debug_assert_eq!(*src1, dst.to_reg());
|
||||
if produces_const {
|
||||
src2.map_as_def(mapper);
|
||||
map_def(mapper, dst);
|
||||
mapper.map_def(dst);
|
||||
*src1 = dst.to_reg();
|
||||
} else if *op == SseOpcode::Pextrb
|
||||
|| *op == SseOpcode::Pextrw
|
||||
@@ -2760,11 +2713,11 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
|| *op == SseOpcode::Roundpd
|
||||
{
|
||||
src2.map_uses(mapper);
|
||||
map_def(mapper, dst);
|
||||
mapper.map_def(dst);
|
||||
*src1 = dst.to_reg();
|
||||
} else {
|
||||
src2.map_uses(mapper);
|
||||
map_mod(mapper, dst);
|
||||
mapper.map_mod(dst);
|
||||
*src1 = dst.to_reg();
|
||||
}
|
||||
}
|
||||
@@ -2777,11 +2730,11 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
debug_assert_eq!(*src1, dst.to_reg());
|
||||
if produces_const {
|
||||
src2.map_as_def(mapper);
|
||||
map_def(mapper, dst);
|
||||
mapper.map_def(dst);
|
||||
*src1 = dst.to_reg();
|
||||
} else {
|
||||
src2.map_uses(mapper);
|
||||
map_mod(mapper, dst);
|
||||
mapper.map_mod(dst);
|
||||
*src1 = dst.to_reg();
|
||||
}
|
||||
}
|
||||
@@ -2793,10 +2746,10 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
..
|
||||
} => {
|
||||
src1.map_uses(mapper);
|
||||
map_use(mapper, src2);
|
||||
mapper.map_use(src2);
|
||||
match *op {
|
||||
Avx512Opcode::Vpermi2b => map_mod(mapper, dst),
|
||||
_ => map_def(mapper, dst),
|
||||
Avx512Opcode::Vpermi2b => mapper.map_mod(dst),
|
||||
_ => mapper.map_def(dst),
|
||||
}
|
||||
}
|
||||
Inst::XmmRmiReg {
|
||||
@@ -2807,29 +2760,29 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
} => {
|
||||
debug_assert_eq!(*src1, dst.to_reg());
|
||||
src2.map_uses(mapper);
|
||||
map_mod(mapper, dst);
|
||||
mapper.map_mod(dst);
|
||||
*src1 = dst.to_reg();
|
||||
}
|
||||
Inst::XmmUninitializedValue { ref mut dst, .. } => {
|
||||
map_def(mapper, dst);
|
||||
mapper.map_def(dst);
|
||||
}
|
||||
Inst::XmmLoadConst { ref mut dst, .. } => {
|
||||
map_def(mapper, dst);
|
||||
mapper.map_def(dst);
|
||||
}
|
||||
Inst::XmmMinMaxSeq {
|
||||
ref mut lhs,
|
||||
ref mut rhs_dst,
|
||||
..
|
||||
} => {
|
||||
map_use(mapper, lhs);
|
||||
map_mod(mapper, rhs_dst);
|
||||
mapper.map_use(lhs);
|
||||
mapper.map_mod(rhs_dst);
|
||||
}
|
||||
Inst::XmmMovRM {
|
||||
ref mut src,
|
||||
ref mut dst,
|
||||
..
|
||||
} => {
|
||||
map_use(mapper, src);
|
||||
mapper.map_use(src);
|
||||
dst.map_uses(mapper);
|
||||
}
|
||||
Inst::XmmCmpRmR {
|
||||
@@ -2838,9 +2791,9 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
..
|
||||
} => {
|
||||
src.map_uses(mapper);
|
||||
map_use(mapper, dst);
|
||||
mapper.map_use(dst);
|
||||
}
|
||||
Inst::Imm { ref mut dst, .. } => map_def(mapper, dst),
|
||||
Inst::Imm { ref mut dst, .. } => mapper.map_def(dst),
|
||||
Inst::MovRR {
|
||||
ref mut src,
|
||||
ref mut dst,
|
||||
@@ -2851,8 +2804,8 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
ref mut dst,
|
||||
..
|
||||
} => {
|
||||
map_use(mapper, src);
|
||||
map_def(mapper, dst);
|
||||
mapper.map_use(src);
|
||||
mapper.map_def(dst);
|
||||
}
|
||||
Inst::GprToXmm {
|
||||
ref mut src,
|
||||
@@ -2860,7 +2813,7 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
..
|
||||
} => {
|
||||
src.map_uses(mapper);
|
||||
map_def(mapper, dst);
|
||||
mapper.map_def(dst);
|
||||
}
|
||||
Inst::CvtUint64ToFloatSeq {
|
||||
ref mut src,
|
||||
@@ -2869,10 +2822,10 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
ref mut tmp_gpr2,
|
||||
..
|
||||
} => {
|
||||
map_mod(mapper, src);
|
||||
map_def(mapper, dst);
|
||||
map_def(mapper, tmp_gpr1);
|
||||
map_def(mapper, tmp_gpr2);
|
||||
mapper.map_mod(src);
|
||||
mapper.map_def(dst);
|
||||
mapper.map_def(tmp_gpr1);
|
||||
mapper.map_def(tmp_gpr2);
|
||||
}
|
||||
Inst::CvtFloatToSintSeq {
|
||||
ref mut src,
|
||||
@@ -2888,10 +2841,10 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
ref mut tmp_xmm,
|
||||
..
|
||||
} => {
|
||||
map_mod(mapper, src);
|
||||
map_def(mapper, dst);
|
||||
map_def(mapper, tmp_gpr);
|
||||
map_def(mapper, tmp_xmm);
|
||||
mapper.map_mod(src);
|
||||
mapper.map_def(dst);
|
||||
mapper.map_def(tmp_gpr);
|
||||
mapper.map_def(tmp_xmm);
|
||||
}
|
||||
Inst::MovzxRmR {
|
||||
ref mut src,
|
||||
@@ -2899,11 +2852,11 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
..
|
||||
} => {
|
||||
src.map_uses(mapper);
|
||||
map_def(mapper, dst);
|
||||
mapper.map_def(dst);
|
||||
}
|
||||
Inst::Mov64MR { src, dst, .. } | Inst::LoadEffectiveAddress { addr: src, dst } => {
|
||||
src.map_uses(mapper);
|
||||
map_def(mapper, dst);
|
||||
mapper.map_def(dst);
|
||||
}
|
||||
Inst::MovsxRmR {
|
||||
ref mut src,
|
||||
@@ -2911,14 +2864,14 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
..
|
||||
} => {
|
||||
src.map_uses(mapper);
|
||||
map_def(mapper, dst);
|
||||
mapper.map_def(dst);
|
||||
}
|
||||
Inst::MovRM {
|
||||
ref mut src,
|
||||
ref mut dst,
|
||||
..
|
||||
} => {
|
||||
map_use(mapper, src);
|
||||
mapper.map_use(src);
|
||||
dst.map_uses(mapper);
|
||||
}
|
||||
Inst::ShiftR {
|
||||
@@ -2927,7 +2880,7 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
..
|
||||
} => {
|
||||
debug_assert_eq!(*src, dst.to_reg());
|
||||
map_mod(mapper, dst);
|
||||
mapper.map_mod(dst);
|
||||
*src = dst.to_reg();
|
||||
}
|
||||
Inst::CmpRmiR {
|
||||
@@ -2936,9 +2889,9 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
..
|
||||
} => {
|
||||
src.map_uses(mapper);
|
||||
map_use(mapper, dst);
|
||||
mapper.map_use(dst);
|
||||
}
|
||||
Inst::Setcc { ref mut dst, .. } => map_def(mapper, dst),
|
||||
Inst::Setcc { ref mut dst, .. } => mapper.map_def(dst),
|
||||
Inst::Cmove {
|
||||
consequent: ref mut src,
|
||||
ref mut dst,
|
||||
@@ -2946,7 +2899,7 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
..
|
||||
} => {
|
||||
src.map_uses(mapper);
|
||||
map_mod(mapper, dst);
|
||||
mapper.map_mod(dst);
|
||||
*alternative = dst.to_reg();
|
||||
}
|
||||
Inst::XmmCmove {
|
||||
@@ -2955,11 +2908,11 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
..
|
||||
} => {
|
||||
src.map_uses(mapper);
|
||||
map_mod(mapper, dst);
|
||||
mapper.map_mod(dst);
|
||||
}
|
||||
Inst::Push64 { ref mut src } => src.map_uses(mapper),
|
||||
Inst::Pop64 { ref mut dst } => {
|
||||
map_def(mapper, dst);
|
||||
mapper.map_def(dst);
|
||||
}
|
||||
|
||||
Inst::CallKnown {
|
||||
@@ -2968,10 +2921,10 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
..
|
||||
} => {
|
||||
for r in uses.iter_mut() {
|
||||
map_use(mapper, r);
|
||||
mapper.map_use(r);
|
||||
}
|
||||
for r in defs.iter_mut() {
|
||||
map_def(mapper, r);
|
||||
mapper.map_def(r);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -2982,10 +2935,10 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
..
|
||||
} => {
|
||||
for r in uses.iter_mut() {
|
||||
map_use(mapper, r);
|
||||
mapper.map_use(r);
|
||||
}
|
||||
for r in defs.iter_mut() {
|
||||
map_def(mapper, r);
|
||||
mapper.map_def(r);
|
||||
}
|
||||
dest.map_uses(mapper);
|
||||
}
|
||||
@@ -2996,25 +2949,25 @@ pub(crate) fn x64_map_regs<RM: RegMapper>(inst: &mut Inst, mapper: &RM) {
|
||||
ref mut tmp2,
|
||||
..
|
||||
} => {
|
||||
map_use(mapper, idx);
|
||||
map_def(mapper, tmp1);
|
||||
map_def(mapper, tmp2);
|
||||
mapper.map_use(idx);
|
||||
mapper.map_def(tmp1);
|
||||
mapper.map_def(tmp2);
|
||||
}
|
||||
|
||||
Inst::JmpUnknown { ref mut target } => target.map_uses(mapper),
|
||||
|
||||
Inst::LoadExtName { ref mut dst, .. } => map_def(mapper, dst),
|
||||
Inst::LoadExtName { ref mut dst, .. } => mapper.map_def(dst),
|
||||
|
||||
Inst::LockCmpxchg {
|
||||
ref mut replacement,
|
||||
ref mut mem,
|
||||
..
|
||||
} => {
|
||||
map_use(mapper, replacement);
|
||||
mapper.map_use(replacement);
|
||||
mem.map_uses(mapper);
|
||||
}
|
||||
|
||||
Inst::ValueLabelMarker { ref mut reg, .. } => map_use(mapper, reg),
|
||||
Inst::ValueLabelMarker { ref mut reg, .. } => mapper.map_use(reg),
|
||||
|
||||
Inst::Ret
|
||||
| Inst::EpiloguePlaceholder
|
||||
|
||||
@@ -6,70 +6,29 @@ mod generated_code;
|
||||
// Types that the generated ISLE code uses via `use super::*`.
|
||||
use super::{
|
||||
is_mergeable_load, lower_to_amode, AluRmiROpcode, Inst as MInst, OperandSize, Reg, RegMemImm,
|
||||
Writable,
|
||||
};
|
||||
use crate::isa::x64::inst::args::SyntheticAmode;
|
||||
use crate::isa::x64::settings as x64_settings;
|
||||
use crate::machinst::isle::*;
|
||||
use crate::{
|
||||
ir::{immediates::*, types::*, Inst, InstructionData, Opcode, Value, ValueList},
|
||||
isa::x64::inst::{
|
||||
args::{
|
||||
Avx512Opcode, CmpOpcode, ExtMode, FcmpImm, Imm8Reg, RegMem, ShiftKind, SseOpcode, CC,
|
||||
},
|
||||
x64_map_regs, RegMapper,
|
||||
x64_map_regs,
|
||||
},
|
||||
machinst::{get_output_reg, InsnInput, InsnOutput, LowerCtx},
|
||||
machinst::{get_output_reg, InsnInput, InsnOutput, LowerCtx, RegRenamer},
|
||||
};
|
||||
use smallvec::SmallVec;
|
||||
use std::convert::TryFrom;
|
||||
|
||||
type Unit = ();
|
||||
type ValueSlice<'a> = &'a [Value];
|
||||
type ValueArray2 = [Value; 2];
|
||||
type ValueArray3 = [Value; 3];
|
||||
type WritableReg = Writable<Reg>;
|
||||
type ValueRegs = crate::machinst::ValueRegs<Reg>;
|
||||
|
||||
pub struct SinkableLoad {
|
||||
inst: Inst,
|
||||
addr_input: InsnInput,
|
||||
offset: i32,
|
||||
}
|
||||
|
||||
#[derive(Default)]
|
||||
struct RegRenamer {
|
||||
// Map of `(old, new)` register names. Use a `SmallVec` because we typically
|
||||
// only have one or two renamings.
|
||||
renames: SmallVec<[(Reg, Reg); 2]>,
|
||||
}
|
||||
|
||||
impl RegRenamer {
|
||||
fn add_rename(&mut self, old: Reg, new: Reg) {
|
||||
self.renames.push((old, new));
|
||||
}
|
||||
|
||||
fn get_rename(&self, reg: Reg) -> Option<Reg> {
|
||||
self.renames
|
||||
.iter()
|
||||
.find(|(old, _)| reg == *old)
|
||||
.map(|(_, new)| *new)
|
||||
}
|
||||
}
|
||||
|
||||
impl RegMapper for RegRenamer {
|
||||
fn get_use(&self, reg: Reg) -> Option<Reg> {
|
||||
self.get_rename(reg)
|
||||
}
|
||||
|
||||
fn get_def(&self, reg: Reg) -> Option<Reg> {
|
||||
self.get_rename(reg)
|
||||
}
|
||||
|
||||
fn get_mod(&self, reg: Reg) -> Option<Reg> {
|
||||
self.get_rename(reg)
|
||||
}
|
||||
}
|
||||
|
||||
/// The main entry point for lowering with ISLE.
|
||||
pub(crate) fn lower<C>(
|
||||
lower_ctx: &mut C,
|
||||
@@ -131,159 +90,7 @@ impl<'a, C> generated_code::Context for IsleContext<'a, C>
|
||||
where
|
||||
C: LowerCtx<I = MInst>,
|
||||
{
|
||||
#[inline]
|
||||
fn unpack_value_array_2(&mut self, arr: &ValueArray2) -> (Value, Value) {
|
||||
let [a, b] = *arr;
|
||||
(a, b)
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn pack_value_array_2(&mut self, a: Value, b: Value) -> ValueArray2 {
|
||||
[a, b]
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn unpack_value_array_3(&mut self, arr: &ValueArray3) -> (Value, Value, Value) {
|
||||
let [a, b, c] = *arr;
|
||||
(a, b, c)
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn pack_value_array_3(&mut self, a: Value, b: Value, c: Value) -> ValueArray3 {
|
||||
[a, b, c]
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn value_reg(&mut self, reg: Reg) -> ValueRegs {
|
||||
ValueRegs::one(reg)
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn value_regs(&mut self, r1: Reg, r2: Reg) -> ValueRegs {
|
||||
ValueRegs::two(r1, r2)
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn temp_writable_reg(&mut self, ty: Type) -> WritableReg {
|
||||
let value_regs = self.lower_ctx.alloc_tmp(ty);
|
||||
value_regs.only_reg().unwrap()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn invalid_reg(&mut self) -> Reg {
|
||||
Reg::invalid()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn put_in_reg(&mut self, val: Value) -> Reg {
|
||||
self.lower_ctx.put_value_in_regs(val).only_reg().unwrap()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn put_in_regs(&mut self, val: Value) -> ValueRegs {
|
||||
self.lower_ctx.put_value_in_regs(val)
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn value_regs_get(&mut self, regs: ValueRegs, i: usize) -> Reg {
|
||||
regs.regs()[i]
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn u8_as_u64(&mut self, x: u8) -> u64 {
|
||||
x.into()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn u16_as_u64(&mut self, x: u16) -> u64 {
|
||||
x.into()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn u32_as_u64(&mut self, x: u32) -> u64 {
|
||||
x.into()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn ty_bits(&mut self, ty: Type) -> u16 {
|
||||
ty.bits()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn fits_in_64(&mut self, ty: Type) -> Option<Type> {
|
||||
if ty.bits() <= 64 {
|
||||
Some(ty)
|
||||
} else {
|
||||
None
|
||||
}
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn value_list_slice(&mut self, list: ValueList) -> ValueSlice {
|
||||
list.as_slice(&self.lower_ctx.dfg().value_lists)
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn unwrap_head_value_list_1(&mut self, list: ValueList) -> (Value, ValueSlice) {
|
||||
match self.value_list_slice(list) {
|
||||
[head, tail @ ..] => (*head, tail),
|
||||
_ => out_of_line_panic("`unwrap_head_value_list_1` on empty `ValueList`"),
|
||||
}
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn unwrap_head_value_list_2(&mut self, list: ValueList) -> (Value, Value, ValueSlice) {
|
||||
match self.value_list_slice(list) {
|
||||
[head1, head2, tail @ ..] => (*head1, *head2, tail),
|
||||
_ => out_of_line_panic(
|
||||
"`unwrap_head_value_list_2` on list without at least two elements",
|
||||
),
|
||||
}
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn writable_reg_to_reg(&mut self, r: WritableReg) -> Reg {
|
||||
r.to_reg()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn u64_from_imm64(&mut self, imm: Imm64) -> u64 {
|
||||
imm.bits() as u64
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn inst_results(&mut self, inst: Inst) -> ValueSlice {
|
||||
self.lower_ctx.dfg().inst_results(inst)
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn first_result(&mut self, inst: Inst) -> Option<Value> {
|
||||
self.lower_ctx.dfg().inst_results(inst).first().copied()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn inst_data(&mut self, inst: Inst) -> InstructionData {
|
||||
self.lower_ctx.dfg()[inst].clone()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn value_type(&mut self, val: Value) -> Type {
|
||||
self.lower_ctx.dfg().value_type(val)
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn multi_lane(&mut self, ty: Type) -> Option<(u8, u16)> {
|
||||
if ty.lane_count() > 1 {
|
||||
Some((ty.lane_bits(), ty.lane_count()))
|
||||
} else {
|
||||
None
|
||||
}
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn def_inst(&mut self, val: Value) -> Option<Inst> {
|
||||
self.lower_ctx.dfg().value_def(val).inst()
|
||||
}
|
||||
isle_prelude_methods!();
|
||||
|
||||
#[inline]
|
||||
fn operand_size_of_type(&mut self, ty: Type) -> OperandSize {
|
||||
@@ -400,14 +207,6 @@ where
|
||||
None
|
||||
}
|
||||
}
|
||||
|
||||
fn u64_from_ieee32(&mut self, val: Ieee32) -> u64 {
|
||||
val.bits().into()
|
||||
}
|
||||
|
||||
fn u64_from_ieee64(&mut self, val: Ieee64) -> u64 {
|
||||
val.bits()
|
||||
}
|
||||
}
|
||||
|
||||
#[inline]
|
||||
@@ -420,10 +219,3 @@ fn to_simm32(constant: i64) -> Option<RegMemImm> {
|
||||
None
|
||||
}
|
||||
}
|
||||
|
||||
#[inline(never)]
|
||||
#[cold]
|
||||
#[track_caller]
|
||||
fn out_of_line_panic(msg: &str) -> ! {
|
||||
panic!("{}", msg);
|
||||
}
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
src/clif.isle 9c0563583e5500de00ec5e226edc0547ac3ea789c8d76f1da0401c80ec619320fdc9a6f17fd76bbcac74a5894f85385c1f51c900c2b83bc9906d03d0f29bf5cb
|
||||
src/prelude.isle 21143c73c97885afd9c34acedb8dbc653dca3b5a3e2e73754a5761762f7fe6ce2eefa520227e64c2b18f5626ca0ffa0a6aff54971099e619a464a23adf365bd2
|
||||
src/prelude.isle a069d14321601afc63959af23086709d67d189dafcdc7d1fc8534b32d89d49008acb8368b7b5a7bc51a353736a378197ac352ccce2bb3be89d93afb6979e480a
|
||||
src/isa/x64/inst.isle fdfbfc6dfad1fc5ed252e0a14ccc69baba51d0538e05cfb9916f6213e5a6fcfc9d22605a29bd684d6a66f6d5e1c8ec36a963660d52c2e8b3fb6e0758f7adb7b5
|
||||
src/isa/x64/lower.isle 8555abdae385431c96aaabc392b7b3a8b1bbe733be08b007ef776850860cb77e85a140db02f427586c155c0b0129f9ffd531abd2e4a772c72667535cc015e609
|
||||
|
||||
@@ -73,7 +73,7 @@ pub enum ConsumesFlags {
|
||||
}
|
||||
|
||||
/// Internal type ExtendKind: defined at src/isa/x64/inst.isle line 433.
|
||||
#[derive(Clone, Debug)]
|
||||
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
|
||||
pub enum ExtendKind {
|
||||
Sign,
|
||||
Zero,
|
||||
@@ -82,7 +82,7 @@ pub enum ExtendKind {
|
||||
// Generated as internal constructor for term temp_reg.
|
||||
pub fn constructor_temp_reg<C: Context>(ctx: &mut C, arg0: Type) -> Option<Reg> {
|
||||
let pattern0_0 = arg0;
|
||||
// Rule at src/prelude.isle line 57.
|
||||
// Rule at src/prelude.isle line 60.
|
||||
let expr0_0 = C::temp_writable_reg(ctx, pattern0_0);
|
||||
let expr1_0 = C::writable_reg_to_reg(ctx, expr0_0);
|
||||
return Some(expr1_0);
|
||||
@@ -91,7 +91,7 @@ pub fn constructor_temp_reg<C: Context>(ctx: &mut C, arg0: Type) -> Option<Reg>
|
||||
// Generated as internal constructor for term lo_reg.
|
||||
pub fn constructor_lo_reg<C: Context>(ctx: &mut C, arg0: Value) -> Option<Reg> {
|
||||
let pattern0_0 = arg0;
|
||||
// Rule at src/prelude.isle line 92.
|
||||
// Rule at src/prelude.isle line 95.
|
||||
let expr0_0 = C::put_in_regs(ctx, pattern0_0);
|
||||
let expr1_0: usize = 0;
|
||||
let expr2_0 = C::value_regs_get(ctx, expr0_0, expr1_0);
|
||||
|
||||
Reference in New Issue
Block a user