Bug: 128608941

Clone this repo:
  1. ad2c0b2 Add janitors to the OWNERS file by Sadaf Ebrahimi · 5 weeks ago main master
  2. eca53ba Merge v0.9.0 into `main`. am: 3de70d0029 am: f66f3d320a am: d64300359d am: cc2079932d am: 085bd65bfb by Roland Levillain · 1 year, 4 months ago android14-qpr3-release android14-qpr3-s10-release android14-qpr3-s11-release android14-qpr3-s12-release android14-qpr3-s13-release android14-qpr3-s14-release android14-qpr3-s15-release android14-qpr3-s2-release android14-qpr3-s3-release android14-qpr3-s4-release android14-qpr3-s5-release android14-qpr3-s6-release android14-qpr3-s7-release android14-qpr3-s8-release android14-qpr3-s9-release android15-automotiveos-dev android15-platform-release android15-prebuilt-test android15-qpr1-release android15-qpr1-s3-release android15-qpr1-s4-release android15-qpr1-s5-release android15-release android15-s1-release android15-security-release android15-tests-dev android15-tests-release aml_cfg_351010000 aml_hef_350921160 aml_hef_351016140 aml_hef_351120040 aml_rkp_350910000 aml_rkp_351011000 aml_tz6_351010000 android-14.0.0_r50 android-14.0.0_r51 android-14.0.0_r52 android-14.0.0_r53 android-14.0.0_r54 android-14.0.0_r55 android-14.0.0_r56 android-14.0.0_r57 android-14.0.0_r58 android-14.0.0_r59 android-14.0.0_r60 android-14.0.0_r61 android-14.0.0_r67 android-14.0.0_r68 android-14.0.0_r69 android-14.0.0_r70 android-14.0.0_r71 android-14.0.0_r72 android-14.0.0_r73 android-14.0.0_r75 android-15.0.0_r1 android-15.0.0_r10 android-15.0.0_r11 android-15.0.0_r12 android-15.0.0_r13 android-15.0.0_r2 android-15.0.0_r3 android-15.0.0_r4 android-15.0.0_r5 android-15.0.0_r6 android-15.0.0_r7 android-15.0.0_r8 android-15.0.0_r9 android-cts-15.0_r1 android-cts-15.0_r2 android-platform-15.0.0_r1 android-platform-15.0.0_r2 android-platform-15.0.0_r3 android-platform-15.0.0_r4 android-security-15.0.0_r1 android-security-15.0.0_r2 android-security-15.0.0_r3 android-security-15.0.0_r4 android-vts-15.0_r1 android-vts-15.0_r2
  3. 085bd65 Merge v0.9.0 into `main`. am: 3de70d0029 am: f66f3d320a am: d64300359d am: cc2079932d by Roland Levillain · 1 year, 4 months ago
  4. cc20799 Merge v0.9.0 into `main`. am: 3de70d0029 am: f66f3d320a am: d64300359d by Roland Levillain · 1 year, 4 months ago
  5. d643003 Merge v0.9.0 into `main`. am: 3de70d0029 am: f66f3d320a by Roland Levillain · 1 year, 4 months ago

cpu_features

A cross-platform C library to retrieve CPU features (such as available instructions) at runtime.

GitHub-CI Status

LinuxFreeBSDMacOSWindows
amd64CMake
Bazel
CMake
Bazel
CMake
Bazel
CMake
Bazel
AArch64CMake
Bazel
CMake
Bazel
CMake
Bazel
CMake
Bazel
ARMCMake
Bazel
CMake
Bazel
CMake
Bazel
CMake
Bazel
MIPSCMake
Bazel
CMake
Bazel
CMake
Bazel
CMake
Bazel
POWERCMake
Bazel
CMake
Bazel
CMake
Bazel
CMake
Bazel
RISCVCMake
Bazel
CMake
Bazel
CMake
Bazel
CMake
Bazel
LOONGARCHCMake
Bazel
CMake
Bazel
CMake
Bazel
CMake
Bazel
s390xCMake
Bazel
CMake
Bazel
CMake
Bazel
CMake
Bazel

Table of Contents

Design Rationale

  • Simple to use. See the snippets below for examples.
  • Extensible. Easy to add missing features or architectures.
  • Compatible with old compilers and available on many architectures so it can be used widely. To ensure that cpu_features works on as many platforms as possible, we implemented it in a highly portable version of C: C99.
  • Sandbox-compatible. The library uses a variety of strategies to cope with sandboxed environments or when cpuid is unavailable. This is useful when running integration tests in hermetic environments.
  • Thread safe, no memory allocation, and raises no exceptions. cpu_features is suitable for implementing fundamental libc functions like malloc, memcpy, and memcmp.
  • Unit tested.

Code samples

Note: For C++ code, the library functions are defined in the cpu_features namespace.

Checking features at runtime

Here's a simple example that executes a codepath if the CPU supports both the AES and the SSE4.2 instruction sets:

#include "cpuinfo_x86.h"

// For C++, add `using namespace cpu_features;`
static const X86Features features = GetX86Info().features;

void Compute(void) {
  if (features.aes && features.sse4_2) {
    // Run optimized code.
  } else {
    // Run standard code.
  }
}

Caching for faster evaluation of complex checks

