blob: 4d0f8993e00200ae9c7ed519c0a9a1304269322a [file] [log] [blame]
Evan Chengb9803a82009-11-06 23:52:48 +00001//===-- ARMExpandPseudoInsts.cpp - Expand pseudo instructions -----*- C++ -*-=//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file contains a pass that expand pseudo instructions into target
11// instructions to allow proper scheduling, if-conversion, and other late
12// optimizations. This pass should be run after register allocation but before
13// post- regalloc scheduling pass.
14//
15//===----------------------------------------------------------------------===//
16
17#define DEBUG_TYPE "arm-pseudo"
18#include "ARM.h"
19#include "ARMBaseInstrInfo.h"
20#include "llvm/CodeGen/MachineFunctionPass.h"
21#include "llvm/CodeGen/MachineInstrBuilder.h"
22
23using namespace llvm;
24
25namespace {
26 class ARMExpandPseudo : public MachineFunctionPass {
27 public:
28 static char ID;
29 ARMExpandPseudo() : MachineFunctionPass(&ID) {}
30
31 const TargetInstrInfo *TII;
32
33 virtual bool runOnMachineFunction(MachineFunction &Fn);
34
35 virtual const char *getPassName() const {
36 return "ARM pseudo instruction expansion pass";
37 }
38
39 private:
40 bool ExpandMBB(MachineBasicBlock &MBB);
41 };
42 char ARMExpandPseudo::ID = 0;
43}
44
45bool ARMExpandPseudo::ExpandMBB(MachineBasicBlock &MBB) {
46 bool Modified = false;
47
48 MachineBasicBlock::iterator MBBI = MBB.begin(), E = MBB.end();
49 while (MBBI != E) {
50 MachineInstr &MI = *MBBI;
51 MachineBasicBlock::iterator NMBBI = next(MBBI);
52
53 unsigned Opcode = MI.getOpcode();
54 switch (Opcode) {
55 default: break;
56 case ARM::tLDRpci_pic:
57 case ARM::t2LDRpci_pic: {
58 unsigned NewLdOpc = (Opcode == ARM::tLDRpci_pic)
59 ? ARM::tLDRpci : ARM::t2LDRpci;
60 unsigned DstReg = MI.getOperand(0).getReg();
61 if (!MI.getOperand(0).isDead()) {
62 MachineInstr *NewMI =
63 AddDefaultPred(BuildMI(MBB, MBBI, MI.getDebugLoc(),
64 TII->get(NewLdOpc), DstReg)
65 .addOperand(MI.getOperand(1)));
66 NewMI->setMemRefs(MI.memoperands_begin(), MI.memoperands_end());
67 BuildMI(MBB, MBBI, MI.getDebugLoc(), TII->get(ARM::tPICADD))
68 .addReg(DstReg, getDefRegState(true))
69 .addReg(DstReg)
70 .addOperand(MI.getOperand(2));
71 }
72 MI.eraseFromParent();
73 Modified = true;
74 break;
75 }
76 case ARM::t2MOVi32imm: {
77 unsigned DstReg = MI.getOperand(0).getReg();
78 unsigned Imm = MI.getOperand(1).getImm();
79 unsigned Lo16 = Imm & 0xffff;
80 unsigned Hi16 = (Imm >> 16) & 0xffff;
81 if (!MI.getOperand(0).isDead()) {
82 AddDefaultPred(BuildMI(MBB, MBBI, MI.getDebugLoc(),
83 TII->get(ARM::t2MOVi16), DstReg)
84 .addImm(Lo16));
85 AddDefaultPred(BuildMI(MBB, MBBI, MI.getDebugLoc(),
86 TII->get(ARM::t2MOVTi16))
87 .addReg(DstReg, getDefRegState(true))
88 .addReg(DstReg).addImm(Hi16));
89 }
90 MI.eraseFromParent();
91 Modified = true;
92 }
93 // FIXME: expand t2MOVi32imm
94 }
95 MBBI = NMBBI;
96 }
97
98 return Modified;
99}
100
101bool ARMExpandPseudo::runOnMachineFunction(MachineFunction &MF) {
102 TII = MF.getTarget().getInstrInfo();
103
104 bool Modified = false;
105 for (MachineFunction::iterator MFI = MF.begin(), E = MF.end(); MFI != E;
106 ++MFI)
107 Modified |= ExpandMBB(*MFI);
108 return Modified;
109}
110
111/// createARMExpandPseudoPass - returns an instance of the pseudo instruction
112/// expansion pass.
113FunctionPass *llvm::createARMExpandPseudoPass() {
114 return new ARMExpandPseudo();
115}