summaryrefslogtreecommitdiffstats
path: root/llvm/tools/llvm-exegesis/lib/X86/Target.cpp
diff options
context:
space:
mode:
authorGuillaume Chatelet <gchatelet@google.com>2018-08-01 14:41:45 +0000
committerGuillaume Chatelet <gchatelet@google.com>2018-08-01 14:41:45 +0000
commitfb94354d2dbee7628770b8b75b8289fa8d225ef3 (patch)
tree37cd240911a715166dc635311122985a7d05e614 /llvm/tools/llvm-exegesis/lib/X86/Target.cpp
parentc8e3924b3b8aacbcd089530e5fe8c1b463177780 (diff)
downloadbcm5719-llvm-fb94354d2dbee7628770b8b75b8289fa8d225ef3.tar.gz
bcm5719-llvm-fb94354d2dbee7628770b8b75b8289fa8d225ef3.zip
[llvm-exegesis] Provide a way to handle memory instructions.
Summary: And implement memory instructions on X86. This fixes PR36906. Reviewers: gchatelet Reviewed By: gchatelet Subscribers: lebedev.ri, filcab, mgorny, tschuett, RKSimon, llvm-commits Differential Revision: https://reviews.llvm.org/D48935 llvm-svn: 338567
Diffstat (limited to 'llvm/tools/llvm-exegesis/lib/X86/Target.cpp')
-rw-r--r--llvm/tools/llvm-exegesis/lib/X86/Target.cpp42
1 files changed, 42 insertions, 0 deletions
diff --git a/llvm/tools/llvm-exegesis/lib/X86/Target.cpp b/llvm/tools/llvm-exegesis/lib/X86/Target.cpp
index 5c417e325f5..bdfa5a753b2 100644
--- a/llvm/tools/llvm-exegesis/lib/X86/Target.cpp
+++ b/llvm/tools/llvm-exegesis/lib/X86/Target.cpp
@@ -109,6 +109,48 @@ class ExegesisX86Target : public ExegesisTarget {
PM.add(llvm::createX86FloatingPointStackifierPass());
}
+ unsigned getScratchMemoryRegister(const llvm::Triple &TT) const override {
+ if (!TT.isArch64Bit()) {
+ // FIXME: This would require popping from the stack, so we would have to
+ // add some additional setup code.
+ return 0;
+ }
+ return TT.isOSWindows() ? llvm::X86::RCX : llvm::X86::RDI;
+ }
+
+ unsigned getMaxMemoryAccessSize() const override { return 64; }
+
+ void fillMemoryOperands(InstructionInstance &II, unsigned Reg,
+ unsigned Offset) const override {
+ // FIXME: For instructions that read AND write to memory, we use the same
+ // value for input and output.
+ for (size_t I = 0, E = II.Instr.Operands.size(); I < E; ++I) {
+ const Operand *Op = &II.Instr.Operands[I];
+ if (Op->IsExplicit && Op->IsMem) {
+ // Case 1: 5-op memory.
+ assert((I + 5 <= E) && "x86 memory references are always 5 ops");
+ II.getValueFor(*Op) = llvm::MCOperand::createReg(Reg); // BaseReg
+ Op = &II.Instr.Operands[++I];
+ assert(Op->IsMem);
+ assert(Op->IsExplicit);
+ II.getValueFor(*Op) = llvm::MCOperand::createImm(1); // ScaleAmt
+ Op = &II.Instr.Operands[++I];
+ assert(Op->IsMem);
+ assert(Op->IsExplicit);
+ II.getValueFor(*Op) = llvm::MCOperand::createReg(0); // IndexReg
+ Op = &II.Instr.Operands[++I];
+ assert(Op->IsMem);
+ assert(Op->IsExplicit);
+ II.getValueFor(*Op) = llvm::MCOperand::createImm(Offset); // Disp
+ Op = &II.Instr.Operands[++I];
+ assert(Op->IsMem);
+ assert(Op->IsExplicit);
+ II.getValueFor(*Op) = llvm::MCOperand::createReg(0); // Segment
+ // Case2: segment:index addressing. We assume that ES is 0.
+ }
+ }
+ }
+
std::vector<llvm::MCInst> setRegToConstant(const llvm::MCSubtargetInfo &STI,
unsigned Reg) const override {
// GPR.
OpenPOWER on IntegriCloud