OLD | NEW |
---|---|
1 //===- subzero/src/IceInstARM32.h - ARM32 machine instructions --*- C++ -*-===// | 1 //===- subzero/src/IceInstARM32.h - ARM32 machine instructions --*- C++ -*-===// |
2 // | 2 // |
3 // The Subzero Code Generator | 3 // The Subzero Code Generator |
4 // | 4 // |
5 // This file is distributed under the University of Illinois Open Source | 5 // This file is distributed under the University of Illinois Open Source |
6 // License. See LICENSE.TXT for details. | 6 // License. See LICENSE.TXT for details. |
7 // | 7 // |
8 //===----------------------------------------------------------------------===// | 8 //===----------------------------------------------------------------------===// |
9 /// | 9 /// |
10 /// \file | 10 /// \file |
(...skipping 302 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
313 Sbc, | 313 Sbc, |
314 Sdiv, | 314 Sdiv, |
315 Str, | 315 Str, |
316 Sub, | 316 Sub, |
317 Sxt, | 317 Sxt, |
318 Trap, | 318 Trap, |
319 Tst, | 319 Tst, |
320 Udiv, | 320 Udiv, |
321 Umull, | 321 Umull, |
322 Uxt, | 322 Uxt, |
323 Vabs, | |
323 Vadd, | 324 Vadd, |
324 Vcmp, | 325 Vcmp, |
325 Vcvt, | 326 Vcvt, |
326 Vdiv, | 327 Vdiv, |
327 Vldr, | |
328 Vmov, | |
329 Vmrs, | 328 Vmrs, |
330 Vmul, | 329 Vmul, |
331 Vsqrt, | 330 Vsqrt, |
332 Vsub | 331 Vsub |
333 }; | 332 }; |
334 | 333 |
335 static const char *getWidthString(Type Ty); | 334 static const char *getWidthString(Type Ty); |
336 static const char *getVecWidthString(Type Ty); | 335 static const char *getVecWidthString(Type Ty); |
337 static CondARM32::Cond getOppositeCondition(CondARM32::Cond Cond); | 336 static CondARM32::Cond getOppositeCondition(CondARM32::Cond Cond); |
338 | 337 |
(...skipping 437 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
776 using InstARM32Sub = InstARM32ThreeAddrGPR<InstARM32::Sub>; | 775 using InstARM32Sub = InstARM32ThreeAddrGPR<InstARM32::Sub>; |
777 using InstARM32Udiv = InstARM32ThreeAddrGPR<InstARM32::Udiv>; | 776 using InstARM32Udiv = InstARM32ThreeAddrGPR<InstARM32::Udiv>; |
778 using InstARM32Vadd = InstARM32ThreeAddrFP<InstARM32::Vadd>; | 777 using InstARM32Vadd = InstARM32ThreeAddrFP<InstARM32::Vadd>; |
779 using InstARM32Vdiv = InstARM32ThreeAddrFP<InstARM32::Vdiv>; | 778 using InstARM32Vdiv = InstARM32ThreeAddrFP<InstARM32::Vdiv>; |
780 using InstARM32Vmul = InstARM32ThreeAddrFP<InstARM32::Vmul>; | 779 using InstARM32Vmul = InstARM32ThreeAddrFP<InstARM32::Vmul>; |
781 using InstARM32Vsub = InstARM32ThreeAddrFP<InstARM32::Vsub>; | 780 using InstARM32Vsub = InstARM32ThreeAddrFP<InstARM32::Vsub>; |
782 using InstARM32Ldr = InstARM32Movlike<InstARM32::Ldr>; | 781 using InstARM32Ldr = InstARM32Movlike<InstARM32::Ldr>; |
783 /// Move instruction (variable <- flex). This is more of a pseudo-inst. If var | 782 /// Move instruction (variable <- flex). This is more of a pseudo-inst. If var |
784 /// is a register, then we use "mov". If var is stack, then we use "str" to | 783 /// is a register, then we use "mov". If var is stack, then we use "str" to |
785 /// store to the stack. | 784 /// store to the stack. |
786 using InstARM32Mov = InstARM32Movlike<InstARM32::Mov>; | 785 //!!!!!!!!!!!!!!!!!!!!!!using InstARM32Mov = InstARM32Movlike<InstARM32::Mov>; |
Jim Stichnoth
2015/09/24 05:57:13
Explain this comment, or delete it?
John
2015/09/24 21:28:39
Done.
| |
787 /// Represents various vector mov instruction forms (simple single source, | 786 |
788 /// single dest forms only, not the 2 GPR <-> 1 D reg forms, etc.). | |
789 using InstARM32Vldr = InstARM32Movlike<InstARM32::Vldr>; | |
790 /// MovT leaves the bottom bits alone so dest is also a source. This helps | 787 /// MovT leaves the bottom bits alone so dest is also a source. This helps |
791 /// indicate that a previous MovW setting dest is not dead code. | 788 /// indicate that a previous MovW setting dest is not dead code. |
792 using InstARM32Movt = InstARM32TwoAddrGPR<InstARM32::Movt>; | 789 using InstARM32Movt = InstARM32TwoAddrGPR<InstARM32::Movt>; |
793 using InstARM32Movw = InstARM32UnaryopGPR<InstARM32::Movw, false>; | 790 using InstARM32Movw = InstARM32UnaryopGPR<InstARM32::Movw, false>; |
794 using InstARM32Clz = InstARM32UnaryopGPR<InstARM32::Clz, false>; | 791 using InstARM32Clz = InstARM32UnaryopGPR<InstARM32::Clz, false>; |
795 using InstARM32Mvn = InstARM32UnaryopGPR<InstARM32::Mvn, false>; | 792 using InstARM32Mvn = InstARM32UnaryopGPR<InstARM32::Mvn, false>; |
796 using InstARM32Rbit = InstARM32UnaryopGPR<InstARM32::Rbit, false>; | 793 using InstARM32Rbit = InstARM32UnaryopGPR<InstARM32::Rbit, false>; |
797 using InstARM32Rev = InstARM32UnaryopGPR<InstARM32::Rev, false>; | 794 using InstARM32Rev = InstARM32UnaryopGPR<InstARM32::Rev, false>; |
798 // Technically, the uxt{b,h} and sxt{b,h} instructions have a rotation operand | 795 // Technically, the uxt{b,h} and sxt{b,h} instructions have a rotation operand |
799 // as well (rotate source by 8, 16, 24 bits prior to extending), but we aren't | 796 // as well (rotate source by 8, 16, 24 bits prior to extending), but we aren't |
(...skipping 313 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
1113 static bool classof(const Inst *Inst) { return isClassof(Inst, Vcvt); } | 1110 static bool classof(const Inst *Inst) { return isClassof(Inst, Vcvt); } |
1114 | 1111 |
1115 private: | 1112 private: |
1116 InstARM32Vcvt(Cfg *Func, Variable *Dest, Variable *Src, VcvtVariant Variant, | 1113 InstARM32Vcvt(Cfg *Func, Variable *Dest, Variable *Src, VcvtVariant Variant, |
1117 CondARM32::Cond Predicate); | 1114 CondARM32::Cond Predicate); |
1118 | 1115 |
1119 const VcvtVariant Variant; | 1116 const VcvtVariant Variant; |
1120 }; | 1117 }; |
1121 | 1118 |
1122 /// Handles (some of) vmov's various formats. | 1119 /// Handles (some of) vmov's various formats. |
1123 class InstARM32Vmov final : public InstARM32Pred { | 1120 class InstARM32Mov final : public InstARM32Pred { |
1124 InstARM32Vmov() = delete; | 1121 InstARM32Mov() = delete; |
1125 InstARM32Vmov(const InstARM32Vmov &) = delete; | 1122 InstARM32Mov(const InstARM32Mov &) = delete; |
1126 InstARM32Vmov &operator=(const InstARM32Vmov &) = delete; | 1123 InstARM32Mov &operator=(const InstARM32Mov &) = delete; |
1127 | 1124 |
1128 public: | 1125 public: |
1129 /// RegisterPair is used to group registers in | 1126 static InstARM32Mov *create(Cfg *Func, Variable *Dest, Operand *Src, |
1130 /// | 1127 CondARM32::Cond Predicate) { |
1131 /// vmov D, (R, R) | 1128 return new (Func->allocate<InstARM32Mov>()) |
1132 /// | 1129 InstARM32Mov(Func, Dest, Src, Predicate); |
1133 /// and | |
1134 /// | |
1135 /// vmov (R, R), D | |
1136 struct RegisterPair { | |
1137 explicit RegisterPair(Variable *V0, Variable *V1) : _0(V0), _1(V1) { | |
1138 assert(V0->getType() == IceType_i32); | |
1139 assert(V1->getType() == IceType_i32); | |
1140 } | |
1141 Variable *_0; | |
1142 Variable *_1; | |
1143 }; | |
1144 | |
1145 static InstARM32Vmov *create(Cfg *Func, Variable *Dest, Operand *Src, | |
1146 CondARM32::Cond Predicate) { | |
1147 return new (Func->allocate<InstARM32Vmov>()) | |
1148 InstARM32Vmov(Func, Dest, Src, Predicate); | |
1149 } | |
1150 static InstARM32Vmov *create(Cfg *Func, const RegisterPair &Dests, | |
1151 Variable *Src, CondARM32::Cond Predicate) { | |
1152 return new (Func->allocate<InstARM32Vmov>()) | |
1153 InstARM32Vmov(Func, Dests, Src, Predicate); | |
1154 } | |
1155 static InstARM32Vmov *create(Cfg *Func, Variable *Dest, | |
1156 const RegisterPair &Srcs, | |
1157 CondARM32::Cond Predicate) { | |
1158 return new (Func->allocate<InstARM32Vmov>()) | |
1159 InstARM32Vmov(Func, Dest, Srcs, Predicate); | |
1160 } | 1130 } |
1161 bool isRedundantAssign() const override { | 1131 bool isRedundantAssign() const override { |
1162 return Dest1 == nullptr && getSrcSize() == 1 && | 1132 return !isMultiDest() && !isMultiSource() && |
1163 checkForRedundantAssign(getDest(), getSrc(0)); | 1133 checkForRedundantAssign(getDest(), getSrc(0)); |
1164 } | 1134 } |
1165 bool isSimpleAssign() const override { return true; } | 1135 bool isSimpleAssign() const override { return true; } |
1166 void emit(const Cfg *Func) const override; | 1136 void emit(const Cfg *Func) const override; |
1167 void emitIAS(const Cfg *Func) const override; | 1137 void emitIAS(const Cfg *Func) const override; |
1168 void dump(const Cfg *Func) const override; | 1138 void dump(const Cfg *Func) const override; |
1169 static bool classof(const Inst *Inst) { return isClassof(Inst, Vmov); } | 1139 static bool classof(const Inst *Inst) { return isClassof(Inst, Mov); } |
1170 | |
1171 private: | |
1172 InstARM32Vmov(Cfg *Func, Variable *Dest, Operand *Src, | |
1173 CondARM32::Cond Predicate) | |
1174 : InstARM32Pred(Func, InstARM32::Vmov, 1, Dest, Predicate) { | |
1175 addSource(Src); | |
1176 } | |
1177 | |
1178 InstARM32Vmov(Cfg *Func, const RegisterPair &Dests, Variable *Src, | |
1179 CondARM32::Cond Predicate) | |
1180 : InstARM32Pred(Func, InstARM32::Vmov, 1, Dests._0, Predicate), | |
1181 Dest1(Dests._1) { | |
1182 addSource(Src); | |
1183 } | |
1184 | |
1185 InstARM32Vmov(Cfg *Func, Variable *Dest, const RegisterPair &Srcs, | |
1186 CondARM32::Cond Predicate) | |
1187 : InstARM32Pred(Func, InstARM32::Vmov, 2, Dest, Predicate) { | |
1188 addSource(Srcs._0); | |
1189 addSource(Srcs._1); | |
1190 } | |
1191 | 1140 |
1192 bool isMultiDest() const { | 1141 bool isMultiDest() const { |
1193 assert(getDest() != nullptr); | 1142 assert(getDest() != nullptr); |
1194 return Dest1 != nullptr; | 1143 return llvm::isa<Variable64On32>(getDest()); |
1195 } | 1144 } |
1196 | 1145 |
1197 bool isMultiSource() const { | 1146 bool isMultiSource() const { |
1198 assert(getSrcSize() >= 1); | 1147 assert(getSrcSize() == 1); |
1199 return getSrcSize() > 1; | 1148 return llvm::isa<Variable64On32>(getSrc(0)); |
1149 } | |
1150 | |
1151 private: | |
1152 InstARM32Mov(Cfg *Func, Variable *Dest, Operand *Src, | |
1153 CondARM32::Cond Predicate) | |
1154 : InstARM32Pred(Func, InstARM32::Mov, 1, Dest, Predicate) { | |
1155 addSource(Src); | |
1200 } | 1156 } |
1201 | 1157 |
1202 void emitMultiDestSingleSource(const Cfg *Func) const; | 1158 void emitMultiDestSingleSource(const Cfg *Func) const; |
1203 void emitSingleDestMultiSource(const Cfg *Func) const; | 1159 void emitSingleDestMultiSource(const Cfg *Func) const; |
1204 void emitSingleDestSingleSource(const Cfg *Func) const; | 1160 void emitSingleDestSingleSource(const Cfg *Func) const; |
1205 | |
1206 Variable *Dest1 = nullptr; | |
1207 }; | 1161 }; |
1208 | 1162 |
1209 class InstARM32Vcmp final : public InstARM32Pred { | 1163 class InstARM32Vcmp final : public InstARM32Pred { |
1210 InstARM32Vcmp() = delete; | 1164 InstARM32Vcmp() = delete; |
1211 InstARM32Vcmp(const InstARM32Vcmp &) = delete; | 1165 InstARM32Vcmp(const InstARM32Vcmp &) = delete; |
1212 InstARM32Vcmp &operator=(const InstARM32Vcmp &) = delete; | 1166 InstARM32Vcmp &operator=(const InstARM32Vcmp &) = delete; |
1213 | 1167 |
1214 public: | 1168 public: |
1215 static InstARM32Vcmp *create(Cfg *Func, Variable *Src0, Variable *Src1, | 1169 static InstARM32Vcmp *create(Cfg *Func, Variable *Src0, Variable *Src1, |
1216 CondARM32::Cond Predicate) { | 1170 CondARM32::Cond Predicate) { |
(...skipping 22 matching lines...) Expand all Loading... | |
1239 } | 1193 } |
1240 void emit(const Cfg *Func) const override; | 1194 void emit(const Cfg *Func) const override; |
1241 void emitIAS(const Cfg *Func) const override; | 1195 void emitIAS(const Cfg *Func) const override; |
1242 void dump(const Cfg *Func) const override; | 1196 void dump(const Cfg *Func) const override; |
1243 static bool classof(const Inst *Inst) { return isClassof(Inst, Vmrs); } | 1197 static bool classof(const Inst *Inst) { return isClassof(Inst, Vmrs); } |
1244 | 1198 |
1245 private: | 1199 private: |
1246 InstARM32Vmrs(Cfg *Func, CondARM32::Cond Predicate); | 1200 InstARM32Vmrs(Cfg *Func, CondARM32::Cond Predicate); |
1247 }; | 1201 }; |
1248 | 1202 |
1203 class InstARM32Vabs final : public InstARM32Pred { | |
1204 InstARM32Vabs() = delete; | |
1205 InstARM32Vabs(const InstARM32Vabs &) = delete; | |
1206 InstARM32Vabs &operator=(const InstARM32Vabs &) = delete; | |
1207 | |
1208 public: | |
1209 static InstARM32Vabs *create(Cfg *Func, Variable *Dest, Variable *Src, | |
1210 CondARM32::Cond Predicate) { | |
1211 return new (Func->allocate<InstARM32Vabs>()) | |
1212 InstARM32Vabs(Func, Dest, Src, Predicate); | |
1213 } | |
1214 void emit(const Cfg *Func) const override; | |
1215 void emitIAS(const Cfg *Func) const override; | |
1216 void dump(const Cfg *Func) const override; | |
1217 static bool classof(const Inst *Inst) { return isClassof(Inst, Vabs); } | |
1218 | |
1219 private: | |
1220 InstARM32Vabs(Cfg *Func, Variable *Dest, Variable *Src, | |
1221 CondARM32::Cond Predicate); | |
1222 }; | |
1249 // Declare partial template specializations of emit() methods that already have | 1223 // Declare partial template specializations of emit() methods that already have |
1250 // default implementations. Without this, there is the possibility of ODR | 1224 // default implementations. Without this, there is the possibility of ODR |
1251 // violations and link errors. | 1225 // violations and link errors. |
1252 | 1226 |
1253 template <> void InstARM32Ldr::emit(const Cfg *Func) const; | 1227 template <> void InstARM32Ldr::emit(const Cfg *Func) const; |
1254 template <> void InstARM32Mov::emit(const Cfg *Func) const; | |
1255 template <> void InstARM32Movw::emit(const Cfg *Func) const; | 1228 template <> void InstARM32Movw::emit(const Cfg *Func) const; |
1256 template <> void InstARM32Movt::emit(const Cfg *Func) const; | 1229 template <> void InstARM32Movt::emit(const Cfg *Func) const; |
1257 template <> void InstARM32Vldr::emit(const Cfg *Func) const; | |
1258 | 1230 |
1259 } // end of namespace Ice | 1231 } // end of namespace Ice |
1260 | 1232 |
1261 #endif // SUBZERO_SRC_ICEINSTARM32_H | 1233 #endif // SUBZERO_SRC_ICEINSTARM32_H |
OLD | NEW |