|
@@ -1,7 +1,7 @@
|
|
//! Assembly instructions
|
|
//! Assembly instructions
|
|
|
|
|
|
macro_rules! instruction {
|
|
macro_rules! instruction {
|
|
- ($(#[$attr:meta])*, $fnname:ident, $asm:expr) => (
|
|
|
|
|
|
+ ($(#[$attr:meta])*, unsafe $fnname:ident, $asm:expr) => (
|
|
$(#[$attr])*
|
|
$(#[$attr])*
|
|
#[inline]
|
|
#[inline]
|
|
pub unsafe fn $fnname() {
|
|
pub unsafe fn $fnname() {
|
|
@@ -13,25 +13,42 @@ macro_rules! instruction {
|
|
() => unimplemented!(),
|
|
() => unimplemented!(),
|
|
}
|
|
}
|
|
}
|
|
}
|
|
- )
|
|
|
|
|
|
+ );
|
|
|
|
+ ($(#[$attr:meta])*, $fnname:ident, $asm:expr) => (
|
|
|
|
+ $(#[$attr])*
|
|
|
|
+ #[inline]
|
|
|
|
+ pub fn $fnname() {
|
|
|
|
+ match () {
|
|
|
|
+ #[cfg(riscv)]
|
|
|
|
+ () => unsafe { core::arch::asm!($asm) },
|
|
|
|
+
|
|
|
|
+ #[cfg(not(riscv))]
|
|
|
|
+ () => unimplemented!(),
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ );
|
|
}
|
|
}
|
|
|
|
|
|
instruction!(
|
|
instruction!(
|
|
/// `nop` instruction wrapper
|
|
/// `nop` instruction wrapper
|
|
///
|
|
///
|
|
- /// Generates a no-operation. Useful to prevent delay loops from being optimized away.
|
|
|
|
|
|
+ /// The `NOP` instruction does not change any architecturally visible state, except for
|
|
|
|
+ /// advancing the pc and incrementing any applicable performance counters.
|
|
|
|
+ ///
|
|
|
|
+ /// This function generates a no-operation; it's useful to prevent delay loops from being
|
|
|
|
+ /// optimized away.
|
|
, nop, "nop");
|
|
, nop, "nop");
|
|
instruction!(
|
|
instruction!(
|
|
/// `EBREAK` instruction wrapper
|
|
/// `EBREAK` instruction wrapper
|
|
///
|
|
///
|
|
/// Generates a breakpoint exception.
|
|
/// Generates a breakpoint exception.
|
|
- , ebreak, "ebreak");
|
|
|
|
|
|
+ , unsafe ebreak, "ebreak");
|
|
instruction!(
|
|
instruction!(
|
|
/// `WFI` instruction wrapper
|
|
/// `WFI` instruction wrapper
|
|
///
|
|
///
|
|
/// Provides a hint to the implementation that the current hart can be stalled until an interrupt might need servicing.
|
|
/// Provides a hint to the implementation that the current hart can be stalled until an interrupt might need servicing.
|
|
/// The WFI instruction is just a hint, and a legal implementation is to implement WFI as a NOP.
|
|
/// The WFI instruction is just a hint, and a legal implementation is to implement WFI as a NOP.
|
|
- , wfi, "wfi");
|
|
|
|
|
|
+ , unsafe wfi, "wfi");
|
|
instruction!(
|
|
instruction!(
|
|
/// `SFENCE.VMA` instruction wrapper (all address spaces and page table levels)
|
|
/// `SFENCE.VMA` instruction wrapper (all address spaces and page table levels)
|
|
///
|
|
///
|
|
@@ -40,7 +57,7 @@ instruction!(
|
|
/// are ordinarily not ordered with respect to loads and stores in the instruction stream.
|
|
/// are ordinarily not ordered with respect to loads and stores in the instruction stream.
|
|
/// Executing an `SFENCE.VMA` instruction guarantees that any stores in the instruction stream prior to the
|
|
/// Executing an `SFENCE.VMA` instruction guarantees that any stores in the instruction stream prior to the
|
|
/// `SFENCE.VMA` are ordered before all implicit references subsequent to the `SFENCE.VMA`.
|
|
/// `SFENCE.VMA` are ordered before all implicit references subsequent to the `SFENCE.VMA`.
|
|
- , sfence_vma_all, "sfence.vma");
|
|
|
|
|
|
+ , unsafe sfence_vma_all, "sfence.vma");
|
|
instruction!(
|
|
instruction!(
|
|
/// `FENCE` instruction wrapper
|
|
/// `FENCE` instruction wrapper
|
|
///
|
|
///
|
|
@@ -54,7 +71,7 @@ instruction!(
|
|
/// The FENCE instruction also orders memory reads and writes made by the hart as observed by
|
|
/// The FENCE instruction also orders memory reads and writes made by the hart as observed by
|
|
/// memory reads and writes made by an external device. However, FENCE does not order observations
|
|
/// memory reads and writes made by an external device. However, FENCE does not order observations
|
|
/// of events made by an external device using any other signaling mechanism.
|
|
/// of events made by an external device using any other signaling mechanism.
|
|
- , fence, "fence");
|
|
|
|
|
|
+ , unsafe fence, "fence");
|
|
instruction!(
|
|
instruction!(
|
|
/// `FENCE.I` instruction wrapper
|
|
/// `FENCE.I` instruction wrapper
|
|
///
|
|
///
|
|
@@ -72,7 +89,7 @@ instruction!(
|
|
/// The unused fields in the FENCE.I instruction, imm\[11:0\], rs1, and rd, are reserved for
|
|
/// The unused fields in the FENCE.I instruction, imm\[11:0\], rs1, and rd, are reserved for
|
|
/// finer-grain fences in future extensions. For forward compatibility, base
|
|
/// finer-grain fences in future extensions. For forward compatibility, base
|
|
/// implementations shall ignore these fields, and standard software shall zero these fields.
|
|
/// implementations shall ignore these fields, and standard software shall zero these fields.
|
|
- , fence_i, "fence.i");
|
|
|
|
|
|
+ , unsafe fence_i, "fence.i");
|
|
|
|
|
|
/// `SFENCE.VMA` instruction wrapper
|
|
/// `SFENCE.VMA` instruction wrapper
|
|
///
|
|
///
|
|
@@ -104,10 +121,10 @@ pub unsafe fn sfence_vma(asid: usize, addr: usize) {
|
|
/// any other case please use a more accurate method to produce a delay.
|
|
/// any other case please use a more accurate method to produce a delay.
|
|
#[inline]
|
|
#[inline]
|
|
#[allow(unused_variables)]
|
|
#[allow(unused_variables)]
|
|
-pub unsafe fn delay(cycles: u32) {
|
|
|
|
|
|
+pub fn delay(cycles: u32) {
|
|
match () {
|
|
match () {
|
|
#[cfg(riscv)]
|
|
#[cfg(riscv)]
|
|
- () => {
|
|
|
|
|
|
+ () => unsafe {
|
|
let real_cyc = 1 + cycles / 2;
|
|
let real_cyc = 1 + cycles / 2;
|
|
core::arch::asm!(
|
|
core::arch::asm!(
|
|
"1:",
|
|
"1:",
|
|
@@ -116,7 +133,7 @@ pub unsafe fn delay(cycles: u32) {
|
|
inout(reg) real_cyc => _,
|
|
inout(reg) real_cyc => _,
|
|
options(nomem, nostack),
|
|
options(nomem, nostack),
|
|
)
|
|
)
|
|
- }
|
|
|
|
|
|
+ },
|
|
|
|
|
|
#[cfg(not(riscv))]
|
|
#[cfg(not(riscv))]
|
|
() => unimplemented!(),
|
|
() => unimplemented!(),
|