1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73
//! Rust wrappers around the x86-family I/O instructions.
use core::arch::asm;
/// Read a `u8`-sized value from `port`.
pub unsafe fn inb(port: u16) -> u8 {
// The registers for the `in` and `out` instructions are always the
// same: `a` for value, and `d` for the port address.
let result: u8;
asm!(
"in al, dx",
in("dx") port,
out("al") result,
options(nomem, nostack)
);
result
}
/// Write a `u8`-sized `value` to `port`.
pub unsafe fn outb(value: u8, port: u16) {
asm!(
"out dx, al",
in("dx") port,
in("al") value,
options(nomem, nostack)
);
}
/// Read a `u16`-sized value from `port`.
pub unsafe fn inw(port: u16) -> u16 {
let result: u16;
asm!(
"in ax, dx",
in("dx") port,
out("ax") result,
options(nomem, nostack)
);
result
}
/// Write a `u8`-sized `value` to `port`.
pub unsafe fn outw(value: u16, port: u16) {
asm!(
"out dx, ax",
in("dx") port,
in("ax") value,
options(nomem, nostack)
);
}
/// Read a `u32`-sized value from `port`.
pub unsafe fn inl(port: u16) -> u32 {
let result: u32;
asm!(
"in eax, dx",
in("dx") port,
out("eax") result,
options(nomem, nostack)
);
result
}
/// Write a `u32`-sized `value` to `port`.
pub unsafe fn outl(value: u32, port: u16) {
asm!(
"out dx, eax",
in("dx") port,
in("eax") value,
options(nomem, nostack)
);
}