1use crate::{Status, sys};
6use bitflags::bitflags;
7
8pub fn system_get_page_size() -> u32 {
14 unsafe { sys::zx_system_get_page_size() }
15}
16
17pub fn system_get_dcache_line_size() -> u32 {
23 unsafe { sys::zx_system_get_dcache_line_size() }
24}
25
26pub fn system_get_physmem() -> u64 {
32 unsafe { sys::zx_system_get_physmem() }
33}
34
35pub fn system_get_num_cpus() -> u32 {
41 unsafe { sys::zx_system_get_num_cpus() }
42}
43
44#[derive(Debug, Copy, Clone, Eq, PartialEq)]
46#[repr(u32)]
47pub enum FeatureKind {
48 HardwareBreakpointCount = sys::ZX_FEATURE_KIND_HW_BREAKPOINT_COUNT,
49 HardwareWatchpointCount = sys::ZX_FEATURE_KIND_HW_WATCHPOINT_COUNT,
50}
51
52impl Into<u32> for FeatureKind {
53 fn into(self) -> u32 {
54 match self {
55 FeatureKind::HardwareBreakpointCount => sys::ZX_FEATURE_KIND_HW_BREAKPOINT_COUNT,
56 FeatureKind::HardwareWatchpointCount => sys::ZX_FEATURE_KIND_HW_WATCHPOINT_COUNT,
57 }
58 }
59}
60
61mod private {
64 pub struct Internal;
65}
66
67pub trait FeatureFlags: bitflags::Flags {
69 fn kind(_: private::Internal) -> u32;
70}
71
72bitflags! {
73 #[repr(transparent)]
74 #[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
75 pub struct VirtualMemoryFeatureFlags: u32 {
76 const VM_CAN_MAP_XOM = (1<<0);
78
79 const _ = !0;
81 }
82}
83
84impl FeatureFlags for VirtualMemoryFeatureFlags {
85 fn kind(_: private::Internal) -> u32 {
86 sys::ZX_FEATURE_KIND_VM
87 }
88}
89
90bitflags! {
91 #[repr(transparent)]
92 #[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
93 pub struct CpuFeatureFlags: u32 {
94
95 const HAS_CPU_FEATURES = (1<<0);
97
98 #[cfg(target_arch = "aarch64")]
100 const ARM64_FEATURE_ISA_FP = sys::ZX_ARM64_FEATURE_ISA_FP;
101 #[cfg(target_arch = "aarch64")]
102 const ARM64_FEATURE_ISA_ASIMD = sys::ZX_ARM64_FEATURE_ISA_ASIMD;
103 #[cfg(target_arch = "aarch64")]
104 const ARM64_FEATURE_ISA_AES = sys::ZX_ARM64_FEATURE_ISA_AES;
105 #[cfg(target_arch = "aarch64")]
106 const ARM64_FEATURE_ISA_PMULL = sys::ZX_ARM64_FEATURE_ISA_PMULL;
107 #[cfg(target_arch = "aarch64")]
108 const ARM64_FEATURE_ISA_SHA1 = sys::ZX_ARM64_FEATURE_ISA_SHA1;
109 #[cfg(target_arch = "aarch64")]
110 const ARM64_FEATURE_ISA_SHA256 = sys::ZX_ARM64_FEATURE_ISA_SHA256;
111 #[cfg(target_arch = "aarch64")]
112 const ARM64_FEATURE_ISA_CRC32 = sys::ZX_ARM64_FEATURE_ISA_CRC32;
113 #[cfg(target_arch = "aarch64")]
114 const ARM64_FEATURE_ISA_ATOMICS = sys::ZX_ARM64_FEATURE_ISA_ATOMICS;
115 #[cfg(target_arch = "aarch64")]
116 const ARM64_FEATURE_ISA_RDM = sys::ZX_ARM64_FEATURE_ISA_RDM;
117 #[cfg(target_arch = "aarch64")]
118 const ARM64_FEATURE_ISA_SHA3 = sys::ZX_ARM64_FEATURE_ISA_SHA3;
119 #[cfg(target_arch = "aarch64")]
120 const ARM64_FEATURE_ISA_SM3 = sys::ZX_ARM64_FEATURE_ISA_SM3;
121 #[cfg(target_arch = "aarch64")]
122 const ARM64_FEATURE_ISA_SM4 = sys::ZX_ARM64_FEATURE_ISA_SM4;
123 #[cfg(target_arch = "aarch64")]
124 const ARM64_FEATURE_ISA_DP = sys::ZX_ARM64_FEATURE_ISA_DP;
125 #[cfg(target_arch = "aarch64")]
126 const ARM64_FEATURE_ISA_DPB = sys::ZX_ARM64_FEATURE_ISA_DPB;
127 #[cfg(target_arch = "aarch64")]
128 const ARM64_FEATURE_ISA_FHM = sys::ZX_ARM64_FEATURE_ISA_FHM;
129 #[cfg(target_arch = "aarch64")]
130 const ARM64_FEATURE_ISA_TS = sys::ZX_ARM64_FEATURE_ISA_TS;
131 #[cfg(target_arch = "aarch64")]
132 const ARM64_FEATURE_ISA_RNDR = sys::ZX_ARM64_FEATURE_ISA_RNDR;
133 #[cfg(target_arch = "aarch64")]
134 const ARM64_FEATURE_ISA_SHA512 = sys::ZX_ARM64_FEATURE_ISA_SHA512;
135 #[cfg(target_arch = "aarch64")]
136 const ARM64_FEATURE_ISA_I8MM = sys::ZX_ARM64_FEATURE_ISA_I8MM;
137 #[cfg(target_arch = "aarch64")]
138 const ARM64_FEATURE_ISA_SVE = sys::ZX_ARM64_FEATURE_ISA_SVE;
139 #[cfg(target_arch = "aarch64")]
140 const ARM64_FEATURE_ISA_ARM32 = sys::ZX_ARM64_FEATURE_ISA_ARM32;
141
142 #[cfg(target_arch = "aarch64")]
144 const ARM64_FEATURE_ISA_SHA2 = sys::ZX_ARM64_FEATURE_ISA_SHA256;
145
146 const _ = !0;
148 }
149}
150
151impl FeatureFlags for CpuFeatureFlags {
152 fn kind(_: private::Internal) -> u32 {
153 sys::ZX_FEATURE_KIND_CPU
154 }
155}
156
157bitflags! {
158 #[repr(transparent)]
159 #[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
160 pub struct AddressTaggingFeatureFlags: u32 {
161 #[cfg(target_arch = "aarch64")]
163 const ARM64_FEATURE_ADDRESS_TAGGING_TBI = sys::ZX_ARM64_FEATURE_ADDRESS_TAGGING_TBI;
164
165 const _ = !0;
167 }
168}
169
170impl FeatureFlags for AddressTaggingFeatureFlags {
171 fn kind(_: private::Internal) -> u32 {
172 sys::ZX_FEATURE_KIND_ADDRESS_TAGGING
173 }
174}
175
176pub fn system_get_feature_flags<F: FeatureFlags>() -> Result<F, Status>
182where
183 F: bitflags::Flags<Bits = u32>,
184{
185 let mut raw_features: u32 = 0;
186 let access = private::Internal;
187 Status::ok(unsafe {
188 sys::zx_system_get_features(F::kind(access), &mut raw_features as *mut u32)
189 })
190 .and_then(|_status| Ok(F::from_bits_retain(raw_features)))
191}
192
193pub fn system_get_feature_count(kind: FeatureKind) -> Result<u32, Status> {
199 let mut raw_features: u32 = 0;
200 Status::ok(unsafe { sys::zx_system_get_features(kind as u32, &mut raw_features as *mut u32) })
201 .and_then(|_status| Ok(raw_features))
202}
203
204#[cfg(test)]
205mod tests {
206 use super::*;
207
208 #[test]
209 fn system_get_features_cpu_kind() {
210 let result = system_get_feature_flags()
212 .map(|flags: CpuFeatureFlags| flags & CpuFeatureFlags::HAS_CPU_FEATURES);
213
214 #[cfg(target_arch = "aarch64")]
216 assert_eq!(result, Ok(CpuFeatureFlags::HAS_CPU_FEATURES));
217
218 #[cfg(not(target_arch = "aarch64"))]
219 assert_eq!(result, Err(Status::NOT_SUPPORTED));
220 }
221
222 #[test]
223 fn system_get_features_other_kinds() {
224 assert!(system_get_feature_flags::<VirtualMemoryFeatureFlags>().is_ok());
225 assert!(system_get_feature_flags::<AddressTaggingFeatureFlags>().is_ok());
226 assert!(system_get_feature_count(FeatureKind::HardwareBreakpointCount).is_ok());
227 assert!(system_get_feature_count(FeatureKind::HardwareWatchpointCount).is_ok());
228 }
229}