• Home
  • History
  • Annotate
Name Date Size #Lines LOC

..--

.github/workflows/23-Nov-2023-3124

cmake/23-Nov-2023-5235

include/23-Nov-2023-1,457954

ndk_compat/23-Nov-2023-606319

scripts/23-Nov-2023-317249

src/23-Nov-2023-3,2412,690

test/23-Nov-2023-2,0151,577

.clang-formatD23-Nov-202351 54

.gitignoreD23-Nov-202327 53

.travis.ymlD23-Nov-20232.6 KiB122105

Android.bpD23-Nov-20238.6 KiB370349

CMakeLists.txtD23-Nov-20238.7 KiB260226

CONTRIBUTING.mdD23-Nov-2023969 2417

LICENSED23-Nov-202312.6 KiB231195

METADATAD23-Nov-2023388 2019

MODULE_LICENSE_APACHE2D23-Nov-20230

OWNERSD23-Nov-202338 43

PREUPLOAD.cfgD23-Nov-202329 32

README.mdD23-Nov-20236.9 KiB200152

TEST_MAPPINGD23-Nov-2023260 1716

WORKSPACED23-Nov-2023194 86

appveyor.ymlD23-Nov-2023600 2518

README.md

1# cpu_features [![Build Status](https://travis-ci.org/google/cpu_features.svg?branch=master)](https://travis-ci.org/google/cpu_features) [![Build status](https://ci.appveyor.com/api/projects/status/46d1owsj7n8dsylq/branch/master?svg=true)](https://ci.appveyor.com/project/gchatelet/cpu-features/branch/master)
2
3A cross-platform C library to retrieve CPU features (such as available
4instructions) at runtime.
5
6## Table of Contents
7
8- [Design Rationale](#rationale)
9- [Code samples](#codesample)
10- [Running sample code](#usagesample)
11- [What's supported](#support)
12- [Android NDK's drop in replacement](#ndk)
13- [License](#license)
14- [Build with cmake](#cmake)
15
16<a name="rationale"></a>
17## Design Rationale
18
19-   **Simple to use.** See the snippets below for examples.
20-   **Extensible.** Easy to add missing features or architectures.
21-   **Compatible with old compilers** and available on many architectures so it
22    can be used widely. To ensure that cpu_features works on as many platforms
23    as possible, we implemented it in a highly portable version of C: C99.
24-   **Sandbox-compatible.** The library uses a variety of strategies to cope
25    with sandboxed environments or when `cpuid` is unavailable. This is useful
26    when running integration tests in hermetic environments.
27-   **Thread safe, no memory allocation, and raises no exceptions.**
28    cpu_features is suitable for implementing fundamental libc functions like
29    `malloc`, `memcpy`, and `memcmp`.
30-   **Unit tested.**
31
32<a name="codesample"></a>
33## Code samples
34
35**Note:** For C++ code, the library functions are defined in the `CpuFeatures` namespace.
36
37### Checking features at runtime
38
39Here's a simple example that executes a codepath if the CPU supports both the
40AES and the SSE4.2 instruction sets:
41
42```c
43#include "cpuinfo_x86.h"
44
45// For C++, add `using namespace CpuFeatures;`
46static const X86Features features = GetX86Info().features;
47
48void Compute(void) {
49  if (features.aes && features.sse4_2) {
50    // Run optimized code.
51  } else {
52    // Run standard code.
53  }
54}
55```
56
57### Caching for faster evaluation of complex checks
58
59If you wish, you can read all the features at once into a global variable, and
60then query for the specific features you care about. Below, we store all the ARM
61features and then check whether AES and NEON are supported.
62
63```c
64#include <stdbool.h>
65#include "cpuinfo_arm.h"
66
67// For C++, add `using namespace CpuFeatures;`
68static const ArmFeatures features = GetArmInfo().features;
69static const bool has_aes_and_neon = features.aes && features.neon;
70
71// use has_aes_and_neon.
72```
73
74This is a good approach to take if you're checking for combinations of features
75when using a compiler that is slow to extract individual bits from bit-packed
76structures.
77
78### Checking compile time flags
79
80The following code determines whether the compiler was told to use the AVX
81instruction set (e.g., `g++ -mavx`) and sets `has_avx` accordingly.
82
83```c
84#include <stdbool.h>
85#include "cpuinfo_x86.h"
86
87// For C++, add `using namespace CpuFeatures;`
88static const X86Features features = GetX86Info().features;
89static const bool has_avx = CPU_FEATURES_COMPILED_X86_AVX || features.avx;
90
91// use has_avx.
92```
93
94`CPU_FEATURES_COMPILED_X86_AVX` is set to 1 if the compiler was instructed to
95use AVX and 0 otherwise, combining compile time and runtime knowledge.
96
97### Rejecting poor hardware implementations based on microarchitecture
98
99On x86, the first incarnation of a feature in a microarchitecture might not be
100the most efficient (e.g. AVX on Sandy Bridge). We provide a function to retrieve
101the underlying microarchitecture so you can decide whether to use it.
102
103Below, `has_fast_avx` is set to 1 if the CPU supports the AVX instruction
104set&mdash;but only if it's not Sandy Bridge.
105
106```c
107#include <stdbool.h>
108#include "cpuinfo_x86.h"
109
110// For C++, add `using namespace CpuFeatures;`
111static const X86Info info = GetX86Info();
112static const X86Microarchitecture uarch = GetX86Microarchitecture(&info);
113static const bool has_fast_avx = info.features.avx && uarch != INTEL_SNB;
114
115// use has_fast_avx.
116```
117
118This feature is currently available only for x86 microarchitectures.
119
120<a name="usagesample"></a>
121### Running sample code
122
123Building `cpu_features` (check [quickstart](#quickstart) below) brings a small executable to test the library.
124
125```shell
126 % ./build/list_cpu_features
127arch            : x86
128brand           :        Intel(R) Xeon(R) CPU E5-1650 0 @ 3.20GHz
129family          :   6 (0x06)
130model           :  45 (0x2D)
131stepping        :   7 (0x07)
132uarch           : INTEL_SNB
133flags           : aes,avx,cx16,smx,sse4_1,sse4_2,ssse3
134```
135
136```shell
137% ./build/list_cpu_features --json
138{"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"]}
139```
140
141<a name="support"></a>
142## What's supported
143
144|         | x86³ |   ARM   | AArch64 |  MIPS⁴ |  POWER  |
145|---------|:----:|:-------:|:-------:|:------:|:-------:|
146| Android | yes² |   yes¹  |   yes¹  |  yes¹  |   N/A   |
147| iOS     |  N/A | not yet | not yet |   N/A  |   N/A   |
148| Linux   | yes² |   yes¹  |   yes¹  |  yes¹  |   yes¹  |
149| MacOs   | yes² |   N/A   | not yet |   N/A  |    no   |
150| Windows | yes² | not yet | not yet |   N/A  |   N/A   |
151
1521.  **Features revealed from Linux.** We gather data from several sources
153    depending on availability:
154    +   from glibc's
155        [getauxval](https://www.gnu.org/software/libc/manual/html_node/Auxiliary-Vector.html)
156    +   by parsing `/proc/self/auxv`
157    +   by parsing `/proc/cpuinfo`
1582.  **Features revealed from CPU.** features are retrieved by using the `cpuid`
159    instruction.
1603.  **Microarchitecture detection.** On x86 some features are not always
161    implemented efficiently in hardware (e.g. AVX on Sandybridge). Exposing the
162    microarchitecture allows the client to reject particular microarchitectures.
1634.  All flavors of Mips are supported, little and big endian as well as 32/64
164    bits.
165
166<a name="ndk"></a>
167## Android NDK's drop in replacement
168
169[cpu_features](https://github.com/google/cpu_features) is now officially
170supporting Android and offers a drop in replacement of for the NDK's [cpu-features.h](https://android.googlesource.com/platform/ndk/+/master/sources/android/cpufeatures/cpu-features.h)
171, see [ndk_compat](ndk_compat) folder for details.
172
173<a name="license"></a>
174## License
175
176The cpu_features library is licensed under the terms of the Apache license.
177See [LICENSE](LICENSE) for more information.
178
179<a name="cmake"></a>
180## Build with CMake
181
182Please check the [CMake build instructions](cmake/README.md).
183
184<a name="quickstart"></a>
185### Quickstart with `Ninja`
186
187 - build `list_cpu_features`
188```
189    cmake -B/tmp/cpu_features -H. -GNinja -DCMAKE_BUILD_TYPE=Release
190    ninja -C/tmp/cpu_features
191    /tmp/cpu_features/list_cpu_features --json
192```
193
194 - run tests
195```
196    cmake -B/tmp/cpu_features -H. -GNinja -DBUILD_TESTING=ON
197    ninja -C/tmp/cpu_features
198    ninja -C/tmp/cpu_features test
199```
200