blob: 0f81212894255c319b4a29a2292243577c0504ab [file] [log] [blame]
niklase@google.com470e71d2011-07-07 08:21:25 +00001/*
2 * Copyright (c) 2011 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
ajm@google.comce7c2a22011-08-04 01:50:00 +000011// Parts of this file derived from Chromium's base/cpu.cc.
12
Niels Möllera12c42a2018-07-25 16:05:48 +020013#include "rtc_base/system/arch.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020014#include "system_wrappers/include/cpu_features_wrapper.h"
Per Åhgrena43178c2020-09-25 12:02:32 +020015#include "system_wrappers/include/field_trial.h"
niklase@google.com470e71d2011-07-07 08:21:25 +000016
phoglund@webrtc.orgb15d2852012-11-21 08:02:57 +000017#if defined(WEBRTC_ARCH_X86_FAMILY) && defined(_MSC_VER)
ajm@google.comce7c2a22011-08-04 01:50:00 +000018#include <intrin.h>
19#endif
phoglund@webrtc.orgb15d2852012-11-21 08:02:57 +000020
Mirko Bonadeibef022b2020-09-06 16:07:15 +020021namespace webrtc {
22
niklase@google.com470e71d2011-07-07 08:21:25 +000023// No CPU feature is available => straight C path.
24int GetCPUInfoNoASM(CPUFeature feature) {
25 (void)feature;
26 return 0;
27}
28
ajm@google.comce7c2a22011-08-04 01:50:00 +000029#if defined(WEBRTC_ARCH_X86_FAMILY)
Zhaoliang Ma72e43212020-08-17 17:13:41 +080030
31#if defined(WEBRTC_ENABLE_AVX2)
32// xgetbv returns the value of an Intel Extended Control Register (XCR).
33// Currently only XCR0 is defined by Intel so |xcr| should always be zero.
34static uint64_t xgetbv(uint32_t xcr) {
35#if defined(_MSC_VER)
36 return _xgetbv(xcr);
37#else
38 uint32_t eax, edx;
39
40 __asm__ volatile("xgetbv" : "=a"(eax), "=d"(edx) : "c"(xcr));
41 return (static_cast<uint64_t>(edx) << 32) | eax;
42#endif // _MSC_VER
43}
44#endif // WEBRTC_ENABLE_AVX2
45
ajm@google.comce7c2a22011-08-04 01:50:00 +000046#ifndef _MSC_VER
niklase@google.com470e71d2011-07-07 08:21:25 +000047// Intrinsic for "cpuid".
48#if defined(__pic__) && defined(__i386__)
ajm@google.comce7c2a22011-08-04 01:50:00 +000049static inline void __cpuid(int cpu_info[4], int info_type) {
phoglund@webrtc.orgb15d2852012-11-21 08:02:57 +000050 __asm__ volatile(
Karl Wiberg79eb1d92017-11-08 12:26:07 +010051 "mov %%ebx, %%edi\n"
52 "cpuid\n"
53 "xchg %%edi, %%ebx\n"
54 : "=a"(cpu_info[0]), "=D"(cpu_info[1]), "=c"(cpu_info[2]),
55 "=d"(cpu_info[3])
56 : "a"(info_type));
niklase@google.com470e71d2011-07-07 08:21:25 +000057}
ajm@google.comce7c2a22011-08-04 01:50:00 +000058#else
59static inline void __cpuid(int cpu_info[4], int info_type) {
Karl Wiberg79eb1d92017-11-08 12:26:07 +010060 __asm__ volatile("cpuid\n"
61 : "=a"(cpu_info[0]), "=b"(cpu_info[1]), "=c"(cpu_info[2]),
62 "=d"(cpu_info[3])
Zhaoliang Ma72e43212020-08-17 17:13:41 +080063 : "a"(info_type), "c"(0));
niklase@google.com470e71d2011-07-07 08:21:25 +000064}
65#endif
ajm@google.comce7c2a22011-08-04 01:50:00 +000066#endif // _MSC_VER
67#endif // WEBRTC_ARCH_X86_FAMILY
niklase@google.com470e71d2011-07-07 08:21:25 +000068
ajm@google.comce7c2a22011-08-04 01:50:00 +000069#if defined(WEBRTC_ARCH_X86_FAMILY)
niklase@google.com470e71d2011-07-07 08:21:25 +000070// Actual feature detection for x86.
Mirko Bonadeibef022b2020-09-06 16:07:15 +020071int GetCPUInfo(CPUFeature feature) {
niklase@google.com470e71d2011-07-07 08:21:25 +000072 int cpu_info[4];
ajm@google.comce7c2a22011-08-04 01:50:00 +000073 __cpuid(cpu_info, 1);
niklase@google.com470e71d2011-07-07 08:21:25 +000074 if (feature == kSSE2) {
75 return 0 != (cpu_info[3] & 0x04000000);
76 }
77 if (feature == kSSE3) {
78 return 0 != (cpu_info[2] & 0x00000001);
79 }
Zhaoliang Ma72e43212020-08-17 17:13:41 +080080#if defined(WEBRTC_ENABLE_AVX2)
Per Åhgrena43178c2020-09-25 12:02:32 +020081 if (feature == kAVX2 &&
82 !webrtc::field_trial::IsEnabled("WebRTC-Avx2SupportKillSwitch")) {
Zhaoliang Ma72e43212020-08-17 17:13:41 +080083 int cpu_info7[4];
84 __cpuid(cpu_info7, 0);
85 int num_ids = cpu_info7[0];
86 if (num_ids < 7) {
87 return 0;
88 }
89 // Interpret CPU feature information.
90 __cpuid(cpu_info7, 7);
91
92 // AVX instructions can be used when
93 // a) AVX are supported by the CPU,
94 // b) XSAVE is supported by the CPU,
95 // c) XSAVE is enabled by the kernel.
96 // See http://software.intel.com/en-us/blogs/2011/04/14/is-avx-enabled
97 // AVX2 support needs (avx_support && (cpu_info7[1] & 0x00000020) != 0;).
98 return (cpu_info[2] & 0x10000000) != 0 &&
99 (cpu_info[2] & 0x04000000) != 0 /* XSAVE */ &&
100 (cpu_info[2] & 0x08000000) != 0 /* OSXSAVE */ &&
101 (xgetbv(0) & 0x00000006) == 6 /* XSAVE enabled by kernel */ &&
102 (cpu_info7[1] & 0x00000020) != 0;
103 }
104#endif // WEBRTC_ENABLE_AVX2
niklase@google.com470e71d2011-07-07 08:21:25 +0000105 return 0;
106}
107#else
108// Default to straight C for other platforms.
Mirko Bonadeibef022b2020-09-06 16:07:15 +0200109int GetCPUInfo(CPUFeature feature) {
niklase@google.com470e71d2011-07-07 08:21:25 +0000110 (void)feature;
111 return 0;
112}
113#endif
114
Mirko Bonadeibef022b2020-09-06 16:07:15 +0200115} // namespace webrtc