Constant-fold icmp instructions (#5666)

We found examples of icmp instructions with both operands constant in
spidermonkey.wasm.
This commit is contained in:
Jamey Sharp
2023-02-01 13:55:36 -08:00
committed by GitHub
parent bdfb746548
commit ac4d28f4dd
5 changed files with 161 additions and 18 deletions

View File

@@ -134,6 +134,28 @@ macro_rules! isle_common_prelude_methods {
x & 0xffff_ffff
}
#[inline]
fn imm64_icmp(&mut self, ty: Type, cc: &IntCC, x: Imm64, y: Imm64) -> Imm64 {
let shift = u32::checked_sub(64, ty.bits()).unwrap_or(0);
let mask = u64::MAX >> shift;
let x = (x.bits() as u64) & mask;
let y = (y.bits() as u64) & mask;
let sext = |v| ((v << shift) as i64) >> shift;
let result = match cc {
IntCC::Equal => x == y,
IntCC::NotEqual => x != y,
IntCC::UnsignedGreaterThanOrEqual => x >= y,
IntCC::UnsignedGreaterThan => x > y,
IntCC::UnsignedLessThanOrEqual => x <= y,
IntCC::UnsignedLessThan => x < y,
IntCC::SignedGreaterThanOrEqual => sext(x) >= sext(y),
IntCC::SignedGreaterThan => sext(x) > sext(y),
IntCC::SignedLessThanOrEqual => sext(x) <= sext(y),
IntCC::SignedLessThan => sext(x) < sext(y),
};
Imm64::new(result.into())
}
#[inline]
fn ty_bits(&mut self, ty: Type) -> u8 {
use std::convert::TryInto;