2017-05-03 23:42:29 +08:00
|
|
|
//===-- Host.cpp - Implement OS Host Concept --------------------*- C++ -*-===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2017-05-03 23:42:29 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implements the operating system Host concept.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "llvm/Support/Host.h"
|
|
|
|
#include "llvm/ADT/SmallSet.h"
|
|
|
|
#include "llvm/ADT/SmallVector.h"
|
2020-03-12 06:30:04 +08:00
|
|
|
#include "llvm/ADT/StringMap.h"
|
2017-05-03 23:42:29 +08:00
|
|
|
#include "llvm/ADT/StringRef.h"
|
|
|
|
#include "llvm/ADT/StringSwitch.h"
|
|
|
|
#include "llvm/ADT/Triple.h"
|
2018-04-30 22:59:11 +08:00
|
|
|
#include "llvm/Config/llvm-config.h"
|
2017-05-03 23:42:29 +08:00
|
|
|
#include "llvm/Support/Debug.h"
|
|
|
|
#include "llvm/Support/FileSystem.h"
|
|
|
|
#include "llvm/Support/MemoryBuffer.h"
|
2020-06-11 12:25:15 +08:00
|
|
|
#include "llvm/Support/X86TargetParser.h"
|
2017-05-03 23:42:29 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
#include <assert.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
// Include the platform-specific parts of this class.
|
|
|
|
#ifdef LLVM_ON_UNIX
|
|
|
|
#include "Unix/Host.inc"
|
2020-04-17 23:43:02 +08:00
|
|
|
#include <sched.h>
|
2017-05-03 23:42:29 +08:00
|
|
|
#endif
|
2018-04-29 08:45:03 +08:00
|
|
|
#ifdef _WIN32
|
2017-05-03 23:42:29 +08:00
|
|
|
#include "Windows/Host.inc"
|
|
|
|
#endif
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
#include <intrin.h>
|
|
|
|
#endif
|
2019-10-31 03:50:04 +08:00
|
|
|
#if defined(__APPLE__) && (!defined(__x86_64__))
|
2017-05-03 23:42:29 +08:00
|
|
|
#include <mach/host_info.h>
|
|
|
|
#include <mach/mach.h>
|
|
|
|
#include <mach/mach_host.h>
|
|
|
|
#include <mach/machine.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define DEBUG_TYPE "host-detection"
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// Implementations of the CPU detection routines
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
static std::unique_ptr<llvm::MemoryBuffer>
|
|
|
|
LLVM_ATTRIBUTE_UNUSED getProcCpuinfoContent() {
|
|
|
|
llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> Text =
|
|
|
|
llvm::MemoryBuffer::getFileAsStream("/proc/cpuinfo");
|
|
|
|
if (std::error_code EC = Text.getError()) {
|
|
|
|
llvm::errs() << "Can't read "
|
|
|
|
<< "/proc/cpuinfo: " << EC.message() << "\n";
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
return std::move(*Text);
|
|
|
|
}
|
|
|
|
|
2018-03-08 01:53:16 +08:00
|
|
|
StringRef sys::detail::getHostCPUNameForPowerPC(StringRef ProcCpuinfoContent) {
|
2017-05-03 23:42:29 +08:00
|
|
|
// Access to the Processor Version Register (PVR) on PowerPC is privileged,
|
|
|
|
// and so we must use an operating-system interface to determine the current
|
|
|
|
// processor type. On Linux, this is exposed through the /proc/cpuinfo file.
|
|
|
|
const char *generic = "generic";
|
|
|
|
|
|
|
|
// The cpu line is second (after the 'processor: 0' line), so if this
|
|
|
|
// buffer is too small then something has changed (or is wrong).
|
|
|
|
StringRef::const_iterator CPUInfoStart = ProcCpuinfoContent.begin();
|
|
|
|
StringRef::const_iterator CPUInfoEnd = ProcCpuinfoContent.end();
|
|
|
|
|
|
|
|
StringRef::const_iterator CIP = CPUInfoStart;
|
|
|
|
|
|
|
|
StringRef::const_iterator CPUStart = 0;
|
|
|
|
size_t CPULen = 0;
|
|
|
|
|
|
|
|
// We need to find the first line which starts with cpu, spaces, and a colon.
|
|
|
|
// After the colon, there may be some additional spaces and then the cpu type.
|
|
|
|
while (CIP < CPUInfoEnd && CPUStart == 0) {
|
|
|
|
if (CIP < CPUInfoEnd && *CIP == '\n')
|
|
|
|
++CIP;
|
|
|
|
|
|
|
|
if (CIP < CPUInfoEnd && *CIP == 'c') {
|
|
|
|
++CIP;
|
|
|
|
if (CIP < CPUInfoEnd && *CIP == 'p') {
|
|
|
|
++CIP;
|
|
|
|
if (CIP < CPUInfoEnd && *CIP == 'u') {
|
|
|
|
++CIP;
|
|
|
|
while (CIP < CPUInfoEnd && (*CIP == ' ' || *CIP == '\t'))
|
|
|
|
++CIP;
|
|
|
|
|
|
|
|
if (CIP < CPUInfoEnd && *CIP == ':') {
|
|
|
|
++CIP;
|
|
|
|
while (CIP < CPUInfoEnd && (*CIP == ' ' || *CIP == '\t'))
|
|
|
|
++CIP;
|
|
|
|
|
|
|
|
if (CIP < CPUInfoEnd) {
|
|
|
|
CPUStart = CIP;
|
|
|
|
while (CIP < CPUInfoEnd && (*CIP != ' ' && *CIP != '\t' &&
|
|
|
|
*CIP != ',' && *CIP != '\n'))
|
|
|
|
++CIP;
|
|
|
|
CPULen = CIP - CPUStart;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (CPUStart == 0)
|
|
|
|
while (CIP < CPUInfoEnd && *CIP != '\n')
|
|
|
|
++CIP;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (CPUStart == 0)
|
|
|
|
return generic;
|
|
|
|
|
|
|
|
return StringSwitch<const char *>(StringRef(CPUStart, CPULen))
|
|
|
|
.Case("604e", "604e")
|
|
|
|
.Case("604", "604")
|
|
|
|
.Case("7400", "7400")
|
|
|
|
.Case("7410", "7400")
|
|
|
|
.Case("7447", "7400")
|
|
|
|
.Case("7455", "7450")
|
|
|
|
.Case("G4", "g4")
|
|
|
|
.Case("POWER4", "970")
|
|
|
|
.Case("PPC970FX", "970")
|
|
|
|
.Case("PPC970MP", "970")
|
|
|
|
.Case("G5", "g5")
|
|
|
|
.Case("POWER5", "g5")
|
|
|
|
.Case("A2", "a2")
|
|
|
|
.Case("POWER6", "pwr6")
|
|
|
|
.Case("POWER7", "pwr7")
|
|
|
|
.Case("POWER8", "pwr8")
|
|
|
|
.Case("POWER8E", "pwr8")
|
|
|
|
.Case("POWER8NVL", "pwr8")
|
|
|
|
.Case("POWER9", "pwr9")
|
[PowerPC] Add support for -mcpu=pwr10 in both clang and llvm
Summary:
This patch simply adds support for the new CPU in anticipation of
Power10. There isn't really any functionality added so there are no
associated test cases at this time.
Reviewers: stefanp, nemanjai, amyk, hfinkel, power-llvm-team, #powerpc
Reviewed By: stefanp, nemanjai, amyk, #powerpc
Subscribers: NeHuang, steven.zhang, hiraditya, llvm-commits, wuzish, shchenz, cfe-commits, kbarton, echristo
Tags: #clang, #powerpc, #llvm
Differential Revision: https://reviews.llvm.org/D80020
2020-05-27 22:50:14 +08:00
|
|
|
.Case("POWER10", "pwr10")
|
2019-11-28 02:50:23 +08:00
|
|
|
// FIXME: If we get a simulator or machine with the capabilities of
|
|
|
|
// mcpu=future, we should revisit this and add the name reported by the
|
|
|
|
// simulator/machine.
|
2017-05-03 23:42:29 +08:00
|
|
|
.Default(generic);
|
|
|
|
}
|
|
|
|
|
2018-03-08 01:53:16 +08:00
|
|
|
StringRef sys::detail::getHostCPUNameForARM(StringRef ProcCpuinfoContent) {
|
2017-05-03 23:42:29 +08:00
|
|
|
// The cpuid register on arm is not accessible from user space. On Linux,
|
|
|
|
// it is exposed through the /proc/cpuinfo file.
|
|
|
|
|
|
|
|
// Read 32 lines from /proc/cpuinfo, which should contain the CPU part line
|
|
|
|
// in all cases.
|
|
|
|
SmallVector<StringRef, 32> Lines;
|
|
|
|
ProcCpuinfoContent.split(Lines, "\n");
|
|
|
|
|
|
|
|
// Look for the CPU implementer line.
|
|
|
|
StringRef Implementer;
|
|
|
|
StringRef Hardware;
|
|
|
|
for (unsigned I = 0, E = Lines.size(); I != E; ++I) {
|
|
|
|
if (Lines[I].startswith("CPU implementer"))
|
|
|
|
Implementer = Lines[I].substr(15).ltrim("\t :");
|
|
|
|
if (Lines[I].startswith("Hardware"))
|
|
|
|
Hardware = Lines[I].substr(8).ltrim("\t :");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Implementer == "0x41") { // ARM Ltd.
|
|
|
|
// MSM8992/8994 may give cpu part for the core that the kernel is running on,
|
|
|
|
// which is undeterministic and wrong. Always return cortex-a53 for these SoC.
|
|
|
|
if (Hardware.endswith("MSM8994") || Hardware.endswith("MSM8996"))
|
|
|
|
return "cortex-a53";
|
|
|
|
|
|
|
|
|
|
|
|
// Look for the CPU part line.
|
|
|
|
for (unsigned I = 0, E = Lines.size(); I != E; ++I)
|
|
|
|
if (Lines[I].startswith("CPU part"))
|
|
|
|
// The CPU part is a 3 digit hexadecimal number with a 0x prefix. The
|
|
|
|
// values correspond to the "Part number" in the CP15/c0 register. The
|
|
|
|
// contents are specified in the various processor manuals.
|
2020-02-12 00:57:25 +08:00
|
|
|
// This corresponds to the Main ID Register in Technical Reference Manuals.
|
|
|
|
// and is used in programs like sys-utils
|
2017-05-03 23:42:29 +08:00
|
|
|
return StringSwitch<const char *>(Lines[I].substr(8).ltrim("\t :"))
|
|
|
|
.Case("0x926", "arm926ej-s")
|
|
|
|
.Case("0xb02", "mpcore")
|
|
|
|
.Case("0xb36", "arm1136j-s")
|
|
|
|
.Case("0xb56", "arm1156t2-s")
|
|
|
|
.Case("0xb76", "arm1176jz-s")
|
|
|
|
.Case("0xc08", "cortex-a8")
|
|
|
|
.Case("0xc09", "cortex-a9")
|
|
|
|
.Case("0xc0f", "cortex-a15")
|
|
|
|
.Case("0xc20", "cortex-m0")
|
|
|
|
.Case("0xc23", "cortex-m3")
|
|
|
|
.Case("0xc24", "cortex-m4")
|
2020-02-14 21:33:32 +08:00
|
|
|
.Case("0xd22", "cortex-m55")
|
2020-02-12 00:57:25 +08:00
|
|
|
.Case("0xd02", "cortex-a34")
|
2017-05-03 23:42:29 +08:00
|
|
|
.Case("0xd04", "cortex-a35")
|
|
|
|
.Case("0xd03", "cortex-a53")
|
|
|
|
.Case("0xd07", "cortex-a57")
|
|
|
|
.Case("0xd08", "cortex-a72")
|
|
|
|
.Case("0xd09", "cortex-a73")
|
2019-06-11 08:05:36 +08:00
|
|
|
.Case("0xd0a", "cortex-a75")
|
|
|
|
.Case("0xd0b", "cortex-a76")
|
[ARM] Add Cortex-A77 Support for Clang and LLVM
This patch upstreams support for the Arm-v8 Cortex-A77
processor for AArch64 and ARM.
In detail:
- Adding cortex-a77 as a cpu option for aarch64 and arm targets in clang
- Cortex-A77 CPU name and ProcessorModel in llvm
details of the CPU can be found here:
https://www.arm.com/products/silicon-ip-cpu/cortex-a/cortex-a77
and a similar submission to GCC can be found here:
https://github.com/gcc-mirror/gcc/commit/e0664b7a63ed8305e9f8539309df7fb3eb13babe
The following people contributed to this patch:
- Luke Geeson
- Mikhail Maltsev
Reviewers: t.p.northover, dmgreen, ostannard, SjoerdMeijer
Reviewed By: dmgreen
Subscribers: dmgreen, kristof.beyls, hiraditya, danielkiss, cfe-commits,
llvm-commits, miyuki
Tags: #clang, #llvm
Differential Revision: https://reviews.llvm.org/D82887
2020-06-30 23:45:36 +08:00
|
|
|
.Case("0xd0d", "cortex-a77")
|
2020-05-29 02:49:12 +08:00
|
|
|
.Case("0xd0c", "neoverse-n1")
|
2017-05-03 23:42:29 +08:00
|
|
|
.Default("generic");
|
|
|
|
}
|
|
|
|
|
2018-10-06 06:23:21 +08:00
|
|
|
if (Implementer == "0x42" || Implementer == "0x43") { // Broadcom | Cavium.
|
|
|
|
for (unsigned I = 0, E = Lines.size(); I != E; ++I) {
|
|
|
|
if (Lines[I].startswith("CPU part")) {
|
|
|
|
return StringSwitch<const char *>(Lines[I].substr(8).ltrim("\t :"))
|
|
|
|
.Case("0x516", "thunderx2t99")
|
|
|
|
.Case("0x0516", "thunderx2t99")
|
|
|
|
.Case("0xaf", "thunderx2t99")
|
|
|
|
.Case("0x0af", "thunderx2t99")
|
|
|
|
.Case("0xa1", "thunderxt88")
|
|
|
|
.Case("0x0a1", "thunderxt88")
|
|
|
|
.Default("generic");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-03 20:52:27 +08:00
|
|
|
if (Implementer == "0x46") { // Fujitsu Ltd.
|
|
|
|
for (unsigned I = 0, E = Lines.size(); I != E; ++I) {
|
|
|
|
if (Lines[I].startswith("CPU part")) {
|
|
|
|
return StringSwitch<const char *>(Lines[I].substr(8).ltrim("\t :"))
|
|
|
|
.Case("0x001", "a64fx")
|
|
|
|
.Default("generic");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
[AArch64] Add NVIDIA Carmel support
Summary:
NVIDIA's Carmel ARM64 cores are used in Tegra194 chips found in Jetson AGX Xavier, DRIVE AGX Xavier and DRIVE AGX Pegasus.
References:
* https://devblogs.nvidia.com/nvidia-jetson-agx-xavier-32-teraops-ai-robotics/#h.huq9xtg75a5e
* NVIDIA Xavier Series System-on-Chip Technical Reference Manual 1.3 (https://developer.nvidia.com/embedded/downloads#?search=Xavier%20Series%20SoC%20Technical%20Reference%20Manual)
Reviewers: sdesmalen, paquette
Reviewed By: sdesmalen
Subscribers: llvm-commits, ianshmean, kristof.beyls, hiraditya, jfb, danielkiss, cfe-commits, t.p.northover
Tags: #clang, #llvm
Differential Revision: https://reviews.llvm.org/D77940
2020-05-04 18:45:35 +08:00
|
|
|
if (Implementer == "0x4e") { // NVIDIA Corporation
|
|
|
|
for (unsigned I = 0, E = Lines.size(); I != E; ++I) {
|
|
|
|
if (Lines[I].startswith("CPU part")) {
|
|
|
|
return StringSwitch<const char *>(Lines[I].substr(8).ltrim("\t :"))
|
|
|
|
.Case("0x004", "carmel")
|
|
|
|
.Default("generic");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-10 03:32:08 +08:00
|
|
|
if (Implementer == "0x48") // HiSilicon Technologies, Inc.
|
|
|
|
// Look for the CPU part line.
|
|
|
|
for (unsigned I = 0, E = Lines.size(); I != E; ++I)
|
|
|
|
if (Lines[I].startswith("CPU part"))
|
|
|
|
// The CPU part is a 3 digit hexadecimal number with a 0x prefix. The
|
|
|
|
// values correspond to the "Part number" in the CP15/c0 register. The
|
|
|
|
// contents are specified in the various processor manuals.
|
|
|
|
return StringSwitch<const char *>(Lines[I].substr(8).ltrim("\t :"))
|
|
|
|
.Case("0xd01", "tsv110")
|
|
|
|
.Default("generic");
|
|
|
|
|
2017-05-03 23:42:29 +08:00
|
|
|
if (Implementer == "0x51") // Qualcomm Technologies, Inc.
|
|
|
|
// Look for the CPU part line.
|
|
|
|
for (unsigned I = 0, E = Lines.size(); I != E; ++I)
|
|
|
|
if (Lines[I].startswith("CPU part"))
|
|
|
|
// The CPU part is a 3 digit hexadecimal number with a 0x prefix. The
|
|
|
|
// values correspond to the "Part number" in the CP15/c0 register. The
|
|
|
|
// contents are specified in the various processor manuals.
|
|
|
|
return StringSwitch<const char *>(Lines[I].substr(8).ltrim("\t :"))
|
|
|
|
.Case("0x06f", "krait") // APQ8064
|
|
|
|
.Case("0x201", "kryo")
|
|
|
|
.Case("0x205", "kryo")
|
2017-09-14 05:48:00 +08:00
|
|
|
.Case("0x211", "kryo")
|
|
|
|
.Case("0x800", "cortex-a73")
|
|
|
|
.Case("0x801", "cortex-a73")
|
2019-06-11 08:05:36 +08:00
|
|
|
.Case("0x802", "cortex-a73")
|
|
|
|
.Case("0x803", "cortex-a73")
|
|
|
|
.Case("0x804", "cortex-a73")
|
|
|
|
.Case("0x805", "cortex-a73")
|
2017-09-23 01:46:36 +08:00
|
|
|
.Case("0xc00", "falkor")
|
2017-09-25 22:05:00 +08:00
|
|
|
.Case("0xc01", "saphira")
|
2017-05-03 23:42:29 +08:00
|
|
|
.Default("generic");
|
|
|
|
|
2017-12-09 05:09:59 +08:00
|
|
|
if (Implementer == "0x53") { // Samsung Electronics Co., Ltd.
|
|
|
|
// The Exynos chips have a convoluted ID scheme that doesn't seem to follow
|
|
|
|
// any predictive pattern across variants and parts.
|
|
|
|
unsigned Variant = 0, Part = 0;
|
|
|
|
|
|
|
|
// Look for the CPU variant line, whose value is a 1 digit hexadecimal
|
|
|
|
// number, corresponding to the Variant bits in the CP15/C0 register.
|
|
|
|
for (auto I : Lines)
|
|
|
|
if (I.consume_front("CPU variant"))
|
|
|
|
I.ltrim("\t :").getAsInteger(0, Variant);
|
|
|
|
|
|
|
|
// Look for the CPU part line, whose value is a 3 digit hexadecimal
|
|
|
|
// number, corresponding to the PartNum bits in the CP15/C0 register.
|
|
|
|
for (auto I : Lines)
|
|
|
|
if (I.consume_front("CPU part"))
|
|
|
|
I.ltrim("\t :").getAsInteger(0, Part);
|
|
|
|
|
|
|
|
unsigned Exynos = (Variant << 12) | Part;
|
|
|
|
switch (Exynos) {
|
|
|
|
default:
|
2019-10-03 05:26:40 +08:00
|
|
|
// Default by falling through to Exynos M3.
|
2017-12-09 05:09:59 +08:00
|
|
|
LLVM_FALLTHROUGH;
|
2019-10-03 05:26:40 +08:00
|
|
|
case 0x1002:
|
|
|
|
return "exynos-m3";
|
|
|
|
case 0x1003:
|
|
|
|
return "exynos-m4";
|
2017-12-09 05:09:59 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-03 23:42:29 +08:00
|
|
|
return "generic";
|
|
|
|
}
|
|
|
|
|
2018-03-08 01:53:16 +08:00
|
|
|
StringRef sys::detail::getHostCPUNameForS390x(StringRef ProcCpuinfoContent) {
|
2017-05-03 23:42:29 +08:00
|
|
|
// STIDP is a privileged operation, so use /proc/cpuinfo instead.
|
|
|
|
|
|
|
|
// The "processor 0:" line comes after a fair amount of other information,
|
|
|
|
// including a cache breakdown, but this should be plenty.
|
|
|
|
SmallVector<StringRef, 32> Lines;
|
|
|
|
ProcCpuinfoContent.split(Lines, "\n");
|
|
|
|
|
|
|
|
// Look for the CPU features.
|
|
|
|
SmallVector<StringRef, 32> CPUFeatures;
|
|
|
|
for (unsigned I = 0, E = Lines.size(); I != E; ++I)
|
|
|
|
if (Lines[I].startswith("features")) {
|
|
|
|
size_t Pos = Lines[I].find(":");
|
|
|
|
if (Pos != StringRef::npos) {
|
|
|
|
Lines[I].drop_front(Pos + 1).split(CPUFeatures, ' ');
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We need to check for the presence of vector support independently of
|
|
|
|
// the machine type, since we may only use the vector register set when
|
|
|
|
// supported by the kernel (and hypervisor).
|
|
|
|
bool HaveVectorSupport = false;
|
|
|
|
for (unsigned I = 0, E = CPUFeatures.size(); I != E; ++I) {
|
|
|
|
if (CPUFeatures[I] == "vx")
|
|
|
|
HaveVectorSupport = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now check the processor machine type.
|
|
|
|
for (unsigned I = 0, E = Lines.size(); I != E; ++I) {
|
|
|
|
if (Lines[I].startswith("processor ")) {
|
|
|
|
size_t Pos = Lines[I].find("machine = ");
|
|
|
|
if (Pos != StringRef::npos) {
|
|
|
|
Pos += sizeof("machine = ") - 1;
|
|
|
|
unsigned int Id;
|
|
|
|
if (!Lines[I].drop_front(Pos).getAsInteger(10, Id)) {
|
2019-07-13 02:13:16 +08:00
|
|
|
if (Id >= 8561 && HaveVectorSupport)
|
2019-09-21 07:04:45 +08:00
|
|
|
return "z15";
|
2017-07-18 01:41:11 +08:00
|
|
|
if (Id >= 3906 && HaveVectorSupport)
|
|
|
|
return "z14";
|
2017-05-03 23:42:29 +08:00
|
|
|
if (Id >= 2964 && HaveVectorSupport)
|
|
|
|
return "z13";
|
|
|
|
if (Id >= 2827)
|
|
|
|
return "zEC12";
|
|
|
|
if (Id >= 2817)
|
|
|
|
return "z196";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return "generic";
|
|
|
|
}
|
|
|
|
|
2017-08-23 12:25:57 +08:00
|
|
|
StringRef sys::detail::getHostCPUNameForBPF() {
|
|
|
|
#if !defined(__linux__) || !defined(__x86_64__)
|
|
|
|
return "generic";
|
|
|
|
#else
|
2019-02-07 18:43:09 +08:00
|
|
|
uint8_t v3_insns[40] __attribute__ ((aligned (8))) =
|
|
|
|
/* BPF_MOV64_IMM(BPF_REG_0, 0) */
|
|
|
|
{ 0xb7, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
|
|
/* BPF_MOV64_IMM(BPF_REG_2, 1) */
|
|
|
|
0xb7, 0x2, 0x0, 0x0, 0x1, 0x0, 0x0, 0x0,
|
|
|
|
/* BPF_JMP32_REG(BPF_JLT, BPF_REG_0, BPF_REG_2, 1) */
|
|
|
|
0xae, 0x20, 0x1, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
|
|
/* BPF_MOV64_IMM(BPF_REG_0, 1) */
|
|
|
|
0xb7, 0x0, 0x0, 0x0, 0x1, 0x0, 0x0, 0x0,
|
|
|
|
/* BPF_EXIT_INSN() */
|
|
|
|
0x95, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0 };
|
|
|
|
|
|
|
|
uint8_t v2_insns[40] __attribute__ ((aligned (8))) =
|
2017-08-23 12:25:57 +08:00
|
|
|
/* BPF_MOV64_IMM(BPF_REG_0, 0) */
|
|
|
|
{ 0xb7, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
|
|
/* BPF_MOV64_IMM(BPF_REG_2, 1) */
|
|
|
|
0xb7, 0x2, 0x0, 0x0, 0x1, 0x0, 0x0, 0x0,
|
|
|
|
/* BPF_JMP_REG(BPF_JLT, BPF_REG_0, BPF_REG_2, 1) */
|
|
|
|
0xad, 0x20, 0x1, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
|
|
/* BPF_MOV64_IMM(BPF_REG_0, 1) */
|
|
|
|
0xb7, 0x0, 0x0, 0x0, 0x1, 0x0, 0x0, 0x0,
|
|
|
|
/* BPF_EXIT_INSN() */
|
|
|
|
0x95, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0 };
|
|
|
|
|
|
|
|
struct bpf_prog_load_attr {
|
|
|
|
uint32_t prog_type;
|
|
|
|
uint32_t insn_cnt;
|
|
|
|
uint64_t insns;
|
|
|
|
uint64_t license;
|
|
|
|
uint32_t log_level;
|
|
|
|
uint32_t log_size;
|
|
|
|
uint64_t log_buf;
|
|
|
|
uint32_t kern_version;
|
|
|
|
uint32_t prog_flags;
|
|
|
|
} attr = {};
|
|
|
|
attr.prog_type = 1; /* BPF_PROG_TYPE_SOCKET_FILTER */
|
|
|
|
attr.insn_cnt = 5;
|
2019-02-07 18:43:09 +08:00
|
|
|
attr.insns = (uint64_t)v3_insns;
|
2017-08-23 12:25:57 +08:00
|
|
|
attr.license = (uint64_t)"DUMMY";
|
|
|
|
|
2019-02-07 18:43:09 +08:00
|
|
|
int fd = syscall(321 /* __NR_bpf */, 5 /* BPF_PROG_LOAD */, &attr,
|
|
|
|
sizeof(attr));
|
|
|
|
if (fd >= 0) {
|
|
|
|
close(fd);
|
|
|
|
return "v3";
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Clear the whole attr in case its content changed by syscall. */
|
|
|
|
memset(&attr, 0, sizeof(attr));
|
|
|
|
attr.prog_type = 1; /* BPF_PROG_TYPE_SOCKET_FILTER */
|
|
|
|
attr.insn_cnt = 5;
|
|
|
|
attr.insns = (uint64_t)v2_insns;
|
|
|
|
attr.license = (uint64_t)"DUMMY";
|
|
|
|
fd = syscall(321 /* __NR_bpf */, 5 /* BPF_PROG_LOAD */, &attr, sizeof(attr));
|
2017-08-24 00:24:31 +08:00
|
|
|
if (fd >= 0) {
|
|
|
|
close(fd);
|
|
|
|
return "v2";
|
|
|
|
}
|
|
|
|
return "v1";
|
2017-08-23 12:25:57 +08:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-05-03 23:42:29 +08:00
|
|
|
#if defined(__i386__) || defined(_M_IX86) || \
|
|
|
|
defined(__x86_64__) || defined(_M_X64)
|
|
|
|
|
|
|
|
enum VendorSignatures {
|
|
|
|
SIG_INTEL = 0x756e6547 /* Genu */,
|
|
|
|
SIG_AMD = 0x68747541 /* Auth */
|
|
|
|
};
|
|
|
|
|
|
|
|
// The check below for i386 was copied from clang's cpuid.h (__get_cpuid_max).
|
|
|
|
// Check motivated by bug reports for OpenSSL crashing on CPUs without CPUID
|
|
|
|
// support. Consequently, for i386, the presence of CPUID is checked first
|
|
|
|
// via the corresponding eflags bit.
|
|
|
|
// Removal of cpuid.h header motivated by PR30384
|
|
|
|
// Header cpuid.h and method __get_cpuid_max are not used in llvm, clang, openmp
|
|
|
|
// or test-suite, but are used in external projects e.g. libstdcxx
|
|
|
|
static bool isCpuIdSupported() {
|
|
|
|
#if defined(__GNUC__) || defined(__clang__)
|
|
|
|
#if defined(__i386__)
|
|
|
|
int __cpuid_supported;
|
|
|
|
__asm__(" pushfl\n"
|
|
|
|
" popl %%eax\n"
|
|
|
|
" movl %%eax,%%ecx\n"
|
|
|
|
" xorl $0x00200000,%%eax\n"
|
|
|
|
" pushl %%eax\n"
|
|
|
|
" popfl\n"
|
|
|
|
" pushfl\n"
|
|
|
|
" popl %%eax\n"
|
|
|
|
" movl $0,%0\n"
|
|
|
|
" cmpl %%eax,%%ecx\n"
|
|
|
|
" je 1f\n"
|
|
|
|
" movl $1,%0\n"
|
|
|
|
"1:"
|
|
|
|
: "=r"(__cpuid_supported)
|
|
|
|
:
|
|
|
|
: "eax", "ecx");
|
|
|
|
if (!__cpuid_supported)
|
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
return true;
|
|
|
|
#endif
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// getX86CpuIDAndInfo - Execute the specified cpuid and return the 4 values in
|
|
|
|
/// the specified arguments. If we can't run cpuid on the host, return true.
|
|
|
|
static bool getX86CpuIDAndInfo(unsigned value, unsigned *rEAX, unsigned *rEBX,
|
|
|
|
unsigned *rECX, unsigned *rEDX) {
|
|
|
|
#if defined(__GNUC__) || defined(__clang__)
|
|
|
|
#if defined(__x86_64__)
|
|
|
|
// gcc doesn't know cpuid would clobber ebx/rbx. Preserve it manually.
|
|
|
|
// FIXME: should we save this for Clang?
|
|
|
|
__asm__("movq\t%%rbx, %%rsi\n\t"
|
|
|
|
"cpuid\n\t"
|
|
|
|
"xchgq\t%%rbx, %%rsi\n\t"
|
|
|
|
: "=a"(*rEAX), "=S"(*rEBX), "=c"(*rECX), "=d"(*rEDX)
|
|
|
|
: "a"(value));
|
2017-07-10 14:04:11 +08:00
|
|
|
return false;
|
2017-05-03 23:42:29 +08:00
|
|
|
#elif defined(__i386__)
|
|
|
|
__asm__("movl\t%%ebx, %%esi\n\t"
|
|
|
|
"cpuid\n\t"
|
|
|
|
"xchgl\t%%ebx, %%esi\n\t"
|
|
|
|
: "=a"(*rEAX), "=S"(*rEBX), "=c"(*rECX), "=d"(*rEDX)
|
|
|
|
: "a"(value));
|
2017-07-10 14:04:11 +08:00
|
|
|
return false;
|
2017-05-03 23:42:29 +08:00
|
|
|
#else
|
2017-07-10 14:04:11 +08:00
|
|
|
return true;
|
2017-05-03 23:42:29 +08:00
|
|
|
#endif
|
|
|
|
#elif defined(_MSC_VER)
|
|
|
|
// The MSVC intrinsic is portable across x86 and x64.
|
|
|
|
int registers[4];
|
|
|
|
__cpuid(registers, value);
|
|
|
|
*rEAX = registers[0];
|
|
|
|
*rEBX = registers[1];
|
|
|
|
*rECX = registers[2];
|
|
|
|
*rEDX = registers[3];
|
|
|
|
return false;
|
|
|
|
#else
|
|
|
|
return true;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/// getX86CpuIDAndInfoEx - Execute the specified cpuid with subleaf and return
|
|
|
|
/// the 4 values in the specified arguments. If we can't run cpuid on the host,
|
|
|
|
/// return true.
|
|
|
|
static bool getX86CpuIDAndInfoEx(unsigned value, unsigned subleaf,
|
|
|
|
unsigned *rEAX, unsigned *rEBX, unsigned *rECX,
|
|
|
|
unsigned *rEDX) {
|
|
|
|
#if defined(__GNUC__) || defined(__clang__)
|
2017-07-17 13:16:16 +08:00
|
|
|
#if defined(__x86_64__)
|
2017-07-10 14:09:22 +08:00
|
|
|
// gcc doesn't know cpuid would clobber ebx/rbx. Preserve it manually.
|
2017-05-03 23:42:29 +08:00
|
|
|
// FIXME: should we save this for Clang?
|
|
|
|
__asm__("movq\t%%rbx, %%rsi\n\t"
|
|
|
|
"cpuid\n\t"
|
|
|
|
"xchgq\t%%rbx, %%rsi\n\t"
|
|
|
|
: "=a"(*rEAX), "=S"(*rEBX), "=c"(*rECX), "=d"(*rEDX)
|
|
|
|
: "a"(value), "c"(subleaf));
|
2017-07-10 14:04:11 +08:00
|
|
|
return false;
|
2017-07-17 13:16:16 +08:00
|
|
|
#elif defined(__i386__)
|
2017-05-03 23:42:29 +08:00
|
|
|
__asm__("movl\t%%ebx, %%esi\n\t"
|
|
|
|
"cpuid\n\t"
|
|
|
|
"xchgl\t%%ebx, %%esi\n\t"
|
|
|
|
: "=a"(*rEAX), "=S"(*rEBX), "=c"(*rECX), "=d"(*rEDX)
|
|
|
|
: "a"(value), "c"(subleaf));
|
2017-07-10 14:04:11 +08:00
|
|
|
return false;
|
2017-05-03 23:42:29 +08:00
|
|
|
#else
|
2017-07-10 14:04:11 +08:00
|
|
|
return true;
|
2017-05-03 23:42:29 +08:00
|
|
|
#endif
|
2017-07-17 13:16:16 +08:00
|
|
|
#elif defined(_MSC_VER)
|
|
|
|
int registers[4];
|
|
|
|
__cpuidex(registers, value, subleaf);
|
|
|
|
*rEAX = registers[0];
|
|
|
|
*rEBX = registers[1];
|
|
|
|
*rECX = registers[2];
|
|
|
|
*rEDX = registers[3];
|
|
|
|
return false;
|
2017-05-03 23:42:29 +08:00
|
|
|
#else
|
|
|
|
return true;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-07-12 14:49:57 +08:00
|
|
|
// Read control register 0 (XCR0). Used to detect features such as AVX.
|
2017-05-03 23:42:29 +08:00
|
|
|
static bool getX86XCR0(unsigned *rEAX, unsigned *rEDX) {
|
|
|
|
#if defined(__GNUC__) || defined(__clang__)
|
|
|
|
// Check xgetbv; this uses a .byte sequence instead of the instruction
|
|
|
|
// directly because older assemblers do not include support for xgetbv and
|
|
|
|
// there is no easy way to conditionally compile based on the assembler used.
|
|
|
|
__asm__(".byte 0x0f, 0x01, 0xd0" : "=a"(*rEAX), "=d"(*rEDX) : "c"(0));
|
|
|
|
return false;
|
|
|
|
#elif defined(_MSC_FULL_VER) && defined(_XCR_XFEATURE_ENABLED_MASK)
|
|
|
|
unsigned long long Result = _xgetbv(_XCR_XFEATURE_ENABLED_MASK);
|
|
|
|
*rEAX = Result;
|
|
|
|
*rEDX = Result >> 32;
|
|
|
|
return false;
|
|
|
|
#else
|
|
|
|
return true;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static void detectX86FamilyModel(unsigned EAX, unsigned *Family,
|
|
|
|
unsigned *Model) {
|
|
|
|
*Family = (EAX >> 8) & 0xf; // Bits 8 - 11
|
|
|
|
*Model = (EAX >> 4) & 0xf; // Bits 4 - 7
|
|
|
|
if (*Family == 6 || *Family == 0xf) {
|
|
|
|
if (*Family == 0xf)
|
|
|
|
// Examine extended family ID if family ID is F.
|
|
|
|
*Family += (EAX >> 20) & 0xff; // Bits 20 - 27
|
|
|
|
// Examine extended model ID if family ID is 6 or F.
|
|
|
|
*Model += ((EAX >> 16) & 0xf) << 4; // Bits 16 - 19
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2017-07-08 13:16:14 +08:00
|
|
|
getIntelProcessorTypeAndSubtype(unsigned Family, unsigned Model,
|
2020-06-23 04:29:43 +08:00
|
|
|
const unsigned *Features,
|
2018-10-20 11:51:43 +08:00
|
|
|
unsigned *Type, unsigned *Subtype) {
|
2020-06-12 12:14:45 +08:00
|
|
|
auto testFeature = [&](unsigned F) {
|
2020-06-13 08:13:49 +08:00
|
|
|
return (Features[F / 32] & (1U << (F % 32))) != 0;
|
2020-06-12 12:14:45 +08:00
|
|
|
};
|
|
|
|
|
2017-05-03 23:42:29 +08:00
|
|
|
switch (Family) {
|
|
|
|
case 3:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_i386;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
case 4:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_i486;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
case 5:
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_MMX)) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_PENTIUM_MMX;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
}
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_PENTIUM;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
case 6:
|
|
|
|
switch (Model) {
|
|
|
|
case 0x01: // Pentium Pro processor
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_PENTIUM_PRO;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
case 0x03: // Intel Pentium II OverDrive processor, Pentium II processor,
|
|
|
|
// model 03
|
|
|
|
case 0x05: // Pentium II processor, model 05, Pentium II Xeon processor,
|
|
|
|
// model 05, and Intel Celeron processor, model 05
|
|
|
|
case 0x06: // Celeron processor, model 06
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_PENTIUM_II;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
case 0x07: // Pentium III processor, model 07, and Pentium III Xeon
|
|
|
|
// processor, model 07
|
|
|
|
case 0x08: // Pentium III processor, model 08, Pentium III Xeon processor,
|
|
|
|
// model 08, and Celeron processor, model 08
|
|
|
|
case 0x0a: // Pentium III Xeon processor, model 0Ah
|
|
|
|
case 0x0b: // Pentium III processor, model 0Bh
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_PENTIUM_III;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
case 0x09: // Intel Pentium M processor, Intel Celeron M processor model 09.
|
|
|
|
case 0x0d: // Intel Pentium M processor, Intel Celeron M processor, model
|
|
|
|
// 0Dh. All processors are manufactured using the 90 nm process.
|
|
|
|
case 0x15: // Intel EP80579 Integrated Processor and Intel EP80579
|
|
|
|
// Integrated Processor with Intel QuickAssist Technology
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_PENTIUM_M;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
case 0x0e: // Intel Core Duo processor, Intel Core Solo processor, model
|
|
|
|
// 0Eh. All processors are manufactured using the 65 nm process.
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_CORE_DUO;
|
2017-05-03 23:42:29 +08:00
|
|
|
break; // yonah
|
|
|
|
case 0x0f: // Intel Core 2 Duo processor, Intel Core 2 Duo mobile
|
|
|
|
// processor, Intel Core 2 Quad processor, Intel Core 2 Quad
|
|
|
|
// mobile processor, Intel Core 2 Extreme processor, Intel
|
|
|
|
// Pentium Dual-Core processor, Intel Xeon processor, model
|
|
|
|
// 0Fh. All processors are manufactured using the 65 nm process.
|
|
|
|
case 0x16: // Intel Celeron processor model 16h. All processors are
|
|
|
|
// manufactured using the 65 nm process
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_CORE2; // "core2"
|
|
|
|
*Subtype = X86::INTEL_CORE2_65;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
case 0x17: // Intel Core 2 Extreme processor, Intel Xeon processor, model
|
|
|
|
// 17h. All processors are manufactured using the 45 nm process.
|
|
|
|
//
|
|
|
|
// 45nm: Penryn , Wolfdale, Yorkfield (XE)
|
|
|
|
case 0x1d: // Intel Xeon processor MP. All processors are manufactured using
|
|
|
|
// the 45 nm process.
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_CORE2; // "penryn"
|
|
|
|
*Subtype = X86::INTEL_CORE2_45;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
case 0x1a: // Intel Core i7 processor and Intel Xeon processor. All
|
|
|
|
// processors are manufactured using the 45 nm process.
|
|
|
|
case 0x1e: // Intel(R) Core(TM) i7 CPU 870 @ 2.93GHz.
|
|
|
|
// As found in a Summer 2010 model iMac.
|
|
|
|
case 0x1f:
|
|
|
|
case 0x2e: // Nehalem EX
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_COREI7; // "nehalem"
|
|
|
|
*Subtype = X86::INTEL_COREI7_NEHALEM;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
case 0x25: // Intel Core i7, laptop version.
|
|
|
|
case 0x2c: // Intel Core i7 processor and Intel Xeon processor. All
|
|
|
|
// processors are manufactured using the 32 nm process.
|
|
|
|
case 0x2f: // Westmere EX
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_COREI7; // "westmere"
|
|
|
|
*Subtype = X86::INTEL_COREI7_WESTMERE;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
case 0x2a: // Intel Core i7 processor. All processors are manufactured
|
|
|
|
// using the 32 nm process.
|
|
|
|
case 0x2d:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_COREI7; //"sandybridge"
|
|
|
|
*Subtype = X86::INTEL_COREI7_SANDYBRIDGE;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
case 0x3a:
|
|
|
|
case 0x3e: // Ivy Bridge EP
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_COREI7; // "ivybridge"
|
|
|
|
*Subtype = X86::INTEL_COREI7_IVYBRIDGE;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
// Haswell:
|
|
|
|
case 0x3c:
|
|
|
|
case 0x3f:
|
|
|
|
case 0x45:
|
|
|
|
case 0x46:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_COREI7; // "haswell"
|
|
|
|
*Subtype = X86::INTEL_COREI7_HASWELL;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
// Broadwell:
|
|
|
|
case 0x3d:
|
|
|
|
case 0x47:
|
|
|
|
case 0x4f:
|
|
|
|
case 0x56:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_COREI7; // "broadwell"
|
|
|
|
*Subtype = X86::INTEL_COREI7_BROADWELL;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
// Skylake:
|
2019-06-01 03:18:07 +08:00
|
|
|
case 0x4e: // Skylake mobile
|
|
|
|
case 0x5e: // Skylake desktop
|
|
|
|
case 0x8e: // Kaby Lake mobile
|
|
|
|
case 0x9e: // Kaby Lake desktop
|
2020-05-20 12:05:31 +08:00
|
|
|
case 0xa5: // Comet Lake-H/S
|
|
|
|
case 0xa6: // Comet Lake-U
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_COREI7; // "skylake"
|
|
|
|
*Subtype = X86::INTEL_COREI7_SKYLAKE;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
// Skylake Xeon:
|
|
|
|
case 0x55:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_COREI7;
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_AVX512BF16))
|
2019-06-07 16:31:35 +08:00
|
|
|
*Subtype = X86::INTEL_COREI7_COOPERLAKE; // "cooperlake"
|
2020-06-12 12:14:45 +08:00
|
|
|
else if (testFeature(X86::FEATURE_AVX512VNNI))
|
2019-06-01 03:18:07 +08:00
|
|
|
*Subtype = X86::INTEL_COREI7_CASCADELAKE; // "cascadelake"
|
|
|
|
else
|
|
|
|
*Subtype = X86::INTEL_COREI7_SKYLAKE_AVX512; // "skylake-avx512"
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
|
2017-11-15 14:02:42 +08:00
|
|
|
// Cannonlake:
|
|
|
|
case 0x66:
|
|
|
|
*Type = X86::INTEL_COREI7;
|
|
|
|
*Subtype = X86::INTEL_COREI7_CANNONLAKE; // "cannonlake"
|
|
|
|
break;
|
|
|
|
|
2019-05-21 00:58:23 +08:00
|
|
|
// Icelake:
|
2019-05-23 03:51:35 +08:00
|
|
|
case 0x7d:
|
2019-05-21 00:58:23 +08:00
|
|
|
case 0x7e:
|
|
|
|
*Type = X86::INTEL_COREI7;
|
|
|
|
*Subtype = X86::INTEL_COREI7_ICELAKE_CLIENT; // "icelake-client"
|
|
|
|
break;
|
|
|
|
|
2019-05-23 03:51:35 +08:00
|
|
|
// Icelake Xeon:
|
|
|
|
case 0x6a:
|
|
|
|
case 0x6c:
|
|
|
|
*Type = X86::INTEL_COREI7;
|
|
|
|
*Subtype = X86::INTEL_COREI7_ICELAKE_SERVER; // "icelake-server"
|
|
|
|
break;
|
|
|
|
|
2017-05-03 23:42:29 +08:00
|
|
|
case 0x1c: // Most 45 nm Intel Atom processors
|
|
|
|
case 0x26: // 45 nm Atom Lincroft
|
|
|
|
case 0x27: // 32 nm Atom Medfield
|
|
|
|
case 0x35: // 32 nm Atom Midview
|
|
|
|
case 0x36: // 32 nm Atom Midview
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_BONNELL;
|
2017-05-03 23:42:29 +08:00
|
|
|
break; // "bonnell"
|
|
|
|
|
|
|
|
// Atom Silvermont codes from the Intel software optimization guide.
|
|
|
|
case 0x37:
|
|
|
|
case 0x4a:
|
|
|
|
case 0x4d:
|
|
|
|
case 0x5a:
|
|
|
|
case 0x5d:
|
|
|
|
case 0x4c: // really airmont
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_SILVERMONT;
|
2017-05-03 23:42:29 +08:00
|
|
|
break; // "silvermont"
|
2017-06-29 18:00:33 +08:00
|
|
|
// Goldmont:
|
2017-11-15 14:02:43 +08:00
|
|
|
case 0x5c: // Apollo Lake
|
|
|
|
case 0x5f: // Denverton
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_GOLDMONT;
|
2017-06-29 18:00:33 +08:00
|
|
|
break; // "goldmont"
|
2018-04-16 15:47:35 +08:00
|
|
|
case 0x7a:
|
|
|
|
*Type = X86::INTEL_GOLDMONT_PLUS;
|
|
|
|
break;
|
2019-05-21 00:58:23 +08:00
|
|
|
case 0x86:
|
|
|
|
*Type = X86::INTEL_TREMONT;
|
|
|
|
break;
|
2019-06-01 03:18:07 +08:00
|
|
|
|
2017-05-03 23:42:29 +08:00
|
|
|
case 0x57:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_KNL; // knl
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
2019-06-01 03:18:07 +08:00
|
|
|
|
2017-10-14 02:10:17 +08:00
|
|
|
case 0x85:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_KNM; // knm
|
2017-10-14 02:10:17 +08:00
|
|
|
break;
|
2017-05-03 23:42:29 +08:00
|
|
|
|
|
|
|
default: // Unknown family 6 CPU, try to guess.
|
2020-06-12 12:14:45 +08:00
|
|
|
// TODO detect tigerlake host from model
|
|
|
|
if (testFeature(X86::FEATURE_AVX512VP2INTERSECT)) {
|
2019-08-12 09:29:46 +08:00
|
|
|
*Type = X86::INTEL_COREI7;
|
|
|
|
*Subtype = X86::INTEL_COREI7_TIGERLAKE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_AVX512VBMI2)) {
|
2018-11-16 02:11:52 +08:00
|
|
|
*Type = X86::INTEL_COREI7;
|
|
|
|
*Subtype = X86::INTEL_COREI7_ICELAKE_CLIENT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_AVX512VBMI)) {
|
2017-11-15 14:02:42 +08:00
|
|
|
*Type = X86::INTEL_COREI7;
|
|
|
|
*Subtype = X86::INTEL_COREI7_CANNONLAKE;
|
2017-07-27 11:26:52 +08:00
|
|
|
break;
|
|
|
|
}
|
2017-11-15 14:02:42 +08:00
|
|
|
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_AVX512BF16)) {
|
2019-06-07 16:31:35 +08:00
|
|
|
*Type = X86::INTEL_COREI7;
|
|
|
|
*Subtype = X86::INTEL_COREI7_COOPERLAKE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_AVX512VNNI)) {
|
2018-11-28 02:05:00 +08:00
|
|
|
*Type = X86::INTEL_COREI7;
|
|
|
|
*Subtype = X86::INTEL_COREI7_CASCADELAKE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_AVX512VL)) {
|
2017-11-15 14:02:42 +08:00
|
|
|
*Type = X86::INTEL_COREI7;
|
|
|
|
*Subtype = X86::INTEL_COREI7_SKYLAKE_AVX512;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_AVX512ER)) {
|
2017-11-15 14:02:42 +08:00
|
|
|
*Type = X86::INTEL_KNL; // knl
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_CLFLUSHOPT)) {
|
|
|
|
if (testFeature(X86::FEATURE_SHA)) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_GOLDMONT;
|
2017-07-27 11:26:52 +08:00
|
|
|
} else {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_COREI7;
|
|
|
|
*Subtype = X86::INTEL_COREI7_SKYLAKE;
|
2017-07-27 11:26:52 +08:00
|
|
|
}
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
}
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_ADX)) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_COREI7;
|
|
|
|
*Subtype = X86::INTEL_COREI7_BROADWELL;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
}
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_AVX2)) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_COREI7;
|
|
|
|
*Subtype = X86::INTEL_COREI7_HASWELL;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
}
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_AVX)) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_COREI7;
|
|
|
|
*Subtype = X86::INTEL_COREI7_SANDYBRIDGE;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
}
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_SSE4_2)) {
|
|
|
|
if (testFeature(X86::FEATURE_MOVBE)) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_SILVERMONT;
|
2017-05-03 23:42:29 +08:00
|
|
|
} else {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_COREI7;
|
|
|
|
*Subtype = X86::INTEL_COREI7_NEHALEM;
|
2017-05-03 23:42:29 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_SSE4_1)) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_CORE2; // "penryn"
|
|
|
|
*Subtype = X86::INTEL_CORE2_45;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
}
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_SSSE3)) {
|
|
|
|
if (testFeature(X86::FEATURE_MOVBE)) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_BONNELL; // "bonnell"
|
2017-05-03 23:42:29 +08:00
|
|
|
} else {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_CORE2; // "core2"
|
|
|
|
*Subtype = X86::INTEL_CORE2_65;
|
2017-05-03 23:42:29 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2020-07-09 23:47:35 +08:00
|
|
|
if (testFeature(X86::FEATURE_EM64T)) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_CORE2; // "core2"
|
|
|
|
*Subtype = X86::INTEL_CORE2_65;
|
2017-11-03 03:13:32 +08:00
|
|
|
break;
|
|
|
|
}
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_SSE3)) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_CORE_DUO;
|
2017-11-03 03:13:32 +08:00
|
|
|
break;
|
2017-05-03 23:42:29 +08:00
|
|
|
}
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_SSE2)) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_PENTIUM_M;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
}
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_SSE)) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_PENTIUM_III;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
}
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_MMX)) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_PENTIUM_II;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
}
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_PENTIUM_PRO;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 15: {
|
2020-07-09 23:47:35 +08:00
|
|
|
if (testFeature(X86::FEATURE_EM64T)) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_NOCONA;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
2017-11-03 03:13:34 +08:00
|
|
|
}
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_SSE3)) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_PRESCOTT;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
}
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::INTEL_PENTIUM_IV;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break; /*"generic"*/
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-08 14:44:34 +08:00
|
|
|
static void getAMDProcessorTypeAndSubtype(unsigned Family, unsigned Model,
|
2020-06-23 04:29:43 +08:00
|
|
|
const unsigned *Features,
|
2020-06-13 08:13:49 +08:00
|
|
|
unsigned *Type, unsigned *Subtype) {
|
2020-06-12 12:14:45 +08:00
|
|
|
auto testFeature = [&](unsigned F) {
|
2020-06-13 08:13:49 +08:00
|
|
|
return (Features[F / 32] & (1U << (F % 32))) != 0;
|
2020-06-12 12:14:45 +08:00
|
|
|
};
|
|
|
|
|
2017-05-03 23:42:29 +08:00
|
|
|
// FIXME: this poorly matches the generated SubtargetFeatureKV table. There
|
|
|
|
// appears to be no way to generate the wide variety of AMD-specific targets
|
|
|
|
// from the information returned from CPUID.
|
|
|
|
switch (Family) {
|
|
|
|
case 4:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::AMD_i486;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
case 5:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::AMDPENTIUM;
|
2017-05-03 23:42:29 +08:00
|
|
|
switch (Model) {
|
|
|
|
case 6:
|
|
|
|
case 7:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Subtype = X86::AMDPENTIUM_K6;
|
2017-05-03 23:42:29 +08:00
|
|
|
break; // "k6"
|
|
|
|
case 8:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Subtype = X86::AMDPENTIUM_K62;
|
2017-05-03 23:42:29 +08:00
|
|
|
break; // "k6-2"
|
|
|
|
case 9:
|
|
|
|
case 13:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Subtype = X86::AMDPENTIUM_K63;
|
2017-05-03 23:42:29 +08:00
|
|
|
break; // "k6-3"
|
|
|
|
case 10:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Subtype = X86::AMDPENTIUM_GEODE;
|
2017-05-03 23:42:29 +08:00
|
|
|
break; // "geode"
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 6:
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_SSE)) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::AMD_ATHLON_XP;
|
2017-05-03 23:42:29 +08:00
|
|
|
break; // "athlon-xp"
|
|
|
|
}
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::AMD_ATHLON;
|
2017-07-13 14:34:10 +08:00
|
|
|
break; // "athlon"
|
2017-05-03 23:42:29 +08:00
|
|
|
case 15:
|
2020-06-12 12:14:45 +08:00
|
|
|
if (testFeature(X86::FEATURE_SSE3)) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::AMD_K8SSE3;
|
2017-05-03 23:42:29 +08:00
|
|
|
break; // "k8-sse3"
|
|
|
|
}
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::AMD_K8;
|
2017-07-13 14:34:10 +08:00
|
|
|
break; // "k8"
|
2017-05-03 23:42:29 +08:00
|
|
|
case 16:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::AMDFAM10H; // "amdfam10"
|
2017-05-03 23:42:29 +08:00
|
|
|
switch (Model) {
|
|
|
|
case 2:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Subtype = X86::AMDFAM10H_BARCELONA;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
case 4:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Subtype = X86::AMDFAM10H_SHANGHAI;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
case 8:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Subtype = X86::AMDFAM10H_ISTANBUL;
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 20:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::AMD_BTVER1;
|
2017-05-03 23:42:29 +08:00
|
|
|
break; // "btver1";
|
|
|
|
case 21:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::AMDFAM15H;
|
2017-07-08 14:44:35 +08:00
|
|
|
if (Model >= 0x60 && Model <= 0x7f) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Subtype = X86::AMDFAM15H_BDVER4;
|
2017-07-12 14:49:56 +08:00
|
|
|
break; // "bdver4"; 60h-7Fh: Excavator
|
2017-05-03 23:42:29 +08:00
|
|
|
}
|
|
|
|
if (Model >= 0x30 && Model <= 0x3f) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Subtype = X86::AMDFAM15H_BDVER3;
|
2017-05-03 23:42:29 +08:00
|
|
|
break; // "bdver3"; 30h-3Fh: Steamroller
|
|
|
|
}
|
[X86][AMD][Bulldozer] Fix Bulldozer Model 2 detection.
Summary:
I have discovered an issue by accident.
```
$ lscpu
Architecture: x86_64
CPU op-mode(s): 32-bit, 64-bit
Byte Order: Little Endian
CPU(s): 8
On-line CPU(s) list: 0-7
Thread(s) per core: 2
Core(s) per socket: 4
Socket(s): 1
NUMA node(s): 1
Vendor ID: AuthenticAMD
CPU family: 21
Model: 2
Model name: AMD FX(tm)-8350 Eight-Core Processor
Stepping: 0
CPU MHz: 3584.018
CPU max MHz: 4000.0000
CPU min MHz: 1400.0000
BogoMIPS: 8027.22
Virtualization: AMD-V
L1d cache: 16K
L1i cache: 64K
L2 cache: 2048K
L3 cache: 8192K
NUMA node0 CPU(s): 0-7
Flags: fpu vme de pse tsc msr pae mce cx8 apic sep mtrr pge mca cmov pat pse36 clflush mmx fxsr sse sse2 ht syscall nx mmxext fxsr_opt pdpe1gb rdtscp lm constant_tsc rep_good nopl nonstop_tsc cpuid extd_apicid aperfmperf pni pclmulqdq monitor ssse3 fma cx16 sse4_1 sse4_2 popcnt aes xsave avx f16c lahf_lm cmp_legacy svm extapic cr8_legacy abm sse4a misalignsse 3dnowprefetch osvw ibs xop skinit wdt lwp fma4 tce nodeid_msr tbm topoext perfctr_core perfctr_nb cpb hw_pstate vmmcall bmi1 arat npt lbrv svm_lock nrip_save tsc_scale vmcb_clean flushbyasid decodeassists pausefilter pfthreshold
```
So this is model-2 bulldozer AMD CPU.
GCC agrees:
```
$ echo | gcc -E - -march=native -###
<...>
/usr/lib/gcc/x86_64-linux-gnu/7/cc1 -E -quiet -imultiarch x86_64-linux-gnu - "-march=bdver2" -mmmx -mno-3dnow -msse -msse2 -msse3 -mssse3 -msse4a -mcx16 -msahf -mno-movbe -maes -mno-sha -mpclmul -mpopcnt -mabm -mlwp -mfma -mfma4 -mxop -mbmi -mno-sgx -mno-bmi2 -mtbm -mavx -mno-avx2 -msse4.2 -msse4.1 -mlzcnt -mno-rtm -mno-hle -mno-rdrnd -mf16c -mno-fsgsbase -mno-rdseed -mprfchw -mno-adx -mfxsr -mxsave -mno-xsaveopt -mno-avx512f -mno-avx512er -mno-avx512cd -mno-avx512pf -mno-prefetchwt1 -mno-clflushopt -mno-xsavec -mno-xsaves -mno-avx512dq -mno-avx512bw -mno-avx512vl -mno-avx512ifma -mno-avx512vbmi -mno-avx5124fmaps -mno-avx5124vnniw -mno-clwb -mno-mwaitx -mno-clzero -mno-pku -mno-rdpid --param "l1-cache-size=16" --param "l1-cache-line-size=64" --param "l2-cache-size=2048" "-mtune=bdver2"
<...>
```
But clang does not: (look for `bdver1`)
```
$ echo | clang -E - -march=native -###
clang version 7.0.0- (trunk)
Target: x86_64-pc-linux-gnu
Thread model: posix
InstalledDir: /usr/local/bin
"/usr/lib/llvm-7/bin/clang" "-cc1" "-triple" "x86_64-pc-linux-gnu" "-E" "-disable-free" "-disable-llvm-verifier" "-discard-value-names" "-main-file-name" "-" "-mrelocation-model" "static" "-mthread-model" "posix" "-mdisable-fp-elim" "-fmath-errno" "-masm-verbose" "-mconstructor-aliases" "-munwind-tables" "-fuse-init-array" "-target-cpu" "bdver1" "-target-feature" "+sse2" "-target-feature" "+cx16" "-target-feature" "+sahf" "-target-feature" "+tbm" "-target-feature" "-avx512ifma" "-target-feature" "-sha" "-target-feature" "-gfni" "-target-feature" "+fma4" "-target-feature" "-vpclmulqdq" "-target-feature" "+prfchw" "-target-feature" "-bmi2" "-target-feature" "-cldemote" "-target-feature" "-fsgsbase" "-target-feature" "-xsavec" "-target-feature" "+popcnt" "-target-feature" "+aes" "-target-feature" "-avx512bitalg" "-target-feature" "-xsaves" "-target-feature" "-avx512er" "-target-feature" "-avx512vnni" "-target-feature" "-avx512vpopcntdq" "-target-feature" "-clwb" "-target-feature" "-avx512f" "-target-feature" "-clzero" "-target-feature" "-pku" "-target-feature" "+mmx" "-target-feature" "+lwp" "-target-feature" "-rdpid" "-target-feature" "+xop" "-target-feature" "-rdseed" "-target-feature" "-waitpkg" "-target-feature" "-ibt" "-target-feature" "+sse4a" "-target-feature" "-avx512bw" "-target-feature" "-clflushopt" "-target-feature" "+xsave" "-target-feature" "-avx512vbmi2" "-target-feature" "-avx512vl" "-target-feature" "-avx512cd" "-target-feature" "+avx" "-target-feature" "-vaes" "-target-feature" "-rtm" "-target-feature" "+fma" "-target-feature" "+bmi" "-target-feature" "-rdrnd" "-target-feature" "-mwaitx" "-target-feature" "+sse4.1" "-target-feature" "+sse4.2" "-target-feature" "-avx2" "-target-feature" "-wbnoinvd" "-target-feature" "+sse" "-target-feature" "+lzcnt" "-target-feature" "+pclmul" "-target-feature" "-prefetchwt1" "-target-feature" "+f16c" "-target-feature" "+ssse3" "-target-feature" "-sgx" "-target-feature" "-shstk" "-target-feature" "+cmov" "-target-feature" "-avx512vbmi" "-target-feature" "-movbe" "-target-feature" "-xsaveopt" "-target-feature" "-avx512dq" "-target-feature" "-adx" "-target-feature" "-avx512pf" "-target-feature" "+sse3" "-dwarf-column-info" "-debugger-tuning=gdb" "-resource-dir" "/usr/lib/llvm-7/lib/clang/7.0.0" "-internal-isystem" "/usr/local/include" "-internal-isystem" "/usr/lib/llvm-7/lib/clang/7.0.0/include" "-internal-externc-isystem" "/usr/include/x86_64-linux-gnu" "-internal-externc-isystem" "/include" "-internal-externc-isystem" "/usr/include" "-fdebug-compilation-dir" "/build/llvm-build-Clang-release" "-ferror-limit" "19" "-fmessage-length" "271" "-fobjc-runtime=gcc" "-fdiagnostics-show-option" "-fcolor-diagnostics" "-o" "-" "-x" "c" "-"
```
So clang, unlike gcc, considers this to be `bdver1`.
After some digging, i've come across `getAMDProcessorTypeAndSubtype()` in `Host.cpp`.
I have added the following debug printf after the call to that function in `sys::getHostCPUName()`:
```
errs() << "Family " << Family << " Model " << Model << " Type " << Type "\n";
```
Which produced:
```
Family 21 Model 2 Type 5
```
Which matches the `lscpu` output.
As it was pointed in the review by @craig.topper:
>>! In D46314#1084123, @craig.topper wrote:
> I dont' think this is right. Here is what I found on wikipedia. https://en.wikipedia.org/wiki/List_of_AMD_CPU_microarchitectures.
>
> AMD Bulldozer Family 15h - the successor of 10h/K10. Bulldozer is designed for processors in the 10 to 220W category, implementing XOP, FMA4 and CVT16 instruction sets. Orochi was the first design which implemented it. For Bulldozer, CPUID model numbers are 00h and 01h.
> AMD Piledriver Family 15h (2nd-gen) - successor to Bulldozer. CPUID model numbers are 02h (earliest "Vishera" Piledrivers) and 10h-1Fh.
> AMD Steamroller Family 15h (3rd-gen) - third-generation Bulldozer derived core. CPUID model numbers are 30h-3Fh.
> AMD Excavator Family 15h (4th-gen) - fourth-generation Bulldozer derived core. CPUID model numbers are 60h-6Fh, later updated revisions have model numbers 70h-7Fh.
>
>
> So there's a weird exception where model 2 should go with 0x10-0x1f.
Though It does not help that the code can't be tested at the moment.
With this logical change, the `bdver2` is properly detected.
```
$ echo | /build/llvm-build-Clang-release/bin/clang -E - -march=native -###
clang version 7.0.0 (trunk 331249) (llvm/trunk 331256)
Target: x86_64-unknown-linux-gnu
Thread model: posix
InstalledDir: /build/llvm-build-Clang-release/bin
"/build/llvm-build-Clang-release/bin/clang-7" "-cc1" "-triple" "x86_64-unknown-linux-gnu" "-E" "-disable-free" "-main-file-name" "-" "-mrelocation-model" "static" "-mthread-model" "posix" "-mdisable-fp-elim" "-fmath-errno" "-masm-verbose" "-mconstructor-aliases" "-munwind-tables" "-fuse-init-array" "-target-cpu" "bdver2" "-target-feature" "+sse2" "-target-feature" "+cx16" "-target-feature" "+sahf" "-target-feature" "+tbm" "-target-feature" "-avx512ifma" "-target-feature" "-sha" "-target-feature" "-gfni" "-target-feature" "+fma4" "-target-feature" "-vpclmulqdq" "-target-feature" "+prfchw" "-target-feature" "-bmi2" "-target-feature" "-cldemote" "-target-feature" "-fsgsbase" "-target-feature" "-xsavec" "-target-feature" "+popcnt" "-target-feature" "+aes" "-target-feature" "-avx512bitalg" "-target-feature" "-movdiri" "-target-feature" "-xsaves" "-target-feature" "-avx512er" "-target-feature" "-avx512vnni" "-target-feature" "-avx512vpopcntdq" "-target-feature" "-clwb" "-target-feature" "-avx512f" "-target-feature" "-clzero" "-target-feature" "-pku" "-target-feature" "+mmx" "-target-feature" "+lwp" "-target-feature" "-rdpid" "-target-feature" "+xop" "-target-feature" "-rdseed" "-target-feature" "-waitpkg" "-target-feature" "-movdir64b" "-target-feature" "-ibt" "-target-feature" "+sse4a" "-target-feature" "-avx512bw" "-target-feature" "-clflushopt" "-target-feature" "+xsave" "-target-feature" "-avx512vbmi2" "-target-feature" "-avx512vl" "-target-feature" "-avx512cd" "-target-feature" "+avx" "-target-feature" "-vaes" "-target-feature" "-rtm" "-target-feature" "+fma" "-target-feature" "+bmi" "-target-feature" "-rdrnd" "-target-feature" "-mwaitx" "-target-feature" "+sse4.1" "-target-feature" "+sse4.2" "-target-feature" "-avx2" "-target-feature" "-wbnoinvd" "-target-feature" "+sse" "-target-feature" "+lzcnt" "-target-feature" "+pclmul" "-target-feature" "-prefetchwt1" "-target-feature" "+f16c" "-target-feature" "+ssse3" "-target-feature" "-sgx" "-target-feature" "-shstk" "-target-feature" "+cmov" "-target-feature" "-avx512vbmi" "-target-feature" "-movbe" "-target-feature" "-xsaveopt" "-target-feature" "-avx512dq" "-target-feature" "-adx" "-target-feature" "-avx512pf" "-target-feature" "+sse3" "-dwarf-column-info" "-debugger-tuning=gdb" "-resource-dir" "/build/llvm-build-Clang-release/lib/clang/7.0.0" "-internal-isystem" "/usr/local/include" "-internal-isystem" "/build/llvm-build-Clang-release/lib/clang/7.0.0/include" "-internal-externc-isystem" "/usr/include/x86_64-linux-gnu" "-internal-externc-isystem" "/include" "-internal-externc-isystem" "/usr/include" "-fdebug-compilation-dir" "/build/llvm-build-Clang-release" "-ferror-limit" "19" "-fmessage-length" "271" "-fobjc-runtime=gcc" "-fdiagnostics-show-option" "-fcolor-diagnostics" "-o" "-" "-x" "c" "-"
```
Reviewers: craig.topper, GBuella, RKSimon, asbirlea, echristo, bkramer, spatel, andreadb, GGanesh
Reviewed By: craig.topper
Subscribers: sdardis, aprantl, arichardson, JDevlieghere, llvm-commits
Differential Revision: https://reviews.llvm.org/D46314
llvm-svn: 331294
2018-05-02 02:39:31 +08:00
|
|
|
if ((Model >= 0x10 && Model <= 0x1f) || Model == 0x02) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Subtype = X86::AMDFAM15H_BDVER2;
|
[X86][AMD][Bulldozer] Fix Bulldozer Model 2 detection.
Summary:
I have discovered an issue by accident.
```
$ lscpu
Architecture: x86_64
CPU op-mode(s): 32-bit, 64-bit
Byte Order: Little Endian
CPU(s): 8
On-line CPU(s) list: 0-7
Thread(s) per core: 2
Core(s) per socket: 4
Socket(s): 1
NUMA node(s): 1
Vendor ID: AuthenticAMD
CPU family: 21
Model: 2
Model name: AMD FX(tm)-8350 Eight-Core Processor
Stepping: 0
CPU MHz: 3584.018
CPU max MHz: 4000.0000
CPU min MHz: 1400.0000
BogoMIPS: 8027.22
Virtualization: AMD-V
L1d cache: 16K
L1i cache: 64K
L2 cache: 2048K
L3 cache: 8192K
NUMA node0 CPU(s): 0-7
Flags: fpu vme de pse tsc msr pae mce cx8 apic sep mtrr pge mca cmov pat pse36 clflush mmx fxsr sse sse2 ht syscall nx mmxext fxsr_opt pdpe1gb rdtscp lm constant_tsc rep_good nopl nonstop_tsc cpuid extd_apicid aperfmperf pni pclmulqdq monitor ssse3 fma cx16 sse4_1 sse4_2 popcnt aes xsave avx f16c lahf_lm cmp_legacy svm extapic cr8_legacy abm sse4a misalignsse 3dnowprefetch osvw ibs xop skinit wdt lwp fma4 tce nodeid_msr tbm topoext perfctr_core perfctr_nb cpb hw_pstate vmmcall bmi1 arat npt lbrv svm_lock nrip_save tsc_scale vmcb_clean flushbyasid decodeassists pausefilter pfthreshold
```
So this is model-2 bulldozer AMD CPU.
GCC agrees:
```
$ echo | gcc -E - -march=native -###
<...>
/usr/lib/gcc/x86_64-linux-gnu/7/cc1 -E -quiet -imultiarch x86_64-linux-gnu - "-march=bdver2" -mmmx -mno-3dnow -msse -msse2 -msse3 -mssse3 -msse4a -mcx16 -msahf -mno-movbe -maes -mno-sha -mpclmul -mpopcnt -mabm -mlwp -mfma -mfma4 -mxop -mbmi -mno-sgx -mno-bmi2 -mtbm -mavx -mno-avx2 -msse4.2 -msse4.1 -mlzcnt -mno-rtm -mno-hle -mno-rdrnd -mf16c -mno-fsgsbase -mno-rdseed -mprfchw -mno-adx -mfxsr -mxsave -mno-xsaveopt -mno-avx512f -mno-avx512er -mno-avx512cd -mno-avx512pf -mno-prefetchwt1 -mno-clflushopt -mno-xsavec -mno-xsaves -mno-avx512dq -mno-avx512bw -mno-avx512vl -mno-avx512ifma -mno-avx512vbmi -mno-avx5124fmaps -mno-avx5124vnniw -mno-clwb -mno-mwaitx -mno-clzero -mno-pku -mno-rdpid --param "l1-cache-size=16" --param "l1-cache-line-size=64" --param "l2-cache-size=2048" "-mtune=bdver2"
<...>
```
But clang does not: (look for `bdver1`)
```
$ echo | clang -E - -march=native -###
clang version 7.0.0- (trunk)
Target: x86_64-pc-linux-gnu
Thread model: posix
InstalledDir: /usr/local/bin
"/usr/lib/llvm-7/bin/clang" "-cc1" "-triple" "x86_64-pc-linux-gnu" "-E" "-disable-free" "-disable-llvm-verifier" "-discard-value-names" "-main-file-name" "-" "-mrelocation-model" "static" "-mthread-model" "posix" "-mdisable-fp-elim" "-fmath-errno" "-masm-verbose" "-mconstructor-aliases" "-munwind-tables" "-fuse-init-array" "-target-cpu" "bdver1" "-target-feature" "+sse2" "-target-feature" "+cx16" "-target-feature" "+sahf" "-target-feature" "+tbm" "-target-feature" "-avx512ifma" "-target-feature" "-sha" "-target-feature" "-gfni" "-target-feature" "+fma4" "-target-feature" "-vpclmulqdq" "-target-feature" "+prfchw" "-target-feature" "-bmi2" "-target-feature" "-cldemote" "-target-feature" "-fsgsbase" "-target-feature" "-xsavec" "-target-feature" "+popcnt" "-target-feature" "+aes" "-target-feature" "-avx512bitalg" "-target-feature" "-xsaves" "-target-feature" "-avx512er" "-target-feature" "-avx512vnni" "-target-feature" "-avx512vpopcntdq" "-target-feature" "-clwb" "-target-feature" "-avx512f" "-target-feature" "-clzero" "-target-feature" "-pku" "-target-feature" "+mmx" "-target-feature" "+lwp" "-target-feature" "-rdpid" "-target-feature" "+xop" "-target-feature" "-rdseed" "-target-feature" "-waitpkg" "-target-feature" "-ibt" "-target-feature" "+sse4a" "-target-feature" "-avx512bw" "-target-feature" "-clflushopt" "-target-feature" "+xsave" "-target-feature" "-avx512vbmi2" "-target-feature" "-avx512vl" "-target-feature" "-avx512cd" "-target-feature" "+avx" "-target-feature" "-vaes" "-target-feature" "-rtm" "-target-feature" "+fma" "-target-feature" "+bmi" "-target-feature" "-rdrnd" "-target-feature" "-mwaitx" "-target-feature" "+sse4.1" "-target-feature" "+sse4.2" "-target-feature" "-avx2" "-target-feature" "-wbnoinvd" "-target-feature" "+sse" "-target-feature" "+lzcnt" "-target-feature" "+pclmul" "-target-feature" "-prefetchwt1" "-target-feature" "+f16c" "-target-feature" "+ssse3" "-target-feature" "-sgx" "-target-feature" "-shstk" "-target-feature" "+cmov" "-target-feature" "-avx512vbmi" "-target-feature" "-movbe" "-target-feature" "-xsaveopt" "-target-feature" "-avx512dq" "-target-feature" "-adx" "-target-feature" "-avx512pf" "-target-feature" "+sse3" "-dwarf-column-info" "-debugger-tuning=gdb" "-resource-dir" "/usr/lib/llvm-7/lib/clang/7.0.0" "-internal-isystem" "/usr/local/include" "-internal-isystem" "/usr/lib/llvm-7/lib/clang/7.0.0/include" "-internal-externc-isystem" "/usr/include/x86_64-linux-gnu" "-internal-externc-isystem" "/include" "-internal-externc-isystem" "/usr/include" "-fdebug-compilation-dir" "/build/llvm-build-Clang-release" "-ferror-limit" "19" "-fmessage-length" "271" "-fobjc-runtime=gcc" "-fdiagnostics-show-option" "-fcolor-diagnostics" "-o" "-" "-x" "c" "-"
```
So clang, unlike gcc, considers this to be `bdver1`.
After some digging, i've come across `getAMDProcessorTypeAndSubtype()` in `Host.cpp`.
I have added the following debug printf after the call to that function in `sys::getHostCPUName()`:
```
errs() << "Family " << Family << " Model " << Model << " Type " << Type "\n";
```
Which produced:
```
Family 21 Model 2 Type 5
```
Which matches the `lscpu` output.
As it was pointed in the review by @craig.topper:
>>! In D46314#1084123, @craig.topper wrote:
> I dont' think this is right. Here is what I found on wikipedia. https://en.wikipedia.org/wiki/List_of_AMD_CPU_microarchitectures.
>
> AMD Bulldozer Family 15h - the successor of 10h/K10. Bulldozer is designed for processors in the 10 to 220W category, implementing XOP, FMA4 and CVT16 instruction sets. Orochi was the first design which implemented it. For Bulldozer, CPUID model numbers are 00h and 01h.
> AMD Piledriver Family 15h (2nd-gen) - successor to Bulldozer. CPUID model numbers are 02h (earliest "Vishera" Piledrivers) and 10h-1Fh.
> AMD Steamroller Family 15h (3rd-gen) - third-generation Bulldozer derived core. CPUID model numbers are 30h-3Fh.
> AMD Excavator Family 15h (4th-gen) - fourth-generation Bulldozer derived core. CPUID model numbers are 60h-6Fh, later updated revisions have model numbers 70h-7Fh.
>
>
> So there's a weird exception where model 2 should go with 0x10-0x1f.
Though It does not help that the code can't be tested at the moment.
With this logical change, the `bdver2` is properly detected.
```
$ echo | /build/llvm-build-Clang-release/bin/clang -E - -march=native -###
clang version 7.0.0 (trunk 331249) (llvm/trunk 331256)
Target: x86_64-unknown-linux-gnu
Thread model: posix
InstalledDir: /build/llvm-build-Clang-release/bin
"/build/llvm-build-Clang-release/bin/clang-7" "-cc1" "-triple" "x86_64-unknown-linux-gnu" "-E" "-disable-free" "-main-file-name" "-" "-mrelocation-model" "static" "-mthread-model" "posix" "-mdisable-fp-elim" "-fmath-errno" "-masm-verbose" "-mconstructor-aliases" "-munwind-tables" "-fuse-init-array" "-target-cpu" "bdver2" "-target-feature" "+sse2" "-target-feature" "+cx16" "-target-feature" "+sahf" "-target-feature" "+tbm" "-target-feature" "-avx512ifma" "-target-feature" "-sha" "-target-feature" "-gfni" "-target-feature" "+fma4" "-target-feature" "-vpclmulqdq" "-target-feature" "+prfchw" "-target-feature" "-bmi2" "-target-feature" "-cldemote" "-target-feature" "-fsgsbase" "-target-feature" "-xsavec" "-target-feature" "+popcnt" "-target-feature" "+aes" "-target-feature" "-avx512bitalg" "-target-feature" "-movdiri" "-target-feature" "-xsaves" "-target-feature" "-avx512er" "-target-feature" "-avx512vnni" "-target-feature" "-avx512vpopcntdq" "-target-feature" "-clwb" "-target-feature" "-avx512f" "-target-feature" "-clzero" "-target-feature" "-pku" "-target-feature" "+mmx" "-target-feature" "+lwp" "-target-feature" "-rdpid" "-target-feature" "+xop" "-target-feature" "-rdseed" "-target-feature" "-waitpkg" "-target-feature" "-movdir64b" "-target-feature" "-ibt" "-target-feature" "+sse4a" "-target-feature" "-avx512bw" "-target-feature" "-clflushopt" "-target-feature" "+xsave" "-target-feature" "-avx512vbmi2" "-target-feature" "-avx512vl" "-target-feature" "-avx512cd" "-target-feature" "+avx" "-target-feature" "-vaes" "-target-feature" "-rtm" "-target-feature" "+fma" "-target-feature" "+bmi" "-target-feature" "-rdrnd" "-target-feature" "-mwaitx" "-target-feature" "+sse4.1" "-target-feature" "+sse4.2" "-target-feature" "-avx2" "-target-feature" "-wbnoinvd" "-target-feature" "+sse" "-target-feature" "+lzcnt" "-target-feature" "+pclmul" "-target-feature" "-prefetchwt1" "-target-feature" "+f16c" "-target-feature" "+ssse3" "-target-feature" "-sgx" "-target-feature" "-shstk" "-target-feature" "+cmov" "-target-feature" "-avx512vbmi" "-target-feature" "-movbe" "-target-feature" "-xsaveopt" "-target-feature" "-avx512dq" "-target-feature" "-adx" "-target-feature" "-avx512pf" "-target-feature" "+sse3" "-dwarf-column-info" "-debugger-tuning=gdb" "-resource-dir" "/build/llvm-build-Clang-release/lib/clang/7.0.0" "-internal-isystem" "/usr/local/include" "-internal-isystem" "/build/llvm-build-Clang-release/lib/clang/7.0.0/include" "-internal-externc-isystem" "/usr/include/x86_64-linux-gnu" "-internal-externc-isystem" "/include" "-internal-externc-isystem" "/usr/include" "-fdebug-compilation-dir" "/build/llvm-build-Clang-release" "-ferror-limit" "19" "-fmessage-length" "271" "-fobjc-runtime=gcc" "-fdiagnostics-show-option" "-fcolor-diagnostics" "-o" "-" "-x" "c" "-"
```
Reviewers: craig.topper, GBuella, RKSimon, asbirlea, echristo, bkramer, spatel, andreadb, GGanesh
Reviewed By: craig.topper
Subscribers: sdardis, aprantl, arichardson, JDevlieghere, llvm-commits
Differential Revision: https://reviews.llvm.org/D46314
llvm-svn: 331294
2018-05-02 02:39:31 +08:00
|
|
|
break; // "bdver2"; 02h, 10h-1Fh: Piledriver
|
2017-05-03 23:42:29 +08:00
|
|
|
}
|
|
|
|
if (Model <= 0x0f) {
|
2017-11-11 01:10:57 +08:00
|
|
|
*Subtype = X86::AMDFAM15H_BDVER1;
|
2017-05-03 23:42:29 +08:00
|
|
|
break; // "bdver1"; 00h-0Fh: Bulldozer
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 22:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::AMD_BTVER2;
|
2017-05-03 23:42:29 +08:00
|
|
|
break; // "btver2"
|
|
|
|
case 23:
|
2017-11-11 01:10:57 +08:00
|
|
|
*Type = X86::AMDFAM17H;
|
2019-11-19 03:31:17 +08:00
|
|
|
if ((Model >= 0x30 && Model <= 0x3f) || Model == 0x71) {
|
2019-02-27 00:55:10 +08:00
|
|
|
*Subtype = X86::AMDFAM17H_ZNVER2;
|
2019-11-19 03:31:17 +08:00
|
|
|
break; // "znver2"; 30h-3fh, 71h: Zen2
|
2019-02-27 00:55:10 +08:00
|
|
|
}
|
|
|
|
if (Model <= 0x0f) {
|
|
|
|
*Subtype = X86::AMDFAM17H_ZNVER1;
|
|
|
|
break; // "znver1"; 00h-0Fh: Zen1
|
|
|
|
}
|
2017-05-03 23:42:29 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break; // "generic"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-12 14:49:58 +08:00
|
|
|
static void getAvailableFeatures(unsigned ECX, unsigned EDX, unsigned MaxLeaf,
|
2020-06-23 04:29:43 +08:00
|
|
|
unsigned *Features) {
|
2017-07-08 13:16:14 +08:00
|
|
|
unsigned EAX, EBX;
|
2017-07-12 14:49:58 +08:00
|
|
|
|
2018-10-20 21:16:31 +08:00
|
|
|
auto setFeature = [&](unsigned F) {
|
2020-06-13 08:13:49 +08:00
|
|
|
Features[F / 32] |= 1U << (F % 32);
|
2018-10-20 21:16:31 +08:00
|
|
|
};
|
2018-10-20 11:51:43 +08:00
|
|
|
|
2017-07-12 14:49:58 +08:00
|
|
|
if ((EDX >> 15) & 1)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_CMOV);
|
2017-07-12 14:49:58 +08:00
|
|
|
if ((EDX >> 23) & 1)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_MMX);
|
2017-07-12 14:49:58 +08:00
|
|
|
if ((EDX >> 25) & 1)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_SSE);
|
2017-07-12 14:49:58 +08:00
|
|
|
if ((EDX >> 26) & 1)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_SSE2);
|
2017-07-12 14:49:58 +08:00
|
|
|
|
|
|
|
if ((ECX >> 0) & 1)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_SSE3);
|
2017-07-12 14:49:58 +08:00
|
|
|
if ((ECX >> 1) & 1)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_PCLMUL);
|
2017-07-12 14:49:58 +08:00
|
|
|
if ((ECX >> 9) & 1)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_SSSE3);
|
2017-07-12 14:49:58 +08:00
|
|
|
if ((ECX >> 12) & 1)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_FMA);
|
2017-07-12 14:49:58 +08:00
|
|
|
if ((ECX >> 19) & 1)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_SSE4_1);
|
2017-07-12 14:49:58 +08:00
|
|
|
if ((ECX >> 20) & 1)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_SSE4_2);
|
2017-07-12 14:49:58 +08:00
|
|
|
if ((ECX >> 23) & 1)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_POPCNT);
|
2017-07-12 14:49:58 +08:00
|
|
|
if ((ECX >> 25) & 1)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_AES);
|
2017-07-12 14:49:58 +08:00
|
|
|
|
|
|
|
if ((ECX >> 22) & 1)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_MOVBE);
|
2017-05-03 23:42:29 +08:00
|
|
|
|
|
|
|
// If CPUID indicates support for XSAVE, XRESTORE and AVX, and XGETBV
|
|
|
|
// indicates that the AVX registers will be saved and restored on context
|
|
|
|
// switch, then we have full AVX support.
|
|
|
|
const unsigned AVXBits = (1 << 27) | (1 << 28);
|
|
|
|
bool HasAVX = ((ECX & AVXBits) == AVXBits) && !getX86XCR0(&EAX, &EDX) &&
|
|
|
|
((EAX & 0x6) == 0x6);
|
2019-11-21 17:03:16 +08:00
|
|
|
#if defined(__APPLE__)
|
|
|
|
// Darwin lazily saves the AVX512 context on first use: trust that the OS will
|
|
|
|
// save the AVX512 context if we use AVX512 instructions, even the bit is not
|
|
|
|
// set right now.
|
|
|
|
bool HasAVX512Save = true;
|
|
|
|
#else
|
|
|
|
// AVX512 requires additional context to be saved by the OS.
|
2017-05-03 23:42:29 +08:00
|
|
|
bool HasAVX512Save = HasAVX && ((EAX & 0xe0) == 0xe0);
|
2019-11-21 17:03:16 +08:00
|
|
|
#endif
|
2017-07-12 14:49:58 +08:00
|
|
|
|
|
|
|
if (HasAVX)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_AVX);
|
2017-07-12 14:49:58 +08:00
|
|
|
|
2017-05-03 23:42:29 +08:00
|
|
|
bool HasLeaf7 =
|
|
|
|
MaxLeaf >= 0x7 && !getX86CpuIDAndInfoEx(0x7, 0x0, &EAX, &EBX, &ECX, &EDX);
|
2017-07-12 14:49:58 +08:00
|
|
|
|
|
|
|
if (HasLeaf7 && ((EBX >> 3) & 1))
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_BMI);
|
2017-07-12 14:49:58 +08:00
|
|
|
if (HasLeaf7 && ((EBX >> 5) & 1) && HasAVX)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_AVX2);
|
2019-08-06 05:25:59 +08:00
|
|
|
if (HasLeaf7 && ((EBX >> 8) & 1))
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_BMI2);
|
2017-07-12 14:49:58 +08:00
|
|
|
if (HasLeaf7 && ((EBX >> 16) & 1) && HasAVX512Save)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_AVX512F);
|
2017-07-12 14:49:58 +08:00
|
|
|
if (HasLeaf7 && ((EBX >> 17) & 1) && HasAVX512Save)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_AVX512DQ);
|
2017-07-12 14:49:58 +08:00
|
|
|
if (HasLeaf7 && ((EBX >> 19) & 1))
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_ADX);
|
2017-07-12 14:49:58 +08:00
|
|
|
if (HasLeaf7 && ((EBX >> 21) & 1) && HasAVX512Save)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_AVX512IFMA);
|
2017-07-27 11:26:52 +08:00
|
|
|
if (HasLeaf7 && ((EBX >> 23) & 1))
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_CLFLUSHOPT);
|
2017-07-12 14:49:58 +08:00
|
|
|
if (HasLeaf7 && ((EBX >> 26) & 1) && HasAVX512Save)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_AVX512PF);
|
2017-07-12 14:49:58 +08:00
|
|
|
if (HasLeaf7 && ((EBX >> 27) & 1) && HasAVX512Save)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_AVX512ER);
|
2017-07-12 14:49:58 +08:00
|
|
|
if (HasLeaf7 && ((EBX >> 28) & 1) && HasAVX512Save)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_AVX512CD);
|
2017-07-27 11:26:52 +08:00
|
|
|
if (HasLeaf7 && ((EBX >> 29) & 1))
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_SHA);
|
2017-07-12 14:49:58 +08:00
|
|
|
if (HasLeaf7 && ((EBX >> 30) & 1) && HasAVX512Save)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_AVX512BW);
|
2017-07-12 14:49:58 +08:00
|
|
|
if (HasLeaf7 && ((EBX >> 31) & 1) && HasAVX512Save)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_AVX512VL);
|
2017-07-12 14:49:58 +08:00
|
|
|
|
|
|
|
if (HasLeaf7 && ((ECX >> 1) & 1) && HasAVX512Save)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_AVX512VBMI);
|
|
|
|
if (HasLeaf7 && ((ECX >> 6) & 1) && HasAVX512Save)
|
|
|
|
setFeature(X86::FEATURE_AVX512VBMI2);
|
|
|
|
if (HasLeaf7 && ((ECX >> 8) & 1))
|
|
|
|
setFeature(X86::FEATURE_GFNI);
|
|
|
|
if (HasLeaf7 && ((ECX >> 10) & 1) && HasAVX)
|
|
|
|
setFeature(X86::FEATURE_VPCLMULQDQ);
|
|
|
|
if (HasLeaf7 && ((ECX >> 11) & 1) && HasAVX512Save)
|
|
|
|
setFeature(X86::FEATURE_AVX512VNNI);
|
|
|
|
if (HasLeaf7 && ((ECX >> 12) & 1) && HasAVX512Save)
|
|
|
|
setFeature(X86::FEATURE_AVX512BITALG);
|
2017-07-12 14:49:58 +08:00
|
|
|
if (HasLeaf7 && ((ECX >> 14) & 1) && HasAVX512Save)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_AVX512VPOPCNTDQ);
|
2017-07-12 14:49:58 +08:00
|
|
|
|
|
|
|
if (HasLeaf7 && ((EDX >> 2) & 1) && HasAVX512Save)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_AVX5124VNNIW);
|
2017-07-12 14:49:58 +08:00
|
|
|
if (HasLeaf7 && ((EDX >> 3) & 1) && HasAVX512Save)
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_AVX5124FMAPS);
|
2019-08-12 09:29:46 +08:00
|
|
|
if (HasLeaf7 && ((EDX >> 8) & 1) && HasAVX512Save)
|
|
|
|
setFeature(X86::FEATURE_AVX512VP2INTERSECT);
|
2017-05-03 23:42:29 +08:00
|
|
|
|
2019-09-05 00:01:43 +08:00
|
|
|
bool HasLeaf7Subleaf1 =
|
|
|
|
MaxLeaf >= 7 && !getX86CpuIDAndInfoEx(0x7, 0x1, &EAX, &EBX, &ECX, &EDX);
|
|
|
|
if (HasLeaf7Subleaf1 && ((EAX >> 5) & 1) && HasAVX512Save)
|
|
|
|
setFeature(X86::FEATURE_AVX512BF16);
|
|
|
|
|
2017-07-08 13:16:13 +08:00
|
|
|
unsigned MaxExtLevel;
|
|
|
|
getX86CpuIDAndInfo(0x80000000, &MaxExtLevel, &EBX, &ECX, &EDX);
|
|
|
|
|
|
|
|
bool HasExtLeaf1 = MaxExtLevel >= 0x80000001 &&
|
|
|
|
!getX86CpuIDAndInfo(0x80000001, &EAX, &EBX, &ECX, &EDX);
|
2017-07-12 14:49:58 +08:00
|
|
|
if (HasExtLeaf1 && ((ECX >> 6) & 1))
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_SSE4_A);
|
2017-07-12 14:49:58 +08:00
|
|
|
if (HasExtLeaf1 && ((ECX >> 11) & 1))
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_XOP);
|
2017-07-12 14:49:58 +08:00
|
|
|
if (HasExtLeaf1 && ((ECX >> 16) & 1))
|
2018-10-20 11:51:43 +08:00
|
|
|
setFeature(X86::FEATURE_FMA4);
|
2017-07-08 13:16:13 +08:00
|
|
|
|
2017-07-12 14:49:58 +08:00
|
|
|
if (HasExtLeaf1 && ((EDX >> 29) & 1))
|
2020-07-09 23:47:35 +08:00
|
|
|
setFeature(X86::FEATURE_EM64T);
|
2017-05-03 23:42:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
StringRef sys::getHostCPUName() {
|
|
|
|
unsigned EAX = 0, EBX = 0, ECX = 0, EDX = 0;
|
|
|
|
unsigned MaxLeaf, Vendor;
|
|
|
|
|
2020-06-12 05:54:41 +08:00
|
|
|
if (!isCpuIdSupported())
|
2017-05-03 23:42:29 +08:00
|
|
|
return "generic";
|
2020-06-12 05:54:41 +08:00
|
|
|
|
2017-07-08 13:16:13 +08:00
|
|
|
if (getX86CpuIDAndInfo(0, &MaxLeaf, &Vendor, &ECX, &EDX) || MaxLeaf < 1)
|
2017-05-03 23:42:29 +08:00
|
|
|
return "generic";
|
2017-07-08 13:16:13 +08:00
|
|
|
getX86CpuIDAndInfo(0x1, &EAX, &EBX, &ECX, &EDX);
|
2017-05-03 23:42:29 +08:00
|
|
|
|
|
|
|
unsigned Family = 0, Model = 0;
|
2020-06-23 04:29:43 +08:00
|
|
|
unsigned Features[(X86::CPU_FEATURE_MAX + 31) / 32] = {0};
|
2017-05-03 23:42:29 +08:00
|
|
|
detectX86FamilyModel(EAX, &Family, &Model);
|
2020-06-13 08:13:49 +08:00
|
|
|
getAvailableFeatures(ECX, EDX, MaxLeaf, Features);
|
2017-05-03 23:42:29 +08:00
|
|
|
|
2017-11-04 02:02:44 +08:00
|
|
|
unsigned Type = 0;
|
|
|
|
unsigned Subtype = 0;
|
2017-05-03 23:42:29 +08:00
|
|
|
|
|
|
|
if (Vendor == SIG_INTEL) {
|
2020-06-13 11:38:30 +08:00
|
|
|
getIntelProcessorTypeAndSubtype(Family, Model, Features, &Type, &Subtype);
|
2017-05-03 23:42:29 +08:00
|
|
|
} else if (Vendor == SIG_AMD) {
|
|
|
|
getAMDProcessorTypeAndSubtype(Family, Model, Features, &Type, &Subtype);
|
|
|
|
}
|
2017-11-11 01:10:57 +08:00
|
|
|
|
|
|
|
// Check subtypes first since those are more specific.
|
|
|
|
#define X86_CPU_SUBTYPE(ARCHNAME, ENUM) \
|
|
|
|
if (Subtype == X86::ENUM) \
|
|
|
|
return ARCHNAME;
|
|
|
|
#include "llvm/Support/X86TargetParser.def"
|
|
|
|
|
|
|
|
// Now check types.
|
2018-03-07 06:45:31 +08:00
|
|
|
#define X86_CPU_TYPE(ARCHNAME, ENUM) \
|
2017-11-11 01:10:57 +08:00
|
|
|
if (Type == X86::ENUM) \
|
|
|
|
return ARCHNAME;
|
|
|
|
#include "llvm/Support/X86TargetParser.def"
|
|
|
|
|
2017-05-03 23:42:29 +08:00
|
|
|
return "generic";
|
|
|
|
}
|
|
|
|
|
|
|
|
#elif defined(__APPLE__) && (defined(__ppc__) || defined(__powerpc__))
|
|
|
|
StringRef sys::getHostCPUName() {
|
|
|
|
host_basic_info_data_t hostInfo;
|
|
|
|
mach_msg_type_number_t infoCount;
|
|
|
|
|
|
|
|
infoCount = HOST_BASIC_INFO_COUNT;
|
2018-09-04 18:54:09 +08:00
|
|
|
mach_port_t hostPort = mach_host_self();
|
|
|
|
host_info(hostPort, HOST_BASIC_INFO, (host_info_t)&hostInfo,
|
2017-05-03 23:42:29 +08:00
|
|
|
&infoCount);
|
2018-09-04 18:54:09 +08:00
|
|
|
mach_port_deallocate(mach_task_self(), hostPort);
|
2017-05-03 23:42:29 +08:00
|
|
|
|
|
|
|
if (hostInfo.cpu_type != CPU_TYPE_POWERPC)
|
|
|
|
return "generic";
|
|
|
|
|
|
|
|
switch (hostInfo.cpu_subtype) {
|
|
|
|
case CPU_SUBTYPE_POWERPC_601:
|
|
|
|
return "601";
|
|
|
|
case CPU_SUBTYPE_POWERPC_602:
|
|
|
|
return "602";
|
|
|
|
case CPU_SUBTYPE_POWERPC_603:
|
|
|
|
return "603";
|
|
|
|
case CPU_SUBTYPE_POWERPC_603e:
|
|
|
|
return "603e";
|
|
|
|
case CPU_SUBTYPE_POWERPC_603ev:
|
|
|
|
return "603ev";
|
|
|
|
case CPU_SUBTYPE_POWERPC_604:
|
|
|
|
return "604";
|
|
|
|
case CPU_SUBTYPE_POWERPC_604e:
|
|
|
|
return "604e";
|
|
|
|
case CPU_SUBTYPE_POWERPC_620:
|
|
|
|
return "620";
|
|
|
|
case CPU_SUBTYPE_POWERPC_750:
|
|
|
|
return "750";
|
|
|
|
case CPU_SUBTYPE_POWERPC_7400:
|
|
|
|
return "7400";
|
|
|
|
case CPU_SUBTYPE_POWERPC_7450:
|
|
|
|
return "7450";
|
|
|
|
case CPU_SUBTYPE_POWERPC_970:
|
|
|
|
return "970";
|
|
|
|
default:;
|
|
|
|
}
|
|
|
|
|
|
|
|
return "generic";
|
|
|
|
}
|
|
|
|
#elif defined(__linux__) && (defined(__ppc__) || defined(__powerpc__))
|
|
|
|
StringRef sys::getHostCPUName() {
|
|
|
|
std::unique_ptr<llvm::MemoryBuffer> P = getProcCpuinfoContent();
|
2018-03-08 01:53:16 +08:00
|
|
|
StringRef Content = P ? P->getBuffer() : "";
|
2017-05-03 23:42:29 +08:00
|
|
|
return detail::getHostCPUNameForPowerPC(Content);
|
|
|
|
}
|
|
|
|
#elif defined(__linux__) && (defined(__arm__) || defined(__aarch64__))
|
|
|
|
StringRef sys::getHostCPUName() {
|
|
|
|
std::unique_ptr<llvm::MemoryBuffer> P = getProcCpuinfoContent();
|
2018-03-08 01:53:16 +08:00
|
|
|
StringRef Content = P ? P->getBuffer() : "";
|
2017-05-03 23:42:29 +08:00
|
|
|
return detail::getHostCPUNameForARM(Content);
|
|
|
|
}
|
|
|
|
#elif defined(__linux__) && defined(__s390x__)
|
|
|
|
StringRef sys::getHostCPUName() {
|
|
|
|
std::unique_ptr<llvm::MemoryBuffer> P = getProcCpuinfoContent();
|
2018-03-08 01:53:16 +08:00
|
|
|
StringRef Content = P ? P->getBuffer() : "";
|
2017-05-03 23:42:29 +08:00
|
|
|
return detail::getHostCPUNameForS390x(Content);
|
|
|
|
}
|
2019-10-31 03:50:04 +08:00
|
|
|
#elif defined(__APPLE__) && defined(__aarch64__)
|
|
|
|
StringRef sys::getHostCPUName() {
|
|
|
|
return "cyclone";
|
|
|
|
}
|
|
|
|
#elif defined(__APPLE__) && defined(__arm__)
|
|
|
|
StringRef sys::getHostCPUName() {
|
|
|
|
host_basic_info_data_t hostInfo;
|
|
|
|
mach_msg_type_number_t infoCount;
|
|
|
|
|
|
|
|
infoCount = HOST_BASIC_INFO_COUNT;
|
|
|
|
mach_port_t hostPort = mach_host_self();
|
|
|
|
host_info(hostPort, HOST_BASIC_INFO, (host_info_t)&hostInfo,
|
|
|
|
&infoCount);
|
|
|
|
mach_port_deallocate(mach_task_self(), hostPort);
|
|
|
|
|
|
|
|
if (hostInfo.cpu_type != CPU_TYPE_ARM) {
|
|
|
|
assert(false && "CPUType not equal to ARM should not be possible on ARM");
|
|
|
|
return "generic";
|
|
|
|
}
|
|
|
|
switch (hostInfo.cpu_subtype) {
|
|
|
|
case CPU_SUBTYPE_ARM_V7S:
|
|
|
|
return "swift";
|
|
|
|
default:;
|
|
|
|
}
|
2020-02-18 10:48:38 +08:00
|
|
|
|
2019-10-31 03:50:04 +08:00
|
|
|
return "generic";
|
|
|
|
}
|
2017-05-03 23:42:29 +08:00
|
|
|
#else
|
|
|
|
StringRef sys::getHostCPUName() { return "generic"; }
|
|
|
|
#endif
|
|
|
|
|
2020-04-17 03:05:53 +08:00
|
|
|
#if defined(__linux__) && (defined(__i386__) || defined(__x86_64__))
|
2017-05-03 23:42:29 +08:00
|
|
|
// On Linux, the number of physical cores can be computed from /proc/cpuinfo,
|
|
|
|
// using the number of unique physical/core id pairs. The following
|
|
|
|
// implementation reads the /proc/cpuinfo format on an x86_64 system.
|
[Support] On Windows, ensure hardware_concurrency() extends to all CPU sockets and all NUMA groups
The goal of this patch is to maximize CPU utilization on multi-socket or high core count systems, so that parallel computations such as LLD/ThinLTO can use all hardware threads in the system. Before this patch, on Windows, a maximum of 64 hardware threads could be used at most, in some cases dispatched only on one CPU socket.
== Background ==
Windows doesn't have a flat cpu_set_t like Linux. Instead, it projects hardware CPUs (or NUMA nodes) to applications through a concept of "processor groups". A "processor" is the smallest unit of execution on a CPU, that is, an hyper-thread if SMT is active; a core otherwise. There's a limit of 32-bit processors on older 32-bit versions of Windows, which later was raised to 64-processors with 64-bit versions of Windows. This limit comes from the affinity mask, which historically is represented by the sizeof(void*). Consequently, the concept of "processor groups" was introduced for dealing with systems with more than 64 hyper-threads.
By default, the Windows OS assigns only one "processor group" to each starting application, in a round-robin manner. If the application wants to use more processors, it needs to programmatically enable it, by assigning threads to other "processor groups". This also means that affinity cannot cross "processor group" boundaries; one can only specify a "preferred" group on start-up, but the application is free to allocate more groups if it wants to.
This creates a peculiar situation, where newer CPUs like the AMD EPYC 7702P (64-cores, 128-hyperthreads) are projected by the OS as two (2) "processor groups". This means that by default, an application can only use half of the cores. This situation could only get worse in the years to come, as dies with more cores will appear on the market.
== The problem ==
The heavyweight_hardware_concurrency() API was introduced so that only *one hardware thread per core* was used. Once that API returns, that original intention is lost, only the number of threads is retained. Consider a situation, on Windows, where the system has 2 CPU sockets, 18 cores each, each core having 2 hyper-threads, for a total of 72 hyper-threads. Both heavyweight_hardware_concurrency() and hardware_concurrency() currently return 36, because on Windows they are simply wrappers over std::thread::hardware_concurrency() -- which can only return processors from the current "processor group".
== The changes in this patch ==
To solve this situation, we capture (and retain) the initial intention until the point of usage, through a new ThreadPoolStrategy class. The number of threads to use is deferred as late as possible, until the moment where the std::threads are created (ThreadPool in the case of ThinLTO).
When using hardware_concurrency(), setting ThreadCount to 0 now means to use all the possible hardware CPU (SMT) threads. Providing a ThreadCount above to the maximum number of threads will have no effect, the maximum will be used instead.
The heavyweight_hardware_concurrency() is similar to hardware_concurrency(), except that only one thread per hardware *core* will be used.
When LLVM_ENABLE_THREADS is OFF, the threading APIs will always return 1, to ensure any caller loops will be exercised at least once.
Differential Revision: https://reviews.llvm.org/D71775
2020-02-14 11:49:57 +08:00
|
|
|
int computeHostNumPhysicalCores() {
|
2020-04-17 03:05:53 +08:00
|
|
|
// Enabled represents the number of physical id/core id pairs with at least
|
|
|
|
// one processor id enabled by the CPU affinity mask.
|
|
|
|
cpu_set_t Affinity, Enabled;
|
|
|
|
if (sched_getaffinity(0, sizeof(Affinity), &Affinity) != 0)
|
|
|
|
return -1;
|
|
|
|
CPU_ZERO(&Enabled);
|
|
|
|
|
2017-05-03 23:42:29 +08:00
|
|
|
// Read /proc/cpuinfo as a stream (until EOF reached). It cannot be
|
|
|
|
// mmapped because it appears to have 0 size.
|
|
|
|
llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> Text =
|
|
|
|
llvm::MemoryBuffer::getFileAsStream("/proc/cpuinfo");
|
|
|
|
if (std::error_code EC = Text.getError()) {
|
|
|
|
llvm::errs() << "Can't read "
|
|
|
|
<< "/proc/cpuinfo: " << EC.message() << "\n";
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
SmallVector<StringRef, 8> strs;
|
|
|
|
(*Text)->getBuffer().split(strs, "\n", /*MaxSplit=*/-1,
|
|
|
|
/*KeepEmpty=*/false);
|
2020-04-17 03:05:53 +08:00
|
|
|
int CurProcessor = -1;
|
2017-05-03 23:42:29 +08:00
|
|
|
int CurPhysicalId = -1;
|
2020-04-17 03:05:53 +08:00
|
|
|
int CurSiblings = -1;
|
2017-05-03 23:42:29 +08:00
|
|
|
int CurCoreId = -1;
|
2020-04-17 03:05:53 +08:00
|
|
|
for (StringRef Line : strs) {
|
2017-05-03 23:42:29 +08:00
|
|
|
std::pair<StringRef, StringRef> Data = Line.split(':');
|
|
|
|
auto Name = Data.first.trim();
|
|
|
|
auto Val = Data.second.trim();
|
2020-04-17 03:05:53 +08:00
|
|
|
// These fields are available if the kernel is configured with CONFIG_SMP.
|
|
|
|
if (Name == "processor")
|
|
|
|
Val.getAsInteger(10, CurProcessor);
|
|
|
|
else if (Name == "physical id")
|
2017-05-03 23:42:29 +08:00
|
|
|
Val.getAsInteger(10, CurPhysicalId);
|
2020-04-17 03:05:53 +08:00
|
|
|
else if (Name == "siblings")
|
|
|
|
Val.getAsInteger(10, CurSiblings);
|
|
|
|
else if (Name == "core id") {
|
2017-05-03 23:42:29 +08:00
|
|
|
Val.getAsInteger(10, CurCoreId);
|
2020-04-17 03:05:53 +08:00
|
|
|
// The processor id corresponds to an index into cpu_set_t.
|
|
|
|
if (CPU_ISSET(CurProcessor, &Affinity))
|
|
|
|
CPU_SET(CurPhysicalId * CurSiblings + CurCoreId, &Enabled);
|
2017-05-03 23:42:29 +08:00
|
|
|
}
|
|
|
|
}
|
2020-04-17 03:05:53 +08:00
|
|
|
return CPU_COUNT(&Enabled);
|
2017-05-03 23:42:29 +08:00
|
|
|
}
|
|
|
|
#elif defined(__APPLE__) && defined(__x86_64__)
|
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/sysctl.h>
|
|
|
|
|
|
|
|
// Gets the number of *physical cores* on the machine.
|
[Support] On Windows, ensure hardware_concurrency() extends to all CPU sockets and all NUMA groups
The goal of this patch is to maximize CPU utilization on multi-socket or high core count systems, so that parallel computations such as LLD/ThinLTO can use all hardware threads in the system. Before this patch, on Windows, a maximum of 64 hardware threads could be used at most, in some cases dispatched only on one CPU socket.
== Background ==
Windows doesn't have a flat cpu_set_t like Linux. Instead, it projects hardware CPUs (or NUMA nodes) to applications through a concept of "processor groups". A "processor" is the smallest unit of execution on a CPU, that is, an hyper-thread if SMT is active; a core otherwise. There's a limit of 32-bit processors on older 32-bit versions of Windows, which later was raised to 64-processors with 64-bit versions of Windows. This limit comes from the affinity mask, which historically is represented by the sizeof(void*). Consequently, the concept of "processor groups" was introduced for dealing with systems with more than 64 hyper-threads.
By default, the Windows OS assigns only one "processor group" to each starting application, in a round-robin manner. If the application wants to use more processors, it needs to programmatically enable it, by assigning threads to other "processor groups". This also means that affinity cannot cross "processor group" boundaries; one can only specify a "preferred" group on start-up, but the application is free to allocate more groups if it wants to.
This creates a peculiar situation, where newer CPUs like the AMD EPYC 7702P (64-cores, 128-hyperthreads) are projected by the OS as two (2) "processor groups". This means that by default, an application can only use half of the cores. This situation could only get worse in the years to come, as dies with more cores will appear on the market.
== The problem ==
The heavyweight_hardware_concurrency() API was introduced so that only *one hardware thread per core* was used. Once that API returns, that original intention is lost, only the number of threads is retained. Consider a situation, on Windows, where the system has 2 CPU sockets, 18 cores each, each core having 2 hyper-threads, for a total of 72 hyper-threads. Both heavyweight_hardware_concurrency() and hardware_concurrency() currently return 36, because on Windows they are simply wrappers over std::thread::hardware_concurrency() -- which can only return processors from the current "processor group".
== The changes in this patch ==
To solve this situation, we capture (and retain) the initial intention until the point of usage, through a new ThreadPoolStrategy class. The number of threads to use is deferred as late as possible, until the moment where the std::threads are created (ThreadPool in the case of ThinLTO).
When using hardware_concurrency(), setting ThreadCount to 0 now means to use all the possible hardware CPU (SMT) threads. Providing a ThreadCount above to the maximum number of threads will have no effect, the maximum will be used instead.
The heavyweight_hardware_concurrency() is similar to hardware_concurrency(), except that only one thread per hardware *core* will be used.
When LLVM_ENABLE_THREADS is OFF, the threading APIs will always return 1, to ensure any caller loops will be exercised at least once.
Differential Revision: https://reviews.llvm.org/D71775
2020-02-14 11:49:57 +08:00
|
|
|
int computeHostNumPhysicalCores() {
|
2017-05-03 23:42:29 +08:00
|
|
|
uint32_t count;
|
|
|
|
size_t len = sizeof(count);
|
|
|
|
sysctlbyname("hw.physicalcpu", &count, &len, NULL, 0);
|
|
|
|
if (count < 1) {
|
|
|
|
int nm[2];
|
|
|
|
nm[0] = CTL_HW;
|
|
|
|
nm[1] = HW_AVAILCPU;
|
|
|
|
sysctl(nm, 2, &count, &len, NULL, 0);
|
|
|
|
if (count < 1)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return count;
|
|
|
|
}
|
2020-02-15 07:31:45 +08:00
|
|
|
#elif defined(_WIN32) && LLVM_ENABLE_THREADS != 0
|
[Support] On Windows, ensure hardware_concurrency() extends to all CPU sockets and all NUMA groups
The goal of this patch is to maximize CPU utilization on multi-socket or high core count systems, so that parallel computations such as LLD/ThinLTO can use all hardware threads in the system. Before this patch, on Windows, a maximum of 64 hardware threads could be used at most, in some cases dispatched only on one CPU socket.
== Background ==
Windows doesn't have a flat cpu_set_t like Linux. Instead, it projects hardware CPUs (or NUMA nodes) to applications through a concept of "processor groups". A "processor" is the smallest unit of execution on a CPU, that is, an hyper-thread if SMT is active; a core otherwise. There's a limit of 32-bit processors on older 32-bit versions of Windows, which later was raised to 64-processors with 64-bit versions of Windows. This limit comes from the affinity mask, which historically is represented by the sizeof(void*). Consequently, the concept of "processor groups" was introduced for dealing with systems with more than 64 hyper-threads.
By default, the Windows OS assigns only one "processor group" to each starting application, in a round-robin manner. If the application wants to use more processors, it needs to programmatically enable it, by assigning threads to other "processor groups". This also means that affinity cannot cross "processor group" boundaries; one can only specify a "preferred" group on start-up, but the application is free to allocate more groups if it wants to.
This creates a peculiar situation, where newer CPUs like the AMD EPYC 7702P (64-cores, 128-hyperthreads) are projected by the OS as two (2) "processor groups". This means that by default, an application can only use half of the cores. This situation could only get worse in the years to come, as dies with more cores will appear on the market.
== The problem ==
The heavyweight_hardware_concurrency() API was introduced so that only *one hardware thread per core* was used. Once that API returns, that original intention is lost, only the number of threads is retained. Consider a situation, on Windows, where the system has 2 CPU sockets, 18 cores each, each core having 2 hyper-threads, for a total of 72 hyper-threads. Both heavyweight_hardware_concurrency() and hardware_concurrency() currently return 36, because on Windows they are simply wrappers over std::thread::hardware_concurrency() -- which can only return processors from the current "processor group".
== The changes in this patch ==
To solve this situation, we capture (and retain) the initial intention until the point of usage, through a new ThreadPoolStrategy class. The number of threads to use is deferred as late as possible, until the moment where the std::threads are created (ThreadPool in the case of ThinLTO).
When using hardware_concurrency(), setting ThreadCount to 0 now means to use all the possible hardware CPU (SMT) threads. Providing a ThreadCount above to the maximum number of threads will have no effect, the maximum will be used instead.
The heavyweight_hardware_concurrency() is similar to hardware_concurrency(), except that only one thread per hardware *core* will be used.
When LLVM_ENABLE_THREADS is OFF, the threading APIs will always return 1, to ensure any caller loops will be exercised at least once.
Differential Revision: https://reviews.llvm.org/D71775
2020-02-14 11:49:57 +08:00
|
|
|
// Defined in llvm/lib/Support/Windows/Threading.inc
|
|
|
|
int computeHostNumPhysicalCores();
|
2017-05-03 23:42:29 +08:00
|
|
|
#else
|
|
|
|
// On other systems, return -1 to indicate unknown.
|
|
|
|
static int computeHostNumPhysicalCores() { return -1; }
|
|
|
|
#endif
|
|
|
|
|
|
|
|
int sys::getHostNumPhysicalCores() {
|
|
|
|
static int NumCores = computeHostNumPhysicalCores();
|
|
|
|
return NumCores;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(__i386__) || defined(_M_IX86) || \
|
|
|
|
defined(__x86_64__) || defined(_M_X64)
|
|
|
|
bool sys::getHostCPUFeatures(StringMap<bool> &Features) {
|
|
|
|
unsigned EAX = 0, EBX = 0, ECX = 0, EDX = 0;
|
|
|
|
unsigned MaxLevel;
|
|
|
|
|
2020-07-09 14:40:37 +08:00
|
|
|
if (getX86CpuIDAndInfo(0, &MaxLevel, &EBX, &ECX, &EDX) || MaxLevel < 1)
|
2017-05-03 23:42:29 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
getX86CpuIDAndInfo(1, &EAX, &EBX, &ECX, &EDX);
|
|
|
|
|
2019-03-21 07:35:49 +08:00
|
|
|
Features["cx8"] = (EDX >> 8) & 1;
|
2017-11-20 07:30:22 +08:00
|
|
|
Features["cmov"] = (EDX >> 15) & 1;
|
|
|
|
Features["mmx"] = (EDX >> 23) & 1;
|
2019-02-14 02:21:36 +08:00
|
|
|
Features["fxsr"] = (EDX >> 24) & 1;
|
2017-11-20 07:30:22 +08:00
|
|
|
Features["sse"] = (EDX >> 25) & 1;
|
|
|
|
Features["sse2"] = (EDX >> 26) & 1;
|
|
|
|
|
|
|
|
Features["sse3"] = (ECX >> 0) & 1;
|
|
|
|
Features["pclmul"] = (ECX >> 1) & 1;
|
|
|
|
Features["ssse3"] = (ECX >> 9) & 1;
|
|
|
|
Features["cx16"] = (ECX >> 13) & 1;
|
2017-05-03 23:42:29 +08:00
|
|
|
Features["sse4.1"] = (ECX >> 19) & 1;
|
|
|
|
Features["sse4.2"] = (ECX >> 20) & 1;
|
2017-11-20 07:30:22 +08:00
|
|
|
Features["movbe"] = (ECX >> 22) & 1;
|
2017-05-03 23:42:29 +08:00
|
|
|
Features["popcnt"] = (ECX >> 23) & 1;
|
2017-11-20 07:30:22 +08:00
|
|
|
Features["aes"] = (ECX >> 25) & 1;
|
|
|
|
Features["rdrnd"] = (ECX >> 30) & 1;
|
2017-05-03 23:42:29 +08:00
|
|
|
|
|
|
|
// If CPUID indicates support for XSAVE, XRESTORE and AVX, and XGETBV
|
|
|
|
// indicates that the AVX registers will be saved and restored on context
|
|
|
|
// switch, then we have full AVX support.
|
2020-07-02 08:36:45 +08:00
|
|
|
bool HasXSave = ((ECX >> 27) & 1) && !getX86XCR0(&EAX, &EDX);
|
|
|
|
bool HasAVXSave = HasXSave && ((ECX >> 28) & 1) && ((EAX & 0x6) == 0x6);
|
2019-11-21 17:03:16 +08:00
|
|
|
#if defined(__APPLE__)
|
|
|
|
// Darwin lazily saves the AVX512 context on first use: trust that the OS will
|
|
|
|
// save the AVX512 context if we use AVX512 instructions, even the bit is not
|
|
|
|
// set right now.
|
|
|
|
bool HasAVX512Save = true;
|
|
|
|
#else
|
2017-05-03 23:42:29 +08:00
|
|
|
// AVX512 requires additional context to be saved by the OS.
|
|
|
|
bool HasAVX512Save = HasAVXSave && ((EAX & 0xe0) == 0xe0);
|
2019-11-21 17:03:16 +08:00
|
|
|
#endif
|
2020-07-02 08:36:45 +08:00
|
|
|
// AMX requires additional context to be saved by the OS.
|
|
|
|
const unsigned AMXBits = (1 << 17) | (1 << 18);
|
|
|
|
bool HasAMXSave = HasXSave && ((EAX & AMXBits) == AMXBits);
|
2017-05-03 23:42:29 +08:00
|
|
|
|
2017-11-20 07:30:22 +08:00
|
|
|
Features["avx"] = HasAVXSave;
|
|
|
|
Features["fma"] = ((ECX >> 12) & 1) && HasAVXSave;
|
|
|
|
// Only enable XSAVE if OS has enabled support for saving YMM state.
|
|
|
|
Features["xsave"] = ((ECX >> 26) & 1) && HasAVXSave;
|
|
|
|
Features["f16c"] = ((ECX >> 29) & 1) && HasAVXSave;
|
|
|
|
|
2017-05-03 23:42:29 +08:00
|
|
|
unsigned MaxExtLevel;
|
|
|
|
getX86CpuIDAndInfo(0x80000000, &MaxExtLevel, &EBX, &ECX, &EDX);
|
|
|
|
|
|
|
|
bool HasExtLeaf1 = MaxExtLevel >= 0x80000001 &&
|
|
|
|
!getX86CpuIDAndInfo(0x80000001, &EAX, &EBX, &ECX, &EDX);
|
2018-02-18 00:52:49 +08:00
|
|
|
Features["sahf"] = HasExtLeaf1 && ((ECX >> 0) & 1);
|
2017-11-20 07:30:22 +08:00
|
|
|
Features["lzcnt"] = HasExtLeaf1 && ((ECX >> 5) & 1);
|
|
|
|
Features["sse4a"] = HasExtLeaf1 && ((ECX >> 6) & 1);
|
|
|
|
Features["prfchw"] = HasExtLeaf1 && ((ECX >> 8) & 1);
|
|
|
|
Features["xop"] = HasExtLeaf1 && ((ECX >> 11) & 1) && HasAVXSave;
|
|
|
|
Features["lwp"] = HasExtLeaf1 && ((ECX >> 15) & 1);
|
|
|
|
Features["fma4"] = HasExtLeaf1 && ((ECX >> 16) & 1) && HasAVXSave;
|
|
|
|
Features["tbm"] = HasExtLeaf1 && ((ECX >> 21) & 1);
|
2017-05-03 23:42:29 +08:00
|
|
|
Features["mwaitx"] = HasExtLeaf1 && ((ECX >> 29) & 1);
|
|
|
|
|
2018-09-25 02:55:41 +08:00
|
|
|
Features["64bit"] = HasExtLeaf1 && ((EDX >> 29) & 1);
|
|
|
|
|
2018-04-12 04:01:57 +08:00
|
|
|
// Miscellaneous memory related features, detected by
|
|
|
|
// using the 0x80000008 leaf of the CPUID instruction
|
2017-05-03 23:42:29 +08:00
|
|
|
bool HasExtLeaf8 = MaxExtLevel >= 0x80000008 &&
|
2017-11-20 07:49:19 +08:00
|
|
|
!getX86CpuIDAndInfo(0x80000008, &EAX, &EBX, &ECX, &EDX);
|
2018-04-12 04:01:57 +08:00
|
|
|
Features["clzero"] = HasExtLeaf8 && ((EBX >> 0) & 1);
|
|
|
|
Features["wbnoinvd"] = HasExtLeaf8 && ((EBX >> 9) & 1);
|
2017-05-03 23:42:29 +08:00
|
|
|
|
|
|
|
bool HasLeaf7 =
|
|
|
|
MaxLevel >= 7 && !getX86CpuIDAndInfoEx(0x7, 0x0, &EAX, &EBX, &ECX, &EDX);
|
|
|
|
|
2017-11-20 07:30:22 +08:00
|
|
|
Features["fsgsbase"] = HasLeaf7 && ((EBX >> 0) & 1);
|
|
|
|
Features["sgx"] = HasLeaf7 && ((EBX >> 2) & 1);
|
|
|
|
Features["bmi"] = HasLeaf7 && ((EBX >> 3) & 1);
|
2017-05-03 23:42:29 +08:00
|
|
|
// AVX2 is only supported if we have the OS save support from AVX.
|
2017-11-20 07:30:22 +08:00
|
|
|
Features["avx2"] = HasLeaf7 && ((EBX >> 5) & 1) && HasAVXSave;
|
|
|
|
Features["bmi2"] = HasLeaf7 && ((EBX >> 8) & 1);
|
2018-05-25 14:32:05 +08:00
|
|
|
Features["invpcid"] = HasLeaf7 && ((EBX >> 10) & 1);
|
2017-11-20 07:30:22 +08:00
|
|
|
Features["rtm"] = HasLeaf7 && ((EBX >> 11) & 1);
|
2017-05-03 23:42:29 +08:00
|
|
|
// AVX512 is only supported if the OS supports the context save for it.
|
2017-11-20 07:30:22 +08:00
|
|
|
Features["avx512f"] = HasLeaf7 && ((EBX >> 16) & 1) && HasAVX512Save;
|
|
|
|
Features["avx512dq"] = HasLeaf7 && ((EBX >> 17) & 1) && HasAVX512Save;
|
|
|
|
Features["rdseed"] = HasLeaf7 && ((EBX >> 18) & 1);
|
|
|
|
Features["adx"] = HasLeaf7 && ((EBX >> 19) & 1);
|
2017-05-03 23:42:29 +08:00
|
|
|
Features["avx512ifma"] = HasLeaf7 && ((EBX >> 21) & 1) && HasAVX512Save;
|
2017-11-20 07:30:22 +08:00
|
|
|
Features["clflushopt"] = HasLeaf7 && ((EBX >> 23) & 1);
|
|
|
|
Features["clwb"] = HasLeaf7 && ((EBX >> 24) & 1);
|
|
|
|
Features["avx512pf"] = HasLeaf7 && ((EBX >> 26) & 1) && HasAVX512Save;
|
|
|
|
Features["avx512er"] = HasLeaf7 && ((EBX >> 27) & 1) && HasAVX512Save;
|
|
|
|
Features["avx512cd"] = HasLeaf7 && ((EBX >> 28) & 1) && HasAVX512Save;
|
|
|
|
Features["sha"] = HasLeaf7 && ((EBX >> 29) & 1);
|
|
|
|
Features["avx512bw"] = HasLeaf7 && ((EBX >> 30) & 1) && HasAVX512Save;
|
|
|
|
Features["avx512vl"] = HasLeaf7 && ((EBX >> 31) & 1) && HasAVX512Save;
|
|
|
|
|
|
|
|
Features["prefetchwt1"] = HasLeaf7 && ((ECX >> 0) & 1);
|
|
|
|
Features["avx512vbmi"] = HasLeaf7 && ((ECX >> 1) & 1) && HasAVX512Save;
|
2017-11-22 02:50:41 +08:00
|
|
|
Features["pku"] = HasLeaf7 && ((ECX >> 4) & 1);
|
2018-04-21 02:42:47 +08:00
|
|
|
Features["waitpkg"] = HasLeaf7 && ((ECX >> 5) & 1);
|
2017-11-21 17:48:44 +08:00
|
|
|
Features["avx512vbmi2"] = HasLeaf7 && ((ECX >> 6) & 1) && HasAVX512Save;
|
2017-11-26 21:02:45 +08:00
|
|
|
Features["shstk"] = HasLeaf7 && ((ECX >> 7) & 1);
|
2017-11-26 17:36:41 +08:00
|
|
|
Features["gfni"] = HasLeaf7 && ((ECX >> 8) & 1);
|
2017-11-22 02:50:41 +08:00
|
|
|
Features["vaes"] = HasLeaf7 && ((ECX >> 9) & 1) && HasAVXSave;
|
|
|
|
Features["vpclmulqdq"] = HasLeaf7 && ((ECX >> 10) & 1) && HasAVXSave;
|
|
|
|
Features["avx512vnni"] = HasLeaf7 && ((ECX >> 11) & 1) && HasAVX512Save;
|
|
|
|
Features["avx512bitalg"] = HasLeaf7 && ((ECX >> 12) & 1) && HasAVX512Save;
|
2017-08-23 12:25:57 +08:00
|
|
|
Features["avx512vpopcntdq"] = HasLeaf7 && ((ECX >> 14) & 1) && HasAVX512Save;
|
2018-01-19 07:52:31 +08:00
|
|
|
Features["rdpid"] = HasLeaf7 && ((ECX >> 22) & 1);
|
2018-04-13 15:35:08 +08:00
|
|
|
Features["cldemote"] = HasLeaf7 && ((ECX >> 25) & 1);
|
2018-05-01 18:01:16 +08:00
|
|
|
Features["movdiri"] = HasLeaf7 && ((ECX >> 27) & 1);
|
|
|
|
Features["movdir64b"] = HasLeaf7 && ((ECX >> 28) & 1);
|
2019-05-30 11:59:16 +08:00
|
|
|
Features["enqcmd"] = HasLeaf7 && ((ECX >> 29) & 1);
|
2018-01-19 07:52:31 +08:00
|
|
|
|
2020-05-28 23:28:12 +08:00
|
|
|
Features["avx512vp2intersect"] =
|
|
|
|
HasLeaf7 && ((EDX >> 8) & 1) && HasAVX512Save;
|
2020-04-02 16:15:34 +08:00
|
|
|
Features["serialize"] = HasLeaf7 && ((EDX >> 14) & 1);
|
2020-04-09 13:15:42 +08:00
|
|
|
Features["tsxldtrk"] = HasLeaf7 && ((EDX >> 16) & 1);
|
2018-05-08 14:47:36 +08:00
|
|
|
// There are two CPUID leafs which information associated with the pconfig
|
|
|
|
// instruction:
|
|
|
|
// EAX=0x7, ECX=0x0 indicates the availability of the instruction (via the 18th
|
|
|
|
// bit of EDX), while the EAX=0x1b leaf returns information on the
|
|
|
|
// availability of specific pconfig leafs.
|
|
|
|
// The target feature here only refers to the the first of these two.
|
|
|
|
// Users might need to check for the availability of specific pconfig
|
|
|
|
// leaves using cpuid, since that information is ignored while
|
|
|
|
// detecting features using the "-march=native" flag.
|
|
|
|
// For more info, see X86 ISA docs.
|
|
|
|
Features["pconfig"] = HasLeaf7 && ((EDX >> 18) & 1);
|
2020-07-02 08:36:45 +08:00
|
|
|
Features["amx-bf16"] = HasLeaf7 && ((EDX >> 22) & 1) && HasAMXSave;
|
|
|
|
Features["amx-tile"] = HasLeaf7 && ((EDX >> 24) & 1) && HasAMXSave;
|
|
|
|
Features["amx-int8"] = HasLeaf7 && ((EDX >> 25) & 1) && HasAMXSave;
|
Enable AVX512_BF16 instructions, which are supported for BFLOAT16 in Cooper Lake
Summary:
1. Enable infrastructure of AVX512_BF16, which is supported for BFLOAT16 in Cooper Lake;
2. Enable VCVTNE2PS2BF16, VCVTNEPS2BF16 and DPBF16PS instructions, which are Vector Neural Network Instructions supporting BFLOAT16 inputs and conversion instructions from IEEE single precision.
VCVTNE2PS2BF16: Convert Two Packed Single Data to One Packed BF16 Data.
VCVTNEPS2BF16: Convert Packed Single Data to Packed BF16 Data.
VDPBF16PS: Dot Product of BF16 Pairs Accumulated into Packed Single Precision.
For more details about BF16 isa, please refer to the latest ISE document: https://software.intel.com/en-us/download/intel-architecture-instruction-set-extensions-programming-reference
Author: LiuTianle
Reviewers: craig.topper, smaslov, LuoYuanke, wxiao3, annita.zhang, RKSimon, spatel
Reviewed By: craig.topper
Subscribers: kristina, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D60550
llvm-svn: 360017
2019-05-06 16:22:37 +08:00
|
|
|
bool HasLeaf7Subleaf1 =
|
|
|
|
MaxLevel >= 7 && !getX86CpuIDAndInfoEx(0x7, 0x1, &EAX, &EBX, &ECX, &EDX);
|
|
|
|
Features["avx512bf16"] = HasLeaf7Subleaf1 && ((EAX >> 5) & 1) && HasAVX512Save;
|
2018-05-08 14:47:36 +08:00
|
|
|
|
2017-05-03 23:42:29 +08:00
|
|
|
bool HasLeafD = MaxLevel >= 0xd &&
|
|
|
|
!getX86CpuIDAndInfoEx(0xd, 0x1, &EAX, &EBX, &ECX, &EDX);
|
|
|
|
|
|
|
|
// Only enable XSAVE if OS has enabled support for saving YMM state.
|
2017-11-20 07:30:22 +08:00
|
|
|
Features["xsaveopt"] = HasLeafD && ((EAX >> 0) & 1) && HasAVXSave;
|
|
|
|
Features["xsavec"] = HasLeafD && ((EAX >> 1) & 1) && HasAVXSave;
|
|
|
|
Features["xsaves"] = HasLeafD && ((EAX >> 3) & 1) && HasAVXSave;
|
2017-05-03 23:42:29 +08:00
|
|
|
|
2018-05-10 15:26:05 +08:00
|
|
|
bool HasLeaf14 = MaxLevel >= 0x14 &&
|
|
|
|
!getX86CpuIDAndInfoEx(0x14, 0x0, &EAX, &EBX, &ECX, &EDX);
|
|
|
|
|
|
|
|
Features["ptwrite"] = HasLeaf14 && ((EBX >> 4) & 1);
|
|
|
|
|
2017-05-03 23:42:29 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#elif defined(__linux__) && (defined(__arm__) || defined(__aarch64__))
|
|
|
|
bool sys::getHostCPUFeatures(StringMap<bool> &Features) {
|
|
|
|
std::unique_ptr<llvm::MemoryBuffer> P = getProcCpuinfoContent();
|
|
|
|
if (!P)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
SmallVector<StringRef, 32> Lines;
|
|
|
|
P->getBuffer().split(Lines, "\n");
|
|
|
|
|
|
|
|
SmallVector<StringRef, 32> CPUFeatures;
|
|
|
|
|
|
|
|
// Look for the CPU features.
|
|
|
|
for (unsigned I = 0, E = Lines.size(); I != E; ++I)
|
|
|
|
if (Lines[I].startswith("Features")) {
|
|
|
|
Lines[I].split(CPUFeatures, ' ');
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(__aarch64__)
|
|
|
|
// Keep track of which crypto features we have seen
|
|
|
|
enum { CAP_AES = 0x1, CAP_PMULL = 0x2, CAP_SHA1 = 0x4, CAP_SHA2 = 0x8 };
|
|
|
|
uint32_t crypto = 0;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
for (unsigned I = 0, E = CPUFeatures.size(); I != E; ++I) {
|
|
|
|
StringRef LLVMFeatureStr = StringSwitch<StringRef>(CPUFeatures[I])
|
|
|
|
#if defined(__aarch64__)
|
|
|
|
.Case("asimd", "neon")
|
|
|
|
.Case("fp", "fp-armv8")
|
|
|
|
.Case("crc32", "crc")
|
|
|
|
#else
|
|
|
|
.Case("half", "fp16")
|
|
|
|
.Case("neon", "neon")
|
|
|
|
.Case("vfpv3", "vfp3")
|
|
|
|
.Case("vfpv3d16", "d16")
|
|
|
|
.Case("vfpv4", "vfp4")
|
|
|
|
.Case("idiva", "hwdiv-arm")
|
|
|
|
.Case("idivt", "hwdiv")
|
|
|
|
#endif
|
|
|
|
.Default("");
|
|
|
|
|
|
|
|
#if defined(__aarch64__)
|
|
|
|
// We need to check crypto separately since we need all of the crypto
|
|
|
|
// extensions to enable the subtarget feature
|
|
|
|
if (CPUFeatures[I] == "aes")
|
|
|
|
crypto |= CAP_AES;
|
|
|
|
else if (CPUFeatures[I] == "pmull")
|
|
|
|
crypto |= CAP_PMULL;
|
|
|
|
else if (CPUFeatures[I] == "sha1")
|
|
|
|
crypto |= CAP_SHA1;
|
|
|
|
else if (CPUFeatures[I] == "sha2")
|
|
|
|
crypto |= CAP_SHA2;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (LLVMFeatureStr != "")
|
|
|
|
Features[LLVMFeatureStr] = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(__aarch64__)
|
|
|
|
// If we have all crypto bits we can add the feature
|
|
|
|
if (crypto == (CAP_AES | CAP_PMULL | CAP_SHA1 | CAP_SHA2))
|
|
|
|
Features["crypto"] = true;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2019-10-02 19:04:55 +08:00
|
|
|
#elif defined(_WIN32) && (defined(__aarch64__) || defined(_M_ARM64))
|
|
|
|
bool sys::getHostCPUFeatures(StringMap<bool> &Features) {
|
|
|
|
if (IsProcessorFeaturePresent(PF_ARM_NEON_INSTRUCTIONS_AVAILABLE))
|
|
|
|
Features["neon"] = true;
|
|
|
|
if (IsProcessorFeaturePresent(PF_ARM_V8_CRC32_INSTRUCTIONS_AVAILABLE))
|
|
|
|
Features["crc"] = true;
|
|
|
|
if (IsProcessorFeaturePresent(PF_ARM_V8_CRYPTO_INSTRUCTIONS_AVAILABLE))
|
|
|
|
Features["crypto"] = true;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2017-05-03 23:42:29 +08:00
|
|
|
#else
|
|
|
|
bool sys::getHostCPUFeatures(StringMap<bool> &Features) { return false; }
|
|
|
|
#endif
|
|
|
|
|
|
|
|
std::string sys::getProcessTriple() {
|
2017-07-07 17:53:47 +08:00
|
|
|
std::string TargetTripleString = updateTripleOSVersion(LLVM_HOST_TRIPLE);
|
|
|
|
Triple PT(Triple::normalize(TargetTripleString));
|
2017-05-03 23:42:29 +08:00
|
|
|
|
|
|
|
if (sizeof(void *) == 8 && PT.isArch32Bit())
|
|
|
|
PT = PT.get64BitArchVariant();
|
|
|
|
if (sizeof(void *) == 4 && PT.isArch64Bit())
|
|
|
|
PT = PT.get32BitArchVariant();
|
|
|
|
|
|
|
|
return PT.str();
|
|
|
|
}
|