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