// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef BASE_CPU_H_ #define BASE_CPU_H_ #include #include #include #include "base/base_export.h" #include "base/time/time.h" #include "build/build_config.h" namespace base { #if defined(ARCH_CPU_X86_FAMILY) namespace internal { // Compute the CPU family and model based on the vendor and CPUID signature. // Returns in order: family, model, extended family, extended model. BASE_EXPORT std::tuple ComputeX86FamilyAndModel( const std::string& vendor, int signature); } // namespace internal #endif // defined(ARCH_CPU_X86_FAMILY) // Query information about the processor. class BASE_EXPORT CPU final { public: CPU(); CPU(CPU&&); CPU(const CPU&) = delete; // Construction path used in very early application startup. The difference // between this and CPU::CPU() is that this doesn't allocate any memory, the // catch is that no CPU model information is available (only features). #if defined(ARCH_CPU_ARM_FAMILY) static CPU CreateNoAllocation() { return CPU(false); } #endif enum IntelMicroArchitecture { PENTIUM, SSE, SSE2, SSE3, SSSE3, SSE41, SSE42, AVX, AVX2, MAX_INTEL_MICRO_ARCHITECTURE }; // Accessors for CPU information. const std::string& vendor_name() const { return cpu_vendor_; } int signature() const { return signature_; } int stepping() const { return stepping_; } int model() const { return model_; } int family() const { return family_; } int type() const { return type_; } int extended_model() const { return ext_model_; } int extended_family() const { return ext_family_; } bool has_mmx() const { return has_mmx_; } bool has_sse() const { return has_sse_; } bool has_sse2() const { return has_sse2_; } bool has_sse3() const { return has_sse3_; } bool has_ssse3() const { return has_ssse3_; } bool has_sse41() const { return has_sse41_; } bool has_sse42() const { return has_sse42_; } bool has_popcnt() const { return has_popcnt_; } bool has_avx() const { return has_avx_; } bool has_avx2() const { return has_avx2_; } bool has_aesni() const { return has_aesni_; } bool has_non_stop_time_stamp_counter() const { return has_non_stop_time_stamp_counter_; } bool is_running_in_vm() const { return is_running_in_vm_; } // The cpuinfo values for ARM cores are from the MIDR_EL1 register, a // bitfield whose format is described in the core-specific manuals. E.g., // ARM Cortex-A57: // https://developer.arm.com/documentation/ddi0488/h/system-control/aarch64-register-descriptions/main-id-register--el1. uint8_t implementer() const { return implementer_; } uint32_t part_number() const { return part_number_; } // Armv8.5-A extensions for control flow and memory safety. bool has_mte() const { return has_mte_; } bool has_bti() const { return has_bti_; } IntelMicroArchitecture GetIntelMicroArchitecture() const; const std::string& cpu_brand() const { return cpu_brand_; } #if defined(OS_LINUX) || defined(OS_CHROMEOS) || defined(OS_ANDROID) || \ defined(OS_AIX) enum class CoreType { kUnknown = 0, kOther, kSymmetric, kBigLittle_Little, kBigLittle_Big, kBigLittleBigger_Little, kBigLittleBigger_Big, kBigLittleBigger_Bigger, kMaxValue = kBigLittleBigger_Bigger }; // Attempts to guess the core types of individual CPU cores based on frequency // information from /sys/devices/system/cpu/cpuN/cpufreq/cpuinfo_max_freq. // Beware that it is kernel/hardware dependent whether the information from // sys is accurate. Returns a reference to a static-storage vector (leaked on // shutdown) with the guessed type for core N at index N. static const std::vector& GetGuessedCoreTypes(); struct TimeInStateEntry { CPU::CoreType core_type; // type of the cores in this cluster. uint32_t cluster_core_index; // index of the first core in the cluster. uint64_t core_frequency_khz; TimeDelta cumulative_time; }; using TimeInState = std::vector; // For each CPU core, emits the cumulative time spent in different frequency // states into the output parameter (replacing its current contents). One // entry in the output parameter is added for each cluster core index // + frequency state combination with a non-zero CPU time value. Returns false // on failure. We return the usage via an output parameter to allow reuse of // TimeInState's std::vector by the caller, e.g. to avoid allocations between // repeated calls to this method. // // NOTE: Currently only supported on Linux/Android, and only on kernels with // cpufreq-stats driver. static bool GetTimeInState(TimeInState&); // For each CPU core, emits the total cumulative wall time spent in any idle // state into the output parameter (replacing its current contents). Returns // false on failure. We return the usage via an output parameter to allow // reuse of TimeInState's std::vector by the caller, e.g. to avoid allocations // between repeated calls to this method. // // NOTE: Currently only supported on Linux/Android, and only on kernels with // cpuidle driver. using CoreIdleTimes = std::vector; static bool GetCumulativeCoreIdleTimes(CoreIdleTimes&); #endif // defined(OS_LINUX) || defined(OS_CHROMEOS) || defined(OS_ANDROID) || // defined(OS_AIX) private: // Query the processor for CPUID information. void Initialize(bool requires_branding); explicit CPU(bool requires_branding); int signature_ = 0; // raw form of type, family, model, and stepping int type_ = 0; // process type int family_ = 0; // family of the processor int model_ = 0; // model of processor int stepping_ = 0; // processor revision number int ext_model_ = 0; int ext_family_ = 0; uint32_t part_number_ = 0; // ARM MIDR part number uint8_t implementer_ = 0; // ARM MIDR implementer identifier bool has_mmx_ = false; bool has_sse_ = false; bool has_sse2_ = false; bool has_sse3_ = false; bool has_ssse3_ = false; bool has_sse41_ = false; bool has_sse42_ = false; bool has_popcnt_ = false; bool has_avx_ = false; bool has_avx2_ = false; bool has_aesni_ = false; bool has_mte_ = false; // Armv8.5-A MTE (Memory Taggging Extension) bool has_bti_ = false; // Armv8.5-A BTI (Branch Target Identification) bool has_non_stop_time_stamp_counter_ = false; bool is_running_in_vm_ = false; std::string cpu_vendor_ = "unknown"; std::string cpu_brand_; }; } // namespace base #endif // BASE_CPU_H_