From bf202a42baef1affc245002af16482c0747fed48 Mon Sep 17 00:00:00 2001 From: Mark Date: Sun, 2 Mar 2025 10:11:19 -0800 Subject: [PATCH] Minor refactor --- tetros/src/idt/stackframe.rs | 100 +---------------------------------- tetros/src/lib.rs | 5 +- tetros/src/os/eflags.rs | 98 ++++++++++++++++++++++++++++++++++ tetros/src/os/mod.rs | 8 ++- tetros/src/os/thunk.rs | 3 +- tetros/src/os/util.rs | 2 +- 6 files changed, 111 insertions(+), 105 deletions(-) create mode 100644 tetros/src/os/eflags.rs diff --git a/tetros/src/idt/stackframe.rs b/tetros/src/idt/stackframe.rs index fb207b5..0360f0c 100644 --- a/tetros/src/idt/stackframe.rs +++ b/tetros/src/idt/stackframe.rs @@ -1,105 +1,9 @@ -use bitflags::bitflags; -use core::arch::asm; use core::{fmt, ops::Deref}; +use crate::os::EFlags; + use super::VirtAddr; -bitflags! { - /// The EFLAGS register. All bit patterns are valid representations for this type. - /// - /// See https://wiki.osdev.org/CPU_Registers_x86#EFLAGS_Register - #[repr(transparent)] - #[derive(PartialEq, Eq, PartialOrd, Ord, Hash, Debug, Clone, Copy)] - pub struct EFlags: u32 { - /// Processor feature identification flag. - /// - /// If this flag is modifiable, the CPU supports CPUID. - const ID = 1 << 21; - - /// Indicates that an external, maskable interrupt is pending. - /// - /// Used when virtual-8086 mode extensions (CR4.VME) or protected-mode virtual - /// interrupts (CR4.PVI) are activated. - const VIRTUAL_INTERRUPT_PENDING = 1 << 20; - - /// Virtual image of the INTERRUPT_FLAG bit. - /// - /// Used when virtual-8086 mode extensions (CR4.VME) or protected-mode virtual - /// interrupts (CR4.PVI) are activated. - const VIRTUAL_INTERRUPT = 1 << 19; - - /// Enable automatic alignment checking if CR0.AM is set. Only works if CPL is 3. - const ALIGNMENT_CHECK = 1 << 18; - - /// Enable the virtual-8086 mode. - const VIRTUAL_8086_MODE = 1 << 17; - - /// Allows to restart an instruction following an instruction breakpoint. - const RESUME_FLAG = 1 << 16; - - /// Used by `iret` in hardware task switch mode to determine if current task is nested. - const NESTED_TASK = 1 << 14; - - /// The high bit of the I/O Privilege Level field. - /// - /// Specifies the privilege level required for executing I/O address-space instructions. - const IOPL_HIGH = 1 << 13; - - /// The low bit of the I/O Privilege Level field. - /// - /// Specifies the privilege level required for executing I/O address-space instructions. - const IOPL_LOW = 1 << 12; - - /// Set by hardware to indicate that the sign bit of the result of the last signed integer - /// operation differs from the source operands. - const OVERFLOW_FLAG = 1 << 11; - - /// Determines the order in which strings are processed. - const DIRECTION_FLAG = 1 << 10; - - /// Enable interrupts. - const INTERRUPT_FLAG = 1 << 9; - - /// Enable single-step mode for debugging. - const TRAP_FLAG = 1 << 8; - - /// Set by hardware if last arithmetic operation resulted in a negative value. - const SIGN_FLAG = 1 << 7; - - /// Set by hardware if last arithmetic operation resulted in a zero value. - const ZERO_FLAG = 1 << 6; - - /// Set by hardware if last arithmetic operation generated a carry ouf of bit 3 of the - /// result. - const AUXILIARY_CARRY_FLAG = 1 << 4; - - /// Set by hardware if last result has an even number of 1 bits (only for some operations). - const PARITY_FLAG = 1 << 2; - - /// Set by hardware if last arithmetic operation generated a carry out of the - /// most-significant bit of the result. - const CARRY_FLAG = 1; - } -} - -impl EFlags { - #[inline] - pub fn read() -> EFlags { - EFlags::from_bits_truncate(EFlags::read_raw()) - } - - #[inline] - fn read_raw() -> u32 { - let r: u32; - - unsafe { - asm!("pushfd; pop {0:e}", out(reg) r, options(nomem, preserves_flags)); - } - - r - } -} - /// Wrapper type for the interrupt stack frame pushed by the CPU. /// /// This type derefs to an [`InterruptStackFrameValue`], which allows reading the actual values. diff --git a/tetros/src/lib.rs b/tetros/src/lib.rs index 2dada67..8b5e649 100644 --- a/tetros/src/lib.rs +++ b/tetros/src/lib.rs @@ -12,8 +12,8 @@ use spin::Mutex; use drivers::{pic::PICDriver, vga::Vga13h}; use idt::{InterruptDescriptorTable, InterruptStackFrame}; use os::{ - thunk::ThunkData, util::{sti, without_interrupts}, + ThunkData, }; use tetrisboard::{FallingTetromino, TetrisBoard}; @@ -148,8 +148,7 @@ pub unsafe extern "C" fn start(thunk10: extern "C" fn()) -> ! { pic.init(); } - // We're ready for interrupts, - // enable them + // We're ready for interrupts, enable them sti(); let mut last_t = 0; diff --git a/tetros/src/os/eflags.rs b/tetros/src/os/eflags.rs new file mode 100644 index 0000000..2395a0b --- /dev/null +++ b/tetros/src/os/eflags.rs @@ -0,0 +1,98 @@ +use bitflags::bitflags; +use core::arch::asm; + +bitflags! { + /// The EFLAGS register. All bit patterns are valid representations for this type. + /// + /// See https://wiki.osdev.org/CPU_Registers_x86#EFLAGS_Register + #[repr(transparent)] + #[derive(PartialEq, Eq, PartialOrd, Ord, Hash, Debug, Clone, Copy)] + pub struct EFlags: u32 { + /// Processor feature identification flag. + /// + /// If this flag is modifiable, the CPU supports CPUID. + const ID = 1 << 21; + + /// Indicates that an external, maskable interrupt is pending. + /// + /// Used when virtual-8086 mode extensions (CR4.VME) or protected-mode virtual + /// interrupts (CR4.PVI) are activated. + const VIRTUAL_INTERRUPT_PENDING = 1 << 20; + + /// Virtual image of the INTERRUPT_FLAG bit. + /// + /// Used when virtual-8086 mode extensions (CR4.VME) or protected-mode virtual + /// interrupts (CR4.PVI) are activated. + const VIRTUAL_INTERRUPT = 1 << 19; + + /// Enable automatic alignment checking if CR0.AM is set. Only works if CPL is 3. + const ALIGNMENT_CHECK = 1 << 18; + + /// Enable the virtual-8086 mode. + const VIRTUAL_8086_MODE = 1 << 17; + + /// Allows to restart an instruction following an instruction breakpoint. + const RESUME_FLAG = 1 << 16; + + /// Used by `iret` in hardware task switch mode to determine if current task is nested. + const NESTED_TASK = 1 << 14; + + /// The high bit of the I/O Privilege Level field. + /// + /// Specifies the privilege level required for executing I/O address-space instructions. + const IOPL_HIGH = 1 << 13; + + /// The low bit of the I/O Privilege Level field. + /// + /// Specifies the privilege level required for executing I/O address-space instructions. + const IOPL_LOW = 1 << 12; + + /// Set by hardware to indicate that the sign bit of the result of the last signed integer + /// operation differs from the source operands. + const OVERFLOW_FLAG = 1 << 11; + + /// Determines the order in which strings are processed. + const DIRECTION_FLAG = 1 << 10; + + /// Enable interrupts. + const INTERRUPT_FLAG = 1 << 9; + + /// Enable single-step mode for debugging. + const TRAP_FLAG = 1 << 8; + + /// Set by hardware if last arithmetic operation resulted in a negative value. + const SIGN_FLAG = 1 << 7; + + /// Set by hardware if last arithmetic operation resulted in a zero value. + const ZERO_FLAG = 1 << 6; + + /// Set by hardware if last arithmetic operation generated a carry ouf of bit 3 of the + /// result. + const AUXILIARY_CARRY_FLAG = 1 << 4; + + /// Set by hardware if last result has an even number of 1 bits (only for some operations). + const PARITY_FLAG = 1 << 2; + + /// Set by hardware if last arithmetic operation generated a carry out of the + /// most-significant bit of the result. + const CARRY_FLAG = 1; + } +} + +impl EFlags { + #[inline] + pub fn read() -> EFlags { + EFlags::from_bits_truncate(EFlags::read_raw()) + } + + #[inline] + fn read_raw() -> u32 { + let r: u32; + + unsafe { + asm!("pushfd; pop {0:e}", out(reg) r, options(nomem, preserves_flags)); + } + + r + } +} diff --git a/tetros/src/os/mod.rs b/tetros/src/os/mod.rs index 5b1be8e..d738fd5 100644 --- a/tetros/src/os/mod.rs +++ b/tetros/src/os/mod.rs @@ -1,7 +1,11 @@ -pub mod thunk; +mod thunk; +pub use thunk::*; + +mod eflags; +pub use eflags::*; + pub mod util; #[macro_use] pub mod panic; -const THUNK_STACK_ADDR: usize = 0x7C00; // Grows downwards diff --git a/tetros/src/os/thunk.rs b/tetros/src/os/thunk.rs index e960d60..11284cf 100644 --- a/tetros/src/os/thunk.rs +++ b/tetros/src/os/thunk.rs @@ -1,6 +1,7 @@ use core::ptr; -use super::THUNK_STACK_ADDR; +// Grows downwards +const THUNK_STACK_ADDR: usize = 0x7C00; #[derive(Clone, Copy, Debug)] #[repr(C, packed)] diff --git a/tetros/src/os/util.rs b/tetros/src/os/util.rs index 56cd62a..bf4f004 100644 --- a/tetros/src/os/util.rs +++ b/tetros/src/os/util.rs @@ -1,6 +1,6 @@ use core::arch::asm; -use crate::idt::EFlags; +use super::EFlags; /// Disable interrupts #[inline]