| Chris Lattner | c9670ef | 2003-07-31 04:43:49 +0000 | [diff] [blame] | 1 | //===- CodeEmitterGen.cpp - Code Emitter Generator ------------------------===// | 
| John Criswell | 01d4582 | 2003-10-20 20:20:30 +0000 | [diff] [blame] | 2 | // | 
|  | 3 | //                     The LLVM Compiler Infrastructure | 
|  | 4 | // | 
|  | 5 | // This file was developed by the LLVM research group and is distributed under | 
|  | 6 | // the University of Illinois Open Source License. See LICENSE.TXT for details. | 
|  | 7 | // | 
|  | 8 | //===----------------------------------------------------------------------===// | 
| Chris Lattner | c9670ef | 2003-07-31 04:43:49 +0000 | [diff] [blame] | 9 | // | 
| Misha Brukman | 4e4f863 | 2004-08-04 22:07:54 +0000 | [diff] [blame] | 10 | // CodeEmitterGen uses the descriptions of instructions and their fields to | 
|  | 11 | // construct an automated code emitter: a function that, given a MachineInstr, | 
|  | 12 | // returns the (currently, 32-bit unsigned) value of the instruction. | 
| Chris Lattner | c9670ef | 2003-07-31 04:43:49 +0000 | [diff] [blame] | 13 | // | 
|  | 14 | //===----------------------------------------------------------------------===// | 
|  | 15 |  | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 16 | #include "CodeEmitterGen.h" | 
| Misha Brukman | d7a5b28 | 2004-08-09 19:10:43 +0000 | [diff] [blame] | 17 | #include "CodeGenTarget.h" | 
| Chris Lattner | c9670ef | 2003-07-31 04:43:49 +0000 | [diff] [blame] | 18 | #include "Record.h" | 
| Reid Spencer | 551ccae | 2004-09-01 22:55:40 +0000 | [diff] [blame] | 19 | #include "llvm/Support/Debug.h" | 
| Chris Lattner | 2082ebe | 2004-08-01 03:55:39 +0000 | [diff] [blame] | 20 | using namespace llvm; | 
| Brian Gaeke | d0fde30 | 2003-11-11 22:41:34 +0000 | [diff] [blame] | 21 |  | 
| Misha Brukman | 28eefa5 | 2004-10-14 05:53:01 +0000 | [diff] [blame^] | 22 | void CodeEmitterGen::emitInstrOpBits(std::ostream &o, | 
|  | 23 | const std::vector<RecordVal> &Vals, | 
|  | 24 | std::map<std::string, unsigned> &OpOrder, | 
|  | 25 | std::map<std::string, bool> &OpContinuous) | 
|  | 26 | { | 
|  | 27 | for (unsigned f = 0, e = Vals.size(); f != e; ++f) { | 
|  | 28 | if (Vals[f].getPrefix()) { | 
|  | 29 | BitsInit *FieldInitializer = (BitsInit*)Vals[f].getValue(); | 
|  | 30 |  | 
|  | 31 | // Scan through the field looking for bit initializers of the current | 
|  | 32 | // variable... | 
|  | 33 | for (int i = FieldInitializer->getNumBits()-1; i >= 0; --i) { | 
|  | 34 | Init *I = FieldInitializer->getBit(i); | 
|  | 35 | if (BitInit *BI = dynamic_cast<BitInit*>(I)) { | 
|  | 36 | DEBUG(o << "      // bit init: f: " << f << ", i: " << i << "\n"); | 
|  | 37 | } else if (UnsetInit *UI = dynamic_cast<UnsetInit*>(I)) { | 
|  | 38 | DEBUG(o << "      // unset init: f: " << f << ", i: " << i << "\n"); | 
|  | 39 | } else if (VarBitInit *VBI = dynamic_cast<VarBitInit*>(I)) { | 
|  | 40 | TypedInit *TI = VBI->getVariable(); | 
|  | 41 | if (VarInit *VI = dynamic_cast<VarInit*>(TI)) { | 
|  | 42 | // If the bits of the field are laid out consecutively in the | 
|  | 43 | // instruction, then instead of separately ORing in bits, just | 
|  | 44 | // mask and shift the entire field for efficiency. | 
|  | 45 | if (OpContinuous[VI->getName()]) { | 
|  | 46 | // already taken care of in the loop above, thus there is no | 
|  | 47 | // need to individually OR in the bits | 
|  | 48 |  | 
|  | 49 | // for debugging, output the regular version anyway, commented | 
|  | 50 | DEBUG(o << "      // Value |= getValueBit(op" | 
|  | 51 | << OpOrder[VI->getName()] << ", " << VBI->getBitNum() | 
|  | 52 | << ")" << " << " << i << ";\n"); | 
|  | 53 | } else { | 
|  | 54 | o << "      Value |= getValueBit(op" << OpOrder[VI->getName()] | 
|  | 55 | << ", " << VBI->getBitNum() | 
|  | 56 | << ")" << " << " << i << ";\n"; | 
|  | 57 | } | 
|  | 58 | } else if (FieldInit *FI = dynamic_cast<FieldInit*>(TI)) { | 
|  | 59 | // FIXME: implement this! | 
|  | 60 | std::cerr << "Error: FieldInit not implemented!\n"; | 
|  | 61 | abort(); | 
|  | 62 | } else { | 
|  | 63 | std::cerr << "Error: unimplemented case in " | 
|  | 64 | << "CodeEmitterGen::emitInstrOpBits()\n"; | 
|  | 65 | abort(); | 
|  | 66 | } | 
|  | 67 | } | 
|  | 68 | } | 
|  | 69 | } | 
|  | 70 | } | 
|  | 71 | } | 
|  | 72 |  | 
|  | 73 |  | 
| Chris Lattner | 048c00d | 2003-08-01 04:38:18 +0000 | [diff] [blame] | 74 | void CodeEmitterGen::run(std::ostream &o) { | 
| Misha Brukman | e2ba778 | 2004-08-10 18:31:01 +0000 | [diff] [blame] | 75 | CodeGenTarget Target; | 
| Chris Lattner | 048c00d | 2003-08-01 04:38:18 +0000 | [diff] [blame] | 76 | std::vector<Record*> Insts = Records.getAllDerivedDefinitions("Instruction"); | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 77 |  | 
| Chris Lattner | 0e5e49e | 2003-08-06 04:36:35 +0000 | [diff] [blame] | 78 | EmitSourceFileHeader("Machine Code Emitter", o); | 
| Chris Lattner | 2c38413 | 2004-08-17 03:08:28 +0000 | [diff] [blame] | 79 | o << "namespace llvm {\n\n"; | 
| Misha Brukman | e2ba778 | 2004-08-10 18:31:01 +0000 | [diff] [blame] | 80 | std::string Namespace = Insts[0]->getValueAsString("Namespace") + "::"; | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 81 |  | 
| Misha Brukman | ad346ad | 2004-08-10 20:54:58 +0000 | [diff] [blame] | 82 | // Emit function declaration | 
| Misha Brukman | e2ba778 | 2004-08-10 18:31:01 +0000 | [diff] [blame] | 83 | o << "unsigned " << Target.getName() << "CodeEmitter::" | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 84 | << "getBinaryCodeForInstr(MachineInstr &MI) {\n" | 
|  | 85 | << "  unsigned Value = 0;\n" | 
| Misha Brukman | f4ef4c8 | 2003-06-06 00:27:02 +0000 | [diff] [blame] | 86 | << "  DEBUG(std::cerr << MI);\n" | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 87 | << "  switch (MI.getOpcode()) {\n"; | 
| Misha Brukman | ad346ad | 2004-08-10 20:54:58 +0000 | [diff] [blame] | 88 |  | 
|  | 89 | // Emit a case statement for each opcode | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 90 | for (std::vector<Record*>::iterator I = Insts.begin(), E = Insts.end(); | 
| Chris Lattner | cf1b585 | 2003-08-01 04:15:25 +0000 | [diff] [blame] | 91 | I != E; ++I) { | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 92 | Record *R = *I; | 
| Misha Brukman | ecc7fd3 | 2003-05-28 18:29:10 +0000 | [diff] [blame] | 93 | o << "    case " << Namespace << R->getName() << ": {\n" | 
| Misha Brukman | f4ef4c8 | 2003-06-06 00:27:02 +0000 | [diff] [blame] | 94 | << "      DEBUG(std::cerr << \"Emitting " << R->getName() << "\\n\");\n"; | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 95 |  | 
| Chris Lattner | 6f334ad | 2003-08-01 04:46:24 +0000 | [diff] [blame] | 96 | BitsInit *BI = R->getValueAsBitsInit("Inst"); | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 97 |  | 
| Misha Brukman | 28eefa5 | 2004-10-14 05:53:01 +0000 | [diff] [blame^] | 98 | // For little-endian instruction bit encodings, reverse the bit order | 
|  | 99 | if (Target.isLittleEndianEncoding()) { | 
|  | 100 | unsigned numBits = BI->getNumBits(); | 
|  | 101 | BitsInit *NewBI = new BitsInit(numBits); | 
|  | 102 | for (unsigned bit = 0, end = numBits / 2; bit != end; ++bit) { | 
|  | 103 | unsigned bitSwapIdx = numBits - bit - 1; | 
|  | 104 | Init *OrigBit = BI->getBit(bit); | 
|  | 105 | Init *BitSwap = BI->getBit(bitSwapIdx); | 
|  | 106 | NewBI->setBit(bit, BitSwap); | 
|  | 107 | NewBI->setBit(bitSwapIdx, OrigBit); | 
|  | 108 | } | 
|  | 109 | if (numBits % 2) { | 
|  | 110 | unsigned middle = (numBits + 1) / 2; | 
|  | 111 | NewBI->setBit(middle, BI->getBit(middle)); | 
|  | 112 | } | 
|  | 113 | BI = NewBI; | 
|  | 114 | } | 
|  | 115 |  | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 116 | unsigned Value = 0; | 
|  | 117 | const std::vector<RecordVal> &Vals = R->getValues(); | 
|  | 118 |  | 
| Misha Brukman | f6e5217 | 2003-07-15 21:26:09 +0000 | [diff] [blame] | 119 | DEBUG(o << "      // prefilling: "); | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 120 | // Start by filling in fixed values... | 
| Misha Brukman | cbfde0a | 2003-05-27 22:19:58 +0000 | [diff] [blame] | 121 | for (unsigned i = 0, e = BI->getNumBits(); i != e; ++i) { | 
|  | 122 | if (BitInit *B = dynamic_cast<BitInit*>(BI->getBit(e-i-1))) { | 
|  | 123 | Value |= B->getValue() << (e-i-1); | 
| Misha Brukman | f6e5217 | 2003-07-15 21:26:09 +0000 | [diff] [blame] | 124 | DEBUG(o << B->getValue()); | 
| Misha Brukman | cbfde0a | 2003-05-27 22:19:58 +0000 | [diff] [blame] | 125 | } else { | 
| Misha Brukman | f6e5217 | 2003-07-15 21:26:09 +0000 | [diff] [blame] | 126 | DEBUG(o << "0"); | 
| Misha Brukman | cbfde0a | 2003-05-27 22:19:58 +0000 | [diff] [blame] | 127 | } | 
|  | 128 | } | 
| Misha Brukman | f6e5217 | 2003-07-15 21:26:09 +0000 | [diff] [blame] | 129 | DEBUG(o << "\n"); | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 130 |  | 
| Chris Lattner | 6f334ad | 2003-08-01 04:46:24 +0000 | [diff] [blame] | 131 | DEBUG(o << "      // " << *R->getValue("Inst") << "\n"); | 
| Misha Brukman | cbfde0a | 2003-05-27 22:19:58 +0000 | [diff] [blame] | 132 | o << "      Value = " << Value << "U;\n\n"; | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 133 |  | 
| Misha Brukman | 28eefa5 | 2004-10-14 05:53:01 +0000 | [diff] [blame^] | 134 | // Loop over all of the fields in the instruction, determining which are the | 
| Chris Lattner | d7efef9 | 2003-08-05 03:53:04 +0000 | [diff] [blame] | 135 | // operands to the instruction. | 
| Misha Brukman | cbfde0a | 2003-05-27 22:19:58 +0000 | [diff] [blame] | 136 | unsigned op = 0; | 
| Chris Lattner | d7efef9 | 2003-08-05 03:53:04 +0000 | [diff] [blame] | 137 | std::map<std::string, unsigned> OpOrder; | 
|  | 138 | std::map<std::string, bool> OpContinuous; | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 139 | for (unsigned i = 0, e = Vals.size(); i != e; ++i) { | 
| Misha Brukman | 28eefa5 | 2004-10-14 05:53:01 +0000 | [diff] [blame^] | 140 | if (!Vals[i].getPrefix() && !Vals[i].getValue()->isComplete()) { | 
| Misha Brukman | 7eac476 | 2003-07-15 21:00:32 +0000 | [diff] [blame] | 141 | // Is the operand continuous? If so, we can just mask and OR it in | 
|  | 142 | // instead of doing it bit-by-bit, saving a lot in runtime cost. | 
| Misha Brukman | 28eefa5 | 2004-10-14 05:53:01 +0000 | [diff] [blame^] | 143 | BitsInit *InstInit = BI; | 
| Chris Lattner | d7efef9 | 2003-08-05 03:53:04 +0000 | [diff] [blame] | 144 | int beginBitInVar = -1, endBitInVar = -1; | 
|  | 145 | int beginBitInInst = -1, endBitInInst = -1; | 
| Misha Brukman | 7eac476 | 2003-07-15 21:00:32 +0000 | [diff] [blame] | 146 | bool continuous = true; | 
|  | 147 |  | 
|  | 148 | for (int bit = InstInit->getNumBits()-1; bit >= 0; --bit) { | 
|  | 149 | if (VarBitInit *VBI = | 
|  | 150 | dynamic_cast<VarBitInit*>(InstInit->getBit(bit))) { | 
|  | 151 | TypedInit *TI = VBI->getVariable(); | 
|  | 152 | if (VarInit *VI = dynamic_cast<VarInit*>(TI)) { | 
|  | 153 | // only process the current variable | 
|  | 154 | if (VI->getName() != Vals[i].getName()) | 
|  | 155 | continue; | 
|  | 156 |  | 
|  | 157 | if (beginBitInVar == -1) | 
|  | 158 | beginBitInVar = VBI->getBitNum(); | 
|  | 159 |  | 
|  | 160 | if (endBitInVar == -1) | 
|  | 161 | endBitInVar = VBI->getBitNum(); | 
|  | 162 | else { | 
|  | 163 | if (endBitInVar == (int)VBI->getBitNum() + 1) | 
|  | 164 | endBitInVar = VBI->getBitNum(); | 
|  | 165 | else { | 
|  | 166 | continuous = false; | 
|  | 167 | break; | 
|  | 168 | } | 
|  | 169 | } | 
|  | 170 |  | 
|  | 171 | if (beginBitInInst == -1) | 
|  | 172 | beginBitInInst = bit; | 
|  | 173 | if (endBitInInst == -1) | 
|  | 174 | endBitInInst = bit; | 
|  | 175 | else { | 
|  | 176 | if (endBitInInst == bit + 1) | 
|  | 177 | endBitInInst = bit; | 
|  | 178 | else { | 
|  | 179 | continuous = false; | 
|  | 180 | break; | 
|  | 181 | } | 
|  | 182 | } | 
|  | 183 |  | 
|  | 184 | // maintain same distance between bits in field and bits in | 
|  | 185 | // instruction. if the relative distances stay the same | 
|  | 186 | // throughout, | 
| Chris Lattner | d7efef9 | 2003-08-05 03:53:04 +0000 | [diff] [blame] | 187 | if (beginBitInVar - (int)VBI->getBitNum() != | 
|  | 188 | beginBitInInst - bit) { | 
| Misha Brukman | 7eac476 | 2003-07-15 21:00:32 +0000 | [diff] [blame] | 189 | continuous = false; | 
|  | 190 | break; | 
|  | 191 | } | 
|  | 192 | } | 
|  | 193 | } | 
|  | 194 | } | 
|  | 195 |  | 
| Chris Lattner | ffaee37 | 2003-08-05 03:59:01 +0000 | [diff] [blame] | 196 | // If we have found no bit in "Inst" which comes from this field, then | 
|  | 197 | // this is not an operand!! | 
| Chris Lattner | d7efef9 | 2003-08-05 03:53:04 +0000 | [diff] [blame] | 198 | if (beginBitInInst != -1) { | 
|  | 199 | o << "      // op" << op << ": " << Vals[i].getName() << "\n" | 
|  | 200 | << "      int64_t op" << op | 
|  | 201 | <<" = getMachineOpValue(MI, MI.getOperand("<<op<<"));\n"; | 
|  | 202 | //<< "   MachineOperand &op" << op <<" = MI.getOperand("<<op<<");\n"; | 
|  | 203 | OpOrder[Vals[i].getName()] = op++; | 
| Misha Brukman | 7eac476 | 2003-07-15 21:00:32 +0000 | [diff] [blame] | 204 |  | 
| Chris Lattner | d7efef9 | 2003-08-05 03:53:04 +0000 | [diff] [blame] | 205 | DEBUG(o << "      // Var: begin = " << beginBitInVar | 
|  | 206 | << ", end = " << endBitInVar | 
|  | 207 | << "; Inst: begin = " << beginBitInInst | 
|  | 208 | << ", end = " << endBitInInst << "\n"); | 
|  | 209 |  | 
|  | 210 | if (continuous) { | 
|  | 211 | DEBUG(o << "      // continuous: op" << OpOrder[Vals[i].getName()] | 
|  | 212 | << "\n"); | 
|  | 213 |  | 
|  | 214 | // Mask off the right bits | 
|  | 215 | // Low mask (ie. shift, if necessary) | 
| Misha Brukman | dfd414a | 2003-08-06 16:28:49 +0000 | [diff] [blame] | 216 | assert(endBitInVar >= 0 && "Negative shift amount in masking!"); | 
| Chris Lattner | d7efef9 | 2003-08-05 03:53:04 +0000 | [diff] [blame] | 217 | if (endBitInVar != 0) { | 
|  | 218 | o << "      op" << OpOrder[Vals[i].getName()] | 
|  | 219 | << " >>= " << endBitInVar << ";\n"; | 
|  | 220 | beginBitInVar -= endBitInVar; | 
|  | 221 | endBitInVar = 0; | 
|  | 222 | } | 
|  | 223 |  | 
|  | 224 | // High mask | 
| Misha Brukman | 7eac476 | 2003-07-15 21:00:32 +0000 | [diff] [blame] | 225 | o << "      op" << OpOrder[Vals[i].getName()] | 
| Chris Lattner | d7efef9 | 2003-08-05 03:53:04 +0000 | [diff] [blame] | 226 | << " &= (1<<" << beginBitInVar+1 << ") - 1;\n"; | 
|  | 227 |  | 
|  | 228 | // Shift the value to the correct place (according to place in inst) | 
| Misha Brukman | 4e4f863 | 2004-08-04 22:07:54 +0000 | [diff] [blame] | 229 | assert(endBitInInst >= 0 && "Negative shift amount!"); | 
| Chris Lattner | d7efef9 | 2003-08-05 03:53:04 +0000 | [diff] [blame] | 230 | if (endBitInInst != 0) | 
|  | 231 | o << "      op" << OpOrder[Vals[i].getName()] | 
| Misha Brukman | 7eac476 | 2003-07-15 21:00:32 +0000 | [diff] [blame] | 232 | << " <<= " << endBitInInst << ";\n"; | 
| Chris Lattner | d7efef9 | 2003-08-05 03:53:04 +0000 | [diff] [blame] | 233 |  | 
|  | 234 | // Just OR in the result | 
|  | 235 | o << "      Value |= op" << OpOrder[Vals[i].getName()] << ";\n"; | 
|  | 236 | } | 
|  | 237 |  | 
|  | 238 | // otherwise, will be taken care of in the loop below using this | 
|  | 239 | // value: | 
|  | 240 | OpContinuous[Vals[i].getName()] = continuous; | 
| Misha Brukman | 7eac476 | 2003-07-15 21:00:32 +0000 | [diff] [blame] | 241 | } | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 242 | } | 
|  | 243 | } | 
|  | 244 |  | 
| Misha Brukman | 28eefa5 | 2004-10-14 05:53:01 +0000 | [diff] [blame^] | 245 | emitInstrOpBits(o, Vals, OpOrder, OpContinuous); | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 246 |  | 
|  | 247 | o << "      break;\n" | 
|  | 248 | << "    }\n"; | 
|  | 249 | } | 
| Misha Brukman | 7eac476 | 2003-07-15 21:00:32 +0000 | [diff] [blame] | 250 |  | 
| Misha Brukman | 28eefa5 | 2004-10-14 05:53:01 +0000 | [diff] [blame^] | 251 | // Default case: unhandled opcode | 
| Misha Brukman | cbfde0a | 2003-05-27 22:19:58 +0000 | [diff] [blame] | 252 | o << "  default:\n" | 
| Misha Brukman | 0bb806b | 2003-09-17 18:21:48 +0000 | [diff] [blame] | 253 | << "    std::cerr << \"Not supported instr: \" << MI << \"\\n\";\n" | 
| Misha Brukman | cbfde0a | 2003-05-27 22:19:58 +0000 | [diff] [blame] | 254 | << "    abort();\n" | 
|  | 255 | << "  }\n" | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 256 | << "  return Value;\n" | 
| Misha Brukman | 28eefa5 | 2004-10-14 05:53:01 +0000 | [diff] [blame^] | 257 | << "}\n\n"; | 
| Brian Gaeke | d0fde30 | 2003-11-11 22:41:34 +0000 | [diff] [blame] | 258 |  | 
| Chris Lattner | 2c38413 | 2004-08-17 03:08:28 +0000 | [diff] [blame] | 259 | o << "} // End llvm namespace \n"; | 
| Misha Brukman | 9fff7e1 | 2003-05-24 00:15:53 +0000 | [diff] [blame] | 260 | } |