rstubs/arch/acpi.rs
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 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267
//! Abstracts the ACPI standard that provides interfaces for hardware detection,
//! device configuration, and energy management.
use core::mem::size_of;
use core::ops::Range;
use core::{ptr, slice, str};
use crate::util::InlineStr;
/// This (usually!) contains the base address of the EBDA (Extended Bios Data Area), shifted right by 4
const EBDA_START_PTR: usize = 0x40e;
/// The earliest (lowest) memory address an EBDA (Extended Bios Data Area) can start
const EBDA_EARLIEST_START: usize = 0x80000;
/// The end of the EBDA (Extended Bios Data Area)
const EBDA_END: usize = 0x9ffff;
/// The start of the main BIOS area below 1mb in which to search for the RSDP (Root System Description Pointer)
const RSDP_BIOS_AREA_START: usize = 0xe0000;
/// The end of the main BIOS area below 1mb in which to search for the RSDP (Root System Description Pointer)
const RSDP_BIOS_AREA_END: usize = 0xfffff;
/// The RSDP (Root System Description Pointer)'s signature (trailing space!)
const RSDP_SIGNATURE: InlineStr<8> = InlineStr(*b"RSD PTR ");
/// Error indicating why the ACPI tables could not be loaded correctly.
#[derive(Clone, Copy, PartialEq, Eq, Debug)]
pub enum ApicError {
/// No valid RSDP was found.
NoValidRsdp,
/// The RSDP signature was incorrect.
IncorrectSignature,
/// The OEM ID was not valid UTF8.
InvalidOemId,
/// The checksum was incorrect.
InvalidChecksum,
}
/// ACPI is the successor to APM (Advanced Power Management), aiming to give
/// the operating system more control over the hardware.
///
/// This extended control, for instance, enables the operating system to assign
/// a particular amount of energy to every device
/// (e.g., by disabling a device or changing to standby mode).
/// For this purpose, BIOS and chipset provide a set of tables that describe
/// the system and its components and provide routines the OS can call.
/// These tables contain details about the system, such as the number of CPU
/// cores and the LAPIC/IOAPIC, which are determined during system boot.
#[derive(Debug, Clone, Copy)]
pub struct Acpi {
root: &'static SysDescTable,
entry_size: usize,
}
impl Acpi {
/// Load the ACPI tables by searching through the BIOS memory area.
pub fn load() -> Result<Self, ApicError> {
let rsdp = unsafe { Rsdp::search_for_on_bios() }?;
serial!(
"found rsdp oem='{}' @ {:?}",
rsdp.oem_id(),
ptr::from_ref(rsdp)
);
let acpi = if rsdp.revision != 0 && rsdp.length >= 36 {
// If the XSDT is present we must use it.
let root = unsafe { &*(rsdp.xsdt_address as *const SysDescTable) };
serial!("xsdt 0x{:?} valid={}", ptr::from_ref(root), root.valid());
Self {
root,
entry_size: 8,
}
} else {
let root = unsafe { &*(rsdp.rsdt_address as *const SysDescTable) };
serial!("rsdt 0x{:?} valid={}", ptr::from_ref(root), root.valid());
Self {
root,
entry_size: 4,
}
};
if acpi.root.valid() {
for i in 0..acpi.len() {
if let Some(entry) = acpi.entry(i) {
serial!("{i}: {}", entry.signature.as_str());
}
}
Ok(acpi)
} else {
Err(ApicError::InvalidChecksum)
}
}
pub const fn len(&self) -> usize {
(self.root.length as usize - size_of::<SysDescTable>()) / self.entry_size
}
/// Get the i-th entry
pub fn entry(&self, i: usize) -> Option<&'static SysDescTable> {
if i >= self.len() {
return None;
}
let offset = size_of::<SysDescTable>() + i * self.entry_size;
let entry_ptr = unsafe { ptr::from_ref(self.root).cast::<u8>().add(offset) };
let table = unsafe { &*entry_ptr.cast::<*const SysDescTable>().read_unaligned() };
if table.valid() {
Some(table)
} else {
serial!("Invalid table: {i}");
None
}
}
/// Find the given system descriptor table
pub fn find(&self, signature: [u8; 4]) -> Option<&'static SysDescTable> {
for i in 0..self.len() {
if let Some(table) = self.entry(i)
&& table.signature.0 == signature
{
return Some(table);
}
}
None
}
}
/// Simple marker trait for the other ACPI table types
pub trait AcpiTable {}
/// Header of an ACPI table
#[derive(Clone, Copy, Debug)]
#[repr(C, packed)]
pub struct SysDescTable {
pub signature: InlineStr<4>,
pub length: u32,
revision: u8,
checksum: u8,
oemid: [u8; 6],
oem_table_id: [u8; 8],
oem_revision: u32,
creator_id: u32,
creator_revision: u32,
}
const _: () = assert!(size_of::<SysDescTable>() == 36);
impl SysDescTable {
pub fn valid(&self) -> bool {
let mem = unsafe { slice::from_raw_parts(ptr::from_ref(self).cast(), self.length as _) };
let mut sum = 0u8;
for &b in mem {
sum = sum.wrapping_add(b);
}
sum == 0
}
pub const fn cast<T: AcpiTable>(&self) -> &T {
unsafe { &*ptr::from_ref(self).cast() }
}
}
/// The first structure found in ACPI. It just tells us where the RSDT is.
///
/// On BIOS systems, it is either found in the first 1KB of the Extended Bios Data Area, or between
/// 0x000E0000 and 0x000FFFFF. The signature is always on a 16 byte boundary. On (U)EFI, it may not
/// be located in these locations, and so an address should be found in the EFI configuration table
/// instead.
///
/// The recommended way of locating the RSDP is to let the bootloader do it - Multiboot2 can pass a
/// tag with the physical address of it. If this is not possible, a manual scan can be done.
///
/// If `revision > 0`, (the hardware ACPI version is Version 2.0 or greater), the RSDP contains
/// some new fields. For ACPI Version 1.0, these fields are not valid and should not be accessed.
#[derive(Clone, Copy, Debug)]
#[repr(C, packed)]
struct Rsdp {
signature: InlineStr<8>,
checksum: u8,
/// Name of the vendor
oem_id: [u8; 6],
/// Version number
revision: u8,
/// Address of the Root System Description Table
rsdt_address: u32,
// These fields are only valid for ACPI Version 2.0 and greater
length: u32,
// For ACPI Version 2.0+, `xsdt_address` should be used instead of `rsdt_address`.
xsdt_address: u64,
ext_checksum: u8,
reserved: [u8; 3],
}
impl Rsdp {
/// This searches for a RSDP on BIOS systems.
unsafe fn search_for_on_bios() -> Result<&'static Self, ApicError> {
for area in find_search_areas() {
serial!("search {:#x}..{:#x}", area.start, area.end);
for address in area.step_by(16) {
let rsdp = unsafe { &*(address as *const Self) };
if rsdp.signature == RSDP_SIGNATURE {
if let Err(err) = rsdp.validate() {
serial!("Invalid RSDP found at {address:#x}: {err:?}");
} else {
return Ok(rsdp);
}
}
}
}
Err(ApicError::NoValidRsdp)
}
/// Checks that:
/// 1) The signature is correct
/// 2) The checksum is correct
/// 3) For Version 2.0+, that the extension checksum is correct
fn validate(&self) -> Result<(), ApicError> {
const RSDP_V1_LENGTH: usize = 20;
// Check the signature
if self.signature != RSDP_SIGNATURE {
return Err(ApicError::IncorrectSignature);
}
// Check the OEM id is valid UTF8 (allows use of unwrap)
if str::from_utf8(&self.oem_id).is_err() {
return Err(ApicError::InvalidOemId);
}
// `self.length` doesn't exist on ACPI version 1.0, so we mustn't rely on it.
// Instead, check for version 1.0 and use a hard-coded length.
let length = if self.revision > 0 {
self.length as usize
} else {
RSDP_V1_LENGTH
};
let bytes = unsafe { slice::from_raw_parts(ptr::from_ref(self).cast(), length) };
let sum = bytes.iter().fold(0u8, |sum, &byte| sum.wrapping_add(byte));
if sum != 0 {
Err(ApicError::InvalidChecksum)
} else {
Ok(())
}
}
fn oem_id(&self) -> &str {
str::from_utf8(&self.oem_id).unwrap()
}
}
/// Find the areas we should search for the RSDP in.
fn find_search_areas() -> [Range<usize>; 2] {
// Read the base address of the EBDA from its location in the BDA (BIOS Data Area).
// Not all BIOSs fill this out unfortunately, so we might not get a sensible result.
// We shift it left 4, as it's a segment address.
let ebda_start = (unsafe { *(EBDA_START_PTR as *const u16) } as usize) << 4;
[
// The main BIOS area below 1MiB. In practice, from my [Restioson's] testing,
// the RSDP seems more often here than the EBDA.
// We also don't want to search the entire possible EBDA range,
// if we've failed to find it from the BDA.
RSDP_BIOS_AREA_START..(RSDP_BIOS_AREA_END + 1),
// Check if base segment ptr is in valid range for EBDA base
if (EBDA_EARLIEST_START..EBDA_END).contains(&ebda_start) {
// First KiB of EBDA
ebda_start..ebda_start + 1024
} else {
// We don't know where the EBDA starts, so just search the largest possible EBDA
EBDA_EARLIEST_START..(EBDA_END + 1)
},
]
}