xref: /aosp_15_r20/external/cronet/base/cpu.h (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1*6777b538SAndroid Build Coastguard Worker // Copyright 2012 The Chromium Authors
2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file.
4*6777b538SAndroid Build Coastguard Worker 
5*6777b538SAndroid Build Coastguard Worker #ifndef BASE_CPU_H_
6*6777b538SAndroid Build Coastguard Worker #define BASE_CPU_H_
7*6777b538SAndroid Build Coastguard Worker 
8*6777b538SAndroid Build Coastguard Worker #include <cstdint>
9*6777b538SAndroid Build Coastguard Worker #include <string>
10*6777b538SAndroid Build Coastguard Worker 
11*6777b538SAndroid Build Coastguard Worker #include "base/base_export.h"
12*6777b538SAndroid Build Coastguard Worker #include "build/build_config.h"
13*6777b538SAndroid Build Coastguard Worker 
14*6777b538SAndroid Build Coastguard Worker namespace base {
15*6777b538SAndroid Build Coastguard Worker 
16*6777b538SAndroid Build Coastguard Worker #if defined(ARCH_CPU_X86_FAMILY)
17*6777b538SAndroid Build Coastguard Worker namespace internal {
18*6777b538SAndroid Build Coastguard Worker 
19*6777b538SAndroid Build Coastguard Worker struct X86ModelInfo {
20*6777b538SAndroid Build Coastguard Worker   int family;
21*6777b538SAndroid Build Coastguard Worker   int model;
22*6777b538SAndroid Build Coastguard Worker   int ext_family;
23*6777b538SAndroid Build Coastguard Worker   int ext_model;
24*6777b538SAndroid Build Coastguard Worker };
25*6777b538SAndroid Build Coastguard Worker 
26*6777b538SAndroid Build Coastguard Worker // Compute the CPU family and model based on the vendor and CPUID signature.
27*6777b538SAndroid Build Coastguard Worker BASE_EXPORT X86ModelInfo ComputeX86FamilyAndModel(const std::string& vendor,
28*6777b538SAndroid Build Coastguard Worker                                                   int signature);
29*6777b538SAndroid Build Coastguard Worker 
30*6777b538SAndroid Build Coastguard Worker }  // namespace internal
31*6777b538SAndroid Build Coastguard Worker #endif  // defined(ARCH_CPU_X86_FAMILY)
32*6777b538SAndroid Build Coastguard Worker 
33*6777b538SAndroid Build Coastguard Worker // Query information about the processor.
34*6777b538SAndroid Build Coastguard Worker class BASE_EXPORT CPU final {
35*6777b538SAndroid Build Coastguard Worker  public:
36*6777b538SAndroid Build Coastguard Worker   CPU();
37*6777b538SAndroid Build Coastguard Worker   CPU(CPU&&);
38*6777b538SAndroid Build Coastguard Worker   CPU(const CPU&) = delete;
39*6777b538SAndroid Build Coastguard Worker 
40*6777b538SAndroid Build Coastguard Worker   // Get a preallocated instance of CPU.
41*6777b538SAndroid Build Coastguard Worker   // It can be used in very early application startup or in memory allocation
42*6777b538SAndroid Build Coastguard Worker   // handlers. The instance of CPU is created without branding, see `CPU(bool
43*6777b538SAndroid Build Coastguard Worker   // requires_branding)` for details and implications.
44*6777b538SAndroid Build Coastguard Worker   //
45*6777b538SAndroid Build Coastguard Worker   // Support for various security features such as Arm's BTI and MTE uses this
46*6777b538SAndroid Build Coastguard Worker   // instance to detect CPU support. To prevent any attempt
47*6777b538SAndroid Build Coastguard Worker   // to disable a feature by attacking this data, base::ProtectedMemory is used
48*6777b538SAndroid Build Coastguard Worker   // to protected it from write accesses.
49*6777b538SAndroid Build Coastguard Worker   // Note that base::ProtectedMemory falls back to unprotected memory if the
50*6777b538SAndroid Build Coastguard Worker   // target OS is not supported.
51*6777b538SAndroid Build Coastguard Worker   static const CPU& GetInstanceNoAllocation();
52*6777b538SAndroid Build Coastguard Worker 
53*6777b538SAndroid Build Coastguard Worker   enum IntelMicroArchitecture {
54*6777b538SAndroid Build Coastguard Worker     PENTIUM = 0,
55*6777b538SAndroid Build Coastguard Worker     SSE = 1,
56*6777b538SAndroid Build Coastguard Worker     SSE2 = 2,
57*6777b538SAndroid Build Coastguard Worker     SSE3 = 3,
58*6777b538SAndroid Build Coastguard Worker     SSSE3 = 4,
59*6777b538SAndroid Build Coastguard Worker     SSE41 = 5,
60*6777b538SAndroid Build Coastguard Worker     SSE42 = 6,
61*6777b538SAndroid Build Coastguard Worker     AVX = 7,
62*6777b538SAndroid Build Coastguard Worker     AVX2 = 8,
63*6777b538SAndroid Build Coastguard Worker     FMA3 = 9,
64*6777b538SAndroid Build Coastguard Worker     MAX_INTEL_MICRO_ARCHITECTURE = 10
65*6777b538SAndroid Build Coastguard Worker   };
66*6777b538SAndroid Build Coastguard Worker 
67*6777b538SAndroid Build Coastguard Worker   // Accessors for CPU information.
vendor_name()68*6777b538SAndroid Build Coastguard Worker   std::string vendor_name() const { return cpu_vendor_; }
signature()69*6777b538SAndroid Build Coastguard Worker   int signature() const { return signature_; }
stepping()70*6777b538SAndroid Build Coastguard Worker   int stepping() const { return stepping_; }
model()71*6777b538SAndroid Build Coastguard Worker   int model() const { return model_; }
family()72*6777b538SAndroid Build Coastguard Worker   int family() const { return family_; }
type()73*6777b538SAndroid Build Coastguard Worker   int type() const { return type_; }
extended_model()74*6777b538SAndroid Build Coastguard Worker   int extended_model() const { return ext_model_; }
extended_family()75*6777b538SAndroid Build Coastguard Worker   int extended_family() const { return ext_family_; }
has_mmx()76*6777b538SAndroid Build Coastguard Worker   bool has_mmx() const { return has_mmx_; }
has_sse()77*6777b538SAndroid Build Coastguard Worker   bool has_sse() const { return has_sse_; }
has_sse2()78*6777b538SAndroid Build Coastguard Worker   bool has_sse2() const { return has_sse2_; }
has_sse3()79*6777b538SAndroid Build Coastguard Worker   bool has_sse3() const { return has_sse3_; }
has_ssse3()80*6777b538SAndroid Build Coastguard Worker   bool has_ssse3() const { return has_ssse3_; }
has_sse41()81*6777b538SAndroid Build Coastguard Worker   bool has_sse41() const { return has_sse41_; }
has_sse42()82*6777b538SAndroid Build Coastguard Worker   bool has_sse42() const { return has_sse42_; }
has_popcnt()83*6777b538SAndroid Build Coastguard Worker   bool has_popcnt() const { return has_popcnt_; }
has_avx()84*6777b538SAndroid Build Coastguard Worker   bool has_avx() const { return has_avx_; }
has_fma3()85*6777b538SAndroid Build Coastguard Worker   bool has_fma3() const { return has_fma3_; }
has_avx2()86*6777b538SAndroid Build Coastguard Worker   bool has_avx2() const { return has_avx2_; }
has_aesni()87*6777b538SAndroid Build Coastguard Worker   bool has_aesni() const { return has_aesni_; }
has_non_stop_time_stamp_counter()88*6777b538SAndroid Build Coastguard Worker   bool has_non_stop_time_stamp_counter() const {
89*6777b538SAndroid Build Coastguard Worker     return has_non_stop_time_stamp_counter_;
90*6777b538SAndroid Build Coastguard Worker   }
is_running_in_vm()91*6777b538SAndroid Build Coastguard Worker   bool is_running_in_vm() const { return is_running_in_vm_; }
92*6777b538SAndroid Build Coastguard Worker 
93*6777b538SAndroid Build Coastguard Worker #if defined(ARCH_CPU_ARM_FAMILY)
94*6777b538SAndroid Build Coastguard Worker   // The cpuinfo values for ARM cores are from the MIDR_EL1 register, a
95*6777b538SAndroid Build Coastguard Worker   // bitfield whose format is described in the core-specific manuals. E.g.,
96*6777b538SAndroid Build Coastguard Worker   // ARM Cortex-A57:
97*6777b538SAndroid Build Coastguard Worker   // https://developer.arm.com/documentation/ddi0488/h/system-control/aarch64-register-descriptions/main-id-register--el1.
implementer()98*6777b538SAndroid Build Coastguard Worker   uint8_t implementer() const { return implementer_; }
part_number()99*6777b538SAndroid Build Coastguard Worker   uint32_t part_number() const { return part_number_; }
100*6777b538SAndroid Build Coastguard Worker #endif
101*6777b538SAndroid Build Coastguard Worker 
102*6777b538SAndroid Build Coastguard Worker   // Armv8.5-A extensions for control flow and memory safety.
103*6777b538SAndroid Build Coastguard Worker #if defined(ARCH_CPU_ARM_FAMILY)
has_mte()104*6777b538SAndroid Build Coastguard Worker   bool has_mte() const { return has_mte_; }
has_bti()105*6777b538SAndroid Build Coastguard Worker   bool has_bti() const { return has_bti_; }
106*6777b538SAndroid Build Coastguard Worker #else
has_mte()107*6777b538SAndroid Build Coastguard Worker   constexpr bool has_mte() const { return false; }
has_bti()108*6777b538SAndroid Build Coastguard Worker   constexpr bool has_bti() const { return false; }
109*6777b538SAndroid Build Coastguard Worker #endif
110*6777b538SAndroid Build Coastguard Worker 
111*6777b538SAndroid Build Coastguard Worker #if defined(ARCH_CPU_X86_FAMILY)
112*6777b538SAndroid Build Coastguard Worker   // Memory protection key support for user-mode pages
has_pku()113*6777b538SAndroid Build Coastguard Worker   bool has_pku() const { return has_pku_; }
114*6777b538SAndroid Build Coastguard Worker #else
has_pku()115*6777b538SAndroid Build Coastguard Worker   constexpr bool has_pku() const { return false; }
116*6777b538SAndroid Build Coastguard Worker #endif
117*6777b538SAndroid Build Coastguard Worker 
118*6777b538SAndroid Build Coastguard Worker #if defined(ARCH_CPU_X86_FAMILY)
119*6777b538SAndroid Build Coastguard Worker   IntelMicroArchitecture GetIntelMicroArchitecture() const;
120*6777b538SAndroid Build Coastguard Worker #endif
cpu_brand()121*6777b538SAndroid Build Coastguard Worker   std::string cpu_brand() const { return cpu_brand_; }
122*6777b538SAndroid Build Coastguard Worker 
123*6777b538SAndroid Build Coastguard Worker  private:
124*6777b538SAndroid Build Coastguard Worker   // Query the processor for CPUID information.
125*6777b538SAndroid Build Coastguard Worker   void Initialize(bool requires_branding);
126*6777b538SAndroid Build Coastguard Worker   explicit CPU(bool requires_branding);
127*6777b538SAndroid Build Coastguard Worker 
128*6777b538SAndroid Build Coastguard Worker   int signature_ = 0;  // raw form of type, family, model, and stepping
129*6777b538SAndroid Build Coastguard Worker   int type_ = 0;       // process type
130*6777b538SAndroid Build Coastguard Worker   int family_ = 0;     // family of the processor
131*6777b538SAndroid Build Coastguard Worker   int model_ = 0;      // model of processor
132*6777b538SAndroid Build Coastguard Worker   int stepping_ = 0;   // processor revision number
133*6777b538SAndroid Build Coastguard Worker   int ext_model_ = 0;
134*6777b538SAndroid Build Coastguard Worker   int ext_family_ = 0;
135*6777b538SAndroid Build Coastguard Worker #if defined(ARCH_CPU_ARM_FAMILY)
136*6777b538SAndroid Build Coastguard Worker   uint32_t part_number_ = 0;  // ARM MIDR part number
137*6777b538SAndroid Build Coastguard Worker   uint8_t implementer_ = 0;   // ARM MIDR implementer identifier
138*6777b538SAndroid Build Coastguard Worker #endif
139*6777b538SAndroid Build Coastguard Worker   bool has_mmx_ = false;
140*6777b538SAndroid Build Coastguard Worker   bool has_sse_ = false;
141*6777b538SAndroid Build Coastguard Worker   bool has_sse2_ = false;
142*6777b538SAndroid Build Coastguard Worker   bool has_sse3_ = false;
143*6777b538SAndroid Build Coastguard Worker   bool has_ssse3_ = false;
144*6777b538SAndroid Build Coastguard Worker   bool has_sse41_ = false;
145*6777b538SAndroid Build Coastguard Worker   bool has_sse42_ = false;
146*6777b538SAndroid Build Coastguard Worker   bool has_popcnt_ = false;
147*6777b538SAndroid Build Coastguard Worker   bool has_avx_ = false;
148*6777b538SAndroid Build Coastguard Worker   bool has_fma3_ = false;
149*6777b538SAndroid Build Coastguard Worker   bool has_avx2_ = false;
150*6777b538SAndroid Build Coastguard Worker   bool has_aesni_ = false;
151*6777b538SAndroid Build Coastguard Worker #if defined(ARCH_CPU_ARM_FAMILY)
152*6777b538SAndroid Build Coastguard Worker   bool has_mte_ = false;  // Armv8.5-A MTE (Memory Taggging Extension)
153*6777b538SAndroid Build Coastguard Worker   bool has_bti_ = false;  // Armv8.5-A BTI (Branch Target Identification)
154*6777b538SAndroid Build Coastguard Worker #endif
155*6777b538SAndroid Build Coastguard Worker #if defined(ARCH_CPU_X86_FAMILY)
156*6777b538SAndroid Build Coastguard Worker   bool has_pku_ = false;
157*6777b538SAndroid Build Coastguard Worker #endif
158*6777b538SAndroid Build Coastguard Worker   bool has_non_stop_time_stamp_counter_ = false;
159*6777b538SAndroid Build Coastguard Worker   bool is_running_in_vm_ = false;
160*6777b538SAndroid Build Coastguard Worker 
161*6777b538SAndroid Build Coastguard Worker   // The CPUID instruction of the X86 instruction set returns the vendor name in
162*6777b538SAndroid Build Coastguard Worker   // 3 32bit registers, which make 12 characters. See "Intel® 64 and IA-32
163*6777b538SAndroid Build Coastguard Worker   // Architectures Software Developer’s Manual - Volume 2".
164*6777b538SAndroid Build Coastguard Worker   static constexpr size_t kVendorNameSize = 12;
165*6777b538SAndroid Build Coastguard Worker   char cpu_vendor_[kVendorNameSize + 1] = "unknown";
166*6777b538SAndroid Build Coastguard Worker   // The CPUID instruction of the X86 instruction set returns the brand name in
167*6777b538SAndroid Build Coastguard Worker   // 3*4 32bit registers, which make 48 characters. See "Intel® 64 and IA-32
168*6777b538SAndroid Build Coastguard Worker   // Architectures Software Developer’s Manual - Volume 2".
169*6777b538SAndroid Build Coastguard Worker   static constexpr size_t kBrandNameSize = 48;
170*6777b538SAndroid Build Coastguard Worker   char cpu_brand_[kBrandNameSize + 1] = "\0";
171*6777b538SAndroid Build Coastguard Worker };
172*6777b538SAndroid Build Coastguard Worker 
173*6777b538SAndroid Build Coastguard Worker }  // namespace base
174*6777b538SAndroid Build Coastguard Worker 
175*6777b538SAndroid Build Coastguard Worker #endif  // BASE_CPU_H_
176