If you wish, you can read all the features at once into a global variable, and then query for the specific features you care about. Below, we store all the ARM features and then check whether AES and NEON are supported.

#include <stdbool.h>
#include "cpuinfo_arm.h"

// For C++, add `using namespace cpu_features;`
static const ArmFeatures features = GetArmInfo().features;
static const bool has_aes_and_neon = features.aes && features.neon;

// use has_aes_and_neon.

This is a good approach to take if you're checking for combinations of features when using a compiler that is slow to extract individual bits from bit-packed structures.

Checking compile time flags

The following code determines whether the compiler was told to use the AVX instruction set (e.g., g++ -mavx) and sets has_avx accordingly.

#include <stdbool.h>
#include "cpuinfo_x86.h"

// For C++, add `using namespace cpu_features;`
static const X86Features features = GetX86Info().features;
static const bool has_avx = CPU_FEATURES_COMPILED_X86_AVX || features.avx;

// use has_avx.

CPU_FEATURES_COMPILED_X86_AVX is set to 1 if the compiler was instructed to use AVX and 0 otherwise, combining compile time and runtime knowledge.

Rejecting poor hardware implementations based on microarchitecture

On x86, the first incarnation of a feature in a microarchitecture might not be the most efficient (e.g. AVX on Sandy Bridge). We provide a function to retrieve the underlying microarchitecture so you can decide whether to use it.

Below, has_fast_avx is set to 1 if the CPU supports the AVX instruction set—but only if it's not Sandy Bridge.

#include <stdbool.h>
#include "cpuinfo_x86.h"

// For C++, add `using namespace cpu_features;`
static const X86Info info = GetX86Info();
static const X86Microarchitecture uarch = GetX86Microarchitecture(&info);
static const bool has_fast_avx = info.features.avx && uarch != INTEL_SNB;

// use has_fast_avx.

This feature is currently available only for x86 microarchitectures.

Running sample code

Building cpu_features (check quickstart below) brings a small executable to test the library.

 % ./build/list_cpu_features
arch            : x86
brand           :        Intel(R) Xeon(R) CPU E5-1650 0 @ 3.20GHz
family          :   6 (0x06)
model           :  45 (0x2D)
stepping        :   7 (0x07)
uarch           : INTEL_SNB
flags           : aes,avx,cx16,smx,sse4_1,sse4_2,ssse3
% ./build/list_cpu_features --json
{"arch":"x86","brand":"       Intel(R) Xeon(R) CPU E5-1650 0 @ 3.20GHz","family":6,"model":45,"stepping":7,"uarch":"INTEL_SNB","flags":["aes","avx","cx16","smx","sse4_1","sse4_2","ssse3"]}

What's supported

x86³AArch64ARMMIPS⁴POWERRISCVLoongarchs390x
Linuxyes²yes¹yes¹yes¹yes¹yes¹yes¹yes¹
FreeBSDyes²not yetnot yetnot yetnot yetN/Anot yetnot yet
MacOsyes²yes⁵N/AN/AN/AN/AN/AN/A
Windowsyes²not yetnot yetN/AN/AN/AN/AN/A
Androidyes²yes¹yes¹yes¹N/AN/AN/AN/A
iOSN/Anot yetnot yetN/AN/AN/AN/AN/A
  1. Features revealed from Linux. We gather data from several sources depending on availability:
    • from glibc's getauxval
    • by parsing /proc/self/auxv
    • by parsing /proc/cpuinfo
  2. Features revealed from CPU. features are retrieved by using the cpuid instruction.
  3. Microarchitecture detection. On x86 some features are not always implemented efficiently in hardware (e.g. AVX on Sandybridge). Exposing the microarchitecture allows the client to reject particular microarchitectures.
  4. All flavors of Mips are supported, little and big endian as well as 32/64 bits.
  5. Features revealed from sysctl. features are retrieved by the sysctl instruction.

Android NDK's drop in replacement

cpu_features is now officially supporting Android and offers a drop in replacement of for the NDK's cpu-features.h , see ndk_compat folder for details.

License

The cpu_features library is licensed under the terms of the Apache license. See LICENSE for more information.

Build with CMake

Please check the CMake build instructions.

Quickstart

  • Run list_cpu_features

    cmake -S. -Bbuild -DBUILD_TESTING=OFF -DCMAKE_BUILD_TYPE=Release
    cmake --build build --config Release -j
    ./build/list_cpu_features --json
    

    Note: Use --target ALL_BUILD on the second line for Visual Studio and XCode.

  • run tests

    cmake -S. -Bbuild -DBUILD_TESTING=ON -DCMAKE_BUILD_TYPE=Debug
    cmake --build build --config Debug -j
    cmake --build build --config Debug --target test
    

    Note: Use --target RUN_TESTS on the last line for Visual Studio and --target RUN_TEST for XCode.

  • install cpu_features

    cmake --build build --config Release --target install -v
    

    Note: Use --target INSTALL for Visual Studio.

    Note: When using Makefile or XCode generator, you can use DESTDIR to install on a local repository.
    e.g.

    cmake --build build --config Release --target install -v -- DESTDIR=install
    

Community bindings

Links provided here are not affiliated with Google but are kindly provided by the OSS Community.

Send PR to showcase your wrapper here