| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 1 | //===---- IA64ISelDAGToDAG.cpp - IA64 pattern matching inst selector ------===// | 
|  | 2 | // | 
|  | 3 | //                     The LLVM Compiler Infrastructure | 
|  | 4 | // | 
|  | 5 | // This file was developed by Duraid Madina and is distributed under | 
|  | 6 | // the University of Illinois Open Source License. See LICENSE.TXT for details. | 
|  | 7 | // | 
|  | 8 | //===----------------------------------------------------------------------===// | 
|  | 9 | // | 
|  | 10 | // This file defines a pattern matching instruction selector for IA64, | 
|  | 11 | // converting a legalized dag to an IA64 dag. | 
|  | 12 | // | 
|  | 13 | //===----------------------------------------------------------------------===// | 
|  | 14 |  | 
|  | 15 | #include "IA64.h" | 
|  | 16 | #include "IA64TargetMachine.h" | 
|  | 17 | #include "IA64ISelLowering.h" | 
|  | 18 | #include "llvm/CodeGen/MachineInstrBuilder.h" | 
|  | 19 | #include "llvm/CodeGen/MachineFunction.h" | 
|  | 20 | #include "llvm/CodeGen/SSARegMap.h" | 
|  | 21 | #include "llvm/CodeGen/SelectionDAG.h" | 
|  | 22 | #include "llvm/CodeGen/SelectionDAGISel.h" | 
|  | 23 | #include "llvm/Target/TargetOptions.h" | 
|  | 24 | #include "llvm/ADT/Statistic.h" | 
|  | 25 | #include "llvm/Constants.h" | 
|  | 26 | #include "llvm/GlobalValue.h" | 
|  | 27 | #include "llvm/Support/Debug.h" | 
|  | 28 | #include "llvm/Support/MathExtras.h" | 
|  | 29 | using namespace llvm; | 
|  | 30 |  | 
|  | 31 | namespace { | 
|  | 32 | Statistic<> FusedFP ("ia64-codegen", "Number of fused fp operations"); | 
|  | 33 | Statistic<> FrameOff("ia64-codegen", "Number of frame idx offsets collapsed"); | 
|  | 34 |  | 
|  | 35 | //===--------------------------------------------------------------------===// | 
|  | 36 | /// IA64DAGToDAGISel - IA64 specific code to select IA64 machine | 
|  | 37 | /// instructions for SelectionDAG operations. | 
|  | 38 | /// | 
|  | 39 | class IA64DAGToDAGISel : public SelectionDAGISel { | 
|  | 40 | IA64TargetLowering IA64Lowering; | 
|  | 41 | unsigned GlobalBaseReg; | 
|  | 42 | public: | 
|  | 43 | IA64DAGToDAGISel(TargetMachine &TM) | 
|  | 44 | : SelectionDAGISel(IA64Lowering), IA64Lowering(TM) {} | 
|  | 45 |  | 
|  | 46 | virtual bool runOnFunction(Function &Fn) { | 
|  | 47 | // Make sure we re-emit a set of the global base reg if necessary | 
|  | 48 | GlobalBaseReg = 0; | 
|  | 49 | return SelectionDAGISel::runOnFunction(Fn); | 
|  | 50 | } | 
|  | 51 |  | 
|  | 52 | /// getI64Imm - Return a target constant with the specified value, of type | 
|  | 53 | /// i64. | 
|  | 54 | inline SDOperand getI64Imm(uint64_t Imm) { | 
|  | 55 | return CurDAG->getTargetConstant(Imm, MVT::i64); | 
|  | 56 | } | 
|  | 57 |  | 
|  | 58 | /// getGlobalBaseReg - insert code into the entry mbb to materialize the PIC | 
|  | 59 | /// base register.  Return the virtual register that holds this value. | 
|  | 60 | // SDOperand getGlobalBaseReg(); TODO: hmm | 
|  | 61 |  | 
|  | 62 | // Select - Convert the specified operand from a target-independent to a | 
|  | 63 | // target-specific node if it hasn't already been changed. | 
|  | 64 | SDOperand Select(SDOperand Op); | 
|  | 65 |  | 
|  | 66 | SDNode *SelectIntImmediateExpr(SDOperand LHS, SDOperand RHS, | 
|  | 67 | unsigned OCHi, unsigned OCLo, | 
|  | 68 | bool IsArithmetic = false, | 
|  | 69 | bool Negate = false); | 
|  | 70 | SDNode *SelectBitfieldInsert(SDNode *N); | 
|  | 71 |  | 
|  | 72 | /// SelectCC - Select a comparison of the specified values with the | 
|  | 73 | /// specified condition code, returning the CR# of the expression. | 
|  | 74 | SDOperand SelectCC(SDOperand LHS, SDOperand RHS, ISD::CondCode CC); | 
|  | 75 |  | 
|  | 76 | /// SelectAddr - Given the specified address, return the two operands for a | 
|  | 77 | /// load/store instruction, and return true if it should be an indexed [r+r] | 
|  | 78 | /// operation. | 
|  | 79 | bool SelectAddr(SDOperand Addr, SDOperand &Op1, SDOperand &Op2); | 
|  | 80 |  | 
|  | 81 | SDOperand BuildSDIVSequence(SDNode *N); | 
|  | 82 | SDOperand BuildUDIVSequence(SDNode *N); | 
|  | 83 |  | 
|  | 84 | /// InstructionSelectBasicBlock - This callback is invoked by | 
|  | 85 | /// SelectionDAGISel when it has created a SelectionDAG for us to codegen. | 
|  | 86 | virtual void InstructionSelectBasicBlock(SelectionDAG &DAG); | 
|  | 87 |  | 
|  | 88 | virtual const char *getPassName() const { | 
|  | 89 | return "IA64 (Itanium) DAG->DAG Instruction Selector"; | 
|  | 90 | } | 
|  | 91 |  | 
|  | 92 | // Include the pieces autogenerated from the target description. | 
|  | 93 | #include "IA64GenDAGISel.inc" | 
|  | 94 |  | 
|  | 95 | private: | 
| Duraid Madina | 8edbf80 | 2005-11-21 14:14:54 +0000 | [diff] [blame] | 96 | SDOperand SelectDIV(SDOperand Op); | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 97 | }; | 
|  | 98 | } | 
|  | 99 |  | 
|  | 100 | /// InstructionSelectBasicBlock - This callback is invoked by | 
|  | 101 | /// SelectionDAGISel when it has created a SelectionDAG for us to codegen. | 
|  | 102 | void IA64DAGToDAGISel::InstructionSelectBasicBlock(SelectionDAG &DAG) { | 
|  | 103 | DEBUG(BB->dump()); | 
|  | 104 |  | 
|  | 105 | // The selection process is inherently a bottom-up recursive process (users | 
|  | 106 | // select their uses before themselves).  Given infinite stack space, we | 
|  | 107 | // could just start selecting on the root and traverse the whole graph.  In | 
|  | 108 | // practice however, this causes us to run out of stack space on large basic | 
|  | 109 | // blocks.  To avoid this problem, select the entry node, then all its uses, | 
|  | 110 | // iteratively instead of recursively. | 
|  | 111 | std::vector<SDOperand> Worklist; | 
|  | 112 | Worklist.push_back(DAG.getEntryNode()); | 
|  | 113 |  | 
|  | 114 | // Note that we can do this in the IA64 target (scanning forward across token | 
|  | 115 | // chain edges) because no nodes ever get folded across these edges.  On a | 
|  | 116 | // target like X86 which supports load/modify/store operations, this would | 
|  | 117 | // have to be more careful. | 
|  | 118 | while (!Worklist.empty()) { | 
|  | 119 | SDOperand Node = Worklist.back(); | 
|  | 120 | Worklist.pop_back(); | 
|  | 121 |  | 
|  | 122 | // Chose from the least deep of the top two nodes. | 
|  | 123 | if (!Worklist.empty() && | 
|  | 124 | Worklist.back().Val->getNodeDepth() < Node.Val->getNodeDepth()) | 
|  | 125 | std::swap(Worklist.back(), Node); | 
|  | 126 |  | 
|  | 127 | if ((Node.Val->getOpcode() >= ISD::BUILTIN_OP_END && | 
|  | 128 | Node.Val->getOpcode() < IA64ISD::FIRST_NUMBER) || | 
|  | 129 | CodeGenMap.count(Node)) continue; | 
|  | 130 |  | 
|  | 131 | for (SDNode::use_iterator UI = Node.Val->use_begin(), | 
|  | 132 | E = Node.Val->use_end(); UI != E; ++UI) { | 
|  | 133 | // Scan the values.  If this use has a value that is a token chain, add it | 
|  | 134 | // to the worklist. | 
|  | 135 | SDNode *User = *UI; | 
|  | 136 | for (unsigned i = 0, e = User->getNumValues(); i != e; ++i) | 
|  | 137 | if (User->getValueType(i) == MVT::Other) { | 
|  | 138 | Worklist.push_back(SDOperand(User, i)); | 
|  | 139 | break; | 
|  | 140 | } | 
|  | 141 | } | 
|  | 142 |  | 
|  | 143 | // Finally, legalize this node. | 
|  | 144 | Select(Node); | 
|  | 145 | } | 
|  | 146 |  | 
|  | 147 | // Select target instructions for the DAG. | 
|  | 148 | DAG.setRoot(Select(DAG.getRoot())); | 
|  | 149 | CodeGenMap.clear(); | 
|  | 150 | DAG.RemoveDeadNodes(); | 
|  | 151 |  | 
|  | 152 | // Emit machine code to BB. | 
|  | 153 | ScheduleAndEmitDAG(DAG); | 
|  | 154 | } | 
|  | 155 |  | 
| Duraid Madina | 8edbf80 | 2005-11-21 14:14:54 +0000 | [diff] [blame] | 156 | SDOperand IA64DAGToDAGISel::SelectDIV(SDOperand Op) { | 
|  | 157 | SDNode *N = Op.Val; | 
|  | 158 | SDOperand Chain = Select(N->getOperand(0)); | 
|  | 159 |  | 
|  | 160 | SDOperand Tmp1 = Select(N->getOperand(0)); | 
|  | 161 | SDOperand Tmp2 = Select(N->getOperand(1)); | 
|  | 162 |  | 
|  | 163 | bool isFP=false; | 
|  | 164 |  | 
|  | 165 | if(MVT::isFloatingPoint(Tmp1.getValueType())) | 
|  | 166 | isFP=true; | 
|  | 167 |  | 
|  | 168 | bool isModulus=false; // is it a division or a modulus? | 
|  | 169 | bool isSigned=false; | 
|  | 170 |  | 
|  | 171 | switch(N->getOpcode()) { | 
|  | 172 | case ISD::FDIV: | 
|  | 173 | case ISD::SDIV:  isModulus=false; isSigned=true;  break; | 
|  | 174 | case ISD::UDIV:  isModulus=false; isSigned=false; break; | 
|  | 175 | case ISD::FREM: | 
|  | 176 | case ISD::SREM:  isModulus=true;  isSigned=true;  break; | 
|  | 177 | case ISD::UREM:  isModulus=true;  isSigned=false; break; | 
|  | 178 | } | 
|  | 179 |  | 
|  | 180 | // TODO: check for integer divides by powers of 2 (or other simple patterns?) | 
|  | 181 |  | 
|  | 182 | SDOperand TmpPR, TmpPR2; | 
|  | 183 | SDOperand TmpF1, TmpF2, TmpF3, TmpF4, TmpF5, TmpF6, TmpF7, TmpF8; | 
|  | 184 | SDOperand TmpF9, TmpF10,TmpF11,TmpF12,TmpF13,TmpF14,TmpF15; | 
|  | 185 | SDOperand Result; | 
|  | 186 |  | 
|  | 187 | // OK, emit some code: | 
|  | 188 |  | 
|  | 189 | if(!isFP) { | 
|  | 190 | // first, load the inputs into FP regs. | 
|  | 191 | TmpF1 = CurDAG->getTargetNode(IA64::SETFSIG, MVT::f64, Tmp1); | 
|  | 192 | Chain = TmpF1.getValue(1); | 
|  | 193 | TmpF2 = CurDAG->getTargetNode(IA64::SETFSIG, MVT::f64, Tmp2); | 
|  | 194 | Chain = TmpF2.getValue(1); | 
|  | 195 |  | 
|  | 196 | // next, convert the inputs to FP | 
|  | 197 | if(isSigned) { | 
|  | 198 | TmpF3 = CurDAG->getTargetNode(IA64::FCVTXF, MVT::f64, TmpF1); | 
|  | 199 | Chain = TmpF3.getValue(1); | 
|  | 200 | TmpF4 = CurDAG->getTargetNode(IA64::FCVTXF, MVT::f64, TmpF2); | 
|  | 201 | Chain = TmpF4.getValue(1); | 
|  | 202 | } else { | 
|  | 203 | TmpF3 = CurDAG->getTargetNode(IA64::FCVTXUFS1, MVT::f64, TmpF1); | 
|  | 204 | Chain = TmpF3.getValue(1); | 
|  | 205 | TmpF4 = CurDAG->getTargetNode(IA64::FCVTXUFS1, MVT::f64, TmpF2); | 
|  | 206 | Chain = TmpF4.getValue(1); | 
|  | 207 | } | 
|  | 208 |  | 
|  | 209 | } else { // this is an FP divide/remainder, so we 'leak' some temp | 
|  | 210 | // regs and assign TmpF3=Tmp1, TmpF4=Tmp2 | 
|  | 211 | TmpF3=Tmp1; | 
|  | 212 | TmpF4=Tmp2; | 
|  | 213 | } | 
|  | 214 |  | 
|  | 215 | // we start by computing an approximate reciprocal (good to 9 bits?) | 
|  | 216 | // note, this instruction writes _both_ TmpF5 (answer) and TmpPR (predicate) | 
|  | 217 | TmpF5 = CurDAG->getTargetNode(IA64::FRCPAS1, MVT::f64, MVT::i1, | 
|  | 218 | TmpF3, TmpF4); | 
|  | 219 | TmpPR = TmpF5.getValue(1); | 
|  | 220 | Chain = TmpF5.getValue(2); | 
|  | 221 |  | 
|  | 222 | if(!isModulus) { // if this is a divide, we worry about div-by-zero | 
|  | 223 | SDOperand bogusPR = CurDAG->getTargetNode(IA64::CMPEQ, MVT::i1, | 
|  | 224 | CurDAG->getRegister(IA64::r0, MVT::i64), | 
|  | 225 | CurDAG->getRegister(IA64::r0, MVT::i64)); | 
|  | 226 | Chain = bogusPR.getValue(1); | 
|  | 227 | TmpPR2 = CurDAG->getTargetNode(IA64::TPCMPNE, MVT::i1, bogusPR, | 
|  | 228 | CurDAG->getRegister(IA64::r0, MVT::i64), | 
|  | 229 | CurDAG->getRegister(IA64::r0, MVT::i64), TmpPR); | 
|  | 230 | Chain = TmpPR2.getValue(1); | 
|  | 231 | } | 
|  | 232 |  | 
|  | 233 | SDOperand F0 = CurDAG->getRegister(IA64::F0, MVT::f64); | 
|  | 234 | SDOperand F1 = CurDAG->getRegister(IA64::F1, MVT::f64); | 
|  | 235 |  | 
|  | 236 | // now we apply newton's method, thrice! (FIXME: this is ~72 bits of | 
|  | 237 | // precision, don't need this much for f32/i32) | 
|  | 238 | TmpF6 = CurDAG->getTargetNode(IA64::CFNMAS1, MVT::f64, | 
|  | 239 | TmpF4, TmpF5, F1, TmpPR); | 
|  | 240 | Chain = TmpF6.getValue(1); | 
|  | 241 | TmpF7 = CurDAG->getTargetNode(IA64::CFMAS1, MVT::f64, | 
|  | 242 | TmpF3, TmpF5, F0, TmpPR); | 
|  | 243 | Chain = TmpF7.getValue(1); | 
|  | 244 | TmpF8 = CurDAG->getTargetNode(IA64::CFMAS1, MVT::f64, | 
|  | 245 | TmpF6, TmpF6, F0, TmpPR); | 
|  | 246 | Chain = TmpF8.getValue(1); | 
|  | 247 | TmpF9 = CurDAG->getTargetNode(IA64::CFMAS1, MVT::f64, | 
|  | 248 | TmpF6, TmpF7, TmpF7, TmpPR); | 
|  | 249 | Chain = TmpF9.getValue(1); | 
|  | 250 | TmpF10 = CurDAG->getTargetNode(IA64::CFMAS1, MVT::f64, | 
|  | 251 | TmpF6, TmpF5, TmpF5, TmpPR); | 
|  | 252 | Chain = TmpF10.getValue(1); | 
|  | 253 | TmpF11 = CurDAG->getTargetNode(IA64::CFMAS1, MVT::f64, | 
|  | 254 | TmpF8, TmpF9, TmpF9, TmpPR); | 
|  | 255 | Chain = TmpF11.getValue(1); | 
|  | 256 | TmpF12 = CurDAG->getTargetNode(IA64::CFMAS1, MVT::f64, | 
|  | 257 | TmpF8, TmpF10, TmpF10, TmpPR); | 
|  | 258 | Chain = TmpF12.getValue(1); | 
|  | 259 | TmpF13 = CurDAG->getTargetNode(IA64::CFNMAS1, MVT::f64, | 
|  | 260 | TmpF4, TmpF11, TmpF3, TmpPR); | 
|  | 261 | Chain = TmpF13.getValue(1); | 
|  | 262 |  | 
|  | 263 | // FIXME: this is unfortunate :( | 
|  | 264 | // the story is that the dest reg of the fnma above and the fma below | 
|  | 265 | // (and therefore possibly the src of the fcvt.fx[u] as well) cannot | 
|  | 266 | // be the same register, or this code breaks if the first argument is | 
|  | 267 | // zero. (e.g. without this hack, 0%8 yields -64, not 0.) | 
|  | 268 | TmpF14 = CurDAG->getTargetNode(IA64::CFMAS1, MVT::f64, | 
|  | 269 | TmpF13, TmpF12, TmpF11, TmpPR); | 
|  | 270 | Chain = TmpF14.getValue(1); | 
|  | 271 |  | 
|  | 272 | if(isModulus) { // XXX: fragile! fixes _only_ mod, *breaks* div! ! | 
|  | 273 | SDOperand bogus = CurDAG->getTargetNode(IA64::IUSE, MVT::Other, TmpF13); // hack :( | 
|  | 274 | Chain = bogus.getValue(0); // hmmm | 
|  | 275 | } | 
|  | 276 |  | 
|  | 277 | if(!isFP) { | 
|  | 278 | // round to an integer | 
|  | 279 | if(isSigned) { | 
|  | 280 | TmpF15 = CurDAG->getTargetNode(IA64::FCVTFXTRUNCS1, MVT::i64, TmpF14); | 
|  | 281 | Chain = TmpF15.getValue(1); | 
|  | 282 | } | 
|  | 283 | else { | 
|  | 284 | TmpF15 = CurDAG->getTargetNode(IA64::FCVTFXUTRUNCS1, MVT::i64, TmpF14); | 
|  | 285 | Chain = TmpF15.getValue(1); | 
|  | 286 | } | 
|  | 287 | } else { | 
|  | 288 | TmpF15 = TmpF14; | 
|  | 289 | // EXERCISE: can you see why TmpF15=TmpF14 does not work here, and | 
|  | 290 | // we really do need the above FMOV? ;) | 
|  | 291 | } | 
|  | 292 |  | 
|  | 293 | if(!isModulus) { | 
|  | 294 | if(isFP) { // extra worrying about div-by-zero | 
|  | 295 | // we do a 'conditional fmov' (of the correct result, depending | 
|  | 296 | // on how the frcpa predicate turned out) | 
|  | 297 | SDOperand bogoResult = CurDAG->getTargetNode(IA64::PFMOV, MVT::f64, | 
|  | 298 | TmpF12, TmpPR2); | 
|  | 299 | Chain = bogoResult.getValue(1); | 
|  | 300 | Result = CurDAG->getTargetNode(IA64::CFMOV, MVT::f64, bogoResult, | 
|  | 301 | TmpF15, TmpPR); | 
|  | 302 | Chain = Result.getValue(1); | 
|  | 303 | } | 
|  | 304 | else { | 
|  | 305 | Result = CurDAG->getTargetNode(IA64::GETFSIG, MVT::i64, TmpF15); | 
|  | 306 | Chain = Result.getValue(1); | 
|  | 307 | } | 
|  | 308 | } else { // this is a modulus | 
|  | 309 | if(!isFP) { | 
|  | 310 | // answer = q * (-b) + a | 
|  | 311 | SDOperand TmpI = CurDAG->getTargetNode(IA64::SUB, MVT::i64, | 
|  | 312 | CurDAG->getRegister(IA64::r0, MVT::i64), Tmp2); | 
|  | 313 | Chain = TmpI.getValue(1); | 
|  | 314 | SDOperand TmpF = CurDAG->getTargetNode(IA64::SETFSIG, MVT::f64, TmpI); | 
|  | 315 | Chain = TmpF.getValue(1); | 
|  | 316 | SDOperand ModulusResult = CurDAG->getTargetNode(IA64::XMAL, MVT::f64, | 
|  | 317 | TmpF15, TmpF, TmpF1); | 
|  | 318 | Chain = ModulusResult.getValue(1); | 
|  | 319 | Result = CurDAG->getTargetNode(IA64::GETFSIG, MVT::i64, ModulusResult); | 
|  | 320 | Chain = Result.getValue(1); | 
|  | 321 | } else { // FP modulus! The horror... the horror.... | 
|  | 322 | assert(0 && "sorry, no FP modulus just yet!\n!\n"); | 
|  | 323 | } | 
|  | 324 | } | 
|  | 325 |  | 
|  | 326 | return Result; | 
|  | 327 | } | 
|  | 328 |  | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 329 | // Select - Convert the specified operand from a target-independent to a | 
|  | 330 | // target-specific node if it hasn't already been changed. | 
|  | 331 | SDOperand IA64DAGToDAGISel::Select(SDOperand Op) { | 
|  | 332 | SDNode *N = Op.Val; | 
|  | 333 | if (N->getOpcode() >= ISD::BUILTIN_OP_END && | 
|  | 334 | N->getOpcode() < IA64ISD::FIRST_NUMBER) | 
|  | 335 | return Op;   // Already selected. | 
|  | 336 |  | 
|  | 337 | // If this has already been converted, use it. | 
|  | 338 | std::map<SDOperand, SDOperand>::iterator CGMI = CodeGenMap.find(Op); | 
|  | 339 | if (CGMI != CodeGenMap.end()) return CGMI->second; | 
|  | 340 |  | 
|  | 341 | switch (N->getOpcode()) { | 
|  | 342 | default: break; | 
|  | 343 |  | 
| Duraid Madina | 8f6c86f | 2005-12-22 03:58:17 +0000 | [diff] [blame] | 344 | case ISD::Register: return Op; // XXX: this is a hack, tblgen one day? | 
|  | 345 |  | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 346 | case ISD::CALL: | 
| Duraid Madina | 8f6c86f | 2005-12-22 03:58:17 +0000 | [diff] [blame] | 347 | case ISD::TAILCALL: { { | 
|  | 348 | // FIXME: This is a workaround for a bug in tblgen. | 
|  | 349 | // Pattern #47: (call:Flag (tglobaladdr:i32):$dst, ICC:Flag) | 
|  | 350 | // Emits: (CALL:void (tglobaladdr:i32):$dst) | 
|  | 351 | // Pattern complexity = 2  cost = 1 | 
|  | 352 | SDOperand N1 = N->getOperand(1); | 
|  | 353 | if (N1.getOpcode() != ISD::TargetGlobalAddress && | 
|  | 354 | N1.getOpcode() != ISD::ExternalSymbol) goto P47Fail; | 
|  | 355 | SDOperand InFlag = SDOperand(0, 0); | 
|  | 356 | SDOperand Chain = N->getOperand(0); | 
|  | 357 | SDOperand Tmp0 = N1; | 
|  | 358 | Chain = Select(Chain); | 
|  | 359 | SDOperand Result; | 
|  | 360 | if (N->getNumOperands() == 3) { | 
|  | 361 | InFlag = Select(N->getOperand(2)); | 
|  | 362 | Result = CurDAG->getTargetNode(IA64::BRCALL, MVT::Other, MVT::Flag, Tmp0, | 
|  | 363 | Chain, InFlag); | 
|  | 364 | } else { | 
|  | 365 | Result = CurDAG->getTargetNode(IA64::BRCALL, MVT::Other, MVT::Flag, Tmp0, | 
|  | 366 | Chain); | 
|  | 367 | } | 
|  | 368 | Chain = CodeGenMap[SDOperand(N, 0)] = Result.getValue(0); | 
|  | 369 | CodeGenMap[SDOperand(N, 1)] = Result.getValue(1); | 
|  | 370 | return Result.getValue(Op.ResNo); | 
|  | 371 | } | 
|  | 372 | P47Fail:; | 
|  | 373 |  | 
|  | 374 | } | 
| Duraid Madina | 8edbf80 | 2005-11-21 14:14:54 +0000 | [diff] [blame] | 375 |  | 
|  | 376 | case ISD::FDIV: | 
|  | 377 | case ISD::SDIV: | 
|  | 378 | case ISD::UDIV: | 
|  | 379 | case ISD::SREM: | 
|  | 380 | case ISD::UREM: return SelectDIV(Op); | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 381 |  | 
| Duraid Madina | 0c88f15 | 2005-11-25 07:49:25 +0000 | [diff] [blame] | 382 | case ISD::DYNAMIC_STACKALLOC: { | 
|  | 383 | if (!isa<ConstantSDNode>(N->getOperand(2)) || | 
|  | 384 | cast<ConstantSDNode>(N->getOperand(2))->getValue() != 0) { | 
|  | 385 | std::cerr << "Cannot allocate stack object with greater alignment than" | 
|  | 386 | << " the stack alignment yet!"; | 
|  | 387 | abort(); | 
|  | 388 | } | 
|  | 389 |  | 
|  | 390 | SDOperand Chain = Select(N->getOperand(0)); | 
|  | 391 | SDOperand Amt   = Select(N->getOperand(1)); | 
|  | 392 | SDOperand Reg = CurDAG->getRegister(IA64::r12, MVT::i64); | 
|  | 393 | SDOperand Val = CurDAG->getCopyFromReg(Chain, IA64::r12, MVT::i64); | 
|  | 394 | Chain = Val.getValue(1); | 
|  | 395 |  | 
|  | 396 | // Subtract the amount (guaranteed to be a multiple of the stack alignment) | 
|  | 397 | // from the stack pointer, giving us the result pointer. | 
|  | 398 | SDOperand Result = Select(CurDAG->getNode(ISD::SUB, MVT::i64, Val, Amt)); | 
|  | 399 |  | 
|  | 400 | // Copy this result back into r12. | 
|  | 401 | Chain = CurDAG->getNode(ISD::CopyToReg, MVT::Other, Chain, Reg, Result); | 
|  | 402 |  | 
|  | 403 | // Copy this result back out of r12 to make sure we're not using the stack | 
|  | 404 | // space without decrementing the stack pointer. | 
|  | 405 | Result = CurDAG->getCopyFromReg(Chain, IA64::r12, MVT::i64); | 
|  | 406 |  | 
|  | 407 | // Finally, replace the DYNAMIC_STACKALLOC with the copyfromreg. | 
|  | 408 | CodeGenMap[Op.getValue(0)] = Result; | 
|  | 409 | CodeGenMap[Op.getValue(1)] = Result.getValue(1); | 
|  | 410 | return SDOperand(Result.Val, Op.ResNo); | 
|  | 411 | } | 
|  | 412 |  | 
| Duraid Madina | 9abf165 | 2005-11-02 02:35:04 +0000 | [diff] [blame] | 413 | case ISD::ConstantFP: { | 
| Duraid Madina | 955ffaf | 2005-11-02 07:32:59 +0000 | [diff] [blame] | 414 | SDOperand Chain = CurDAG->getEntryNode(); // this is a constant, so.. | 
|  | 415 |  | 
| Duraid Madina | 9abf165 | 2005-11-02 02:35:04 +0000 | [diff] [blame] | 416 | if (cast<ConstantFPSDNode>(N)->isExactlyValue(+0.0)) | 
| Duraid Madina | 955ffaf | 2005-11-02 07:32:59 +0000 | [diff] [blame] | 417 | return CurDAG->getCopyFromReg(Chain, IA64::F0, MVT::f64); | 
| Duraid Madina | 9abf165 | 2005-11-02 02:35:04 +0000 | [diff] [blame] | 418 | else if (cast<ConstantFPSDNode>(N)->isExactlyValue(+1.0)) | 
| Duraid Madina | 955ffaf | 2005-11-02 07:32:59 +0000 | [diff] [blame] | 419 | return CurDAG->getCopyFromReg(Chain, IA64::F1, MVT::f64); | 
| Duraid Madina | 9abf165 | 2005-11-02 02:35:04 +0000 | [diff] [blame] | 420 | else | 
|  | 421 | assert(0 && "Unexpected FP constant!"); | 
|  | 422 | } | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 423 |  | 
|  | 424 | case ISD::FrameIndex: { // TODO: reduce creepyness | 
|  | 425 | int FI = cast<FrameIndexSDNode>(N)->getIndex(); | 
| Chris Lattner | a75694a | 2005-11-30 23:02:08 +0000 | [diff] [blame] | 426 | if (N->hasOneUse()) | 
|  | 427 | return CurDAG->SelectNodeTo(N, IA64::MOV, MVT::i64, | 
|  | 428 | CurDAG->getTargetFrameIndex(FI, MVT::i64)); | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 429 | return CurDAG->getTargetNode(IA64::MOV, MVT::i64, | 
|  | 430 | CurDAG->getTargetFrameIndex(FI, MVT::i64)); | 
|  | 431 | } | 
|  | 432 |  | 
| Duraid Madina | 7abaf90 | 2005-10-29 16:08:30 +0000 | [diff] [blame] | 433 | case ISD::ConstantPool: { | 
|  | 434 | Constant *C = cast<ConstantPoolSDNode>(N)->get(); | 
|  | 435 | SDOperand CPI = CurDAG->getTargetConstantPool(C, MVT::i64); | 
|  | 436 | return CurDAG->getTargetNode(IA64::ADDL_GA, MVT::i64, // ? | 
|  | 437 | CurDAG->getRegister(IA64::r1, MVT::i64), CPI); | 
|  | 438 | } | 
|  | 439 |  | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 440 | case ISD::GlobalAddress: { | 
|  | 441 | GlobalValue *GV = cast<GlobalAddressSDNode>(N)->getGlobal(); | 
|  | 442 | SDOperand GA = CurDAG->getTargetGlobalAddress(GV, MVT::i64); | 
|  | 443 | SDOperand Tmp = CurDAG->getTargetNode(IA64::ADDL_GA, MVT::i64, | 
|  | 444 | CurDAG->getRegister(IA64::r1, MVT::i64), GA); | 
|  | 445 | return CurDAG->getTargetNode(IA64::LD8, MVT::i64, Tmp); | 
|  | 446 | } | 
| Duraid Madina | 8f6c86f | 2005-12-22 03:58:17 +0000 | [diff] [blame] | 447 |  | 
|  | 448 | /* XXX  case ISD::ExternalSymbol: { | 
|  | 449 | SDOperand EA = CurDAG->getTargetExternalSymbol(cast<ExternalSymbolSDNode>(N)->getSymbol(), | 
|  | 450 | MVT::i64); | 
|  | 451 | SDOperand Tmp = CurDAG->getTargetNode(IA64::ADDL_EA, MVT::i64, | 
|  | 452 | CurDAG->getRegister(IA64::r1, MVT::i64), EA); | 
|  | 453 | return CurDAG->getTargetNode(IA64::LD8, MVT::i64, Tmp); | 
|  | 454 | } | 
|  | 455 | */ | 
|  | 456 |  | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 457 | case ISD::LOAD: | 
|  | 458 | case ISD::EXTLOAD: | 
|  | 459 | case ISD::ZEXTLOAD: { | 
|  | 460 | SDOperand Chain = Select(N->getOperand(0)); | 
|  | 461 | SDOperand Address = Select(N->getOperand(1)); | 
|  | 462 |  | 
|  | 463 | MVT::ValueType TypeBeingLoaded = (N->getOpcode() == ISD::LOAD) ? | 
|  | 464 | N->getValueType(0) : cast<VTSDNode>(N->getOperand(3))->getVT(); | 
|  | 465 | unsigned Opc; | 
|  | 466 | switch (TypeBeingLoaded) { | 
|  | 467 | default: N->dump(); assert(0 && "Cannot load this type!"); | 
| Duraid Madina | fc1d1b2 | 2005-11-04 09:59:06 +0000 | [diff] [blame] | 468 | case MVT::i1: { // this is a bool | 
|  | 469 | Opc = IA64::LD1; // first we load a byte, then compare for != 0 | 
| Duraid Madina | 8f6c86f | 2005-12-22 03:58:17 +0000 | [diff] [blame] | 470 | if(N->getValueType(0) == MVT::i1) // XXX: early exit! | 
|  | 471 | return CurDAG->SelectNodeTo(N, IA64::CMPNE, MVT::i1, MVT::Other, | 
| Chris Lattner | a75694a | 2005-11-30 23:02:08 +0000 | [diff] [blame] | 472 | CurDAG->getTargetNode(Opc, MVT::i64, Address), | 
|  | 473 | CurDAG->getRegister(IA64::r0, MVT::i64), | 
|  | 474 | Chain).getValue(Op.ResNo); | 
| Duraid Madina | 8f6c86f | 2005-12-22 03:58:17 +0000 | [diff] [blame] | 475 | /* otherwise, we want to load a bool into something bigger: LD1 | 
|  | 476 | will do that for us, so we just fall through */ | 
| Chris Lattner | a75694a | 2005-11-30 23:02:08 +0000 | [diff] [blame] | 477 | } | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 478 | case MVT::i8:  Opc = IA64::LD1; break; | 
|  | 479 | case MVT::i16: Opc = IA64::LD2; break; | 
|  | 480 | case MVT::i32: Opc = IA64::LD4; break; | 
|  | 481 | case MVT::i64: Opc = IA64::LD8; break; | 
|  | 482 |  | 
|  | 483 | case MVT::f32: Opc = IA64::LDF4; break; | 
|  | 484 | case MVT::f64: Opc = IA64::LDF8; break; | 
|  | 485 | } | 
|  | 486 |  | 
| Chris Lattner | a75694a | 2005-11-30 23:02:08 +0000 | [diff] [blame] | 487 | // TODO: comment this | 
|  | 488 | return CurDAG->SelectNodeTo(N, Opc, N->getValueType(0), MVT::Other, | 
|  | 489 | Address, Chain).getValue(Op.ResNo); | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 490 | } | 
|  | 491 |  | 
|  | 492 | case ISD::TRUNCSTORE: | 
|  | 493 | case ISD::STORE: { | 
|  | 494 | SDOperand Address = Select(N->getOperand(2)); | 
| Duraid Madina | 3c1c8c5 | 2005-11-07 03:11:02 +0000 | [diff] [blame] | 495 | SDOperand Chain = Select(N->getOperand(0)); | 
|  | 496 |  | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 497 | unsigned Opc; | 
|  | 498 | if (N->getOpcode() == ISD::STORE) { | 
|  | 499 | switch (N->getOperand(1).getValueType()) { | 
| Duraid Madina | 3c1c8c5 | 2005-11-07 03:11:02 +0000 | [diff] [blame] | 500 | default: assert(0 && "unknown type in store"); | 
|  | 501 | case MVT::i1: { // this is a bool | 
|  | 502 | Opc = IA64::ST1; // we store either 0 or 1 as a byte | 
| Chris Lattner | a75694a | 2005-11-30 23:02:08 +0000 | [diff] [blame] | 503 | SDOperand Tmp = | 
|  | 504 | CurDAG->getTargetNode(IA64::PADDS, MVT::i64, | 
|  | 505 | CurDAG->getRegister(IA64::r0, MVT::i64), | 
|  | 506 | CurDAG->getConstant(1, MVT::i64), | 
|  | 507 | Select(N->getOperand(1))); | 
|  | 508 | return CurDAG->SelectNodeTo(N, Opc, MVT::Other, Address, Tmp, Chain); | 
|  | 509 | } | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 510 | case MVT::i64: Opc = IA64::ST8;  break; | 
|  | 511 | case MVT::f64: Opc = IA64::STF8; break; | 
| Duraid Madina | 3c1c8c5 | 2005-11-07 03:11:02 +0000 | [diff] [blame] | 512 | } | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 513 | } else { //ISD::TRUNCSTORE | 
|  | 514 | switch(cast<VTSDNode>(N->getOperand(4))->getVT()) { | 
| Duraid Madina | 3c1c8c5 | 2005-11-07 03:11:02 +0000 | [diff] [blame] | 515 | default: assert(0 && "unknown type in truncstore"); | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 516 | case MVT::i8:  Opc = IA64::ST1;  break; | 
|  | 517 | case MVT::i16: Opc = IA64::ST2;  break; | 
|  | 518 | case MVT::i32: Opc = IA64::ST4;  break; | 
|  | 519 | case MVT::f32: Opc = IA64::STF4; break; | 
|  | 520 | } | 
|  | 521 | } | 
|  | 522 |  | 
| Chris Lattner | a75694a | 2005-11-30 23:02:08 +0000 | [diff] [blame] | 523 | return CurDAG->SelectNodeTo(N, Opc, MVT::Other, Select(N->getOperand(2)), | 
|  | 524 | Select(N->getOperand(1)), Chain); | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 525 | } | 
|  | 526 |  | 
|  | 527 | case ISD::BRCOND: { | 
|  | 528 | SDOperand Chain = Select(N->getOperand(0)); | 
|  | 529 | SDOperand CC = Select(N->getOperand(1)); | 
|  | 530 | MachineBasicBlock *Dest = | 
|  | 531 | cast<BasicBlockSDNode>(N->getOperand(2))->getBasicBlock(); | 
|  | 532 | //FIXME - we do NOT need long branches all the time | 
| Chris Lattner | a75694a | 2005-11-30 23:02:08 +0000 | [diff] [blame] | 533 | return CurDAG->SelectNodeTo(N, IA64::BRLCOND_NOTCALL, MVT::Other, CC, | 
|  | 534 | CurDAG->getBasicBlock(Dest), Chain); | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 535 | } | 
|  | 536 |  | 
|  | 537 | case ISD::CALLSEQ_START: | 
|  | 538 | case ISD::CALLSEQ_END: { | 
|  | 539 | int64_t Amt = cast<ConstantSDNode>(N->getOperand(1))->getValue(); | 
|  | 540 | unsigned Opc = N->getOpcode() == ISD::CALLSEQ_START ? | 
|  | 541 | IA64::ADJUSTCALLSTACKDOWN : IA64::ADJUSTCALLSTACKUP; | 
| Chris Lattner | a75694a | 2005-11-30 23:02:08 +0000 | [diff] [blame] | 542 | return CurDAG->SelectNodeTo(N, Opc, MVT::Other, | 
|  | 543 | getI64Imm(Amt), Select(N->getOperand(0))); | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 544 | } | 
|  | 545 |  | 
|  | 546 | case ISD::RET: { | 
|  | 547 | SDOperand Chain = Select(N->getOperand(0));     // Token chain. | 
| Duraid Madina | 8f6c86f | 2005-12-22 03:58:17 +0000 | [diff] [blame] | 548 | SDOperand InFlag; | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 549 |  | 
|  | 550 | switch (N->getNumOperands()) { | 
|  | 551 | default: | 
|  | 552 | assert(0 && "Unknown return instruction!"); | 
|  | 553 | case 2: { | 
|  | 554 | SDOperand RetVal = Select(N->getOperand(1)); | 
|  | 555 | switch (RetVal.getValueType()) { | 
|  | 556 | default: assert(0 && "I don't know how to return this type! (promote?)"); | 
|  | 557 | // FIXME: do I need to add support for bools here? | 
|  | 558 | // (return '0' or '1' in r8, basically...) | 
|  | 559 | // | 
|  | 560 | // FIXME: need to round floats - 80 bits is bad, the tester | 
|  | 561 | // told me so | 
|  | 562 | case MVT::i64: | 
|  | 563 | // we mark r8 as live on exit up above in LowerArguments() | 
|  | 564 | // BuildMI(BB, IA64::MOV, 1, IA64::r8).addReg(Tmp1); | 
|  | 565 | Chain = CurDAG->getCopyToReg(Chain, IA64::r8, RetVal); | 
| Duraid Madina | 8f6c86f | 2005-12-22 03:58:17 +0000 | [diff] [blame] | 566 | InFlag = Chain.getValue(1); | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 567 | break; | 
|  | 568 | case MVT::f64: | 
|  | 569 | // we mark F8 as live on exit up above in LowerArguments() | 
|  | 570 | // BuildMI(BB, IA64::FMOV, 1, IA64::F8).addReg(Tmp1); | 
|  | 571 | Chain = CurDAG->getCopyToReg(Chain, IA64::F8, RetVal); | 
| Duraid Madina | 8f6c86f | 2005-12-22 03:58:17 +0000 | [diff] [blame] | 572 | InFlag = Chain.getValue(1); | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 573 | break; | 
|  | 574 | } | 
|  | 575 | break; | 
|  | 576 | } | 
|  | 577 | case 1: | 
|  | 578 | break; | 
|  | 579 | } | 
|  | 580 |  | 
|  | 581 | // we need to copy VirtGPR (the vreg (to become a real reg)) that holds | 
|  | 582 | // the output of this function's alloc instruction back into ar.pfs | 
|  | 583 | // before we return. this copy must not float up above the last | 
|  | 584 | // outgoing call in this function!!! | 
|  | 585 | SDOperand AR_PFSVal = CurDAG->getCopyFromReg(Chain, IA64Lowering.VirtGPR, | 
|  | 586 | MVT::i64); | 
|  | 587 | Chain = AR_PFSVal.getValue(1); | 
|  | 588 | Chain = CurDAG->getCopyToReg(Chain, IA64::AR_PFS, AR_PFSVal); | 
|  | 589 |  | 
| Chris Lattner | a75694a | 2005-11-30 23:02:08 +0000 | [diff] [blame] | 590 | // and then just emit a 'ret' instruction | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 591 | // before returning, restore the ar.pfs register (set by the 'alloc' up top) | 
|  | 592 | // BuildMI(BB, IA64::MOV, 1).addReg(IA64::AR_PFS).addReg(IA64Lowering.VirtGPR); | 
|  | 593 | // | 
| Chris Lattner | a75694a | 2005-11-30 23:02:08 +0000 | [diff] [blame] | 594 | return CurDAG->SelectNodeTo(N, IA64::RET, MVT::Other, Chain); | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 595 | } | 
|  | 596 |  | 
|  | 597 | case ISD::BR: | 
|  | 598 | // FIXME: we don't need long branches all the time! | 
| Chris Lattner | a75694a | 2005-11-30 23:02:08 +0000 | [diff] [blame] | 599 | return CurDAG->SelectNodeTo(N, IA64::BRL_NOTCALL, MVT::Other, | 
|  | 600 | N->getOperand(1), Select(N->getOperand(0))); | 
| Duraid Madina | f221c26 | 2005-10-28 17:46:35 +0000 | [diff] [blame] | 601 | } | 
|  | 602 |  | 
|  | 603 | return SelectCode(Op); | 
|  | 604 | } | 
|  | 605 |  | 
|  | 606 |  | 
|  | 607 | /// createIA64DAGToDAGInstructionSelector - This pass converts a legalized DAG | 
|  | 608 | /// into an IA64-specific DAG, ready for instruction scheduling. | 
|  | 609 | /// | 
|  | 610 | FunctionPass *llvm::createIA64DAGToDAGInstructionSelector(TargetMachine &TM) { | 
|  | 611 | return new IA64DAGToDAGISel(TM); | 
|  | 612 | } | 
|  | 613 |  |