aboutsummaryrefslogtreecommitdiff
path: root/ffi/src
diff options
context:
space:
mode:
authoriximeow <me@iximeow.net>2021-03-13 14:19:55 -0800
committeriximeow <me@iximeow.net>2021-03-13 14:54:47 -0800
commit93c53657c2289e979672ee6c4612af7e9eac109c (patch)
treea0acdc143595f64a796827bc849a122dd49cc4e3 /ffi/src
parent6232e8b1daf7067cb2e8065687530d5f88ecb46d (diff)
split ffi crate to support distinct 16, 32, and 64-bit builds
initial work to optionally discard any instruction printing support when using `-Z build-std` to fully remove .eh_frame, a stripped long_mode_no_fmt .so is 61kb!
Diffstat (limited to 'ffi/src')
-rw-r--r--ffi/src/lib.rs55
-rw-r--r--ffi/src/long_mode.rs37
-rw-r--r--ffi/src/protected_mode.rs35
-rw-r--r--ffi/src/real_mode.rs35
-rw-r--r--ffi/src/write_sink.rs20
5 files changed, 127 insertions, 55 deletions
diff --git a/ffi/src/lib.rs b/ffi/src/lib.rs
deleted file mode 100644
index ca0bf5c..0000000
--- a/ffi/src/lib.rs
+++ /dev/null
@@ -1,55 +0,0 @@
-#![no_std]
-#![feature(lang_items)]
-
-#[panic_handler]
-#[cold]
-fn panic(_panic: &core::panic::PanicInfo) -> ! {
- loop {}
-}
-
-#[lang = "eh_personality"] extern fn eh_personality() {}
-
-use yaxpeax_arch::{Arch, Decoder, LengthedInstruction, AddressBase};
-use yaxpeax_x86::long_mode as amd64;
-
-use core::fmt::Write;
-
-#[no_mangle]
-pub unsafe extern "C" fn yaxpeax_decode_x86_64_optimistic(data: *const u8, length: u64, inst: *mut amd64::Instruction) -> bool {
- let inst: &mut amd64::Instruction = core::mem::transmute(inst);
- <amd64::Arch as Arch>::Decoder::default().decode_into(inst, core::slice::from_raw_parts(data as *const u8, length as usize).iter().cloned()).is_err()
-}
-
-#[no_mangle]
-pub unsafe extern "C" fn yaxpeax_instr_length_x86_64(inst: *mut amd64::Instruction) -> usize {
- let inst: &mut amd64::Instruction = core::mem::transmute(inst);
- 0.wrapping_offset(inst.len()).to_linear()
-}
-
-struct InstructionSink<'buf> {
- buf: &'buf mut [u8],
- offs: usize,
-}
-
-impl<'a> core::fmt::Write for InstructionSink<'a> {
- fn write_str(&mut self, s: &str) -> core::fmt::Result {
- for b in s.bytes() {
- if self.offs < self.buf.len() {
- self.buf[self.offs] = b;
- self.offs += 1;
- } else {
- break;
- }
- }
-
- Ok(())
- }
-}
-
-#[no_mangle]
-pub unsafe extern "C" fn yaxpeax_instr_fmt(inst: *mut amd64::Instruction, text: *mut u8, len: usize) {
- let inst: &mut amd64::Instruction = core::mem::transmute(inst);
- let res = core::slice::from_raw_parts_mut(text, len);
-
- write!(InstructionSink { buf: res, offs: 0 }, "{}", inst).unwrap();
-}
diff --git a/ffi/src/long_mode.rs b/ffi/src/long_mode.rs
new file mode 100644
index 0000000..a3fd784
--- /dev/null
+++ b/ffi/src/long_mode.rs
@@ -0,0 +1,37 @@
+use yaxpeax_arch::{Arch, Decoder, LengthedInstruction, AddressBase};
+use yaxpeax_x86::long_mode;
+
+#[no_mangle]
+pub unsafe extern "C" fn yaxpeax_x86_64_decode_optimistic(data: *const u8, length: u64, inst: *mut long_mode::Instruction) -> bool {
+ let inst: &mut long_mode::Instruction = core::mem::transmute(inst);
+ <long_mode::Arch as Arch>::Decoder::default().decode_into(inst, core::slice::from_raw_parts(data as *const u8, length as usize).iter().cloned()).is_err()
+}
+
+#[no_mangle]
+pub unsafe extern "C" fn yaxpeax_x86_64_instr_length(inst: *mut long_mode::Instruction) -> usize {
+ let inst: &mut long_mode::Instruction = core::mem::transmute(inst);
+ 0.wrapping_offset(inst.len()).to_linear()
+}
+
+#[cfg(feature = "fmt")]
+mod write_sink;
+
+#[cfg(feature = "fmt")]
+mod fmt {
+ use super::write_sink::InstructionSink;
+
+ use core::fmt::Write;
+
+ use yaxpeax_x86::long_mode;
+
+ #[no_mangle]
+ pub unsafe extern "C" fn yaxpeax_x86_64_fmt(inst: *mut long_mode::Instruction, text: *mut u8, len: usize) {
+ let inst: &mut long_mode::Instruction = core::mem::transmute(inst);
+ let res = core::slice::from_raw_parts_mut(text, len);
+
+ write!(InstructionSink { buf: res, offs: 0 }, "{}", inst).unwrap();
+ }
+}
+
+#[cfg(feature = "fmt")]
+pub use fmt::yaxpeax_x86_64_fmt;
diff --git a/ffi/src/protected_mode.rs b/ffi/src/protected_mode.rs
new file mode 100644
index 0000000..fc0c6a9
--- /dev/null
+++ b/ffi/src/protected_mode.rs
@@ -0,0 +1,35 @@
+use yaxpeax_arch::{Arch, Decoder, LengthedInstruction, AddressBase};
+use yaxpeax_x86::protected_mode;
+
+#[no_mangle]
+pub unsafe extern "C" fn yaxpeax_x86_32_decode_optimistic(data: *const u8, length: u64, inst: *mut protected_mode::Instruction) -> bool {
+ let inst: &mut protected_mode::Instruction = core::mem::transmute(inst);
+ <protected_mode::Arch as Arch>::Decoder::default().decode_into(inst, core::slice::from_raw_parts(data as *const u8, length as usize).iter().cloned()).is_err()
+}
+
+#[no_mangle]
+pub unsafe extern "C" fn yaxpeax_x86_32_instr_length(inst: *mut protected_mode::Instruction) -> usize {
+ let inst: &mut protected_mode::Instruction = core::mem::transmute(inst);
+ 0.wrapping_offset(inst.len()).to_linear()
+}
+
+#[cfg(fmt)]
+mod write_sink;
+
+#[cfg(fmt)]
+mod fmt {
+ use write_sink::InstructionSink;
+
+ use core::fmt::Write;
+
+ #[no_mangle]
+ pub unsafe extern "C" fn yaxpeax_x86_32_fmt(inst: *mut protected_mode::Instruction, text: *mut u8, len: usize) {
+ let inst: &mut protected_mode::Instruction = core::mem::transmute(inst);
+ let res = core::slice::from_raw_parts_mut(text, len);
+
+ write!(InstructionSink { buf: res, offs: 0 }, "{}", inst).unwrap();
+ }
+}
+
+#[cfg(fmt)]
+pub use fmt::yaxpeax_x86_32_fmt;
diff --git a/ffi/src/real_mode.rs b/ffi/src/real_mode.rs
new file mode 100644
index 0000000..6a5f866
--- /dev/null
+++ b/ffi/src/real_mode.rs
@@ -0,0 +1,35 @@
+use yaxpeax_arch::{Arch, Decoder, LengthedInstruction, AddressBase};
+use yaxpeax_x86::real_mode;
+
+#[no_mangle]
+pub unsafe extern "C" fn yaxpeax_x86_16_decode_optimistic(data: *const u8, length: u64, inst: *mut real_mode::Instruction) -> bool {
+ let inst: &mut real_mode::Instruction = core::mem::transmute(inst);
+ <real_mode::Arch as Arch>::Decoder::default().decode_into(inst, core::slice::from_raw_parts(data as *const u8, length as usize).iter().cloned()).is_err()
+}
+
+#[no_mangle]
+pub unsafe extern "C" fn yaxpeax_x86_16_instr_length(inst: *mut real_mode::Instruction) -> usize {
+ let inst: &mut real_mode::Instruction = core::mem::transmute(inst);
+ 0.wrapping_offset(inst.len()).to_linear()
+}
+
+#[cfg(fmt)]
+mod write_sink;
+
+#[cfg(fmt)]
+mod fmt {
+ use write_sink::InstructionSink;
+
+ use core::fmt::Write;
+
+ #[no_mangle]
+ pub unsafe extern "C" fn yaxpeax_x86_16_fmt(inst: *mut real_mode::Instruction, text: *mut u8, len: usize) {
+ let inst: &mut real_mode::Instruction = core::mem::transmute(inst);
+ let res = core::slice::from_raw_parts_mut(text, len);
+
+ write!(InstructionSink { buf: res, offs: 0 }, "{}", inst).unwrap();
+ }
+}
+
+#[cfg(fmt)]
+pub use fmt::yaxpeax_x86_16_fmt;
diff --git a/ffi/src/write_sink.rs b/ffi/src/write_sink.rs
new file mode 100644
index 0000000..b188498
--- /dev/null
+++ b/ffi/src/write_sink.rs
@@ -0,0 +1,20 @@
+pub struct InstructionSink<'buf> {
+ pub buf: &'buf mut [u8],
+ pub offs: usize,
+}
+
+impl<'a> core::fmt::Write for InstructionSink<'a> {
+ fn write_str(&mut self, s: &str) -> core::fmt::Result {
+ for b in s.bytes() {
+ if self.offs < self.buf.len() {
+ self.buf[self.offs] = b;
+ self.offs += 1;
+ } else {
+ break;
+ }
+ }
+
+ Ok(())
+ }
+}
+