cranelift_assembler_x64/
imm.rs
#![allow(clippy::module_name_repetitions, reason = "'imm::Imm*' is fine")]
use crate::api::CodeSink;
use std::fmt;
macro_rules! hexify {
($n:expr) => {
format!("$0x{:x}", $n)
};
}
macro_rules! hexify_sign_extend {
($n:expr, $from:ty => $to:ty) => {{
let from: $from = $n; let to = <$to>::from(from);
format!("$0x{:x}", to)
}};
}
#[derive(Clone, Copy, Debug)]
#[cfg_attr(any(test, feature = "fuzz"), derive(arbitrary::Arbitrary))]
pub struct Imm8(u8);
impl Imm8 {
#[must_use]
pub fn new(value: u8) -> Self {
Self(value)
}
#[must_use]
pub fn value(&self) -> u8 {
self.0
}
pub fn encode(&self, sink: &mut impl CodeSink) {
sink.put1(self.0);
}
}
impl fmt::Display for Imm8 {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "$0x{:x}", self.0)
}
}
#[derive(Clone, Copy, Debug)]
#[cfg_attr(any(test, feature = "fuzz"), derive(arbitrary::Arbitrary))]
pub struct Simm8(i8);
impl Simm8 {
#[must_use]
pub fn new(value: i8) -> Self {
Self(value)
}
#[must_use]
pub fn value(&self) -> i8 {
self.0
}
pub fn encode(&self, sink: &mut impl CodeSink) {
#[allow(clippy::cast_sign_loss, reason = "bit conversion is intended here")]
sink.put1(self.0 as u8);
}
#[must_use]
pub fn to_string(&self, extend: Extension) -> String {
use Extension::{None, SignExtendLong, SignExtendQuad, SignExtendWord};
match extend {
None => hexify!(self.0),
SignExtendWord => hexify_sign_extend!(self.0, i8 => i16),
SignExtendLong => hexify_sign_extend!(self.0, i8 => i32),
SignExtendQuad => hexify_sign_extend!(self.0, i8 => i64),
}
}
}
#[derive(Clone, Debug)]
#[cfg_attr(any(test, feature = "fuzz"), derive(arbitrary::Arbitrary))]
pub struct Imm16(u16);
impl Imm16 {
#[must_use]
pub fn new(value: u16) -> Self {
Self(value)
}
#[must_use]
pub fn value(&self) -> u16 {
self.0
}
pub fn encode(&self, sink: &mut impl CodeSink) {
sink.put2(self.0);
}
}
impl fmt::Display for Imm16 {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "$0x{:x}", self.0)
}
}
#[derive(Clone, Debug)]
#[cfg_attr(any(test, feature = "fuzz"), derive(arbitrary::Arbitrary))]
pub struct Simm16(i16);
impl Simm16 {
#[must_use]
pub fn new(value: i16) -> Self {
Self(value)
}
#[must_use]
pub fn value(&self) -> i16 {
self.0
}
pub fn encode(&self, sink: &mut impl CodeSink) {
#[allow(clippy::cast_sign_loss, reason = "bit conversion is intended here")]
sink.put2(self.0 as u16);
}
#[must_use]
pub fn to_string(&self, extend: Extension) -> String {
use Extension::{None, SignExtendLong, SignExtendQuad, SignExtendWord};
match extend {
None => hexify!(self.0),
SignExtendWord => unreachable!("the 16-bit value is already 16 bits"),
SignExtendLong => hexify_sign_extend!(self.0, i16 => i32),
SignExtendQuad => hexify_sign_extend!(self.0, i16 => i64),
}
}
}
#[derive(Clone, Debug)]
#[cfg_attr(any(test, feature = "fuzz"), derive(arbitrary::Arbitrary))]
pub struct Imm32(u32);
impl Imm32 {
#[must_use]
pub fn new(value: u32) -> Self {
Self(value)
}
#[must_use]
pub fn value(&self) -> u32 {
self.0
}
pub fn encode(&self, sink: &mut impl CodeSink) {
sink.put4(self.0);
}
}
impl fmt::Display for Imm32 {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "$0x{:x}", self.0)
}
}
#[derive(Clone, Debug)]
#[cfg_attr(any(test, feature = "fuzz"), derive(arbitrary::Arbitrary))]
pub struct Simm32(i32);
impl Simm32 {
#[must_use]
pub fn new(value: i32) -> Self {
Self(value)
}
#[must_use]
pub fn value(&self) -> i32 {
self.0
}
pub fn encode(&self, sink: &mut impl CodeSink) {
#[allow(clippy::cast_sign_loss, reason = "bit conversion is intended here")]
sink.put4(self.0 as u32);
}
#[must_use]
pub fn to_string(&self, extend: Extension) -> String {
use Extension::{None, SignExtendLong, SignExtendQuad, SignExtendWord};
match extend {
None => hexify!(self.0),
SignExtendWord => unreachable!("cannot sign extend a 32-bit value to 16 bits"),
SignExtendLong => unreachable!("the 32-bit value is already 32 bits"),
SignExtendQuad => hexify_sign_extend!(self.0, i32 => i64),
}
}
}
#[derive(Clone, Copy, Debug)]
pub enum Extension {
None,
SignExtendQuad,
SignExtendLong,
SignExtendWord,
}