blob: 8cd701eb413416cc8f8bf50392a73022a04c595b [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).
Artem Titovf0671922021-07-27 12:40:17 +020033// Currently only XCR0 is defined by Intel so `xcr` should always be zero.
Zhaoliang Ma72e43212020-08-17 17:13:41 +080034static 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.
Gustaf Ullberg8b8d0b52022-04-29 08:54:29 +020096 // Compiling with MSVC and /arch:AVX2 surprisingly generates BMI2
97 // instructions (see crbug.com/1315519).
98 return (cpu_info[2] & 0x10000000) != 0 /* AVX */ &&
Zhaoliang Ma72e43212020-08-17 17:13:41 +080099 (cpu_info[2] & 0x04000000) != 0 /* XSAVE */ &&
100 (cpu_info[2] & 0x08000000) != 0 /* OSXSAVE */ &&
101 (xgetbv(0) & 0x00000006) == 6 /* XSAVE enabled by kernel */ &&
Gustaf Ullberg8b8d0b52022-04-29 08:54:29 +0200102 (cpu_info7[1] & 0x00000020) != 0 /* AVX2 */ &&
103 (cpu_info7[1] & 0x00000100) != 0 /* BMI2 */;
Zhaoliang Ma72e43212020-08-17 17:13:41 +0800104 }
105#endif // WEBRTC_ENABLE_AVX2
Henrik Lundincb4b0a62023-01-31 15:49:55 +0000106 if (feature == kFMA3) {
107 return 0 != (cpu_info[2] & 0x00001000);
108 }
niklase@google.com470e71d2011-07-07 08:21:25 +0000109 return 0;
110}
111#else
112// Default to straight C for other platforms.
Mirko Bonadeibef022b2020-09-06 16:07:15 +0200113int GetCPUInfo(CPUFeature feature) {
niklase@google.com470e71d2011-07-07 08:21:25 +0000114 (void)feature;
115 return 0;
116}
117#endif
118
Mirko Bonadeibef022b2020-09-06 16:07:15 +0200119} // namespace webrtc