#![feature(prelude_import, asm)]
#![no_std]
#![crate_name = "raw_cpuid"]
#![crate_type = "lib"]
#[macro_use]
mod bitflags;
#[cfg(test)]
#[macro_use]
extern crate std;
use core::str;
use core::mem::transmute;
use core::fmt;
use core::slice;
#[cfg(not(test))]
mod std {
pub use core::ops;
pub use core::option;
}
#[macro_export]
macro_rules! cpuid {
($eax:expr)
=> ( $crate::cpuid1($eax as u32) );
($eax:expr, $ecx:expr)
=> ( $crate::cpuid2($eax as u32, $ecx as u32) );
}
pub fn cpuid2(eax: u32, ecx: u32) -> CpuIdResult {
let mut res = CpuIdResult { eax: 0, ebx: 0, ecx: 0, edx: 0 };
unsafe {
asm!("movl $0, %eax" : : "{eax}" (eax) : "eax");
asm!("movl $0, %ecx" : : "{ecx}" (ecx) : "ecx");
asm!("cpuid" : "={eax}"(res.eax) "={ebx}"(res.ebx)
"={ecx}"(res.ecx) "={edx}"(res.edx)
:: "eax", "ebx", "ecx", "edx");
}
res
}
pub fn cpuid1(eax: u32) -> CpuIdResult {
let mut res = CpuIdResult { eax: 0, ebx: 0, ecx: 0, edx: 0 };
unsafe {
asm!("movl $0, %eax" : : "{eax}" (eax) : "eax");
asm!("cpuid" : "={eax}"(res.eax) "={ebx}"(res.ebx)
"={ecx}"(res.ecx) "={edx}"(res.edx)
:: "eax", "ebx", "ecx", "edx");
}
res
}
fn as_bytes(v: &u32) -> &[u8] {
let start = v as *const u32 as *const u8;
unsafe { slice::from_raw_parts(start, 4) }
}
fn get_bits(r: u32, from: u32, to: u32) -> u32 {
assert!(from <= 31);
assert!(to <= 31);
assert!(from <= to);
let mask = match to {
31 => 0xffffffff,
_ => (1 << (to+1)) - 1,
};
(r & mask) >> from
}
macro_rules! check_flag {
($doc:meta, $fun:ident, $flags:ident, $flag:ident) => (
#[$doc]
pub fn $fun(&self) -> bool {
self.$flags.contains($flag)
}
)
}
macro_rules! is_bit_set {
($field:expr, $bit:expr) => (
$field & (1 << $bit) > 0
)
}
macro_rules! check_bit_fn {
($doc:meta, $fun:ident, $field:ident, $bit:expr) => (
#[$doc]
pub fn $fun(&self) -> bool {
is_bit_set!(self.$field, $bit)
}
)
}
#[derive(Debug)]
pub struct CpuId {
max_eax_value: u32,
}
#[derive(Debug, Copy, Clone, Default)]
pub struct CpuIdResult {
pub eax: u32,
pub ebx: u32,
pub ecx: u32,
pub edx: u32,
}
const EAX_VENDOR_INFO: u32 = 0x0;
const EAX_FEATURE_INFO: u32 = 0x1;
const EAX_CACHE_INFO: u32 = 0x2;
const EAX_PROCESSOR_SERIAL: u32 = 0x3;
const EAX_CACHE_PARAMETERS: u32 = 0x4;
const EAX_MONITOR_MWAIT_INFO: u32 = 0x5;
const EAX_THERMAL_POWER_INFO: u32 = 0x6;
const EAX_STRUCTURED_EXTENDED_FEATURE_INFO: u32 = 0x7;
const EAX_DIRECT_CACHE_ACCESS_INFO: u32 = 0x9;
const EAX_PERFORMANCE_MONITOR_INFO: u32 = 0xA;
const EAX_EXTENDED_TOPOLOGY_INFO: u32 = 0xB;
const EAX_EXTENDED_STATE_INFO: u32 = 0xD;
const EAX_QOS_INFO: u32 = 0xF;
const EAX_QOS_ENFORCEMENT_INFO: u32 = 0x10;
const EAX_TRACE_INFO: u32 = 0x14;
const EAX_TIME_STAMP_COUNTER_INFO: u32 = 0x15;
const EAX_FREQUENCY_INFO: u32 = 0x16;
const EAX_EXTENDED_FUNCTION_INFO: u32 = 0x80000000;
impl CpuId {
pub fn new() -> CpuId {
let res = cpuid!(EAX_VENDOR_INFO);
CpuId { max_eax_value: res.eax }
}
fn leaf_is_supported(&self, val: u32) -> bool {
val <= self.max_eax_value
}
pub fn get_vendor_info(&self) -> Option<VendorInfo> {
if self.leaf_is_supported(EAX_VENDOR_INFO) {
let res = cpuid!(EAX_VENDOR_INFO);
Some(VendorInfo { ebx: res.ebx, ecx: res.ecx, edx: res.edx })
}
else {
None
}
}
pub fn get_feature_info(&self) -> Option<FeatureInfo> {
if self.leaf_is_supported(EAX_FEATURE_INFO) {
let res = cpuid!(EAX_FEATURE_INFO);
Some(FeatureInfo { eax: res.eax,
ebx: res.ebx,
ecx: FeatureInfoEcx { bits: res.ecx },
edx: FeatureInfoEdx { bits: res.edx },
})
}
else {
None
}
}
pub fn get_cache_info(&self) -> Option<CacheInfoIter> {
if self.leaf_is_supported(EAX_CACHE_INFO) {
let res = cpuid!(EAX_CACHE_INFO);
Some(CacheInfoIter { current: 1,
eax: res.eax,
ebx: res.ebx,
ecx: res.ecx,
edx: res.edx })
}
else {
None
}
}
pub fn get_processor_serial(&self) -> Option<ProcessorSerial> {
if self.leaf_is_supported(EAX_PROCESSOR_SERIAL) {
let res = cpuid!(EAX_PROCESSOR_SERIAL);
Some(ProcessorSerial { ecx: res.ecx, edx: res.edx })
}
else {
None
}
}
pub fn get_cache_parameters(&self) -> Option<CacheParametersIter> {
if self.leaf_is_supported(EAX_CACHE_PARAMETERS) {
Some(CacheParametersIter { current: 0 })
}
else {
None
}
}
pub fn get_monitor_mwait_info(&self) -> Option<MonitorMwaitInfo> {
if self.leaf_is_supported(EAX_MONITOR_MWAIT_INFO) {
let res = cpuid!(EAX_MONITOR_MWAIT_INFO);
Some(MonitorMwaitInfo { eax: res.eax, ebx: res.ebx, ecx: res.ecx, edx: res.edx })
}
else {
None
}
}
pub fn get_thermal_power_info(&self) -> Option<ThermalPowerInfo> {
if self.leaf_is_supported(EAX_THERMAL_POWER_INFO) {
let res = cpuid!(EAX_THERMAL_POWER_INFO);
Some(ThermalPowerInfo { eax: ThermalPowerFeaturesEax { bits: res.eax },
ebx: res.ebx,
ecx: ThermalPowerFeaturesEcx { bits: res.ecx },
edx: res.edx })
}
else {
None
}
}
pub fn get_extended_feature_info(&self) -> Option<ExtendedFeatures> {
if self.leaf_is_supported(EAX_STRUCTURED_EXTENDED_FEATURE_INFO) {
let res = cpuid!(EAX_STRUCTURED_EXTENDED_FEATURE_INFO);
assert!(res.eax == 0);
Some(ExtendedFeatures { eax: res.eax,
ebx: ExtendedFeaturesEbx { bits: res.ebx },
ecx: res.ecx,
edx: res.edx })
}
else {
None
}
}
pub fn get_direct_cache_access_info(&self) -> Option<DirectCacheAccessInfo> {
if self.leaf_is_supported(EAX_DIRECT_CACHE_ACCESS_INFO) {
let res = cpuid!(EAX_DIRECT_CACHE_ACCESS_INFO);
Some(DirectCacheAccessInfo{ eax: res.eax })
}
else {
None
}
}
pub fn get_performance_monitoring_info(&self) -> Option<PerformanceMonitoringInfo> {
if self.leaf_is_supported(EAX_PERFORMANCE_MONITOR_INFO) {
let res = cpuid!(EAX_PERFORMANCE_MONITOR_INFO);
Some(PerformanceMonitoringInfo{ eax: res.eax,
ebx: PerformanceMonitoringFeaturesEbx{ bits: res.ebx },
ecx: res.ecx,
edx: res.edx })
}
else {
None
}
}
pub fn get_extended_topology_info(&self) -> Option<ExtendedTopologyIter> {
if self.leaf_is_supported(EAX_EXTENDED_TOPOLOGY_INFO) {
Some(ExtendedTopologyIter { level: 0 })
}
else {
None
}
}
pub fn get_extended_state_info(&self) -> Option<ExtendedStateInfo> {
if self.leaf_is_supported(EAX_EXTENDED_STATE_INFO) {
let res = cpuid!(EAX_EXTENDED_STATE_INFO, 0);
let res1 = cpuid!(EAX_EXTENDED_STATE_INFO, 1);
Some(ExtendedStateInfo { eax: res.eax, ebx: res.ebx,
ecx: res.ecx, edx: res.edx,
eax1: res1.eax })
}
else {
None
}
}
pub fn get_qos_info(&self) -> Option<QoSInfo> {
let res = cpuid!(EAX_QOS_INFO, 0);
let res1 = cpuid!(EAX_QOS_INFO, 1);
if self.leaf_is_supported(EAX_QOS_INFO) {
Some(QoSInfo { ebx0: res.ebx, edx0: res.edx,
ebx1: res1.ebx, ecx1: res1.ecx,
edx1: res1.edx })
}
else {
None
}
}
pub fn get_qos_enforcement_info(&self) -> Option<QoSEnforcementInfo> {
let res = cpuid!(EAX_QOS_ENFORCEMENT_INFO, 0);
let res1 = cpuid!(EAX_QOS_ENFORCEMENT_INFO, 1);
if self.leaf_is_supported(EAX_QOS_ENFORCEMENT_INFO) {
Some(QoSEnforcementInfo { ebx0: res.ebx, eax1: res1.eax,
ebx1: res1.ebx, ecx1: res1.ecx,
edx1: res1.edx })
}
else {
None
}
}
pub fn get_processor_trace_info(&self) -> Option<ProcessorTraceInfo> {
let res = cpuid!(EAX_TRACE_INFO, 0);
if self.leaf_is_supported(EAX_TRACE_INFO) {
Some(ProcessorTraceInfo { eax: res.eax,
ebx: res.ebx,
ecx: res.ecx,
edx: res.edx })
}
else {
None
}
}
pub fn get_tsc_info(&self) -> Option<TscInfo> {
let res = cpuid!(EAX_TIME_STAMP_COUNTER_INFO, 0);
if self.leaf_is_supported(EAX_TIME_STAMP_COUNTER_INFO) {
Some(TscInfo { eax: res.eax,
ebx: res.ebx })
}
else {
None
}
}
pub fn get_processor_frequency_info(&self) -> Option<ProcessorFrequencyInfo> {
let res = cpuid!(EAX_FREQUENCY_INFO, 0);
if self.leaf_is_supported(EAX_FREQUENCY_INFO) {
Some(ProcessorFrequencyInfo { eax: res.eax,
ebx: res.ebx,
ecx: res.ecx })
}
else {
None
}
}
pub fn get_extended_function_info(&self) -> Option<ExtendedFunctionInfo> {
let res = cpuid!(EAX_EXTENDED_FUNCTION_INFO);
if res.eax == 0 {
return None;
}
let mut ef = ExtendedFunctionInfo { max_eax_value: res.eax - EAX_EXTENDED_FUNCTION_INFO,
data: [
CpuIdResult{eax: res.eax, ebx: res.ebx, ecx: res.ecx, edx: res.edx},
CpuIdResult{eax: 0, ebx: 0, ecx: 0, edx: 0},
CpuIdResult{eax: 0, ebx: 0, ecx: 0, edx: 0},
CpuIdResult{eax: 0, ebx: 0, ecx: 0, edx: 0},
CpuIdResult{eax: 0, ebx: 0, ecx: 0, edx: 0},
CpuIdResult{eax: 0, ebx: 0, ecx: 0, edx: 0},
CpuIdResult{eax: 0, ebx: 0, ecx: 0, edx: 0},
CpuIdResult{eax: 0, ebx: 0, ecx: 0, edx: 0},
CpuIdResult{eax: 0, ebx: 0, ecx: 0, edx: 0}
], };
for i in 1..ef.max_eax_value+1 {
ef.data[i as usize] = cpuid!(EAX_EXTENDED_FUNCTION_INFO + i);
}
Some(ef)
}
}
#[derive(Debug)]
pub struct VendorInfo {
ebx: u32,
edx: u32,
ecx: u32,
}
impl VendorInfo {
pub fn as_string(&self) -> &str {
unsafe {
let brand_string_start = transmute::<&VendorInfo, *const u8>(&self);
let slice = slice::from_raw_parts(brand_string_start, 3*4);
let byte_array: &'static [u8] = transmute(slice);
str::from_utf8_unchecked(byte_array)
}
}
}
#[derive(Debug)]
pub struct CacheInfoIter {
current: u32,
eax: u32,
ebx: u32,
ecx: u32,
edx: u32,
}
impl Iterator for CacheInfoIter {
type Item = CacheInfo;
fn next(&mut self) -> Option<CacheInfo> {
if self.current >= 4*4 {
return None;
}
let reg_index = self.current % 4;
let byte_index = self.current / 4;
let reg = match reg_index {
0 => self.eax,
1 => self.ebx,
2 => self.ecx,
3 => self.edx,
_ => unreachable!()
};
let byte = as_bytes(®)[byte_index as usize];
if byte == 0 {
self.current += 1;
return self.next();
}
for cache_info in CACHE_INFO_TABLE.into_iter() {
if cache_info.num == byte {
self.current += 1;
return Some(*cache_info);
}
}
None
}
}
#[derive(Copy, Clone, Debug)]
pub enum CacheInfoType {
GENERAL,
CACHE,
TLB,
STLB,
DTLB,
PREFETCH,
}
#[derive(Copy, Clone, Debug)]
pub struct CacheInfo {
pub num: u8,
pub typ: CacheInfoType,
pub desc: &'static str,
}
impl fmt::Display for CacheInfo {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
let typ = match self.typ {
CacheInfoType::GENERAL => "N/A",
CacheInfoType::CACHE => "Cache",
CacheInfoType::TLB => "TLB",
CacheInfoType::STLB => "STLB",
CacheInfoType::DTLB => "DTLB",
CacheInfoType::PREFETCH => "Prefetcher"
};
write!(f, "{:x}:\t {}: {}", self.num, typ, self.desc)
}
}
pub const CACHE_INFO_TABLE: [CacheInfo; 103] = [
CacheInfo{num: 0x00, typ: CacheInfoType::GENERAL, desc: "Null descriptor, this byte contains no information"},
CacheInfo{num: 0x01, typ: CacheInfoType::TLB, desc: "Instruction TLB: 4 KByte pages, 4-way set associative, 32 entries"},
CacheInfo{num: 0x02, typ: CacheInfoType::TLB, desc: "Instruction TLB: 4 MByte pages, fully associative, 2 entries"},
CacheInfo{num: 0x03, typ: CacheInfoType::TLB, desc: "Data TLB: 4 KByte pages, 4-way set associative, 64 entries"},
CacheInfo{num: 0x04, typ: CacheInfoType::TLB, desc: "Data TLB: 4 MByte pages, 4-way set associative, 8 entries"},
CacheInfo{num: 0x05, typ: CacheInfoType::TLB, desc: "Data TLB1: 4 MByte pages, 4-way set associative, 32 entries"},
CacheInfo{num: 0x06, typ: CacheInfoType::CACHE, desc: "1st-level instruction cache: 8 KBytes, 4-way set associative, 32 byte line size"},
CacheInfo{num: 0x08, typ: CacheInfoType::CACHE, desc: "1st-level instruction cache: 16 KBytes, 4-way set associative, 32 byte line size"},
CacheInfo{num: 0x09, typ: CacheInfoType::CACHE, desc: "1st-level instruction cache: 32KBytes, 4-way set associative, 64 byte line size"},
CacheInfo{num: 0x0A, typ: CacheInfoType::CACHE, desc: "1st-level data cache: 8 KBytes, 2-way set associative, 32 byte line size"},
CacheInfo{num: 0x0B, typ: CacheInfoType::TLB, desc: "Instruction TLB: 4 MByte pages, 4-way set associative, 4 entries"},
CacheInfo{num: 0x0C, typ: CacheInfoType::CACHE, desc: "1st-level data cache: 16 KBytes, 4-way set associative, 32 byte line size"},
CacheInfo{num: 0x0D, typ: CacheInfoType::CACHE, desc: "1st-level data cache: 16 KBytes, 4-way set associative, 64 byte line size"},
CacheInfo{num: 0x0E, typ: CacheInfoType::CACHE, desc: "1st-level data cache: 24 KBytes, 6-way set associative, 64 byte line size"},
CacheInfo{num: 0x21, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 256 KBytes, 8-way set associative, 64 byte line size"},
CacheInfo{num: 0x22, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 512 KBytes, 4-way set associative, 64 byte line size, 2 lines per sector"},
CacheInfo{num: 0x23, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 1 MBytes, 8-way set associative, 64 byte line size, 2 lines per sector"},
CacheInfo{num: 0x24, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 1 MBytes, 16-way set associative, 64 byte line size"},
CacheInfo{num: 0x25, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 2 MBytes, 8-way set associative, 64 byte line size, 2 lines per sector"},
CacheInfo{num: 0x29, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 4 MBytes, 8-way set associative, 64 byte line size, 2 lines per sector"},
CacheInfo{num: 0x2C, typ: CacheInfoType::CACHE, desc: "1st-level data cache: 32 KBytes, 8-way set associative, 64 byte line size"},
CacheInfo{num: 0x30, typ: CacheInfoType::CACHE, desc: "1st-level instruction cache: 32 KBytes, 8-way set associative, 64 byte line size"},
CacheInfo{num: 0x40, typ: CacheInfoType::CACHE, desc: "No 2nd-level cache or, if processor contains a valid 2nd-level cache, no 3rd-level cache"},
CacheInfo{num: 0x41, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 128 KBytes, 4-way set associative, 32 byte line size"},
CacheInfo{num: 0x42, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 256 KBytes, 4-way set associative, 32 byte line size"},
CacheInfo{num: 0x43, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 512 KBytes, 4-way set associative, 32 byte line size"},
CacheInfo{num: 0x44, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 1 MByte, 4-way set associative, 32 byte line size"},
CacheInfo{num: 0x45, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 2 MByte, 4-way set associative, 32 byte line size"},
CacheInfo{num: 0x46, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 4 MByte, 4-way set associative, 64 byte line size"},
CacheInfo{num: 0x47, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 8 MByte, 8-way set associative, 64 byte line size"},
CacheInfo{num: 0x48, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 3MByte, 12-way set associative, 64 byte line size"},
CacheInfo{num: 0x49, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 4MB, 16-way set associative, 64-byte line size (Intel Xeon processor MP, Family 0FH, Model 06H); 2nd-level cache: 4 MByte, 16-way set associative, 64 byte line size"},
CacheInfo{num: 0x4A, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 6MByte, 12-way set associative, 64 byte line size"},
CacheInfo{num: 0x4B, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 8MByte, 16-way set associative, 64 byte line size"},
CacheInfo{num: 0x4C, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 12MByte, 12-way set associative, 64 byte line size"},
CacheInfo{num: 0x4D, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 16MByte, 16-way set associative, 64 byte line size"},
CacheInfo{num: 0x4E, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 6MByte, 24-way set associative, 64 byte line size"},
CacheInfo{num: 0x4F, typ: CacheInfoType::TLB, desc: "Instruction TLB: 4 KByte pages, 32 entries"},
CacheInfo{num: 0x50, typ: CacheInfoType::TLB, desc: "Instruction TLB: 4 KByte and 2-MByte or 4-MByte pages, 64 entries"},
CacheInfo{num: 0x51, typ: CacheInfoType::TLB, desc: "Instruction TLB: 4 KByte and 2-MByte or 4-MByte pages, 128 entries"},
CacheInfo{num: 0x52, typ: CacheInfoType::TLB, desc: "Instruction TLB: 4 KByte and 2-MByte or 4-MByte pages, 256 entries"},
CacheInfo{num: 0x55, typ: CacheInfoType::TLB, desc: "Instruction TLB: 2-MByte or 4-MByte pages, fully associative, 7 entries"},
CacheInfo{num: 0x56, typ: CacheInfoType::TLB, desc: "Data TLB0: 4 MByte pages, 4-way set associative, 16 entries"},
CacheInfo{num: 0x57, typ: CacheInfoType::TLB, desc: "Data TLB0: 4 KByte pages, 4-way associative, 16 entries"},
CacheInfo{num: 0x59, typ: CacheInfoType::TLB, desc: "Data TLB0: 4 KByte pages, fully associative, 16 entries"},
CacheInfo{num: 0x5A, typ: CacheInfoType::TLB, desc: "Data TLB0: 2-MByte or 4 MByte pages, 4-way set associative, 32 entries"},
CacheInfo{num: 0x5B, typ: CacheInfoType::TLB, desc: "Data TLB: 4 KByte and 4 MByte pages, 64 entries"},
CacheInfo{num: 0x5C, typ: CacheInfoType::TLB, desc: "Data TLB: 4 KByte and 4 MByte pages,128 entries"},
CacheInfo{num: 0x5D, typ: CacheInfoType::TLB, desc: "Data TLB: 4 KByte and 4 MByte pages,256 entries"},
CacheInfo{num: 0x60, typ: CacheInfoType::CACHE, desc: "1st-level data cache: 16 KByte, 8-way set associative, 64 byte line size"},
CacheInfo{num: 0x61, typ: CacheInfoType::TLB, desc: "Instruction TLB: 4 KByte pages, fully associative, 48 entries"},
CacheInfo{num: 0x63, typ: CacheInfoType::TLB, desc: "Data TLB: 1 GByte pages, 4-way set associative, 4 entries"},
CacheInfo{num: 0x66, typ: CacheInfoType::CACHE, desc: "1st-level data cache: 8 KByte, 4-way set associative, 64 byte line size"},
CacheInfo{num: 0x67, typ: CacheInfoType::CACHE, desc: "1st-level data cache: 16 KByte, 4-way set associative, 64 byte line size"},
CacheInfo{num: 0x68, typ: CacheInfoType::CACHE, desc: "1st-level data cache: 32 KByte, 4-way set associative, 64 byte line size"},
CacheInfo{num: 0x70, typ: CacheInfoType::CACHE, desc: "Trace cache: 12 K-μop, 8-way set associative"},
CacheInfo{num: 0x71, typ: CacheInfoType::CACHE, desc: "Trace cache: 16 K-μop, 8-way set associative"},
CacheInfo{num: 0x72, typ: CacheInfoType::CACHE, desc: "Trace cache: 32 K-μop, 8-way set associative"},
CacheInfo{num: 0x76, typ: CacheInfoType::TLB, desc: "Instruction TLB: 2M/4M pages, fully associative, 8 entries"},
CacheInfo{num: 0x78, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 1 MByte, 4-way set associative, 64byte line size"},
CacheInfo{num: 0x79, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 128 KByte, 8-way set associative, 64 byte line size, 2 lines per sector"},
CacheInfo{num: 0x7A, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 256 KByte, 8-way set associative, 64 byte line size, 2 lines per sector"},
CacheInfo{num: 0x7B, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 512 KByte, 8-way set associative, 64 byte line size, 2 lines per sector"},
CacheInfo{num: 0x7C, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 1 MByte, 8-way set associative, 64 byte line size, 2 lines per sector"},
CacheInfo{num: 0x7D, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 2 MByte, 8-way set associative, 64byte line size"},
CacheInfo{num: 0x7F, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 512 KByte, 2-way set associative, 64-byte line size"},
CacheInfo{num: 0x80, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 512 KByte, 8-way set associative, 64-byte line size"},
CacheInfo{num: 0x82, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 256 KByte, 8-way set associative, 32 byte line size"},
CacheInfo{num: 0x83, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 512 KByte, 8-way set associative, 32 byte line size"},
CacheInfo{num: 0x84, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 1 MByte, 8-way set associative, 32 byte line size"},
CacheInfo{num: 0x85, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 2 MByte, 8-way set associative, 32 byte line size"},
CacheInfo{num: 0x86, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 512 KByte, 4-way set associative, 64 byte line size"},
CacheInfo{num: 0x87, typ: CacheInfoType::CACHE, desc: "2nd-level cache: 1 MByte, 8-way set associative, 64 byte line size"},
CacheInfo{num: 0xB0, typ: CacheInfoType::TLB, desc: "Instruction TLB: 4 KByte pages, 4-way set associative, 128 entries"},
CacheInfo{num: 0xB1, typ: CacheInfoType::TLB, desc: "Instruction TLB: 2M pages, 4-way, 8 entries or 4M pages, 4-way, 4 entries"},
CacheInfo{num: 0xB2, typ: CacheInfoType::TLB, desc: "Instruction TLB: 4KByte pages, 4-way set associative, 64 entries"},
CacheInfo{num: 0xB3, typ: CacheInfoType::TLB, desc: "Data TLB: 4 KByte pages, 4-way set associative, 128 entries"},
CacheInfo{num: 0xB4, typ: CacheInfoType::TLB, desc: "Data TLB1: 4 KByte pages, 4-way associative, 256 entries"},
CacheInfo{num: 0xB5, typ: CacheInfoType::TLB, desc: "Instruction TLB: 4KByte pages, 8-way set associative, 64 entries"},
CacheInfo{num: 0xB6, typ: CacheInfoType::TLB, desc: "Instruction TLB: 4KByte pages, 8-way set associative, 128 entries"},
CacheInfo{num: 0xBA, typ: CacheInfoType::TLB, desc: "Data TLB1: 4 KByte pages, 4-way associative, 64 entries"},
CacheInfo{num: 0xC0, typ: CacheInfoType::TLB, desc: "Data TLB: 4 KByte and 4 MByte pages, 4-way associative, 8 entries"},
CacheInfo{num: 0xC1, typ: CacheInfoType::STLB, desc: "Shared 2nd-Level TLB: 4 KByte/2MByte pages, 8-way associative, 1024 entries"},
CacheInfo{num: 0xC2, typ: CacheInfoType::DTLB, desc: "DTLB: 2 MByte/$MByte pages, 4-way associative, 16 entries"},
CacheInfo{num: 0xCA, typ: CacheInfoType::STLB, desc: "Shared 2nd-Level TLB: 4 KByte pages, 4-way associative, 512 entries"},
CacheInfo{num: 0xD0, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 512 KByte, 4-way set associative, 64 byte line size"},
CacheInfo{num: 0xD1, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 1 MByte, 4-way set associative, 64 byte line size"},
CacheInfo{num: 0xD2, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 2 MByte, 4-way set associative, 64 byte line size"},
CacheInfo{num: 0xD6, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 1 MByte, 8-way set associative, 64 byte line size"},
CacheInfo{num: 0xD7, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 2 MByte, 8-way set associative, 64 byte line size"},
CacheInfo{num: 0xD8, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 4 MByte, 8-way set associative, 64 byte line size"},
CacheInfo{num: 0xDC, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 1.5 MByte, 12-way set associative, 64 byte line size"},
CacheInfo{num: 0xDD, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 3 MByte, 12-way set associative, 64 byte line size"},
CacheInfo{num: 0xDE, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 6 MByte, 12-way set associative, 64 byte line size"},
CacheInfo{num: 0xE2, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 2 MByte, 16-way set associative, 64 byte line size"},
CacheInfo{num: 0xE3, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 4 MByte, 16-way set associative, 64 byte line size"},
CacheInfo{num: 0xE4, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 8 MByte, 16-way set associative, 64 byte line size"},
CacheInfo{num: 0xEA, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 12MByte, 24-way set associative, 64 byte line size"},
CacheInfo{num: 0xEB, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 18MByte, 24-way set associative, 64 byte line size"},
CacheInfo{num: 0xEC, typ: CacheInfoType::CACHE, desc: "3rd-level cache: 24MByte, 24-way set associative, 64 byte line size"},
CacheInfo{num: 0xF0, typ: CacheInfoType::PREFETCH, desc: "64-Byte prefetching"},
CacheInfo{num: 0xF1, typ: CacheInfoType::PREFETCH, desc: "128-Byte prefetching"},
CacheInfo{num: 0xFF, typ: CacheInfoType::GENERAL, desc: "CPUID leaf 2 does not report cache descriptor information, use CPUID leaf 4 to query cache parameters"},
];
impl fmt::Display for VendorInfo {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "{}", self.as_string())
}
}
pub struct ProcessorSerial {
ecx: u32,
edx: u32,
}
impl ProcessorSerial {
pub fn serial_lower(&self) -> u32 {
self.ecx
}
pub fn serial_middle(&self) -> u32 {
self.edx
}
}
#[derive(Debug)]
pub struct FeatureInfo {
eax: u32,
ebx: u32,
ecx: FeatureInfoEcx,
edx: FeatureInfoEdx,
}
impl FeatureInfo {
pub fn extended_family_id(&self) -> u8 {
get_bits(self.eax, 20, 27) as u8
}
pub fn extended_model_id(&self) -> u8 {
get_bits(self.eax, 16, 19) as u8
}
pub fn family_id(&self) -> u8 {
get_bits(self.eax, 8, 11) as u8
}
pub fn model_id(&self) -> u8 {
get_bits(self.eax, 4, 7) as u8
}
pub fn stepping_id(&self) -> u8 {
get_bits(self.eax, 0, 3) as u8
}
pub fn brand_index(&self) -> u8 {
get_bits(self.ebx, 0, 7) as u8
}
pub fn cflush_cache_line_size(&self) -> u8 {
get_bits(self.ebx, 8, 15) as u8
}
pub fn initial_local_apic_id(&self) -> u8 {
get_bits(self.ebx, 24, 31) as u8
}
pub fn max_logical_processor_ids(&self) -> u8 {
get_bits(self.ebx, 16, 23) as u8
}
check_flag!(doc = "Streaming SIMD Extensions 3 (SSE3). A value of 1 indicates the processor supports this technology.",
has_sse3, ecx, CPU_FEATURE_SSE3);
check_flag!(doc = "PCLMULQDQ. A value of 1 indicates the processor supports the PCLMULQDQ instruction",
has_pclmulqdq, ecx, CPU_FEATURE_PCLMULQDQ);
check_flag!(doc = "64-bit DS Area. A value of 1 indicates the processor supports DS area using 64-bit layout",
has_ds_area, ecx, CPU_FEATURE_DTES64);
check_flag!(doc = "MONITOR/MWAIT. A value of 1 indicates the processor supports this feature.",
has_monitor_mwait, ecx, CPU_FEATURE_MONITOR);
check_flag!(doc = "CPL Qualified Debug Store. A value of 1 indicates the processor supports the extensions to the Debug Store feature to allow for branch message storage qualified by CPL.",
has_cpl, ecx, CPU_FEATURE_DSCPL);
check_flag!(doc = "Virtual Machine Extensions. A value of 1 indicates that the processor supports this technology.",
has_vmx, ecx, CPU_FEATURE_VMX);
check_flag!(doc = "Safer Mode Extensions. A value of 1 indicates that the processor supports this technology. See Chapter 5, Safer Mode Extensions Reference.",
has_smx, ecx, CPU_FEATURE_SMX);
check_flag!(doc = "Enhanced Intel SpeedStep® technology. A value of 1 indicates that the processor supports this technology.",
has_eist, ecx, CPU_FEATURE_EIST);
check_flag!(doc = "Thermal Monitor 2. A value of 1 indicates whether the processor supports this technology.",
has_tm2, ecx, CPU_FEATURE_TM2);
check_flag!(doc = "A value of 1 indicates the presence of the Supplemental Streaming SIMD Extensions 3 (SSSE3). A value of 0 indicates the instruction extensions are not present in the processor",
has_ssse3, ecx, CPU_FEATURE_SSSE3);
check_flag!(doc = "L1 Context ID. A value of 1 indicates the L1 data cache mode can be set to either adaptive mode or shared mode. A value of 0 indicates this feature is not supported. See definition of the IA32_MISC_ENABLE MSR Bit 24 (L1 Data Cache Context Mode) for details.",
has_cnxtid, ecx, CPU_FEATURE_CNXTID);
check_flag!(doc = "A value of 1 indicates the processor supports FMA extensions using YMM state.",
has_fma, ecx, CPU_FEATURE_FMA);
check_flag!(doc = "CMPXCHG16B Available. A value of 1 indicates that the feature is available. See the CMPXCHG8B/CMPXCHG16B Compare and Exchange Bytes section. 14",
has_cmpxchg16b, ecx, CPU_FEATURE_CMPXCHG16B);
check_flag!(doc = "Perfmon and Debug Capability: A value of 1 indicates the processor supports the performance and debug feature indication MSR IA32_PERF_CAPABILITIES.",
has_pdcm, ecx, CPU_FEATURE_PDCM);
check_flag!(doc = "Process-context identifiers. A value of 1 indicates that the processor supports PCIDs and the software may set CR4.PCIDE to 1.",
has_pcid, ecx, CPU_FEATURE_PCID);
check_flag!(doc = "A value of 1 indicates the processor supports the ability to prefetch data from a memory mapped device.",
has_dca, ecx, CPU_FEATURE_DCA);
check_flag!(doc = "A value of 1 indicates that the processor supports SSE4.1.",
has_sse41, ecx, CPU_FEATURE_SSE41);
check_flag!(doc = "A value of 1 indicates that the processor supports SSE4.2.",
has_sse42, ecx, CPU_FEATURE_SSE42);
check_flag!(doc = "A value of 1 indicates that the processor supports x2APIC feature.",
has_x2apic, ecx, CPU_FEATURE_X2APIC);
check_flag!(doc = "A value of 1 indicates that the processor supports MOVBE instruction.",
has_movbe, ecx, CPU_FEATURE_MOVBE);
check_flag!(doc = "A value of 1 indicates that the processor supports the POPCNT instruction.",
has_popcnt, ecx, CPU_FEATURE_POPCNT);
check_flag!(doc = "A value of 1 indicates that the processors local APIC timer supports one-shot operation using a TSC deadline value.",
has_tsc_deadline, ecx, CPU_FEATURE_TSC_DEADLINE);
check_flag!(doc = "A value of 1 indicates that the processor supports the AESNI instruction extensions.",
has_aesni, ecx, CPU_FEATURE_AESNI);
check_flag!(doc = "A value of 1 indicates that the processor supports the XSAVE/XRSTOR processor extended states feature, the XSETBV/XGETBV instructions, and XCR0.",
has_xsave, ecx, CPU_FEATURE_XSAVE);
check_flag!(doc = "A value of 1 indicates that the OS has enabled XSETBV/XGETBV instructions to access XCR0, and support for processor extended state management using XSAVE/XRSTOR.",
has_oxsave, ecx, CPU_FEATURE_OSXSAVE);
check_flag!(doc = "A value of 1 indicates the processor supports the AVX instruction extensions.",
has_avx, ecx, CPU_FEATURE_AVX);
check_flag!(doc = "A value of 1 indicates that processor supports 16-bit floating-point conversion instructions.",
has_f16c, ecx, CPU_FEATURE_F16C);
check_flag!(doc = "A value of 1 indicates that processor supports RDRAND instruction.",
has_rdrand, ecx, CPU_FEATURE_RDRAND);
check_flag!(doc = "Floating Point Unit On-Chip. The processor contains an x87 FPU.",
has_fpu, edx, CPU_FEATURE_FPU);
check_flag!(doc = "Virtual 8086 Mode Enhancements. Virtual 8086 mode enhancements, including CR4.VME for controlling the feature, CR4.PVI for protected mode virtual interrupts, software interrupt indirection, expansion of the TSS with the software indirection bitmap, and EFLAGS.VIF and EFLAGS.VIP flags.",
has_vme, edx, CPU_FEATURE_VME);
check_flag!(doc = "Debugging Extensions. Support for I/O breakpoints, including CR4.DE for controlling the feature, and optional trapping of accesses to DR4 and DR5.",
has_de, edx, CPU_FEATURE_DE);
check_flag!(doc = "Page Size Extension. Large pages of size 4 MByte are supported, including CR4.PSE for controlling the feature, the defined dirty bit in PDE (Page Directory Entries), optional reserved bit trapping in CR3, PDEs, and PTEs.",
has_pse, edx, CPU_FEATURE_PSE);
check_flag!(doc = "Time Stamp Counter. The RDTSC instruction is supported, including CR4.TSD for controlling privilege.",
has_tsc, edx, CPU_FEATURE_TSC);
check_flag!(doc = "Model Specific Registers RDMSR and WRMSR Instructions. The RDMSR and WRMSR instructions are supported. Some of the MSRs are implementation dependent.",
has_msr, edx, CPU_FEATURE_MSR);
check_flag!(doc = "Physical Address Extension. Physical addresses greater than 32 bits are supported: extended page table entry formats, an extra level in the page translation tables is defined, 2-MByte pages are supported instead of 4 Mbyte pages if PAE bit is 1.",
has_pae, edx, CPU_FEATURE_PAE);
check_flag!(doc = "Machine Check Exception. Exception 18 is defined for Machine Checks, including CR4.MCE for controlling the feature. This feature does not define the model-specific implementations of machine-check error logging, reporting, and processor shutdowns. Machine Check exception handlers may have to depend on processor version to do model specific processing of the exception, or test for the presence of the Machine Check feature.",
has_mce, edx, CPU_FEATURE_MCE);
check_flag!(doc = "CMPXCHG8B Instruction. The compare-and-exchange 8 bytes (64 bits) instruction is supported (implicitly locked and atomic).",
has_cmpxchg8b, edx, CPU_FEATURE_CX8);
check_flag!(doc = "APIC On-Chip. The processor contains an Advanced Programmable Interrupt Controller (APIC), responding to memory mapped commands in the physical address range FFFE0000H to FFFE0FFFH (by default - some processors permit the APIC to be relocated).",
has_apic, edx, CPU_FEATURE_APIC);
check_flag!(doc = "SYSENTER and SYSEXIT Instructions. The SYSENTER and SYSEXIT and associated MSRs are supported.",
has_sysenter_sysexit, edx, CPU_FEATURE_SEP);
check_flag!(doc = "Memory Type Range Registers. MTRRs are supported. The MTRRcap MSR contains feature bits that describe what memory types are supported, how many variable MTRRs are supported, and whether fixed MTRRs are supported.",
has_mtrr, edx, CPU_FEATURE_MTRR);
check_flag!(doc = "Page Global Bit. The global bit is supported in paging-structure entries that map a page, indicating TLB entries that are common to different processes and need not be flushed. The CR4.PGE bit controls this feature.",
has_pge, edx, CPU_FEATURE_PGE);
check_flag!(doc = "Machine Check Architecture. The Machine Check Architecture, which provides a compatible mechanism for error reporting in P6 family, Pentium 4, Intel Xeon processors, and future processors, is supported. The MCG_CAP MSR contains feature bits describing how many banks of error reporting MSRs are supported.",
has_mca, edx, CPU_FEATURE_MCA);
check_flag!(doc = "Conditional Move Instructions. The conditional move instruction CMOV is supported. In addition, if x87 FPU is present as indicated by the CPUID.FPU feature bit, then the FCOMI and FCMOV instructions are supported",
has_cmov, edx, CPU_FEATURE_CMOV);
check_flag!(doc = "Page Attribute Table. Page Attribute Table is supported. This feature augments the Memory Type Range Registers (MTRRs), allowing an operating system to specify attributes of memory accessed through a linear address on a 4KB granularity.",
has_pat, edx, CPU_FEATURE_PAT);
check_flag!(doc = "36-Bit Page Size Extension. 4-MByte pages addressing physical memory beyond 4 GBytes are supported with 32-bit paging. This feature indicates that upper bits of the physical address of a 4-MByte page are encoded in bits 20:13 of the page-directory entry. Such physical addresses are limited by MAXPHYADDR and may be up to 40 bits in size.",
has_pse36, edx, CPU_FEATURE_PSE36);
check_flag!(doc = "Processor Serial Number. The processor supports the 96-bit processor identification number feature and the feature is enabled.",
has_psn, edx, CPU_FEATURE_PSN);
check_flag!(doc = "CLFLUSH Instruction. CLFLUSH Instruction is supported.",
has_clflush, edx, CPU_FEATURE_CLFSH);
check_flag!(doc = "Debug Store. The processor supports the ability to write debug information into a memory resident buffer. This feature is used by the branch trace store (BTS) and precise event-based sampling (PEBS) facilities (see Chapter 23, Introduction to Virtual-Machine Extensions, in the Intel® 64 and IA-32 Architectures Software Developers Manual, Volume 3C).",
has_ds, edx, CPU_FEATURE_DS);
check_flag!(doc = "Thermal Monitor and Software Controlled Clock Facilities. The processor implements internal MSRs that allow processor temperature to be monitored and processor performance to be modulated in predefined duty cycles under software control.",
has_acpi, edx, CPU_FEATURE_ACPI);
check_flag!(doc = "Intel MMX Technology. The processor supports the Intel MMX technology.",
has_mmx, edx, CPU_FEATURE_MMX);
check_flag!(doc = "FXSAVE and FXRSTOR Instructions. The FXSAVE and FXRSTOR instructions are supported for fast save and restore of the floating point context. Presence of this bit also indicates that CR4.OSFXSR is available for an operating system to indicate that it supports the FXSAVE and FXRSTOR instructions.",
has_fxsave_fxstor, edx, CPU_FEATURE_FXSR);
check_flag!(doc = "SSE. The processor supports the SSE extensions.",
has_sse, edx, CPU_FEATURE_SSE);
check_flag!(doc = "SSE2. The processor supports the SSE2 extensions.",
has_sse2, edx, CPU_FEATURE_SSE2);
check_flag!(doc = "Self Snoop. The processor supports the management of conflicting memory types by performing a snoop of its own cache structure for transactions issued to the bus.",
has_ss, edx, CPU_FEATURE_SS);
check_flag!(doc = "Max APIC IDs reserved field is Valid. A value of 0 for HTT indicates there is only a single logical processor in the package and software should assume only a single APIC ID is reserved. A value of 1 for HTT indicates the value in CPUID.1.EBX[23:16] (the Maximum number of addressable IDs for logical processors in this package) is valid for the package.",
has_htt, edx, CPU_FEATURE_HTT);
check_flag!(doc = "Thermal Monitor. The processor implements the thermal monitor automatic thermal control circuitry (TCC).",
has_tm, edx, CPU_FEATURE_TM);
check_flag!(doc = "Pending Break Enable. The processor supports the use of the FERR#/PBE# pin when the processor is in the stop-clock state (STPCLK# is asserted) to signal the processor that an interrupt is pending and that the processor should return to normal operation to handle the interrupt. Bit 10 (PBE enable) in the IA32_MISC_ENABLE MSR enables this capability.",
has_pbe, edx, CPU_FEATURE_PBE);
}
bitflags! {
#[doc(hidden)]
#[derive(Debug)]
flags FeatureInfoEcx: u32 {
#[doc(hidden)]
const CPU_FEATURE_SSE3 = 1 << 0,
#[doc(hidden)]
const CPU_FEATURE_PCLMULQDQ = 1 << 1,
#[doc(hidden)]
const CPU_FEATURE_DTES64 = 1 << 2,
#[doc(hidden)]
const CPU_FEATURE_MONITOR = 1 << 3,
#[doc(hidden)]
const CPU_FEATURE_DSCPL = 1 << 4,
#[doc(hidden)]
const CPU_FEATURE_VMX = 1 << 5,
#[doc(hidden)]
const CPU_FEATURE_SMX = 1 << 6,
#[doc(hidden)]
const CPU_FEATURE_EIST = 1 << 7,
#[doc(hidden)]
const CPU_FEATURE_TM2 = 1 << 8,
#[doc(hidden)]
const CPU_FEATURE_SSSE3 = 1 << 9,
#[doc(hidden)]
const CPU_FEATURE_CNXTID = 1 << 10,
#[doc(hidden)]
const CPU_FEATURE_FMA = 1 << 12,
#[doc(hidden)]
const CPU_FEATURE_CMPXCHG16B = 1 << 13,
#[doc(hidden)]
const CPU_FEATURE_PDCM = 1 << 15,
#[doc(hidden)]
const CPU_FEATURE_PCID = 1 << 17,
#[doc(hidden)]
const CPU_FEATURE_DCA = 1 << 18,
#[doc(hidden)]
const CPU_FEATURE_SSE41 = 1 << 19,
#[doc(hidden)]
const CPU_FEATURE_SSE42 = 1 << 20,
#[doc(hidden)]
const CPU_FEATURE_X2APIC = 1 << 21,
#[doc(hidden)]
const CPU_FEATURE_MOVBE = 1 << 22,
#[doc(hidden)]
const CPU_FEATURE_POPCNT = 1 << 23,
#[doc(hidden)]
const CPU_FEATURE_TSC_DEADLINE = 1 << 24,
#[doc(hidden)]
const CPU_FEATURE_AESNI = 1 << 25,
#[doc(hidden)]
const CPU_FEATURE_XSAVE = 1 << 26,
#[doc(hidden)]
const CPU_FEATURE_OSXSAVE = 1 << 27,
#[doc(hidden)]
const CPU_FEATURE_AVX = 1 << 28,
#[doc(hidden)]
const CPU_FEATURE_F16C = 1 << 29,
#[doc(hidden)]
const CPU_FEATURE_RDRAND = 1 << 30,
}
}
bitflags! {
#[doc(hidden)]
#[derive(Debug)]
flags FeatureInfoEdx: u32 {
#[doc(hidden)]
const CPU_FEATURE_FPU = 1 << 0,
#[doc(hidden)]
const CPU_FEATURE_VME = 1 << 1,
#[doc(hidden)]
const CPU_FEATURE_DE = 1 << 2,
#[doc(hidden)]
const CPU_FEATURE_PSE = 1 << 3,
#[doc(hidden)]
const CPU_FEATURE_TSC = 1 << 4,
#[doc(hidden)]
const CPU_FEATURE_MSR = 1 << 5,
#[doc(hidden)]
const CPU_FEATURE_PAE = 1 << 6,
#[doc(hidden)]
const CPU_FEATURE_MCE = 1 << 7,
#[doc(hidden)]
const CPU_FEATURE_CX8 = 1 << 8,
#[doc(hidden)]
const CPU_FEATURE_APIC = 1 << 9,
#[doc(hidden)]
const CPU_FEATURE_SEP = 1 << 11,
#[doc(hidden)]
const CPU_FEATURE_MTRR = 1 << 12,
#[doc(hidden)]
const CPU_FEATURE_PGE = 1 << 13,
#[doc(hidden)]
const CPU_FEATURE_MCA = 1 << 14,
#[doc(hidden)]
const CPU_FEATURE_CMOV = 1 << 15,
#[doc(hidden)]
const CPU_FEATURE_PAT = 1 << 16,
#[doc(hidden)]
const CPU_FEATURE_PSE36 = 1 << 17,
#[doc(hidden)]
const CPU_FEATURE_PSN = 1 << 18,
#[doc(hidden)]
const CPU_FEATURE_CLFSH = 1 << 19,
#[doc(hidden)]
const CPU_FEATURE_DS = 1 << 21,
#[doc(hidden)]
const CPU_FEATURE_ACPI = 1 << 22,
#[doc(hidden)]
const CPU_FEATURE_MMX = 1 << 23,
#[doc(hidden)]
const CPU_FEATURE_FXSR = 1 << 24,
#[doc(hidden)]
const CPU_FEATURE_SSE = 1 << 25,
#[doc(hidden)]
const CPU_FEATURE_SSE2 = 1 << 26,
#[doc(hidden)]
const CPU_FEATURE_SS = 1 << 27,
#[doc(hidden)]
const CPU_FEATURE_HTT = 1 << 28,
#[doc(hidden)]
const CPU_FEATURE_TM = 1 << 29,
#[doc(hidden)]
const CPU_FEATURE_PBE = 1 << 31,
}
}
pub struct CacheParametersIter {
current: u32,
}
impl Iterator for CacheParametersIter {
type Item = CacheParameter;
fn next(&mut self) -> Option<CacheParameter> {
let res = cpuid!(EAX_CACHE_PARAMETERS, self.current);
let cp = CacheParameter { eax: res.eax, ebx: res.ebx, ecx: res.ecx, edx: res.edx };
match cp.cache_type() {
CacheType::NULL => None,
CacheType::RESERVED => None,
_ => {
self.current += 1;
Some(cp)
}
}
}
}
#[derive(Copy, Clone, Debug)]
pub struct CacheParameter {
eax: u32,
ebx: u32,
ecx: u32,
edx: u32,
}
#[derive(PartialEq, Eq)]
pub enum CacheType {
NULL = 0,
DATA,
INSTRUCTION,
UNIFIED,
RESERVED,
}
impl CacheParameter {
pub fn cache_type(&self) -> CacheType {
let typ = get_bits(self.eax, 0, 4) as u8;
match typ {
0 => CacheType::NULL,
1 => CacheType::DATA,
2 => CacheType::INSTRUCTION,
3 => CacheType::UNIFIED,
_ => CacheType::RESERVED
}
}
pub fn level(&self) -> u8 {
get_bits(self.eax, 5, 7) as u8
}
pub fn is_self_initializing(&self) -> bool {
get_bits(self.eax, 8, 8) == 1
}
pub fn is_fully_associative(&self) -> bool {
get_bits(self.eax, 9, 9) == 1
}
pub fn max_cores_for_cache(&self) -> usize {
(get_bits(self.eax, 14, 25) + 1) as usize
}
pub fn max_cores_for_package(&self) -> usize {
(get_bits(self.eax, 26, 31) + 1) as usize
}
pub fn coherency_line_size(&self) -> usize {
(get_bits(self.ebx, 0, 11) + 1) as usize
}
pub fn physical_line_partitions(&self) -> usize {
(get_bits(self.ebx, 12, 21) + 1) as usize
}
pub fn associativity(&self) -> usize {
(get_bits(self.ebx, 22, 31) + 1) as usize
}
pub fn sets(&self) -> usize {
(self.ecx + 1) as usize
}
pub fn is_write_back_invalidate(&self) -> bool {
get_bits(self.edx, 0, 0) == 1
}
pub fn is_inclusive(&self) -> bool {
get_bits(self.edx, 1, 1) == 1
}
pub fn has_complex_indexing(&self) -> bool {
get_bits(self.edx, 2, 2) == 1
}
}
#[derive(Debug)]
pub struct MonitorMwaitInfo {
eax: u32,
ebx: u32,
ecx: u32,
edx: u32,
}
impl MonitorMwaitInfo {
pub fn smallest_monitor_line(&self) -> u16 {
get_bits(self.eax, 0, 15) as u16
}
pub fn largest_monitor_line(&self) -> u16 {
get_bits(self.ebx, 0, 15) as u16
}
pub fn extensions_supported(&self) -> bool {
get_bits(self.ecx, 0, 0) == 1
}
pub fn interrupts_as_break_event(&self) -> bool {
get_bits(self.ecx, 1, 1) == 1
}
pub fn supported_c0_states(&self) -> u16 {
get_bits(self.edx, 0, 3) as u16
}
pub fn supported_c1_states(&self) -> u16 {
get_bits(self.edx, 4, 7) as u16
}
pub fn supported_c2_states(&self) -> u16 {
get_bits(self.edx, 8, 11) as u16
}
pub fn supported_c3_states(&self) -> u16 {
get_bits(self.edx, 12, 15) as u16
}
pub fn supported_c4_states(&self) -> u16 {
get_bits(self.edx, 16, 19) as u16
}
pub fn supported_c5_states(&self) -> u16 {
get_bits(self.edx, 20, 23) as u16
}
pub fn supported_c6_states(&self) -> u16 {
get_bits(self.edx, 24, 27) as u16
}
pub fn supported_c7_states(&self) -> u16 {
get_bits(self.edx, 28, 31) as u16
}
}
#[derive(Debug)]
pub struct ThermalPowerInfo {
eax: ThermalPowerFeaturesEax,
ebx: u32,
ecx: ThermalPowerFeaturesEcx,
edx: u32,
}
impl ThermalPowerInfo {
check_flag!(doc = "Digital temperature sensor is supported if set.",
has_dts, eax, CPU_FEATURE_DTS);
check_flag!(doc = "Intel Turbo Boost Technology Available (see description of IA32_MISC_ENABLE[38]).",
has_turbo_boost, eax, CPU_FEATURE_TURBO_BOOST);
check_flag!(doc = "ARAT. APIC-Timer-always-running feature is supported if set.",
has_arat, eax, CPU_FEATURE_ARAT);
check_flag!(doc = "PLN. Power limit notification controls are supported if set.",
has_pln, eax, CPU_FEATURE_PLN);
check_flag!(doc = "ECMD. Clock modulation duty cycle extension is supported if set.",
has_ecmd, eax, CPU_FEATURE_ECMD);
check_flag!(doc = "PTM. Package thermal management is supported if set.",
has_ptm, eax, CPU_FEATURE_PTM);
check_flag!(doc = "Hardware Coordination Feedback Capability (Presence of IA32_MPERF and IA32_APERF). The capability to provide a measure of delivered processor performance (since last reset of the counters), as a percentage of expected processor performance at frequency specified in CPUID Brand String Bits 02 - 01",
has_hw_coord_feedback, ecx, CPU_FEATURE_HW_COORD_FEEDBACK);
check_flag!(doc = "The processor supports performance-energy bias preference if CPUID.06H:ECX.SETBH[bit 3] is set and it also implies the presence of a new architectural MSR called IA32_ENERGY_PERF_BIAS (1B0H)",
has_energy_bias_pref, ecx, CPU_FEATURE_ENERGY_BIAS_PREF);
}
bitflags! {
#[doc(hidden)]
#[derive(Debug)]
flags ThermalPowerFeaturesEax: u32 {
#[doc(hidden)]
const CPU_FEATURE_DTS = 1 << 0,
#[doc(hidden)]
const CPU_FEATURE_TURBO_BOOST = 1 << 1,
#[doc(hidden)]
const CPU_FEATURE_ARAT = 1 << 2,
#[doc(hidden)]
const CPU_FEATURE_PLN = 1 << 4,
#[doc(hidden)]
const CPU_FEATURE_ECMD = 1 << 5,
#[doc(hidden)]
const CPU_FEATURE_PTM = 1 << 6,
}
}
bitflags! {
#[doc(hidden)]
#[derive(Debug)]
flags ThermalPowerFeaturesEcx: u32 {
#[doc(hidden)]
const CPU_FEATURE_HW_COORD_FEEDBACK = 1 << 0,
#[doc(hidden)]
const CPU_FEATURE_ENERGY_BIAS_PREF = 1 << 3,
}
}
impl ThermalPowerInfo {
pub fn dts_irq_threshold(&self) -> u8 {
get_bits(self.ebx, 0, 3) as u8
}
}
#[derive(Debug)]
pub struct ExtendedFeatures {
eax: u32,
ebx: ExtendedFeaturesEbx,
ecx: u32,
edx: u32,
}
impl ExtendedFeatures {
check_flag!(doc = "FSGSBASE. Supports RDFSBASE/RDGSBASE/WRFSBASE/WRGSBASE if 1.",
has_fsgsbase, ebx, CPU_FEATURE_FSGSBASE);
check_flag!(doc = "IA32_TSC_ADJUST MSR is supported if 1.",
has_tsc_adjust_msr, ebx, CPU_FEATURE_ADJUST_MSR);
check_flag!(doc = "BMI1",
has_bmi1, ebx, CPU_FEATURE_BMI1);
check_flag!(doc = "HLE",
has_hle, ebx, CPU_FEATURE_HLE);
check_flag!(doc = "AVX2",
has_avx2, ebx, CPU_FEATURE_AVX2);
check_flag!(doc = "SMEP. Supports Supervisor-Mode Execution Prevention if 1.",
has_smep, ebx, CPU_FEATURE_SMEP);
check_flag!(doc = "BMI2",
has_bmi2, ebx, CPU_FEATURE_BMI2);
check_flag!(doc = "Supports Enhanced REP MOVSB/STOSB if 1.",
has_rep_movsb_stosb, ebx, CPU_FEATURE_REP_MOVSB_STOSB);
check_flag!(doc = "INVPCID. If 1, supports INVPCID instruction for system software that manages process-context identifiers.",
has_invpcid, ebx, CPU_FEATURE_INVPCID);
check_flag!(doc = "RTM",
has_rtm, ebx, CPU_FEATURE_RTM);
check_flag!(doc = "Supports Quality of Service Monitoring (QM) capability if 1.",
has_qm, ebx, CPU_FEATURE_QM);
check_flag!(doc = "Deprecates FPU CS and FPU DS values if 1.",
has_fpu_cs_ds_deprecated, ebx, CPU_FEATURE_DEPRECATE_FPU_CS_DS);
check_flag!(doc = "MPX. Supports Intel Memory Protection Extensions if 1.",
has_mpx, ebx, CPU_FEATURE_MPX);
}
bitflags! {
#[doc(hidden)]
#[derive(Debug)]
flags ExtendedFeaturesEbx: u32 {
#[doc(hidden)]
const CPU_FEATURE_FSGSBASE = 1 << 0,
#[doc(hidden)]
const CPU_FEATURE_ADJUST_MSR = 1 << 1,
#[doc(hidden)]
const CPU_FEATURE_BMI1 = 1 << 3,
#[doc(hidden)]
const CPU_FEATURE_HLE = 1 << 4,
#[doc(hidden)]
const CPU_FEATURE_AVX2 = 1 << 5,
#[doc(hidden)]
const CPU_FEATURE_SMEP = 1 << 7,
#[doc(hidden)]
const CPU_FEATURE_BMI2 = 1 << 8,
#[doc(hidden)]
const CPU_FEATURE_REP_MOVSB_STOSB = 1 << 9,
#[doc(hidden)]
const CPU_FEATURE_INVPCID = 1 << 10,
#[doc(hidden)]
const CPU_FEATURE_RTM = 1 << 11,
#[doc(hidden)]
const CPU_FEATURE_QM = 1 << 12,
#[doc(hidden)]
const CPU_FEATURE_DEPRECATE_FPU_CS_DS = 1 << 13,
#[doc(hidden)]
const CPU_FEATURE_MPX = 1 << 14,
}
}
#[derive(Debug)]
pub struct DirectCacheAccessInfo {
eax: u32,
}
impl DirectCacheAccessInfo {
pub fn get_dca_cap_value(&self) -> u32 {
self.eax
}
}
#[derive(Debug)]
pub struct PerformanceMonitoringInfo {
eax: u32,
ebx: PerformanceMonitoringFeaturesEbx,
ecx: u32,
edx: u32,
}
impl PerformanceMonitoringInfo {
pub fn version_id(&self) -> u8 {
get_bits(self.eax, 0, 7) as u8
}
pub fn number_of_counters(&self) -> u8 {
get_bits(self.eax, 8, 15) as u8
}
pub fn counter_bit_width(&self) -> u8 {
get_bits(self.eax, 16, 23) as u8
}
pub fn ebx_length(&self) -> u8 {
get_bits(self.eax, 24, 31) as u8
}
pub fn fixed_function_counters(&self) -> u8 {
get_bits(self.edx, 0, 4) as u8
}
pub fn fixed_function_counters_bit_width(&self) -> u8 {
get_bits(self.edx, 5, 12) as u8
}
check_flag!(doc = "Core cycle event not available if 1.",
is_core_cyc_ev_unavailable, ebx, CPU_FEATURE_CORE_CYC_EV_UNAVAILABLE);
check_flag!(doc = "Instruction retired event not available if 1.",
is_inst_ret_ev_unavailable, ebx, CPU_FEATURE_INST_RET_EV_UNAVAILABLE);
check_flag!(doc = "Reference cycles event not available if 1.",
is_ref_cycle_ev_unavailable, ebx, CPU_FEATURE_REF_CYC_EV_UNAVAILABLE);
check_flag!(doc = "Last-level cache reference event not available if 1.",
is_cache_ref_ev_unavailable, ebx, CPU_FEATURE_CACHE_REF_EV_UNAVAILABLE);
check_flag!(doc = "Last-level cache misses event not available if 1.",
is_ll_cache_miss_ev_unavailable, ebx, CPU_FEATURE_LL_CACHE_MISS_EV_UNAVAILABLE);
check_flag!(doc = "Branch instruction retired event not available if 1.",
is_branch_inst_ret_ev_unavailable, ebx, CPU_FEATURE_BRANCH_INST_RET_EV_UNAVAILABLE);
check_flag!(doc = "Branch mispredict retired event not available if 1.",
is_branch_midpred_ev_unavailable, ebx, CPU_FEATURE_BRANCH_MISPRED_EV_UNAVAILABLE);
}
bitflags! {
#[doc(hidden)]
#[derive(Debug)]
flags PerformanceMonitoringFeaturesEbx: u32 {
#[doc(hidden)]
const CPU_FEATURE_CORE_CYC_EV_UNAVAILABLE = 1 << 0,
#[doc(hidden)]
const CPU_FEATURE_INST_RET_EV_UNAVAILABLE = 1 << 1,
#[doc(hidden)]
const CPU_FEATURE_REF_CYC_EV_UNAVAILABLE = 1 << 2,
#[doc(hidden)]
const CPU_FEATURE_CACHE_REF_EV_UNAVAILABLE = 1 << 3,
#[doc(hidden)]
const CPU_FEATURE_LL_CACHE_MISS_EV_UNAVAILABLE = 1 << 4,
#[doc(hidden)]
const CPU_FEATURE_BRANCH_INST_RET_EV_UNAVAILABLE = 1 << 5,
#[doc(hidden)]
const CPU_FEATURE_BRANCH_MISPRED_EV_UNAVAILABLE = 1 << 6,
}
}
#[derive(Debug)]
pub struct ExtendedTopologyIter {
level: u32,
}
#[derive(Debug)]
pub struct ExtendedTopologyLevel {
eax: u32,
ebx: u32,
ecx: u32,
edx: u32,
}
impl ExtendedTopologyLevel {
pub fn processors(&self) -> u16 {
get_bits(self.ebx, 0, 15) as u16
}
pub fn level_number(&self) -> u8 {
get_bits(self.ecx, 0, 7) as u8
}
pub fn level_type(&self) -> TopologyType {
match get_bits(self.ecx, 8, 15) {
0 => TopologyType::INVALID,
1 => TopologyType::SMT,
2 => TopologyType::CORE,
_ => unreachable!()
}
}
pub fn x2apic_id(&self) -> u32 {
self.edx
}
pub fn shift_right_for_next_apic_id(&self) -> u32 {
get_bits(self.eax, 0, 4)
}
}
#[derive(PartialEq, Eq, Debug)]
pub enum TopologyType {
INVALID = 0,
SMT = 1,
CORE = 2,
}
impl Iterator for ExtendedTopologyIter {
type Item = ExtendedTopologyLevel;
fn next(&mut self) -> Option<ExtendedTopologyLevel> {
let res = cpuid!(EAX_EXTENDED_TOPOLOGY_INFO, self.level);
self.level += 1;
let et = ExtendedTopologyLevel { eax: res.eax, ebx: res.ebx, ecx: res.ecx, edx: res.edx };
match et.level_type() {
TopologyType::INVALID => None,
_ => Some(et)
}
}
}
#[derive(PartialEq, Eq, Debug)]
#[allow(non_camel_case_types)]
enum ExtendedStateIdent {
Legacy87 = 1 << 0,
SSE128 = 1 << 1,
AVX256 = 1 << 2,
MPX = 0b11 << 3,
AVX512 = 0b111 << 5,
IA32_XSS = 1 << 8,
PKRU = 1 << 9,
}
#[derive(Debug)]
pub struct ExtendedStateInfo {
eax: u32,
ebx: u32,
ecx: u32,
edx: u32,
eax1: u32,
}
macro_rules! check_xcr_flag {
($doc:meta, $fun:ident, $flag:ident) => (
#[$doc]
pub fn $fun(&self) -> bool {
self.xcr0_supported() & (ExtendedStateIdent::$flag as u64) > 0
}
)
}
impl ExtendedStateInfo {
pub fn xcr0_supported(&self) -> u64 {
(self.edx as u64) << 32 | self.eax as u64
}
check_xcr_flag!(doc = "Legacy x87.",
has_legacy_x87, Legacy87);
check_xcr_flag!(doc = "SSE 128-bit.",
has_sse_128, SSE128);
check_xcr_flag!(doc = "AVX 256-bit.",
has_avx_256, AVX256);
check_xcr_flag!(doc = "MPX.",
has_mpx, MPX);
check_xcr_flag!(doc = "AVX 512-bit.",
has_avx_512, AVX512);
check_xcr_flag!(doc = "IA32_XSS.",
has_ia32_xss, IA32_XSS);
check_xcr_flag!(doc = "PKRU.",
has_pkru, PKRU);
pub fn maximum_size_enabled_features(&self) -> u32 {
self.ebx
}
pub fn maximum_size_supported_features(&self) -> u32 {
self.ecx
}
pub fn has_xsaveopt(&self) -> bool {
self.eax1 & 0x1 > 0
}
pub fn has_xsavec(&self) -> bool {
self.eax1 & 0x0b10 > 0
}
pub fn has_xgetbv(&self) -> bool {
self.eax1 & 0x0b100 > 0
}
pub fn has_xsaves_xrstors(&self) -> bool {
self.eax1 & 0x0b1000 > 0
}
pub fn iter(&self) -> ExtendedStateIter {
ExtendedStateIter { level: 1, xcr0_supported: self.xcr0_supported() }
}
}
pub struct ExtendedStateIter {
level: u32,
xcr0_supported: u64,
}
impl Iterator for ExtendedStateIter {
type Item = ExtendedState;
fn next(&mut self) -> Option<ExtendedState> {
if self.level > 62 {
return None;
}
self.level += 1;
let bit = 1 << self.level;
if self.xcr0_supported & bit > 0 {
let res = cpuid!(EAX_EXTENDED_STATE_INFO, self.level);
return Some(ExtendedState { subleaf: self.level, eax: res.eax, ebx: res.ebx, ecx: res.ecx });
}
self.next()
}
}
#[derive(Debug)]
pub struct ExtendedState {
pub subleaf: u32,
eax: u32,
ebx: u32,
ecx: u32,
}
impl ExtendedState {
pub fn size(&self) -> u32 {
self.eax
}
pub fn offset(&self) -> u32 {
self.ebx
}
pub fn is_in_ia32_xss(&self) -> bool {
self.ecx & 0b1 > 0
}
pub fn is_in_xcr0(&self) -> bool {
self.ecx & 0b1 == 0
}
pub fn is_compacted_format(&self) -> bool {
self.ecx & 0b10 > 0
}
}
#[derive(Debug)]
pub struct QoSInfo {
ebx0: u32,
edx0: u32,
ebx1: u32,
ecx1: u32,
edx1: u32,
}
impl QoSInfo {
pub fn maximum_rmid_range(&self) -> u32 {
self.ebx0
}
pub fn has_l3_qos(&self) -> bool {
self.edx0 & (1 << 1) > 0
}
pub fn conversion_factor(&self) -> u32 {
self.ebx1
}
pub fn maximum_range_l3_rmid(&self) -> u32 {
self.ecx1
}
pub fn has_l3_occupancy_monitoring(&self) -> bool {
self.edx1 & 0x1 > 0
}
}
#[derive(Debug)]
pub struct QoSEnforcementInfo {
ebx0: u32,
eax1: u32,
ebx1: u32,
ecx1: u32,
edx1: u32
}
impl QoSEnforcementInfo {
check_bit_fn!(doc = "Supports L3 Cache QoS enforcement if true.", has_l3_qos_enforcement, ebx0, 0);
pub fn iter(&self) -> QoSEnforcementIter {
QoSEnforcementIter { current: 0, ebx0: self.ebx0 }
}
}
pub struct QoSEnforcementIter {
current: u8,
ebx0: u32,
}
impl Iterator for QoSEnforcementIter {
type Item = QoSEnforcement;
fn next(&mut self) -> Option<QoSEnforcement> {
if self.current > 31 {
return None;
}
self.current += 1;
if is_bit_set!(self.ebx0, self.current) {
let res = cpuid!(EAX_QOS_ENFORCEMENT_INFO, self.current);
return Some(QoSEnforcement { eax: res.eax, ebx: res.ebx, ecx: res.ecx, edx: res.edx });
}
self.next()
}
}
#[derive(Debug)]
pub struct QoSEnforcement {
eax: u32,
ebx: u32,
ecx: u32,
edx: u32
}
impl QoSEnforcement {
pub fn capacity_mask_length(&self) -> u8 {
get_bits(self.eax, 0, 4) as u8
}
pub fn allocation_unit_isolation(&self) -> u32 {
self.ebx
}
pub fn highest_cos_number(&self) -> u16 {
get_bits(self.edx, 0, 15) as u16
}
check_bit_fn!(doc = "Updates of COS should be infrequent if true.",
has_infrequent_cos_updates, ecx, 1);
check_bit_fn!(doc = "Is Code and Data Prioritization Technology supported?",
has_code_data_prioritization, ecx, 2);
}
#[derive(Debug)]
pub struct ProcessorTraceInfo {
eax: u32,
ebx: u32,
ecx: u32,
edx: u32
}
impl ProcessorTraceInfo {
check_bit_fn!(doc = "If true, Indicates that IA32_RTIT_CTL.CR3Filter can be set to 1, and that IA32_RTIT_CR3_MATCH MSR can be accessed.",
has_rtit_cr3_match, ebx, 0);
check_bit_fn!(doc = "If true, Indicates support of Configurable PSB and Cycle-Accurate Mode.",
has_configurable_psb_and_cycle_accurate_mode, ebx, 1);
check_bit_fn!(doc = "If true, Indicates support of IP Filtering, TraceStop filtering, and preservation of Intel PT MSRs across warm reset.",
has_ip_tracestop_filtering, ebx, 2);
check_bit_fn!(doc = "If true, Indicates support of MTC timing packet and suppression of COFI-based packets.",
has_mtc_timing_packet_coefi_suppression, ebx, 3);
check_bit_fn!(doc = "If true, Tracing can be enabled with IA32_RTIT_CTL.ToPA = 1, hence utilizing the ToPA output scheme; IA32_RTIT_OUTPUT_BASE and IA32_RTIT_OUTPUT_MASK_PTRS MSRs can be accessed.",
has_topa, ecx, 0);
check_bit_fn!(doc = "If true, ToPA tables can hold any number of output entries, up to the maximum allowed by the MaskOrTableOffset field of IA32_RTIT_OUTPUT_MASK_PTRS.",
has_topa_maximum_entries, ecx, 1);
check_bit_fn!(doc = "If true, Indicates support of Single-Range Output scheme.",
has_single_range_output_scheme, ecx, 2);
check_bit_fn!(doc = "If true, Indicates support of output to Trace Transport subsystem.",
has_trace_transport_subsystem, ecx, 3);
check_bit_fn!(doc = "If true, Generated packets which contain IP payloads have LIP values, which include the CS base component.",
has_lip_with_cs_base, ecx, 31);
pub fn iter(&self) -> ProcessorTraceIter {
ProcessorTraceIter { current: 0, count: self.eax }
}
}
#[derive(Debug)]
pub struct ProcessorTraceIter {
current: u32,
count: u32,
}
impl Iterator for ProcessorTraceIter {
type Item = ProcessorTrace;
fn next(&mut self) -> Option<ProcessorTrace> {
if self.current <= self.count {
self.current += 1;
let res = cpuid!(EAX_TRACE_INFO, self.current);
return Some(ProcessorTrace { eax: res.eax, ebx: res.ebx });
}
None
}
}
#[derive(Debug)]
pub struct ProcessorTrace {
eax: u32,
ebx: u32,
}
impl ProcessorTrace {
pub fn configurable_address_ranges(&self) -> u8 {
get_bits(self.eax, 0, 2) as u8
}
pub fn supported_mtc_period_encodings(&self) -> u16 {
get_bits(self.eax, 16, 31) as u16
}
pub fn supported_cycle_threshold_value_encodings(&self) -> u16 {
get_bits(self.ebx, 0, 15) as u16
}
pub fn supported_psb_frequency_encodings(&self) -> u16 {
get_bits(self.ebx, 16, 31) as u16
}
}
#[derive(Debug)]
pub struct TscInfo {
eax: u32,
ebx: u32
}
impl TscInfo {
pub fn get_tsc_ratio_denominator(&self) -> u32 {
self.eax
}
pub fn get_tsc_ratio_numerator(&self) -> u32 {
self.ebx
}
}
#[derive(Debug)]
pub struct ProcessorFrequencyInfo {
eax: u32,
ebx: u32,
ecx: u32
}
impl ProcessorFrequencyInfo {
pub fn processor_base_frequency(&self) -> u16 {
get_bits(self.eax, 0, 15) as u16
}
pub fn processor_max_frequency(&self) -> u16 {
get_bits(self.ebx, 0, 15) as u16
}
pub fn bus_frequency(&self) -> u16 {
get_bits(self.ecx, 0, 15) as u16
}
}
#[derive(Debug)]
pub struct ExtendedFunctionInfo {
max_eax_value: u32,
data: [CpuIdResult; 9],
}
#[derive(PartialEq, Eq, Debug)]
pub enum L2Associativity {
Disabled = 0x0,
DirectMapped = 0x1,
TwoWay = 0x2,
FourWay = 0x4,
EightWay = 0x6,
SixteenWay = 0x8,
FullyAssiciative = 0xF,
Unknown,
}
const EAX_EXTENDED_PROC_SIGNATURE: u32 = 0x1;
const EAX_EXTENDED_BRAND_STRING: u32 = 0x4;
const EAX_EXTENDED_CACHE_INFO: u32 = 0x6;
impl ExtendedFunctionInfo {
fn leaf_is_supported(&self, val: u32) -> bool {
val <= self.max_eax_value
}
pub fn processor_brand_string(&self) -> Option<&str> {
if self.leaf_is_supported(EAX_EXTENDED_BRAND_STRING) {
Some(unsafe {
let brand_string_start = transmute::<&CpuIdResult, *const u8>(&self.data[2]);
let slice = slice::from_raw_parts(brand_string_start, 3*4*4);
let byte_array: &'static [u8] = transmute(slice);
str::from_utf8_unchecked(byte_array)
})
}
else {
None
}
}
pub fn extended_signature(&self) -> Option<u32> {
if self.leaf_is_supported(EAX_EXTENDED_PROC_SIGNATURE) {
Some(self.data[1].eax)
}
else {
None
}
}
pub fn cache_line_size(&self) -> Option<u8> {
if self.leaf_is_supported(EAX_EXTENDED_CACHE_INFO) {
Some(get_bits(self.data[6].ecx, 0, 7) as u8)
}
else {
None
}
}
pub fn l2_associativity(&self) -> Option<L2Associativity> {
if self.leaf_is_supported(EAX_EXTENDED_CACHE_INFO) {
Some(match get_bits(self.data[6].ecx, 12, 15) {
0x0 => L2Associativity::Disabled,
0x1 => L2Associativity::DirectMapped,
0x2 => L2Associativity::TwoWay,
0x4 => L2Associativity::FourWay,
0x6 => L2Associativity::EightWay,
0x8 => L2Associativity::SixteenWay,
0xF => L2Associativity::FullyAssiciative,
_ => L2Associativity::Unknown,
})
}
else {
None
}
}
pub fn cache_size(&self) -> Option<u16> {
if self.leaf_is_supported(EAX_EXTENDED_CACHE_INFO) {
Some(get_bits(self.data[6].ecx, 16, 31) as u16)
}
else {
None
}
}
pub fn physical_address_bits(&self) -> Option<u8> {
if self.leaf_is_supported(8) {
Some(get_bits(self.data[8].eax, 0, 7) as u8)
}
else {
None
}
}
pub fn linear_address_bits(&self) -> Option<u8> {
if self.leaf_is_supported(8) {
Some(get_bits(self.data[8].eax, 8, 15) as u8)
}
else {
None
}
}
pub fn has_invariant_tsc(&self) -> bool {
self.leaf_is_supported(7) && self.data[7].edx & (1 << 8) > 0
}
pub fn has_lahf_sahf(&self) -> bool {
self.leaf_is_supported(1) &&
ExtendedFunctionInfoEcx{ bits: self.data[1].ecx }.contains(CPU_FEATURE_LAHF_SAHF)
}
pub fn has_lzcnt(&self) -> bool {
self.leaf_is_supported(1) &&
ExtendedFunctionInfoEcx{ bits: self.data[1].ecx }.contains(CPU_FEATURE_LZCNT)
}
pub fn has_prefetchw(&self) -> bool {
self.leaf_is_supported(1) &&
ExtendedFunctionInfoEcx{ bits: self.data[1].ecx }.contains(CPU_FEATURE_PREFETCHW)
}
pub fn has_syscall_sysret(&self) -> bool {
self.leaf_is_supported(1) &&
ExtendedFunctionInfoEdx{ bits: self.data[1].edx }.contains(CPU_FEATURE_SYSCALL_SYSRET)
}
pub fn has_execute_disable(&self) -> bool {
self.leaf_is_supported(1) &&
ExtendedFunctionInfoEdx{ bits: self.data[1].edx }.contains(CPU_FEATURE_EXECUTE_DISABLE)
}
pub fn has_1gib_pages(&self) -> bool {
self.leaf_is_supported(1) &&
ExtendedFunctionInfoEdx{ bits: self.data[1].edx }.contains(CPU_FEATURE_1GIB_PAGES)
}
pub fn has_rdtscp(&self) -> bool {
self.leaf_is_supported(1) &&
ExtendedFunctionInfoEdx{ bits: self.data[1].edx }.contains(CPU_FEATURE_RDTSCP)
}
pub fn has_64bit_mode(&self) -> bool {
self.leaf_is_supported(1) &&
ExtendedFunctionInfoEdx{ bits: self.data[1].edx }.contains(CPU_FEATURE_64BIT_MODE)
}
}
#[doc(hidden)]
bitflags! {
#[doc(hidden)]
#[derive(Debug)]
flags ExtendedFunctionInfoEcx: u32 {
#[doc(hidden)]
const CPU_FEATURE_LAHF_SAHF = 1 << 0,
#[doc(hidden)]
const CPU_FEATURE_LZCNT = 1 << 5,
#[doc(hidden)]
const CPU_FEATURE_PREFETCHW = 1 << 8,
}
}
bitflags! {
#[doc(hidden)]
#[derive(Debug)]
flags ExtendedFunctionInfoEdx: u32 {
#[doc(hidden)]
const CPU_FEATURE_SYSCALL_SYSRET = 1 << 11,
#[doc(hidden)]
const CPU_FEATURE_EXECUTE_DISABLE = 1 << 20,
#[doc(hidden)]
const CPU_FEATURE_1GIB_PAGES = 1 << 26,
#[doc(hidden)]
const CPU_FEATURE_RDTSCP = 1 << 27,
#[doc(hidden)]
const CPU_FEATURE_64BIT_MODE = 1 << 29,
}
}
#[cfg(test)]
#[test]
fn genuine_intel() {
let vf = VendorInfo { ebx: 1970169159, edx: 1231384169, ecx: 1818588270 };
assert!(vf.as_string() == "GenuineIntel");
}
#[test]
fn feature_info() {
let finfo = FeatureInfo { eax: 198313,
ebx: 34605056,
ecx: FeatureInfoEcx { bits: 2109399999 },
edx: FeatureInfoEdx { bits: 3219913727 }, };
assert!(finfo.model_id() == 10);
assert!(finfo.extended_model_id() == 3);
assert!(finfo.stepping_id() == 9);
assert!(finfo.extended_family_id() == 0);
assert!(finfo.family_id() == 6);
assert!(finfo.stepping_id() == 9);
assert!(finfo.brand_index() == 0);
assert!(finfo.edx.contains(CPU_FEATURE_SSE2));
assert!(finfo.ecx.contains(CPU_FEATURE_SSE41));
}
#[test]
fn cache_info() {
let cinfos = CacheInfoIter { current: 1,
eax: 1979931137,
ebx: 15774463,
ecx: 0,
edx: 13238272, };
for (idx, cache) in cinfos.enumerate() {
match idx {
0 => assert!(cache.num == 0xff),
1 => assert!(cache.num == 0x5a),
2 => assert!(cache.num == 0xb2),
3 => assert!(cache.num == 0x03),
4 => assert!(cache.num == 0xf0),
5 => assert!(cache.num == 0xca),
6 => assert!(cache.num == 0x76),
_ => unreachable!(),
}
}
}
#[test]
fn cache_parameters() {
let caches: [CacheParameter; 4] = [
CacheParameter { eax: 469778721, ebx: 29360191, ecx: 63, edx: 0 },
CacheParameter { eax: 469778722, ebx: 29360191, ecx: 63, edx: 0 },
CacheParameter { eax: 469778755, ebx: 29360191, ecx: 511, edx: 0 },
CacheParameter { eax: 470008163, ebx: 46137407, ecx: 4095, edx: 6 },
];
for (idx, cache) in caches.into_iter().enumerate() {
match idx {
0 => {
assert!(cache.cache_type() == CacheType::DATA);
assert!(cache.level() == 1);
assert!(cache.is_self_initializing());
assert!(!cache.is_fully_associative());
assert!(cache.max_cores_for_cache() == 2);
assert!(cache.max_cores_for_package() == 8);
assert!(cache.coherency_line_size() == 64);
assert!(cache.physical_line_partitions() == 1);
assert!(cache.associativity() == 8);
assert!(!cache.is_write_back_invalidate());
assert!(!cache.is_inclusive());
assert!(!cache.has_complex_indexing());
assert!(cache.sets() == 64);
},
1 => {
assert!(cache.cache_type() == CacheType::INSTRUCTION);
assert!(cache.level() == 1);
assert!(cache.is_self_initializing());
assert!(!cache.is_fully_associative());
assert!(cache.max_cores_for_cache() == 2);
assert!(cache.max_cores_for_package() == 8);
assert!(cache.coherency_line_size() == 64);
assert!(cache.physical_line_partitions() == 1);
assert!(cache.associativity() == 8);
assert!(!cache.is_write_back_invalidate());
assert!(!cache.is_inclusive());
assert!(!cache.has_complex_indexing());
assert!(cache.sets() == 64);
},
2 => {
assert!(cache.cache_type() == CacheType::UNIFIED);
assert!(cache.level() == 2);
assert!(cache.is_self_initializing());
assert!(!cache.is_fully_associative());
assert!(cache.max_cores_for_cache() == 2);
assert!(cache.max_cores_for_package() == 8);
assert!(cache.coherency_line_size() == 64);
assert!(cache.physical_line_partitions() == 1);
assert!(cache.associativity() == 8);
assert!(!cache.is_write_back_invalidate());
assert!(!cache.is_inclusive());
assert!(!cache.has_complex_indexing());
assert!(cache.sets() == 512);
},
3 => {
assert!(cache.cache_type() == CacheType::UNIFIED);
assert!(cache.level() == 3);
assert!(cache.is_self_initializing());
assert!(!cache.is_fully_associative());
assert!(cache.max_cores_for_cache() == 16);
assert!(cache.max_cores_for_package() == 8);
assert!(cache.coherency_line_size() == 64);
assert!(cache.physical_line_partitions() == 1);
assert!(cache.associativity() == 12);
assert!(!cache.is_write_back_invalidate());
assert!(cache.is_inclusive());
assert!(cache.has_complex_indexing());
assert!(cache.sets() == 4096);
},
_ => unreachable!()
}
}
}
#[test]
fn monitor_mwait_features() {
let mmfeatures = MonitorMwaitInfo { eax: 64, ebx: 64, ecx: 3, edx: 135456 };
assert!(mmfeatures.smallest_monitor_line() == 64);
assert!(mmfeatures.largest_monitor_line() == 64);
assert!(mmfeatures.extensions_supported());
assert!(mmfeatures.interrupts_as_break_event());
assert!(mmfeatures.supported_c0_states() == 0);
assert!(mmfeatures.supported_c1_states() == 2);
assert!(mmfeatures.supported_c2_states() == 1);
assert!(mmfeatures.supported_c3_states() == 1);
assert!(mmfeatures.supported_c4_states() == 2);
assert!(mmfeatures.supported_c5_states() == 0);
assert!(mmfeatures.supported_c6_states() == 0);
assert!(mmfeatures.supported_c7_states() == 0);
}
#[test]
fn thermal_power_features() {
let tpfeatures = ThermalPowerInfo { eax: ThermalPowerFeaturesEax { bits: 119 },
ebx: 2,
ecx: ThermalPowerFeaturesEcx { bits: 9 },
edx: 0, };
assert!(tpfeatures.eax.contains(CPU_FEATURE_DTS));
assert!(tpfeatures.eax.contains(CPU_FEATURE_TURBO_BOOST));
assert!(tpfeatures.eax.contains(CPU_FEATURE_ARAT));
assert!(tpfeatures.eax.contains(CPU_FEATURE_PLN));
assert!(tpfeatures.eax.contains(CPU_FEATURE_ECMD));
assert!(tpfeatures.eax.contains(CPU_FEATURE_PTM));
assert!(tpfeatures.ecx.contains(CPU_FEATURE_HW_COORD_FEEDBACK));
assert!(tpfeatures.ecx.contains(CPU_FEATURE_ENERGY_BIAS_PREF));
assert!(tpfeatures.dts_irq_threshold() == 0x2);
}
#[test]
fn extended_features() {
let tpfeatures = ExtendedFeatures { eax: 0,
ebx: ExtendedFeaturesEbx { bits: 641 },
ecx: 0,
edx: 0, };
assert!(tpfeatures.eax == 0);
assert!(tpfeatures.ebx.contains(CPU_FEATURE_FSGSBASE));
assert!(!tpfeatures.ebx.contains(CPU_FEATURE_ADJUST_MSR));
assert!(!tpfeatures.ebx.contains(CPU_FEATURE_BMI1));
assert!(!tpfeatures.ebx.contains(CPU_FEATURE_HLE));
assert!(!tpfeatures.ebx.contains(CPU_FEATURE_AVX2));
assert!(tpfeatures.ebx.contains(CPU_FEATURE_SMEP));
assert!(!tpfeatures.ebx.contains(CPU_FEATURE_BMI2));
assert!(tpfeatures.ebx.contains(CPU_FEATURE_REP_MOVSB_STOSB));
assert!(!tpfeatures.ebx.contains(CPU_FEATURE_INVPCID));
assert!(!tpfeatures.ebx.contains(CPU_FEATURE_RTM));
assert!(!tpfeatures.ebx.contains(CPU_FEATURE_QM));
assert!(!tpfeatures.ebx.contains(CPU_FEATURE_DEPRECATE_FPU_CS_DS));
}
#[test]
fn direct_cache_access_info() {
let dca = DirectCacheAccessInfo { eax: 0x1 };
assert!(dca.get_dca_cap_value() == 0x1);
}
#[test]
fn performance_monitoring_info() {
let pm = PerformanceMonitoringInfo { eax: 120587267,
ebx: PerformanceMonitoringFeaturesEbx { bits: 0 },
ecx: 0,
edx: 1539, };
assert!(pm.version_id() == 3);
assert!(pm.number_of_counters() == 4);
assert!(pm.counter_bit_width() == 48);
assert!(pm.ebx_length() == 7);
assert!(pm.fixed_function_counters() == 3);
assert!(pm.fixed_function_counters_bit_width() == 48);
assert!(!pm.ebx.contains(CPU_FEATURE_CORE_CYC_EV_UNAVAILABLE));
assert!(!pm.ebx.contains(CPU_FEATURE_INST_RET_EV_UNAVAILABLE));
assert!(!pm.ebx.contains(CPU_FEATURE_REF_CYC_EV_UNAVAILABLE));
assert!(!pm.ebx.contains(CPU_FEATURE_CACHE_REF_EV_UNAVAILABLE));
assert!(!pm.ebx.contains(CPU_FEATURE_LL_CACHE_MISS_EV_UNAVAILABLE));
assert!(!pm.ebx.contains(CPU_FEATURE_BRANCH_INST_RET_EV_UNAVAILABLE));
assert!(!pm.ebx.contains(CPU_FEATURE_BRANCH_MISPRED_EV_UNAVAILABLE));
}
#[cfg(test)]
#[test]
fn extended_topology_info() {
let l1 = ExtendedTopologyLevel { eax: 1, ebx: 2, ecx: 256, edx: 3 };
let l2 = ExtendedTopologyLevel { eax: 4, ebx: 4, ecx: 513, edx: 3 };
assert!(l1.processors() == 2);
assert!(l1.level_number() == 0);
assert!(l1.level_type() == TopologyType::SMT);
assert!(l1.x2apic_id() == 3);
assert!(l1.shift_right_for_next_apic_id() == 1);
assert!(l2.processors() == 4);
assert!(l2.level_number() == 1);
assert!(l2.level_type() == TopologyType::CORE);
assert!(l2.x2apic_id() == 3);
assert!(l2.shift_right_for_next_apic_id() == 4);
}
#[test]
fn extended_state_info() {
let es = ExtendedStateInfo { eax: 7, ebx: 832, ecx: 832, edx: 0, eax1: 1 };
assert!(es.xcr0_supported() == 7);
assert!(es.maximum_size_enabled_features() == 832);
assert!(es.maximum_size_supported_features() == 832);
assert!(es.has_xsaveopt());
for (idx, e) in es.iter().enumerate() {
match idx {
0 => {
assert!(e.subleaf == 2);
assert!(e.size() == 256);
assert!(e.offset() == 576);
}
_ => unreachable!()
}
}
}
#[test]
fn quality_of_service_info() {
let qos = QoSInfo { ebx0: 832, edx0: 0, ebx1: 0, ecx1: 0, edx1: 0 };
assert!(qos.maximum_rmid_range() == 832);
assert!(!qos.has_l3_qos());
assert!(qos.conversion_factor() == 0x0);
assert!(qos.maximum_range_l3_rmid() == 0x0);
assert!(!qos.has_l3_occupancy_monitoring());
}
#[test]
fn extended_functions() {
let ef = ExtendedFunctionInfo { max_eax_value: 8,
data: [
CpuIdResult { eax: 2147483656, ebx: 0, ecx: 0, edx: 0 },
CpuIdResult { eax: 0, ebx: 0, ecx: 1, edx: 672139264 },
CpuIdResult { eax: 538976288, ebx: 1226842144, ecx: 1818588270, edx: 539578920 },
CpuIdResult { eax: 1701998403, ebx: 692933672, ecx: 758475040, edx: 926102323 },
CpuIdResult { eax: 1346576469, ebx: 541073493, ecx: 808988209, edx: 8013895 },
CpuIdResult { eax: 0, ebx: 0, ecx: 0, edx: 0 },
CpuIdResult { eax: 0, ebx: 0, ecx: 16801856, edx: 0 },
CpuIdResult { eax: 0, ebx: 0, ecx: 0, edx: 256 },
CpuIdResult { eax: 12324, ebx: 0, ecx: 0, edx: 0 }
], };
assert!(ef.processor_brand_string().unwrap() == " Intel(R) Core(TM) i5-3337U CPU @ 1.80GHz\0");
assert!(ef.has_lahf_sahf());
assert!(!ef.has_lzcnt());
assert!(!ef.has_prefetchw());
assert!(ef.has_syscall_sysret());
assert!(ef.has_execute_disable());
assert!(!ef.has_1gib_pages());
assert!(ef.has_rdtscp());
assert!(ef.has_64bit_mode());
assert!(ef.has_invariant_tsc());
assert!(ef.extended_signature().unwrap() == 0x0);
assert!(ef.cache_line_size().unwrap() == 64);
assert!(ef.l2_associativity().unwrap() == L2Associativity::EightWay);
assert!(ef.cache_size().unwrap() == 256);
assert!(ef.physical_address_bits().unwrap() == 36);
assert!(ef.linear_address_bits().unwrap() == 48);
}
#[cfg(test)]
#[test]
fn readme_test() {
}