ostd/arch/x86/cpu/
cpuid.rs

1// SPDX-License-Identifier: MPL-2.0
2
3//! CPU information from the CPUID instruction.
4
5use core::arch::x86_64::CpuidResult;
6
7use spin::Once;
8
9static MAX_LEAF: Once<u32> = Once::new();
10static MAX_HYPERVISOR_LEAF: Once<u32> = Once::new();
11static MAX_EXTENDED_LEAF: Once<u32> = Once::new();
12
13#[repr(u32)]
14enum Leaf {
15    Base = 0x00,
16    Xstate = 0x0d,
17    Tsc = 0x15,
18
19    HypervisorBase = 0x40000000,
20    ExtBase = 0x80000000,
21}
22
23/// Executes the CPUID instruction for the given leaf and subleaf.
24///
25/// This method will return `None` if the leaf is not supported.
26pub fn cpuid(leaf: u32, subleaf: u32) -> Option<CpuidResult> {
27    fn raw_cpuid(leaf: u32, subleaf: u32) -> CpuidResult {
28        // SAFETY: It is safe to execute the CPUID instruction.
29        unsafe { core::arch::x86_64::__cpuid_count(leaf, subleaf) }
30    }
31
32    let max_leaf = if leaf < Leaf::HypervisorBase as u32 {
33        // Standard leaves (0x0000_0000 - 0x3FFF_FFFF)
34        *MAX_LEAF.call_once(|| raw_cpuid(Leaf::Base as u32, 0).eax)
35    } else if leaf < Leaf::ExtBase as u32 {
36        // Hypervisor leaves (0x4000_0000 - 0x7FFF_FFFF)
37        *MAX_HYPERVISOR_LEAF.call_once(|| raw_cpuid(Leaf::HypervisorBase as u32, 0).eax)
38    } else {
39        // Extended leaves (0x8000_0000 - 0xFFFF_FFFF)
40        *MAX_EXTENDED_LEAF.call_once(|| raw_cpuid(Leaf::ExtBase as u32, 0).eax)
41    };
42
43    if leaf > max_leaf {
44        None
45    } else {
46        Some(raw_cpuid(leaf, subleaf))
47    }
48}
49
50/// Queries the frequency in Hz of the Time Stamp Counter (TSC).
51///
52/// This is based on the information given by the CPUID instruction in the Time Stamp Counter and
53/// Nominal Core Crystal Clock Information Leaf.
54///
55/// Note that the CPUID leaf is currently only supported by new Intel CPUs. This method will return
56/// `None` if it is not supported.
57pub(in crate::arch) fn query_tsc_freq() -> Option<u64> {
58    let CpuidResult {
59        eax: denominator,
60        ebx: numerator,
61        ecx: crystal_freq,
62        ..
63    } = cpuid(Leaf::Tsc as u32, 0)?;
64
65    if denominator == 0 || numerator == 0 {
66        return None;
67    }
68
69    // If the nominal core crystal clock frequency is not enumerated, we can either obtain that
70    // information from a hardcoded table or rely on the processor base frequency. The Intel
71    // documentation recommends the first approach [1], but Linux uses the second approach because
72    // the first approach is difficult to implement correctly for all corner cases [2]. However,
73    // the second approach does not provide 100% accurate frequencies, so Linux must adjust them at
74    // runtime [2]. For now, we avoid these headaches by faithfully reporting that the TSC
75    // frequency is unavailable.
76    //
77    // [1]: Intel(R) 64 and IA-32 Architectures Software Developer’s Manual,
78    //      Section 20.7.3, Determining the Processor Base Frequency
79    // [2]: https://github.com/torvalds/linux/commit/604dc9170f2435d27da5039a3efd757dceadc684
80    if crystal_freq == 0 {
81        return None;
82    }
83
84    Some((crystal_freq as u64) * (numerator as u64) / (denominator as u64))
85}
86
87/// Queries the supported XSTATE features, i.e., the supported bits of `XCR0` and `IA32_XSS`.
88pub(in crate::arch) fn query_xstate_max_features() -> Option<u64> {
89    let res0 = cpuid(Leaf::Xstate as u32, 0)?;
90    let res1 = cpuid(Leaf::Xstate as u32, 1)?;
91
92    // Supported bits in `XCR0`.
93    let xcr_bits = (res0.eax as u64) | ((res0.edx as u64) << 32);
94    // Supported bits in `IA32_XSS`.
95    let xss_bits = (res1.ecx as u64) | ((res1.edx as u64) << 32);
96
97    Some(xcr_bits | xss_bits)
98}
99
100/// Queries the size in bytes of the XSAVE area containing states enabled by `XCRO` and `IA32_XSS`.
101pub(in crate::arch) fn query_xsave_area_size() -> Option<u32> {
102    cpuid(Leaf::Xstate as u32, 1).map(|res| res.ebx)
103}
104
105/// Queries if the system is running in QEMU.
106///
107/// This function uses the CPUID instruction to detect the QEMU hypervisor signature.
108pub(in crate::arch) fn query_is_running_in_qemu() -> bool {
109    let Some(result) = cpuid(Leaf::HypervisorBase as u32, 0) else {
110        return false;
111    };
112
113    let mut signature = [0u8; 12];
114    signature[0..4].copy_from_slice(&result.ebx.to_ne_bytes());
115    signature[4..8].copy_from_slice(&result.ecx.to_ne_bytes());
116    signature[8..12].copy_from_slice(&result.edx.to_ne_bytes());
117
118    // Check for the QEMU hypervisor signature: "TCGTCGTCGTCG" or "KVMKVMKVM\0\0\0".
119    // Reference: <https://wiki.osdev.org/QEMU_fw_cfg#Detecting_QEMU>
120    matches!(&signature, b"TCGTCGTCGTCG" | b"KVMKVMKVM\0\0\0")
121}