Compare commits
2
Commits
84e184518f
...
d66f8f02b7
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
d66f8f02b7 | ||
|
|
4e06e474ea |
+33
-25
@@ -27,7 +27,7 @@ impl Code {
|
|||||||
self.prefix16(width);
|
self.prefix16(width);
|
||||||
self.rex(width, src, 0, dst);
|
self.rex(width, src, 0, dst);
|
||||||
self.bytes.push(0x88 | width.not8());
|
self.bytes.push(0x88 | width.not8());
|
||||||
self.bytes.push(modrm_regs(src, dst));
|
self.modrm(src, dst);
|
||||||
}
|
}
|
||||||
RegImmMem::Imm(src) => {
|
RegImmMem::Imm(src) => {
|
||||||
let src_width = src.width_unsigned()?;
|
let src_width = src.width_unsigned()?;
|
||||||
@@ -60,7 +60,7 @@ impl Code {
|
|||||||
self.prefix16(dst);
|
self.prefix16(dst);
|
||||||
self.rex(dst, dst, 0, src);
|
self.rex(dst, dst, 0, src);
|
||||||
self.bytes.push(0x8a | dst.not8());
|
self.bytes.push(0x8a | dst.not8());
|
||||||
self.modrm_regdisp(dst, src);
|
self.modrm(dst, src);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
RegMem::Mem(dst) => match src {
|
RegMem::Mem(dst) => match src {
|
||||||
@@ -73,9 +73,9 @@ impl Code {
|
|||||||
}
|
}
|
||||||
self.prefix32(&dst)?;
|
self.prefix32(&dst)?;
|
||||||
self.prefix16(src);
|
self.prefix16(src);
|
||||||
self.rex(src, src, 0, dst);
|
self.rex(dst, src, 0, dst);
|
||||||
self.bytes.push(0x88 | src.not8());
|
self.bytes.push(0x88 | src.not8());
|
||||||
self.modrm_regdisp(src, dst);
|
self.modrm(src, dst);
|
||||||
}
|
}
|
||||||
RegImmMem::Imm(src) => {
|
RegImmMem::Imm(src) => {
|
||||||
let encode_width = dst.width.min(Width::B32);
|
let encode_width = dst.width.min(Width::B32);
|
||||||
@@ -94,7 +94,7 @@ impl Code {
|
|||||||
self.prefix16(encode_width);
|
self.prefix16(encode_width);
|
||||||
self.rex(dst, 0, 0, dst);
|
self.rex(dst, 0, 0, dst);
|
||||||
self.bytes.push(0xc6 | encode_width.not8());
|
self.bytes.push(0xc6 | encode_width.not8());
|
||||||
self.modrm_regdisp(None, 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()),
|
RegImmMem::Mem(_) => return Err("cannot move memory to memory".into()),
|
||||||
@@ -145,9 +145,9 @@ impl Code {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn lea(&mut self, dst: Reg, sym: Symbol) {
|
pub fn lea(&mut self, dst: Reg, sym: Symbol) {
|
||||||
self.bytes
|
self.rex(1, dst, 0, 0);
|
||||||
.extend([rex(1, dst, 0, 0), 0x8d, modrm_disp32(dst)]);
|
self.bytes.push(0x8d);
|
||||||
self.sym_offset4(sym);
|
self.modrm(dst, sym);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn int(&mut self, code: u8) {
|
pub fn int(&mut self, code: u8) {
|
||||||
@@ -172,7 +172,7 @@ impl Code {
|
|||||||
self.bytes.push(0xc3);
|
self.bytes.push(0xc3);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn sub(&mut self, dst: Reg, src: impl Into<Imm>) -> ERes {
|
fn add_sub(&mut self, dst: Reg, src: impl Into<Imm>, ext: u8) -> ERes {
|
||||||
let mut src = src.into();
|
let mut src = src.into();
|
||||||
let mut width = src.width_signed()?;
|
let mut width = src.width_signed()?;
|
||||||
let dst_width = dst.width().min(Width::B32);
|
let dst_width = dst.width().min(Width::B32);
|
||||||
@@ -198,11 +198,19 @@ impl Code {
|
|||||||
width = dst_width;
|
width = dst_width;
|
||||||
}
|
}
|
||||||
|
|
||||||
self.bytes.push(modrm(0b11, 0b101, dst.base()));
|
self.modrm(ext, dst);
|
||||||
self.imm(src, width);
|
self.imm(src, width);
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn add(&mut self, dst: Reg, src: impl Into<Imm>) -> ERes {
|
||||||
|
self.add_sub(dst, src, 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn sub(&mut self, dst: Reg, src: impl Into<Imm>) -> ERes {
|
||||||
|
self.add_sub(dst, src, 5)
|
||||||
|
}
|
||||||
|
|
||||||
fn prefix16(&mut self, width: impl Into<Width>) {
|
fn prefix16(&mut self, width: impl Into<Width>) {
|
||||||
if width.into() == Width::B16 {
|
if width.into() == Width::B16 {
|
||||||
self.bytes.push(0x66);
|
self.bytes.push(0x66);
|
||||||
@@ -224,25 +232,25 @@ impl Code {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn modrm_regdisp(&mut self, reg: impl Into<Option<Reg>>, mem: Mem) {
|
fn modrm(&mut self, reg: impl ModRMReg, rm: impl ModRMRM) {
|
||||||
const I8_MIN: i32 = i8::MIN as i32;
|
let addr = rm.addr();
|
||||||
const I8_MAX: i32 = i8::MAX as i32;
|
let mod_ = match addr {
|
||||||
let mod_ = match mem.disp {
|
EffAddr::Mem0 | EffAddr::Sym(_) => 0b00,
|
||||||
0 => 0b00,
|
EffAddr::Mem8(_) => 0b01,
|
||||||
I8_MIN..=I8_MAX => 0b01,
|
EffAddr::Mem32(_) => 0b10,
|
||||||
_ => 0b10,
|
EffAddr::None => 0b11,
|
||||||
};
|
};
|
||||||
let r = reg.into().map(|r| Reg::base(&r)).unwrap_or(0);
|
self.bytes
|
||||||
self.bytes.push(modrm(mod_, r, mem.reg.base()));
|
.push(((mod_ as u8) << 6) | (reg.val() << 3) | rm.rm());
|
||||||
if mem.reg.base() == rsp.base() {
|
if !matches!(addr, EffAddr::None) && rm.rm() == 0b100 {
|
||||||
// SIB
|
// SIB
|
||||||
self.bytes.push(0x24);
|
self.bytes.push(0x24);
|
||||||
}
|
}
|
||||||
match mod_ {
|
match addr {
|
||||||
0b00 => (),
|
EffAddr::Mem8(disp) => self.bytes.push(disp as u8),
|
||||||
0b01 => self.bytes.push(mem.disp as u8),
|
EffAddr::Mem32(disp) => self.bytes.extend(disp.to_le_bytes()),
|
||||||
0b10 => self.bytes.extend(mem.disp.to_le_bytes()),
|
EffAddr::Sym(sym) => self.sym_offset4(sym),
|
||||||
_ => unreachable!(),
|
_ => (),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -37,9 +37,15 @@ fn mov() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn sub() {
|
fn add_sub() {
|
||||||
let c = &mut TestCtx::new("mov");
|
let c = &mut TestCtx::new("mov");
|
||||||
|
|
||||||
|
for dst in regs() {
|
||||||
|
for src in imms() {
|
||||||
|
eq(c, format!("add {dst}, {src}"), |c| c.add(dst, src))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
for dst in regs() {
|
for dst in regs() {
|
||||||
for src in imms() {
|
for src in imms() {
|
||||||
eq(c, format!("sub {dst}, {src}"), |c| c.sub(dst, src))
|
eq(c, format!("sub {dst}, {src}"), |c| c.sub(dst, src))
|
||||||
|
|||||||
+75
-9
@@ -1,18 +1,84 @@
|
|||||||
|
use crate::backend::Symbol;
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|
||||||
#[inline(always)]
|
pub trait ModRMRM {
|
||||||
pub fn modrm_regs(reg: impl Into<Reg>, reg_rm: impl Into<Reg>) -> u8 {
|
fn rm(&self) -> u8;
|
||||||
modrm(0b11, reg.into().base(), reg_rm.into().base())
|
fn addr(&self) -> EffAddr;
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline(always)]
|
pub enum EffAddr {
|
||||||
pub fn modrm_disp32(reg: impl Into<Reg>) -> u8 {
|
Mem0,
|
||||||
modrm(0b00, reg.into().base(), 0b101)
|
Mem8(i8),
|
||||||
|
Mem32(i32),
|
||||||
|
Sym(Symbol),
|
||||||
|
None,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline(always)]
|
impl ModRMRM for Reg {
|
||||||
pub fn modrm(mod_: u8, reg: u8, rm: u8) -> u8 {
|
fn rm(&self) -> u8 {
|
||||||
(mod_ << 6) | (reg << 3) | rm
|
self.base()
|
||||||
|
}
|
||||||
|
fn addr(&self) -> EffAddr {
|
||||||
|
EffAddr::None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ModRMRM for Mem {
|
||||||
|
fn rm(&self) -> u8 {
|
||||||
|
self.reg.base()
|
||||||
|
}
|
||||||
|
fn addr(&self) -> EffAddr {
|
||||||
|
const I8_MIN: i32 = i8::MIN as i32;
|
||||||
|
const I8_MAX: i32 = i8::MAX as i32;
|
||||||
|
let disp = self.disp;
|
||||||
|
match disp {
|
||||||
|
0 => {
|
||||||
|
if self.reg.base() == 0b101 {
|
||||||
|
EffAddr::Mem8(0)
|
||||||
|
} else {
|
||||||
|
EffAddr::Mem0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
I8_MIN..=I8_MAX => EffAddr::Mem8(disp as i8),
|
||||||
|
_ => EffAddr::Mem32(disp),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ModRMRM for i32 {
|
||||||
|
fn rm(&self) -> u8 {
|
||||||
|
0b101
|
||||||
|
}
|
||||||
|
fn addr(&self) -> EffAddr {
|
||||||
|
EffAddr::Mem32(*self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ModRMRM for Symbol {
|
||||||
|
fn rm(&self) -> u8 {
|
||||||
|
0b101
|
||||||
|
}
|
||||||
|
|
||||||
|
fn addr(&self) -> EffAddr {
|
||||||
|
EffAddr::Sym(*self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ModRMReg for u8 {
|
||||||
|
fn val(&self) -> u8 {
|
||||||
|
*self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ModRMReg for Reg {
|
||||||
|
fn val(&self) -> u8 {
|
||||||
|
self.base()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait ModRMReg {
|
||||||
|
fn val(&self) -> u8;
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline(always)]
|
#[inline(always)]
|
||||||
|
|||||||
Binary file not shown.
Reference in New Issue
Block a user