4 #if !defined(__APPLE__) && defined(VGA_arm64)
8 #if defined(VGO_freebsd)
12 unsigned long getauxval(unsigned long type
);
14 unsigned long getauxval(unsigned long type
)
16 extern char** environ
;
17 char** envp
= environ
;
19 while(*envp
++ != NULL
)
21 for (auxp
= (Elf_Auxinfo
*)envp
; auxp
->a_type
!= AT_NULL
; auxp
++)
23 if (type
== auxp
->a_type
)
25 return (unsigned long)auxp
->a_un
.a_val
;
33 // This file determines arm64 features a processor supports.
34 // Arm processors do not have a x86-like cpuinfo instruction. Instead the
35 // getauxval() syscall is used with capabilities parameters: getauxval(AT_HWCAP)
36 // and getauxval(AT_HWCAP2).
39 // - 0 if the machine has the asked-for feature.
40 // - 1 if the machine doesn't have the asked-for feature.
41 // - 2 if the asked-for feature isn't recognised (this will always be the case
42 // for any feature if run on a non-arm64 machine).
43 // - 3 if there was a usage error (it also prints an error message).
44 #define FEATURE_PRESENT 0
45 #define FEATURE_NOT_PRESENT 1
46 #define UNRECOGNISED_FEATURE 2
53 #if defined(VGA_arm64)
55 // The processor's capabilities/features are returned by getauxval() as an
56 // unsigned long with each bit representing a capability/feature.
58 #define HWCAP_FP (1 << 0)
61 #define HWCAP_ASIMD (1 << 1)
64 #define HWCAP_EVTSTRM (1 << 2)
67 #define HWCAP_AES (1 << 3)
70 #define HWCAP_PMULL (1 << 4)
75 #define HWCAP_SHA1 (1 << 5)
78 #define HWCAP_SHA2 (1 << 6)
81 #define HWCAP_CRC32 (1 << 7)
84 #define HWCAP_ATOMICS (1 << 8)
87 #define HWCAP_FPHP (1 << 9)
90 #define HWCAP_ASIMDHP (1 << 10)
93 #define HWCAP_CPUID (1 << 11)
95 #ifndef HWCAP_ASIMDRDM
96 #define HWCAP_ASIMDRDM (1 << 12)
99 #define HWCAP_JSCVT (1 << 13)
102 #define HWCAP_FCMA (1 << 14)
105 #define HWCAP_LRCPC (1 << 15)
108 #define HWCAP_DCPOP (1 << 16)
111 #define HWCAP_SHA3 (1 << 17)
114 #define HWCAP_SM3 (1 << 18)
117 #define HWCAP_SM4 (1 << 19)
119 #ifndef HWCAP_ASIMDDP
120 #define HWCAP_ASIMDDP (1 << 20)
123 #define HWCAP_SHA512 (1 << 21)
126 #define HWCAP_SVE (1 << 22)
128 #ifndef HWCAP_ASIMDFHM
129 #define HWCAP_ASIMDFHM (1 << 23)
132 #define HWCAP_DIT (1 << 24)
135 #define HWCAP_USCAT (1 << 25)
138 #define HWCAP_ILRCPC (1 << 26)
141 #define HWCAP_FLAGM (1 << 27)
144 #define HWCAP_SSBS (1 << 28)
147 #define HWCAP_SB (1 << 29)
150 #define HWCAP_PACA (1 << 30)
153 #define HWCAP_PACG (1UL << 31)
156 #ifndef HWCAP2_DCPODP
157 #define HWCAP2_DCPODP (1 << 0)
160 #define HWCAP2_SVE2 (1 << 1)
162 #ifndef HWCAP2_SVEAES
163 #define HWCAP2_SVEAES (1 << 2)
165 #ifndef HWCAP2_SVEPMULL
166 #define HWCAP2_SVEPMULL (1 << 3)
168 #ifndef HWCAP2_SVEBITPERM
169 #define HWCAP2_SVEBITPERM (1 << 4)
171 #ifndef HWCAP2_SVESHA3
172 #define HWCAP2_SVESHA3 (1 << 5)
174 #ifndef HWCAP2_SVESM4
175 #define HWCAP2_SVESM4 (1 << 6)
177 #ifndef HWCAP2_FLAGM2
178 #define HWCAP2_FLAGM2 (1 << 7)
181 #define HWCAP2_FRINT (1 << 8)
184 unsigned long hwcaps
[] = {
185 HWCAP_FP
, HWCAP_ASIMD
, HWCAP_EVTSTRM
, HWCAP_AES
, HWCAP_PMULL
,
186 HWCAP_SHA1
, HWCAP_SHA2
, HWCAP_CRC32
, HWCAP_ATOMICS
, HWCAP_FPHP
,
187 HWCAP_ASIMDHP
,HWCAP_CPUID
, HWCAP_ASIMDRDM
,HWCAP_JSCVT
, HWCAP_FCMA
,
188 HWCAP_LRCPC
, HWCAP_DCPOP
, HWCAP_SHA3
, HWCAP_SM3
, HWCAP_SM4
,
189 HWCAP_ASIMDDP
,HWCAP_SHA512
, HWCAP_SVE
, HWCAP_ASIMDFHM
,HWCAP_DIT
,
190 HWCAP_USCAT
, HWCAP_ILRCPC
, HWCAP_FLAGM
, HWCAP_SSBS
, HWCAP_SB
,
191 HWCAP_PACA
, HWCAP_PACG
, 0ul};
193 unsigned long hwcaps2
[] = {
194 HWCAP2_DCPODP
, HWCAP2_SVE2
, HWCAP2_SVEAES
, HWCAP2_SVEPMULL
,
195 HWCAP2_SVEBITPERM
, HWCAP2_SVESHA3
, HWCAP2_SVESM4
, HWCAP2_FLAGM2
,
201 unsigned long cap_bit
;
204 capability capabilities
[] = {
205 {"fp", HWCAP_FP
}, {"asimd", HWCAP_ASIMD
},
206 {"evtstrm", HWCAP_EVTSTRM
}, {"aes", HWCAP_AES
},
207 {"pmull", HWCAP_PMULL
}, {"sha1", HWCAP_SHA1
},
208 {"sha2", HWCAP_SHA2
}, {"crc32", HWCAP_CRC32
},
209 {"atomics", HWCAP_ATOMICS
}, {"fphp", HWCAP_FPHP
},
210 {"asimdhp", HWCAP_ASIMDHP
}, {"cpuid", HWCAP_CPUID
},
211 {"asimdrdm", HWCAP_ASIMDRDM
}, {"jscvt", HWCAP_JSCVT
},
212 {"fcma", HWCAP_FCMA
}, {"lrcpc", HWCAP_LRCPC
},
213 {"dcpop", HWCAP_DCPOP
}, {"sha3", HWCAP_SHA3
},
214 {"sm3", HWCAP_SM3
}, {"sm4", HWCAP_SM4
},
215 {"asimddp", HWCAP_ASIMDDP
}, {"sha512", HWCAP_SHA512
},
216 {"sve", HWCAP_SVE
}, {"asimdfhm", HWCAP_ASIMDFHM
},
217 {"dit", HWCAP_DIT
}, {"uscat", HWCAP_USCAT
},
218 {"ilrcpc", HWCAP_ILRCPC
}, {"flagm", HWCAP_FLAGM
},
219 {"ssbs", HWCAP_SSBS
}, {"sb", HWCAP_SB
},
220 {"paca", HWCAP_PACA
}, {"pacg", HWCAP_PACG
},
224 capability capabilities2
[] = {
225 {"dcpodp", HWCAP2_DCPODP
}, {"sve2", HWCAP2_SVE2
},
226 {"sveaes", HWCAP2_SVEAES
}, {"svepmull", HWCAP2_SVEPMULL
},
227 {"svebitperm", HWCAP2_SVEBITPERM
}, {"svesha3", HWCAP2_SVESHA3
},
228 {"svesm4", HWCAP2_SVESM4
}, {"flagm2", HWCAP2_FLAGM2
},
229 {"frint", HWCAP2_FRINT
}, {"", 0ul}
235 unsigned long hwcap2
;
238 #define CAPABILITIES_SEARCH_LOOP(hwcversion) \
239 for (int i = 0; capabilities ## hwcversion[i].cap_bit; ++i) \
240 if (strcmp(name, capabilities ## hwcversion[i].name) == 0) { \
241 caps->hwcap ## hwcversion = capabilities ## hwcversion[i].cap_bit; \
245 static Bool get_feature_from_string(const char *name, hwc *caps)
247 caps
->hwcap
= caps
->hwcap2
= 0;
248 CAPABILITIES_SEARCH_LOOP()
249 CAPABILITIES_SEARCH_LOOP(2)
253 static int go(const char* feature_name
)
256 unsigned long hwcap
= getauxval(AT_HWCAP
);
257 unsigned long hwcap2
= getauxval(AT_HWCAP2
);
259 if (!get_feature_from_string(feature_name
, &hw
))
260 return UNRECOGNISED_FEATURE
;
262 if ((hw
.hwcap
& hwcap
) || (hw
.hwcap2
& hwcap2
))
263 return FEATURE_PRESENT
;
265 return FEATURE_NOT_PRESENT
;
270 static Bool
go(const char* feature_name
)
272 // Feature not recognised (non-arm64 machine!)
273 return UNRECOGNISED_FEATURE
;
276 #endif // defined(VGA_arm64)
279 //---------------------------------------------------------------------------
281 //---------------------------------------------------------------------------
282 int main(int argc
, char **argv
)
285 fprintf(stderr
, "usage: arm64_features <feature>\n");