This commit is contained in:
2026-06-17 00:53:26 -04:00
parent 026aec8565
commit 4fe4b50c8b
3 changed files with 28 additions and 28 deletions
+11 -11
View File
@@ -11,11 +11,11 @@ pub struct Code {
} }
impl Code { impl Code {
pub fn mov(&mut self, dst: impl RegMem, src: impl Into<RegImmMem>) -> ERes { pub fn mov(&mut self, dst: impl RegMem, src: impl Into<RegMemImm>) -> ERes {
let src = src.into(); let src = src.into();
match dst.kind() { match dst.kind() {
RegMemKind::Reg(mut dst) => match src { RegMemKind::Reg(mut dst) => match src {
RegImmMem::Reg(src) => { RegMemImm::Reg(src) => {
if dst.width() != src.width() { if dst.width() != src.width() {
return Err("src and dst are not same width".into()); return Err("src and dst are not same width".into());
} }
@@ -27,7 +27,7 @@ impl Code {
self.bytes.push(0x88 | dst.not8()); self.bytes.push(0x88 | dst.not8());
self.modrm(src, dst); self.modrm(src, dst);
} }
RegImmMem::Imm(src) => { RegMemImm::Imm(src) => {
let src_width = src.width_unsigned()?; let src_width = src.width_unsigned()?;
if src_width > dst.width() { if src_width > dst.width() {
return Err("immediate cannot fit in register".into()); return Err("immediate cannot fit in register".into());
@@ -47,7 +47,7 @@ impl Code {
self.imm(src, dst.width()); self.imm(src, dst.width());
} }
} }
RegImmMem::Mem(src) => { RegMemImm::Mem(src) => {
if src.width != dst.width() { if src.width != dst.width() {
return Err("register & memory sizes don't match".into()); return Err("register & memory sizes don't match".into());
} }
@@ -62,7 +62,7 @@ impl Code {
} }
}, },
RegMemKind::Mem(dst) => match src { RegMemKind::Mem(dst) => match src {
RegImmMem::Reg(src) => { RegMemImm::Reg(src) => {
if src.width() != dst.width { if src.width() != dst.width {
return Err("register & memory sizes don't match".into()); return Err("register & memory sizes don't match".into());
} }
@@ -75,7 +75,7 @@ impl Code {
self.bytes.push(0x88 | src.not8()); self.bytes.push(0x88 | src.not8());
self.modrm(src, dst); self.modrm(src, dst);
} }
RegImmMem::Imm(src) => { RegMemImm::Imm(src) => {
let encode_width = dst.width.min(Width::B32); let encode_width = dst.width.min(Width::B32);
let src_width = if dst.width == Width::B64 { let src_width = if dst.width == Width::B64 {
src.width_signed() src.width_signed()
@@ -95,15 +95,15 @@ impl Code {
self.modrm(0, dst); self.modrm(0, dst);
self.imm(src, encode_width); self.imm(src, encode_width);
} }
RegImmMem::Mem(_) => return Err("cannot move memory to memory".into()), RegMemImm::Mem(_) => return Err("cannot move memory to memory".into()),
}, },
} }
Ok(()) Ok(())
} }
pub fn push(&mut self, reg: impl Into<RegImmMem>) -> ERes { pub fn push(&mut self, reg: impl Into<RegMemImm>) -> ERes {
match reg.into() { match reg.into() {
RegImmMem::Reg(reg) => match reg.width() { RegMemImm::Reg(reg) => match reg.width() {
Width::B64 => { Width::B64 => {
if reg.gt8() { if reg.gt8() {
self.bytes.push(0x41); self.bytes.push(0x41);
@@ -113,7 +113,7 @@ impl Code {
Width::B16 => todo!(), Width::B16 => todo!(),
_ => return Err("register must be 64 or 16 bit".into()), _ => return Err("register must be 64 or 16 bit".into()),
}, },
RegImmMem::Imm(imm) => match imm.width_unsigned()? { RegMemImm::Imm(imm) => match imm.width_unsigned()? {
Width::B8 => { Width::B8 => {
self.bytes.push(0x6a); self.bytes.push(0x6a);
self.bytes.push(imm.0 as u8); self.bytes.push(imm.0 as u8);
@@ -124,7 +124,7 @@ impl Code {
} }
Width::B64 => return Err("immediate must be 32 bit or less".into()), Width::B64 => return Err("immediate must be 32 bit or less".into()),
}, },
RegImmMem::Mem(mem) => todo!(), RegMemImm::Mem(mem) => todo!(),
} }
Ok(()) Ok(())
} }
+14 -14
View File
@@ -1,11 +1,9 @@
use super::*; use super::*;
use crate::backend::Symbol; use crate::backend::Symbol;
#[derive(Clone, Copy)] pub trait RegMem: RexBit + RexW + ModRMRM + Copy + MaybeMem {
pub enum RegImmMem { fn width(&self) -> Width;
Reg(Reg), fn kind(self) -> RegMemKind;
Imm(Imm),
Mem(Mem),
} }
#[derive(Clone, Copy)] #[derive(Clone, Copy)]
@@ -14,9 +12,11 @@ pub enum RegMemKind {
Mem(Mem), Mem(Mem),
} }
pub trait RegMem: RexBit + RexW + ModRMRM + Copy + MaybeMem { #[derive(Clone, Copy)]
fn width(&self) -> Width; pub enum RegMemImm {
fn kind(self) -> RegMemKind; Reg(Reg),
Imm(Imm),
Mem(Mem),
} }
pub trait MaybeMem { pub trait MaybeMem {
@@ -54,7 +54,7 @@ impl MaybeMem for Mem {
} }
// fromrot // fromrot
impl From<Reg> for RegImmMem { impl From<Reg> for RegMemImm {
fn from(value: Reg) -> Self { fn from(value: Reg) -> Self {
Self::Reg(value) Self::Reg(value)
} }
@@ -66,7 +66,7 @@ impl From<Reg> for RegMemKind {
} }
} }
impl From<Mem> for RegImmMem { impl From<Mem> for RegMemImm {
fn from(value: Mem) -> Self { fn from(value: Mem) -> Self {
Self::Mem(value) Self::Mem(value)
} }
@@ -78,25 +78,25 @@ impl From<Mem> for RegMemKind {
} }
} }
impl From<u64> for RegImmMem { impl From<u64> for RegMemImm {
fn from(value: u64) -> Self { fn from(value: u64) -> Self {
Self::Imm(value.into()) Self::Imm(value.into())
} }
} }
impl From<i64> for RegImmMem { impl From<i64> for RegMemImm {
fn from(value: i64) -> Self { fn from(value: i64) -> Self {
Self::Imm(value.into()) Self::Imm(value.into())
} }
} }
impl From<i32> for RegImmMem { impl From<i32> for RegMemImm {
fn from(value: i32) -> Self { fn from(value: i32) -> Self {
Self::Imm(value.into()) Self::Imm(value.into())
} }
} }
impl From<i128> for RegImmMem { impl From<i128> for RegMemImm {
fn from(value: i128) -> Self { fn from(value: i128) -> Self {
Self::Imm(value.into()) Self::Imm(value.into())
} }
+3 -3
View File
@@ -64,12 +64,12 @@ pub fn parse_imm(mut s: &str, span: Span) -> Result<Imm, CompilerMsg> {
Ok(Imm(val)) Ok(Imm(val))
} }
pub fn parse_rmi(ctx: &mut crate::parser::ParseCtx) -> Result<RegImmMem, CompilerMsg> { pub fn parse_rmi(ctx: &mut crate::parser::ParseCtx) -> Result<RegMemImm, CompilerMsg> {
let next = ctx.expect_next()?; let next = ctx.expect_next()?;
let err = || CompilerMsg::unexpected_token(&next, ctx.span, "a register or immediate"); let err = || CompilerMsg::unexpected_token(&next, ctx.span, "a register or immediate");
Ok(match &next { Ok(match &next {
Token::Ident(ident) => RegImmMem::Reg(Reg::parse(ident).ok_or_else(err)?), Token::Ident(ident) => RegMemImm::Reg(Reg::parse(ident).ok_or_else(err)?),
Token::Lit(LitTy::Number(num)) => RegImmMem::Imm(parse_imm(num, ctx.span)?), Token::Lit(LitTy::Number(num)) => RegMemImm::Imm(parse_imm(num, ctx.span)?),
_ => return Err(err()), _ => return Err(err()),
}) })
} }