blob: b7dea2eb19d947bec65c9ecb77ecbf654be51ae1 [file] [log] [blame]
/*
* Copyright (C) 2014 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "berberis/runtime_primitives/platform.h"
#include "berberis/base/config_globals.h"
#if defined(__i386__) || defined(__x86_64__)
#include <cpuid.h>
#endif
#include <cinttypes>
namespace berberis::host_platform {
namespace {
#if defined(__i386__) || defined(__x86_64__)
auto Init() {
PlatformCapabilities platform_capabilities = {};
unsigned int eax, ebx, ecx, edx;
// Technically Zen,Zen+/Zen2 AMD CPUs support BMI2 and thus PDEP/PEXT instruction, but they are
// not usable there: https://twitter.com/instlatx64/status/1322503571288559617
// That's why we need special emulated CPUID flag for these instructions.
bool use_pdep_if_present = true;
__cpuid(0, eax, ebx, ecx, edx);
if (((ebx == signature_AMD_ebx && ecx == signature_AMD_ecx && edx == signature_AMD_edx) ||
(ebx == signature_HYGON_ebx && ecx == signature_HYGON_ecx && edx == signature_HYGON_edx))) {
platform_capabilities.kIsAuthenticAMD = true;
__cpuid(1, eax, ebx, ecx, edx);
uint8_t family = (eax >> 8) & 0b1111;
if (family == 0b1111) {
family += (eax >> 20) & 0b11111111;
if (family < 0x19) {
use_pdep_if_present = false;
}
}
} else {
__cpuid(1, eax, ebx, ecx, edx);
}
platform_capabilities.kHasAES = ecx & bit_AES;
platform_capabilities.kHasAVX = ecx & bit_AVX;
platform_capabilities.kHasCLMUL = ecx & bit_PCLMUL;
platform_capabilities.kHasF16C = ecx & bit_F16C;
platform_capabilities.kHasFMA = ecx & bit_FMA;
platform_capabilities.kHasPOPCNT = ecx & bit_POPCNT;
platform_capabilities.kHasSSE3 = ecx & bit_SSE3;
platform_capabilities.kHasSSSE3 = ecx & bit_SSSE3;
platform_capabilities.kHasSSE4_1 = ecx & bit_SSE4_1;
platform_capabilities.kHasSSE4_2 = ecx & bit_SSE4_2;
__cpuid(0x80000001, eax, ebx, ecx, edx);
platform_capabilities.kHasFMA4 = ecx & bit_FMA4;
platform_capabilities.kHasLZCNT = ecx & bit_LZCNT;
platform_capabilities.kHasSSE4a = ecx & bit_SSE4a;
__cpuid_count(7, 0, eax, ebx, ecx, edx);
platform_capabilities.kHasBMI = ebx & bit_BMI;
platform_capabilities.kHasBMI2 = ebx & bit_BMI2;
platform_capabilities.kHasPDEP = ebx & bit_BMI2 && use_pdep_if_present;
platform_capabilities.kHasSHA = ebx & bit_SHA;
platform_capabilities.kHasCustomCapability = IsConfigFlagSet(kPlatformCustomCPUCapability);
return platform_capabilities;
}
#endif
} // namespace
#if defined(__i386__) || defined(__x86_64__)
const PlatformCapabilities kPlatformCapabilities = Init();
#endif
} // namespace berberis::host_platform