ostd 0.8.4

Rust OS framework that facilitates the development of and innovation in OS kernels
// SPDX-License-Identifier: MPL-2.0

//! Architecture dependent CPU-local information utilities.

use x86_64::registers::segmentation::{Segment64, GS};

/// Sets the base address for the CPU local storage by writing to the GS base model-specific register.
/// This operation is marked as `unsafe` because it directly interfaces with low-level CPU registers.
///
/// # Safety
///
///  - This function is safe to call provided that the GS register is dedicated entirely for CPU local storage
///    and is not concurrently accessed for other purposes.
///  - The caller must ensure that `addr` is a valid address and properly aligned, as required by the CPU.
///  - This function should only be called in contexts where the CPU is in a state to accept such changes,
///    such as during processor initialization.
pub(crate) unsafe fn set_base(addr: u64) {
    GS::write_base(x86_64::addr::VirtAddr::new(addr));
}

/// Gets the base address for the CPU local storage by reading the GS base model-specific register.
pub(crate) fn get_base() -> u64 {
    GS::read_base().as_u64()
}

use crate::cpu::local::single_instr::{
    SingleInstructionAddAssign, SingleInstructionBitAndAssign, SingleInstructionBitOrAssign,
    SingleInstructionBitXorAssign, SingleInstructionLoad, SingleInstructionStore,
    SingleInstructionSubAssign,
};

/// The GDT ensures that the GS segment is initialized to zero on boot.
/// This assertion checks that the base address has been set.
macro_rules! debug_assert_initialized {
    () => {
        // The compiler may think that [`super::get_base`] has side effects
        // so it may not be optimized out. We make sure that it will be
        // conditionally compiled only in debug builds.
        #[cfg(debug_assertions)]
        debug_assert_ne!(get_base(), 0);
    };
}

macro_rules! impl_numeric_single_instruction_for {
    ($([$typ: ty, $inout_type: ident, $register_format: expr])*) => {$(

        impl SingleInstructionAddAssign<$typ> for $typ {
            unsafe fn add_assign(offset: *mut Self, val: Self) {
                debug_assert_initialized!();

                core::arch::asm!(
                    concat!("add gs:[{0}], {1", $register_format, "}"),
                    in(reg) offset,
                    in($inout_type) val,
                    options(nostack),
                );
            }
        }

        impl SingleInstructionSubAssign<$typ> for $typ {
            unsafe fn sub_assign(offset: *mut Self, val: Self) {
                debug_assert_initialized!();

                core::arch::asm!(
                    concat!("sub gs:[{0}], {1", $register_format, "}"),
                    in(reg) offset,
                    in($inout_type) val,
                    options(nostack),
                );
            }
        }

        impl SingleInstructionBitAndAssign<$typ> for $typ {
            unsafe fn bitand_assign(offset: *mut Self, val: Self) {
                debug_assert_initialized!();

                core::arch::asm!(
                    concat!("and gs:[{0}], {1", $register_format, "}"),
                    in(reg) offset,
                    in($inout_type) val,
                    options(nostack),
                );
            }
        }

        impl SingleInstructionBitOrAssign<$typ> for $typ {
            unsafe fn bitor_assign(offset: *mut Self, val: Self) {
                debug_assert_initialized!();

                core::arch::asm!(
                    concat!("or gs:[{0}], {1", $register_format, "}"),
                    in(reg) offset,
                    in($inout_type) val,
                    options(nostack),
                );
            }
        }

        impl SingleInstructionBitXorAssign<$typ> for $typ {
            unsafe fn bitxor_assign(offset: *mut Self, val: Self) {
                debug_assert_initialized!();

                core::arch::asm!(
                    concat!("xor gs:[{0}], {1", $register_format, "}"),
                    in(reg) offset,
                    in($inout_type) val,
                    options(nostack),
                );
            }
        }

        impl SingleInstructionLoad for $typ {
            unsafe fn load(offset: *const Self) -> Self {
                debug_assert_initialized!();

                let val: Self;
                core::arch::asm!(
                    concat!("mov {0", $register_format, "}, gs:[{1}]"),
                    out($inout_type) val,
                    in(reg) offset,
                    options(nostack, readonly),
                );
                val
            }
        }

        impl SingleInstructionStore for $typ {
            unsafe fn store(offset: *mut Self, val: Self) {
                debug_assert_initialized!();

                core::arch::asm!(
                    concat!("mov gs:[{0}], {1", $register_format, "}"),
                    in(reg) offset,
                    in($inout_type) val,
                    options(nostack),
                );
            }
        }

    )*};
}

impl_numeric_single_instruction_for!(
    [u64,   reg,    ":r"]
    [usize, reg,    ":r"]
    [u32,   reg,    ":e"]
    [u16,   reg,    ":x"]
    [u8,    reg_byte, ""]
    [i64,   reg,    ":r"]
    [isize, reg,    ":r"]
    [i32,   reg,    ":e"]
    [i16,   reg,    ":x"]
    [i8,    reg_byte, ""]
);

macro_rules! impl_generic_single_instruction_for {
    ($([<$gen_type:ident $(, $more_gen_type:ident)*>, $typ:ty])*) => {$(

        impl<$gen_type $(, $more_gen_type)*> SingleInstructionLoad for $typ {
            unsafe fn load(offset: *const Self) -> Self {
                debug_assert_initialized!();

                let val: Self;
                core::arch::asm!(
                    concat!("mov {0}, gs:[{1}]"),
                    out(reg) val,
                    in(reg) offset,
                    options(nostack, readonly),
                );
                val
            }
        }

        impl<$gen_type $(, $more_gen_type)*> SingleInstructionStore for $typ {
            unsafe fn store(offset: *mut Self, val: Self) {
                debug_assert_initialized!();

                core::arch::asm!(
                    concat!("mov gs:[{0}], {1}"),
                    in(reg) offset,
                    in(reg) val,
                    options(nostack),
                );
            }
        }
    )*}
}

impl_generic_single_instruction_for!(
    [<T>, *const T]
    [<T>, *mut T]
    [<T, R>, fn(T) -> R]
);

// In this module, booleans are represented by the least significant bit of a
// `u8` type. Other bits must be zero. This definition is compatible with the
// Rust reference: <https://doc.rust-lang.org/reference/types/boolean.html>.

impl SingleInstructionLoad for bool {
    unsafe fn load(offset: *const Self) -> Self {
        debug_assert_initialized!();

        let val: u8;
        core::arch::asm!(
            "mov {0}, gs:[{1}]",
            out(reg_byte) val,
            in(reg) offset,
            options(nostack, readonly),
        );
        debug_assert!(val == 1 || val == 0);
        val == 1
    }
}

impl SingleInstructionStore for bool {
    unsafe fn store(offset: *mut Self, val: Self) {
        debug_assert_initialized!();

        let val: u8 = if val { 1 } else { 0 };
        core::arch::asm!(
            "mov gs:[{0}], {1}",
            in(reg) offset,
            in(reg_byte) val,
            options(nostack),
        );
    }
}