diff options
| author | Manman Ren <mren@apple.com> | 2012-05-31 17:20:29 +0000 |
|---|---|---|
| committer | Manman Ren <mren@apple.com> | 2012-05-31 17:20:29 +0000 |
| commit | 9bccb64e561a1cf8f38f336e26c59cb814b11b91 (patch) | |
| tree | f0e5fee9631c4738d24ba1d12fc6efec8ec18477 /llvm/lib/Target | |
| parent | fa9d7db17b76a6c139e28613a743950c719a1d36 (diff) | |
| download | bcm5719-llvm-9bccb64e561a1cf8f38f336e26c59cb814b11b91.tar.gz bcm5719-llvm-9bccb64e561a1cf8f38f336e26c59cb814b11b91.zip | |
X86: replace SUB with CMP if possible
This patch will optimize the following
movq %rdi, %rax
subq %rsi, %rax
cmovsq %rsi, %rdi
movq %rdi, %rax
to
cmpq %rsi, %rdi
cmovsq %rsi, %rdi
movq %rdi, %rax
Perform this optimization if the actual result of SUB is not used.
rdar: 11540023
llvm-svn: 157755
Diffstat (limited to 'llvm/lib/Target')
| -rw-r--r-- | llvm/lib/Target/X86/X86InstrInfo.cpp | 38 | ||||
| -rw-r--r-- | llvm/lib/Target/X86/X86InstrInfo.h | 3 |
2 files changed, 41 insertions, 0 deletions
diff --git a/llvm/lib/Target/X86/X86InstrInfo.cpp b/llvm/lib/Target/X86/X86InstrInfo.cpp index 7254ddf56cb..2dd5c12c65d 100644 --- a/llvm/lib/Target/X86/X86InstrInfo.cpp +++ b/llvm/lib/Target/X86/X86InstrInfo.cpp @@ -2709,6 +2709,44 @@ void X86InstrInfo::loadRegFromAddr(MachineFunction &MF, unsigned DestReg, NewMIs.push_back(MIB); } +bool X86InstrInfo:: +OptimizeSubInstr(MachineInstr *SubInstr, const MachineRegisterInfo *MRI) const { + // If destination is a memory operand, do not perform this optimization. + if ((SubInstr->getOpcode() != X86::SUB64rr) && + (SubInstr->getOpcode() != X86::SUB32rr) && + (SubInstr->getOpcode() != X86::SUB16rr) && + (SubInstr->getOpcode() != X86::SUB8rr) && + (SubInstr->getOpcode() != X86::SUB64ri32) && + (SubInstr->getOpcode() != X86::SUB64ri8) && + (SubInstr->getOpcode() != X86::SUB32ri) && + (SubInstr->getOpcode() != X86::SUB32ri8) && + (SubInstr->getOpcode() != X86::SUB16ri) && + (SubInstr->getOpcode() != X86::SUB16ri8) && + (SubInstr->getOpcode() != X86::SUB8ri)) + return false; + unsigned DestReg = SubInstr->getOperand(0).getReg(); + if (MRI->use_begin(DestReg) != MRI->use_end()) + return false; + + // There is no use of the destination register, we can replace SUB with CMP. + switch (SubInstr->getOpcode()) { + default: break; + case X86::SUB64rr: SubInstr->setDesc(get(X86::CMP64rr)); break; + case X86::SUB32rr: SubInstr->setDesc(get(X86::CMP32rr)); break; + case X86::SUB16rr: SubInstr->setDesc(get(X86::CMP16rr)); break; + case X86::SUB8rr: SubInstr->setDesc(get(X86::CMP8rr)); break; + case X86::SUB64ri32: SubInstr->setDesc(get(X86::CMP64ri32)); break; + case X86::SUB64ri8: SubInstr->setDesc(get(X86::CMP64ri8)); break; + case X86::SUB32ri: SubInstr->setDesc(get(X86::CMP32ri)); break; + case X86::SUB32ri8: SubInstr->setDesc(get(X86::CMP32ri8)); break; + case X86::SUB16ri: SubInstr->setDesc(get(X86::CMP16ri)); break; + case X86::SUB16ri8: SubInstr->setDesc(get(X86::CMP16ri8)); break; + case X86::SUB8ri: SubInstr->setDesc(get(X86::CMP8ri)); break; + } + SubInstr->RemoveOperand(0); + return true; +} + /// Expand2AddrUndef - Expand a single-def pseudo instruction to a two-addr /// instruction with two undef reads of the register being defined. This is /// used for mapping: diff --git a/llvm/lib/Target/X86/X86InstrInfo.h b/llvm/lib/Target/X86/X86InstrInfo.h index 856f3be57ce..9c4d46522c7 100644 --- a/llvm/lib/Target/X86/X86InstrInfo.h +++ b/llvm/lib/Target/X86/X86InstrInfo.h @@ -365,6 +365,9 @@ public: const MachineInstr *DefMI, unsigned DefIdx, const MachineInstr *UseMI, unsigned UseIdx) const; + virtual bool OptimizeSubInstr(MachineInstr *SubInstr, + const MachineRegisterInfo *MRI) const; + private: MachineInstr * convertToThreeAddressWithLEA(unsigned MIOpc, MachineFunction::iterator &MFI, |

