madt.rs 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542
  1. use crate::{
  2. platform::{
  3. interrupt::{
  4. Apic,
  5. InterruptModel,
  6. InterruptSourceOverride,
  7. IoApic,
  8. LocalInterruptLine,
  9. NmiLine,
  10. NmiProcessor,
  11. NmiSource,
  12. Polarity,
  13. TriggerMode,
  14. },
  15. Processor,
  16. ProcessorInfo,
  17. ProcessorState,
  18. },
  19. sdt::SdtHeader,
  20. AcpiError,
  21. AcpiTable,
  22. };
  23. use alloc::vec::Vec;
  24. use bit_field::BitField;
  25. use core::{marker::PhantomData, mem};
  26. #[derive(Debug)]
  27. pub enum MadtError {
  28. UnexpectedEntry,
  29. InterruptOverrideEntryHasInvalidBus,
  30. InvalidLocalNmiLine,
  31. MpsIntiInvalidPolarity,
  32. MpsIntiInvalidTriggerMode,
  33. }
  34. /// Represents the MADT - this contains the MADT header fields. You can then iterate over a `Madt`
  35. /// to read each entry from it.
  36. ///
  37. /// In modern versions of ACPI, the MADT can detail one of four interrupt models:
  38. /// * The ancient dual-i8259 legacy PIC model
  39. /// * The Advanced Programmable Interrupt Controller (APIC) model
  40. /// * The Streamlined Advanced Programmable Interrupt Controller (SAPIC) model (for Itanium systems)
  41. /// * The Generic Interrupt Controller (GIC) model (for ARM systems)
  42. #[repr(C, packed)]
  43. pub struct Madt {
  44. header: SdtHeader,
  45. local_apic_address: u32,
  46. flags: u32,
  47. }
  48. impl AcpiTable for Madt {
  49. fn header(&self) -> &SdtHeader {
  50. &self.header
  51. }
  52. }
  53. impl Madt {
  54. pub fn parse_interrupt_model(&self) -> Result<(InterruptModel, Option<ProcessorInfo>), AcpiError> {
  55. /*
  56. * We first do a pass through the MADT to determine which interrupt model is being used.
  57. */
  58. for entry in self.entries() {
  59. match entry {
  60. MadtEntry::LocalApic(_) |
  61. MadtEntry::IoApic(_) |
  62. MadtEntry::InterruptSourceOverride(_) |
  63. MadtEntry::NmiSource(_) | // TODO: is this one used by more than one model?
  64. MadtEntry::LocalApicNmi(_) |
  65. MadtEntry::LocalApicAddressOverride(_) => {
  66. return self.parse_apic_model();
  67. }
  68. MadtEntry::IoSapic(_) |
  69. MadtEntry::LocalSapic(_) |
  70. MadtEntry::PlatformInterruptSource(_) => {
  71. unimplemented!();
  72. }
  73. MadtEntry::LocalX2Apic(_) |
  74. MadtEntry::X2ApicNmi(_) => {
  75. unimplemented!();
  76. }
  77. MadtEntry::Gicc(_) |
  78. MadtEntry::Gicd(_) |
  79. MadtEntry::GicMsiFrame(_) |
  80. MadtEntry::GicRedistributor(_) |
  81. MadtEntry::GicInterruptTranslationService(_) => {
  82. unimplemented!();
  83. }
  84. MadtEntry::MultiprocessorWakeup(_) => ()
  85. }
  86. }
  87. Ok((InterruptModel::Unknown, None))
  88. }
  89. fn parse_apic_model(&self) -> Result<(InterruptModel, Option<ProcessorInfo>), AcpiError> {
  90. let mut local_apic_address = self.local_apic_address as u64;
  91. let mut io_apic_count = 0;
  92. let mut iso_count = 0;
  93. let mut nmi_source_count = 0;
  94. let mut local_nmi_line_count = 0;
  95. let mut processor_count = 0usize;
  96. // Do a pass over the entries so we know how much space we should reserve in the vectors
  97. for entry in self.entries() {
  98. match entry {
  99. MadtEntry::IoApic(_) => io_apic_count += 1,
  100. MadtEntry::InterruptSourceOverride(_) => iso_count += 1,
  101. MadtEntry::NmiSource(_) => nmi_source_count += 1,
  102. MadtEntry::LocalApicNmi(_) => local_nmi_line_count += 1,
  103. MadtEntry::LocalApic(_) => processor_count += 1,
  104. _ => (),
  105. }
  106. }
  107. let mut io_apics = Vec::with_capacity(io_apic_count);
  108. let mut interrupt_source_overrides = Vec::with_capacity(iso_count);
  109. let mut nmi_sources = Vec::with_capacity(nmi_source_count);
  110. let mut local_apic_nmi_lines = Vec::with_capacity(local_nmi_line_count);
  111. let mut boot_processor = None;
  112. let mut application_processors = Vec::with_capacity(processor_count.saturating_sub(1)); // Subtract one for the BSP
  113. for entry in self.entries() {
  114. match entry {
  115. MadtEntry::LocalApic(entry) => {
  116. /*
  117. * The first processor is the BSP. Subsequent ones are APs. If we haven't found
  118. * the BSP yet, this must be it.
  119. */
  120. let is_ap = boot_processor.is_some();
  121. let is_disabled = !{ entry.flags }.get_bit(0);
  122. let state = match (is_ap, is_disabled) {
  123. (_, true) => ProcessorState::Disabled,
  124. (true, false) => ProcessorState::WaitingForSipi,
  125. (false, false) => ProcessorState::Running,
  126. };
  127. let processor = Processor {
  128. processor_uid: entry.processor_id,
  129. local_apic_id: entry.apic_id,
  130. state,
  131. is_ap,
  132. };
  133. if is_ap {
  134. application_processors.push(processor);
  135. } else {
  136. boot_processor = Some(processor);
  137. }
  138. }
  139. MadtEntry::IoApic(entry) => {
  140. io_apics.push(IoApic {
  141. id: entry.io_apic_id,
  142. address: entry.io_apic_address,
  143. global_system_interrupt_base: entry.global_system_interrupt_base,
  144. });
  145. }
  146. MadtEntry::InterruptSourceOverride(entry) => {
  147. if entry.bus != 0 {
  148. return Err(AcpiError::InvalidMadt(MadtError::InterruptOverrideEntryHasInvalidBus));
  149. }
  150. let (polarity, trigger_mode) = parse_mps_inti_flags(entry.flags)?;
  151. interrupt_source_overrides.push(InterruptSourceOverride {
  152. isa_source: entry.irq,
  153. global_system_interrupt: entry.global_system_interrupt,
  154. polarity,
  155. trigger_mode,
  156. });
  157. }
  158. MadtEntry::NmiSource(entry) => {
  159. let (polarity, trigger_mode) = parse_mps_inti_flags(entry.flags)?;
  160. nmi_sources.push(NmiSource {
  161. global_system_interrupt: entry.global_system_interrupt,
  162. polarity,
  163. trigger_mode,
  164. });
  165. }
  166. MadtEntry::LocalApicNmi(entry) => local_apic_nmi_lines.push(NmiLine {
  167. processor: if entry.processor_id == 0xff {
  168. NmiProcessor::All
  169. } else {
  170. NmiProcessor::ProcessorUid(entry.processor_id as u32)
  171. },
  172. line: match entry.nmi_line {
  173. 0 => LocalInterruptLine::Lint0,
  174. 1 => LocalInterruptLine::Lint1,
  175. _ => return Err(AcpiError::InvalidMadt(MadtError::InvalidLocalNmiLine)),
  176. },
  177. }),
  178. MadtEntry::LocalApicAddressOverride(entry) => {
  179. local_apic_address = entry.local_apic_address;
  180. }
  181. _ => {
  182. return Err(AcpiError::InvalidMadt(MadtError::UnexpectedEntry));
  183. }
  184. }
  185. }
  186. Ok((
  187. InterruptModel::Apic(Apic {
  188. local_apic_address,
  189. io_apics,
  190. local_apic_nmi_lines,
  191. interrupt_source_overrides,
  192. nmi_sources,
  193. also_has_legacy_pics: self.supports_8259(),
  194. }),
  195. Some(ProcessorInfo { boot_processor: boot_processor.unwrap(), application_processors }),
  196. ))
  197. }
  198. pub fn entries(&self) -> MadtEntryIter {
  199. MadtEntryIter {
  200. pointer: unsafe { (self as *const Madt as *const u8).add(mem::size_of::<Madt>()) },
  201. remaining_length: self.header.length - mem::size_of::<Madt>() as u32,
  202. _phantom: PhantomData,
  203. }
  204. }
  205. pub fn supports_8259(&self) -> bool {
  206. { self.flags }.get_bit(0)
  207. }
  208. }
  209. pub struct MadtEntryIter<'a> {
  210. pointer: *const u8,
  211. /*
  212. * The iterator can only have at most `u32::MAX` remaining bytes, because the length of the
  213. * whole SDT can only be at most `u32::MAX`.
  214. */
  215. remaining_length: u32,
  216. _phantom: PhantomData<&'a ()>,
  217. }
  218. pub enum MadtEntry<'a> {
  219. LocalApic(&'a LocalApicEntry),
  220. IoApic(&'a IoApicEntry),
  221. InterruptSourceOverride(&'a InterruptSourceOverrideEntry),
  222. NmiSource(&'a NmiSourceEntry),
  223. LocalApicNmi(&'a LocalApicNmiEntry),
  224. LocalApicAddressOverride(&'a LocalApicAddressOverrideEntry),
  225. IoSapic(&'a IoSapicEntry),
  226. LocalSapic(&'a LocalSapicEntry),
  227. PlatformInterruptSource(&'a PlatformInterruptSourceEntry),
  228. LocalX2Apic(&'a LocalX2ApicEntry),
  229. X2ApicNmi(&'a X2ApicNmiEntry),
  230. Gicc(&'a GiccEntry),
  231. Gicd(&'a GicdEntry),
  232. GicMsiFrame(&'a GicMsiFrameEntry),
  233. GicRedistributor(&'a GicRedistributorEntry),
  234. GicInterruptTranslationService(&'a GicInterruptTranslationServiceEntry),
  235. MultiprocessorWakeup(&'a MultiprocessorWakeupEntry),
  236. }
  237. impl<'a> Iterator for MadtEntryIter<'a> {
  238. type Item = MadtEntry<'a>;
  239. fn next(&mut self) -> Option<Self::Item> {
  240. while self.remaining_length > 0 {
  241. let entry_pointer = self.pointer;
  242. let header = unsafe { *(self.pointer as *const EntryHeader) };
  243. self.pointer = unsafe { self.pointer.offset(header.length as isize) };
  244. self.remaining_length -= header.length as u32;
  245. macro_rules! construct_entry {
  246. ($entry_type:expr,
  247. $entry_pointer:expr,
  248. $(($value:expr => $variant:path as $type:ty)),*
  249. ) => {
  250. match $entry_type {
  251. $(
  252. $value => {
  253. return Some($variant(unsafe {
  254. &*($entry_pointer as *const $type)
  255. }))
  256. }
  257. )*
  258. /*
  259. * These entry types are reserved by the ACPI standard. We should skip them
  260. * if they appear in a real MADT.
  261. */
  262. 0x11..=0x7f => {}
  263. /*
  264. * These entry types are reserved for OEM use. Atm, we just skip them too.
  265. * TODO: work out if we should ever do anything else here
  266. */
  267. 0x80..=0xff => {}
  268. }
  269. }
  270. }
  271. #[rustfmt::skip]
  272. construct_entry!(
  273. header.entry_type,
  274. entry_pointer,
  275. (0x0 => MadtEntry::LocalApic as LocalApicEntry),
  276. (0x1 => MadtEntry::IoApic as IoApicEntry),
  277. (0x2 => MadtEntry::InterruptSourceOverride as InterruptSourceOverrideEntry),
  278. (0x3 => MadtEntry::NmiSource as NmiSourceEntry),
  279. (0x4 => MadtEntry::LocalApicNmi as LocalApicNmiEntry),
  280. (0x5 => MadtEntry::LocalApicAddressOverride as LocalApicAddressOverrideEntry),
  281. (0x6 => MadtEntry::IoSapic as IoSapicEntry),
  282. (0x7 => MadtEntry::LocalSapic as LocalSapicEntry),
  283. (0x8 => MadtEntry::PlatformInterruptSource as PlatformInterruptSourceEntry),
  284. (0x9 => MadtEntry::LocalX2Apic as LocalX2ApicEntry),
  285. (0xa => MadtEntry::X2ApicNmi as X2ApicNmiEntry),
  286. (0xb => MadtEntry::Gicc as GiccEntry),
  287. (0xc => MadtEntry::Gicd as GicdEntry),
  288. (0xd => MadtEntry::GicMsiFrame as GicMsiFrameEntry),
  289. (0xe => MadtEntry::GicRedistributor as GicRedistributorEntry),
  290. (0xf => MadtEntry::GicInterruptTranslationService as GicInterruptTranslationServiceEntry),
  291. (0x10 => MadtEntry::MultiprocessorWakeup as MultiprocessorWakeupEntry)
  292. );
  293. }
  294. None
  295. }
  296. }
  297. #[derive(Clone, Copy)]
  298. #[repr(C, packed)]
  299. pub struct EntryHeader {
  300. entry_type: u8,
  301. length: u8,
  302. }
  303. #[repr(C, packed)]
  304. pub struct LocalApicEntry {
  305. header: EntryHeader,
  306. processor_id: u8,
  307. apic_id: u8,
  308. flags: u32,
  309. }
  310. #[repr(C, packed)]
  311. pub struct IoApicEntry {
  312. header: EntryHeader,
  313. io_apic_id: u8,
  314. _reserved: u8,
  315. io_apic_address: u32,
  316. global_system_interrupt_base: u32,
  317. }
  318. #[repr(C, packed)]
  319. pub struct InterruptSourceOverrideEntry {
  320. header: EntryHeader,
  321. bus: u8, // 0 - ISA bus
  322. irq: u8, // This is bus-relative
  323. global_system_interrupt: u32,
  324. flags: u16,
  325. }
  326. #[repr(C, packed)]
  327. pub struct NmiSourceEntry {
  328. header: EntryHeader,
  329. flags: u16,
  330. global_system_interrupt: u32,
  331. }
  332. #[repr(C, packed)]
  333. pub struct LocalApicNmiEntry {
  334. header: EntryHeader,
  335. processor_id: u8,
  336. flags: u16,
  337. nmi_line: u8, // Describes which LINTn is the NMI connected to
  338. }
  339. #[repr(C, packed)]
  340. pub struct LocalApicAddressOverrideEntry {
  341. header: EntryHeader,
  342. _reserved: u16,
  343. local_apic_address: u64,
  344. }
  345. /// If this entry is present, the system has an I/O SAPIC, which must be used instead of the I/O
  346. /// APIC.
  347. #[repr(C, packed)]
  348. pub struct IoSapicEntry {
  349. header: EntryHeader,
  350. io_apic_id: u8,
  351. _reserved: u8,
  352. global_system_interrupt_base: u32,
  353. io_sapic_address: u64,
  354. }
  355. #[repr(C, packed)]
  356. pub struct LocalSapicEntry {
  357. header: EntryHeader,
  358. processor_id: u8,
  359. local_sapic_id: u8,
  360. local_sapic_eid: u8,
  361. _reserved: [u8; 3],
  362. flags: u32,
  363. processor_uid: u32,
  364. /// This string can be used to associate this local SAPIC to a processor defined in the
  365. /// namespace when the `_UID` object is a string. It is a null-terminated ASCII string, and so
  366. /// this field will be `'\0'` if the string is not present, otherwise it extends from the
  367. /// address of this field.
  368. processor_uid_string: u8,
  369. }
  370. #[repr(C, packed)]
  371. pub struct PlatformInterruptSourceEntry {
  372. header: EntryHeader,
  373. flags: u16,
  374. interrupt_type: u8,
  375. processor_id: u8,
  376. processor_eid: u8,
  377. io_sapic_vector: u8,
  378. global_system_interrupt: u32,
  379. platform_interrupt_source_flags: u32,
  380. }
  381. #[repr(C, packed)]
  382. pub struct LocalX2ApicEntry {
  383. header: EntryHeader,
  384. _reserved: u16,
  385. x2apic_id: u32,
  386. flags: u32,
  387. processor_uid: u32,
  388. }
  389. #[repr(C, packed)]
  390. pub struct X2ApicNmiEntry {
  391. header: EntryHeader,
  392. flags: u16,
  393. processor_uid: u32,
  394. nmi_line: u8,
  395. _reserved: [u8; 3],
  396. }
  397. /// This field will appear for ARM processors that support ACPI and use the Generic Interrupt
  398. /// Controller. In the GICC interrupt model, each logical process has a Processor Device object in
  399. /// the namespace, and uses this structure to convey its GIC information.
  400. #[repr(C, packed)]
  401. pub struct GiccEntry {
  402. header: EntryHeader,
  403. _reserved1: u16,
  404. cpu_interface_number: u32,
  405. processor_uid: u32,
  406. flags: u32,
  407. parking_protocol_version: u32,
  408. performance_interrupt_gsiv: u32,
  409. parked_address: u64,
  410. gic_registers_address: u64,
  411. gic_control_block_address: u64,
  412. vgic_maintenance_interrupt: u32,
  413. gicr_base_address: u64,
  414. mpidr: u64,
  415. processor_power_efficiency_class: u8,
  416. _reserved2: [u8; 3],
  417. }
  418. #[repr(C, packed)]
  419. pub struct GicdEntry {
  420. header: EntryHeader,
  421. _reserved1: u16,
  422. gic_id: u32,
  423. physical_base_address: u64,
  424. system_vector_base: u32,
  425. /// The GIC version
  426. /// 0x00: Fall back to hardware discovery
  427. /// 0x01: GICv1
  428. /// 0x02: GICv2
  429. /// 0x03: GICv3
  430. /// 0x04: GICv4
  431. /// 0x05-0xff: Reserved for future use
  432. gic_version: u8,
  433. _reserved2: [u8; 3],
  434. }
  435. #[repr(C, packed)]
  436. pub struct GicMsiFrameEntry {
  437. header: EntryHeader,
  438. _reserved: u16,
  439. frame_id: u32,
  440. physical_base_address: u64,
  441. flags: u32,
  442. spi_count: u16,
  443. spi_base: u16,
  444. }
  445. #[repr(C, packed)]
  446. pub struct GicRedistributorEntry {
  447. header: EntryHeader,
  448. _reserved: u16,
  449. discovery_range_base_address: u64,
  450. discovery_range_length: u32,
  451. }
  452. #[repr(C, packed)]
  453. pub struct GicInterruptTranslationServiceEntry {
  454. header: EntryHeader,
  455. _reserved1: u16,
  456. id: u32,
  457. physical_base_address: u64,
  458. _reserved2: u32,
  459. }
  460. #[repr(C, packed)]
  461. pub struct MultiprocessorWakeupEntry {
  462. header: EntryHeader,
  463. mailbox_version: u16,
  464. _reserved: u32,
  465. mailbox_address: u64,
  466. }
  467. fn parse_mps_inti_flags(flags: u16) -> Result<(Polarity, TriggerMode), AcpiError> {
  468. let polarity = match flags.get_bits(0..2) {
  469. 0b00 => Polarity::SameAsBus,
  470. 0b01 => Polarity::ActiveHigh,
  471. 0b11 => Polarity::ActiveLow,
  472. _ => return Err(AcpiError::InvalidMadt(MadtError::MpsIntiInvalidPolarity)),
  473. };
  474. let trigger_mode = match flags.get_bits(2..4) {
  475. 0b00 => TriggerMode::SameAsBus,
  476. 0b01 => TriggerMode::Edge,
  477. 0b11 => TriggerMode::Level,
  478. _ => return Err(AcpiError::InvalidMadt(MadtError::MpsIntiInvalidTriggerMode)),
  479. };
  480. Ok((polarity, trigger_mode))
  481. }