blob: 57de7019be537af98a6f88b3fa71d99c711babe1 [file] [log] [blame]
Clement Courbet44b4c542018-06-19 11:28:59 +00001//===-- Target.cpp ----------------------------------------------*- C++ -*-===//
2//
Chandler Carruth2946cd72019-01-19 08:50:56 +00003// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
Clement Courbet44b4c542018-06-19 11:28:59 +00006//
7//===----------------------------------------------------------------------===//
8#include "Target.h"
9
Clement Courbet4860b982018-06-26 08:49:30 +000010#include "Latency.h"
11#include "Uops.h"
12
Fangrui Song32401af2018-10-22 17:10:47 +000013namespace llvm {
Clement Courbet44b4c542018-06-19 11:28:59 +000014namespace exegesis {
15
Clement Courbetcff2caa2018-06-25 11:22:23 +000016ExegesisTarget::~ExegesisTarget() {} // anchor.
Clement Courbet44b4c542018-06-19 11:28:59 +000017
Clement Courbetcff2caa2018-06-25 11:22:23 +000018static ExegesisTarget *FirstTarget = nullptr;
Clement Courbet44b4c542018-06-19 11:28:59 +000019
Clement Courbet50cdd562019-10-09 11:58:42 +000020const ExegesisTarget *ExegesisTarget::lookup(Triple TT) {
Clement Courbetcff2caa2018-06-25 11:22:23 +000021 for (const ExegesisTarget *T = FirstTarget; T != nullptr; T = T->Next) {
Clement Courbet6fd00e32018-06-20 11:54:35 +000022 if (T->matchesArch(TT.getArch()))
23 return T;
Clement Courbet44b4c542018-06-19 11:28:59 +000024 }
25 return nullptr;
26}
27
Clement Courbetcff2caa2018-06-25 11:22:23 +000028void ExegesisTarget::registerTarget(ExegesisTarget *Target) {
Clement Courbet44b4c542018-06-19 11:28:59 +000029 if (FirstTarget == nullptr) {
30 FirstTarget = Target;
31 return;
32 }
Clement Courbet44b4c542018-06-19 11:28:59 +000033 if (Target->Next != nullptr)
Guillaume Chateletfb943542018-08-01 14:41:45 +000034 return; // Already registered.
Clement Courbet44b4c542018-06-19 11:28:59 +000035 Target->Next = FirstTarget;
36 FirstTarget = Target;
37}
Clement Courbet4860b982018-06-26 08:49:30 +000038
Clement Courbet2cd0f282019-10-08 14:30:24 +000039std::unique_ptr<SnippetGenerator> ExegesisTarget::createSnippetGenerator(
40 InstructionBenchmark::ModeE Mode, const LLVMState &State,
41 const SnippetGenerator::Options &Opts) const {
Clement Courbetd939f6d2018-09-13 07:40:53 +000042 switch (Mode) {
43 case InstructionBenchmark::Unknown:
44 return nullptr;
45 case InstructionBenchmark::Latency:
Clement Courbet2cd0f282019-10-08 14:30:24 +000046 return createLatencySnippetGenerator(State, Opts);
Clement Courbetd939f6d2018-09-13 07:40:53 +000047 case InstructionBenchmark::Uops:
Clement Courbet362653f2019-01-30 16:02:20 +000048 case InstructionBenchmark::InverseThroughput:
Clement Courbet2cd0f282019-10-08 14:30:24 +000049 return createUopsSnippetGenerator(State, Opts);
Clement Courbetd939f6d2018-09-13 07:40:53 +000050 }
51 return nullptr;
52}
53
Clement Courbet4860b982018-06-26 08:49:30 +000054std::unique_ptr<BenchmarkRunner>
55ExegesisTarget::createBenchmarkRunner(InstructionBenchmark::ModeE Mode,
56 const LLVMState &State) const {
57 switch (Mode) {
58 case InstructionBenchmark::Unknown:
59 return nullptr;
60 case InstructionBenchmark::Latency:
Clement Courbet362653f2019-01-30 16:02:20 +000061 case InstructionBenchmark::InverseThroughput:
62 return createLatencyBenchmarkRunner(State, Mode);
Clement Courbet4860b982018-06-26 08:49:30 +000063 case InstructionBenchmark::Uops:
64 return createUopsBenchmarkRunner(State);
65 }
66 return nullptr;
67}
68
Clement Courbet2cd0f282019-10-08 14:30:24 +000069std::unique_ptr<SnippetGenerator> ExegesisTarget::createLatencySnippetGenerator(
70 const LLVMState &State, const SnippetGenerator::Options &Opts) const {
71 return std::make_unique<LatencySnippetGenerator>(State, Opts);
Clement Courbet0d79aaf2018-11-08 12:09:45 +000072}
73
Clement Courbet2cd0f282019-10-08 14:30:24 +000074std::unique_ptr<SnippetGenerator> ExegesisTarget::createUopsSnippetGenerator(
75 const LLVMState &State, const SnippetGenerator::Options &Opts) const {
76 return std::make_unique<UopsSnippetGenerator>(State, Opts);
Clement Courbet0d79aaf2018-11-08 12:09:45 +000077}
78
Clement Courbet362653f2019-01-30 16:02:20 +000079std::unique_ptr<BenchmarkRunner> ExegesisTarget::createLatencyBenchmarkRunner(
80 const LLVMState &State, InstructionBenchmark::ModeE Mode) const {
Jonas Devlieghere0eaee542019-08-15 15:54:37 +000081 return std::make_unique<LatencyBenchmarkRunner>(State, Mode);
Clement Courbet0d79aaf2018-11-08 12:09:45 +000082}
83
84std::unique_ptr<BenchmarkRunner>
85ExegesisTarget::createUopsBenchmarkRunner(const LLVMState &State) const {
Jonas Devlieghere0eaee542019-08-15 15:54:37 +000086 return std::make_unique<UopsBenchmarkRunner>(State);
Clement Courbet0d79aaf2018-11-08 12:09:45 +000087}
88
Clement Courbet50cdd562019-10-09 11:58:42 +000089void ExegesisTarget::randomizeMCOperand(const Instruction &Instr,
90 const Variable &Var,
91 MCOperand &AssignedValue,
92 const BitVector &ForbiddenRegs) const {
Roman Lebedev404bdb12019-04-06 14:16:26 +000093 const Operand &Op = Instr.getPrimaryOperand(Var);
94 switch (Op.getExplicitOperandInfo().OperandType) {
Clement Courbet50cdd562019-10-09 11:58:42 +000095 case MCOI::OperandType::OPERAND_IMMEDIATE:
Roman Lebedev404bdb12019-04-06 14:16:26 +000096 // FIXME: explore immediate values too.
Clement Courbet50cdd562019-10-09 11:58:42 +000097 AssignedValue = MCOperand::createImm(1);
Roman Lebedev404bdb12019-04-06 14:16:26 +000098 break;
Clement Courbet50cdd562019-10-09 11:58:42 +000099 case MCOI::OperandType::OPERAND_REGISTER: {
Roman Lebedev404bdb12019-04-06 14:16:26 +0000100 assert(Op.isReg());
101 auto AllowedRegs = Op.getRegisterAliasing().sourceBits();
102 assert(AllowedRegs.size() == ForbiddenRegs.size());
103 for (auto I : ForbiddenRegs.set_bits())
104 AllowedRegs.reset(I);
Clement Courbet50cdd562019-10-09 11:58:42 +0000105 AssignedValue = MCOperand::createReg(randomBit(AllowedRegs));
Roman Lebedev404bdb12019-04-06 14:16:26 +0000106 break;
107 }
108 default:
109 break;
110 }
111}
112
Clement Courbet41c8af32018-10-25 07:44:01 +0000113static_assert(std::is_pod<PfmCountersInfo>::value,
114 "We shouldn't have dynamic initialization here");
Clement Courbeteee2e062018-11-09 13:15:32 +0000115const PfmCountersInfo PfmCountersInfo::Default = {nullptr, nullptr, nullptr,
116 0u};
Clement Courbet41c8af32018-10-25 07:44:01 +0000117
Clement Courbet50cdd562019-10-09 11:58:42 +0000118const PfmCountersInfo &ExegesisTarget::getPfmCounters(StringRef CpuName) const {
Clement Courbet41c8af32018-10-25 07:44:01 +0000119 assert(std::is_sorted(
120 CpuPfmCounters.begin(), CpuPfmCounters.end(),
121 [](const CpuAndPfmCounters &LHS, const CpuAndPfmCounters &RHS) {
122 return strcmp(LHS.CpuName, RHS.CpuName) < 0;
123 }) &&
124 "CpuPfmCounters table is not sorted");
125
126 // Find entry
127 auto Found =
128 std::lower_bound(CpuPfmCounters.begin(), CpuPfmCounters.end(), CpuName);
Clement Courbet50cdd562019-10-09 11:58:42 +0000129 if (Found == CpuPfmCounters.end() || StringRef(Found->CpuName) != CpuName) {
Clement Courbeteee2e062018-11-09 13:15:32 +0000130 // Use the default.
131 if (CpuPfmCounters.begin() != CpuPfmCounters.end() &&
132 CpuPfmCounters.begin()->CpuName[0] == '\0') {
133 Found = CpuPfmCounters.begin(); // The target specifies a default.
134 } else {
135 return PfmCountersInfo::Default; // No default for the target.
136 }
Clement Courbet41c8af32018-10-25 07:44:01 +0000137 }
138 assert(Found->PCI && "Missing counters");
139 return *Found->PCI;
140}
141
Clement Courbet4860b982018-06-26 08:49:30 +0000142namespace {
143
144// Default implementation.
145class ExegesisDefaultTarget : public ExegesisTarget {
Clement Courbet41c8af32018-10-25 07:44:01 +0000146public:
147 ExegesisDefaultTarget() : ExegesisTarget({}) {}
148
Clement Courbet4860b982018-06-26 08:49:30 +0000149private:
Clement Courbet50cdd562019-10-09 11:58:42 +0000150 std::vector<MCInst> setRegTo(const MCSubtargetInfo &STI, unsigned Reg,
151 const APInt &Value) const override {
Guillaume Chatelet5ad29092018-09-18 11:26:27 +0000152 llvm_unreachable("Not yet implemented");
153 }
154
Clement Courbet50cdd562019-10-09 11:58:42 +0000155 bool matchesArch(Triple::ArchType Arch) const override {
Clement Courbet4860b982018-06-26 08:49:30 +0000156 llvm_unreachable("never called");
157 return false;
158 }
159};
160
161} // namespace
162
163const ExegesisTarget &ExegesisTarget::getDefault() {
164 static ExegesisDefaultTarget Target;
165 return Target;
166}
167
Clement Courbetcff2caa2018-06-25 11:22:23 +0000168} // namespace exegesis
Fangrui Song32401af2018-10-22 17:10:47 +0000169} // namespace llvm