Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(389)

Side by Side Diff: src/IceInstARM32.h

Issue 1151663004: Subzero ARM: do lowerIcmp, lowerBr, and a bit of lowerCall. (Closed) Base URL: https://chromium.googlesource.com/native_client/pnacl-subzero.git@master
Patch Set: clang-format Created 5 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
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 // This file declares the InstARM32 and OperandARM32 classes and 10 // This file declares the InstARM32 and OperandARM32 classes and
11 // their subclasses. This represents the machine instructions and 11 // their subclasses. This represents the machine instructions and
12 // operands used for ARM32 code selection. 12 // operands used for ARM32 code selection.
13 // 13 //
14 //===----------------------------------------------------------------------===// 14 //===----------------------------------------------------------------------===//
15 15
16 #ifndef SUBZERO_SRC_ICEINSTARM32_H 16 #ifndef SUBZERO_SRC_ICEINSTARM32_H
17 #define SUBZERO_SRC_ICEINSTARM32_H 17 #define SUBZERO_SRC_ICEINSTARM32_H
18 18
19 #include "IceConditionCodesARM32.h"
19 #include "IceDefs.h" 20 #include "IceDefs.h"
20 #include "IceInst.h" 21 #include "IceInst.h"
21 #include "IceInstARM32.def" 22 #include "IceInstARM32.def"
22 #include "IceOperand.h" 23 #include "IceOperand.h"
23 24
24 namespace Ice { 25 namespace Ice {
25 26
26 class TargetARM32; 27 class TargetARM32;
27 28
28 // OperandARM32 extends the Operand hierarchy. Its subclasses are 29 // OperandARM32 extends the Operand hierarchy. Its subclasses are
(...skipping 201 matching lines...) Expand 10 before | Expand all | Expand 10 after
230 private: 231 private:
231 OperandARM32FlexReg(Cfg *Func, Type Ty, Variable *Reg, ShiftKind ShiftOp, 232 OperandARM32FlexReg(Cfg *Func, Type Ty, Variable *Reg, ShiftKind ShiftOp,
232 Operand *ShiftAmt); 233 Operand *ShiftAmt);
233 ~OperandARM32FlexReg() override {} 234 ~OperandARM32FlexReg() override {}
234 235
235 Variable *Reg; 236 Variable *Reg;
236 ShiftKind ShiftOp; 237 ShiftKind ShiftOp;
237 Operand *ShiftAmt; 238 Operand *ShiftAmt;
238 }; 239 };
239 240
241 // Base class for ARM instructions. While most ARM instructions can be
242 // conditionally executed, a few of them are not predicable (halt,
243 // memory barriers, etc.).
240 class InstARM32 : public InstTarget { 244 class InstARM32 : public InstTarget {
241 InstARM32() = delete; 245 InstARM32() = delete;
242 InstARM32(const InstARM32 &) = delete; 246 InstARM32(const InstARM32 &) = delete;
243 InstARM32 &operator=(const InstARM32 &) = delete; 247 InstARM32 &operator=(const InstARM32 &) = delete;
244 248
245 public: 249 public:
246 enum InstKindARM32 { 250 enum InstKindARM32 {
247 k__Start = Inst::Target, 251 k__Start = Inst::Target,
248 Adc, 252 Adc,
249 Add, 253 Add,
250 And, 254 And,
255 Br,
256 Call,
257 Cmp,
251 Eor, 258 Eor,
252 Ldr, 259 Ldr,
260 Lsl,
253 Mla, 261 Mla,
254 Mov, 262 Mov,
255 Movt, 263 Movt,
256 Movw, 264 Movw,
257 Mul, 265 Mul,
258 Mvn, 266 Mvn,
259 Orr, 267 Orr,
260 Ret, 268 Ret,
261 Sbc, 269 Sbc,
262 Sub, 270 Sub,
263 Umull 271 Umull
264 }; 272 };
265 273
266 static const char *getWidthString(Type Ty); 274 static const char *getWidthString(Type Ty);
275 static CondARM32::Cond getOppositeCondition(CondARM32::Cond Cond);
267 276
268 void dump(const Cfg *Func) const override; 277 void dump(const Cfg *Func) const override;
269 278
270 protected: 279 protected:
271 InstARM32(Cfg *Func, InstKindARM32 Kind, SizeT Maxsrcs, Variable *Dest) 280 InstARM32(Cfg *Func, InstKindARM32 Kind, SizeT Maxsrcs, Variable *Dest)
272 : InstTarget(Func, static_cast<InstKind>(Kind), Maxsrcs, Dest) {} 281 : InstTarget(Func, static_cast<InstKind>(Kind), Maxsrcs, Dest) {}
273 ~InstARM32() override {} 282 ~InstARM32() override {}
274 static bool isClassof(const Inst *Inst, InstKindARM32 MyKind) { 283 static bool isClassof(const Inst *Inst, InstKindARM32 MyKind) {
275 return Inst->getKind() == static_cast<InstKind>(MyKind); 284 return Inst->getKind() == static_cast<InstKind>(MyKind);
276 } 285 }
277 }; 286 };
278 287
279 void emitTwoAddr(const char *Opcode, const Inst *Inst, const Cfg *Func); 288 // A predicable ARM instruction.
280 void emitThreeAddr(const char *Opcode, const Inst *Inst, const Cfg *Func, 289 class InstARM32Pred : public InstARM32 {
281 bool SetFlags); 290 InstARM32Pred() = delete;
291 InstARM32Pred(const InstARM32Pred &) = delete;
292 InstARM32Pred &operator=(const InstARM32Pred &) = delete;
282 293
283 // TODO(jvoung): add condition codes if instruction can be predicated. 294 public:
295 InstARM32Pred(Cfg *Func, InstKindARM32 Kind, SizeT Maxsrcs, Variable *Dest,
296 CondARM32::Cond Predicate)
297 : InstARM32(Func, Kind, Maxsrcs, Dest), Predicate(Predicate) {}
298
299 CondARM32::Cond getPredicate() const { return Predicate; }
300 void setPredicate(CondARM32::Cond Pred) { Predicate = Pred; }
301
302 static const char *predString(CondARM32::Cond Predicate);
303 void dumpOpcodePred(Ostream &Str, const char *Opcode, Type Ty) const;
304
305 protected:
306 CondARM32::Cond Predicate;
307 };
308
309 template <typename StreamType>
310 inline StreamType &operator<<(StreamType &Stream, CondARM32::Cond Predicate) {
311 Stream << InstARM32Pred::predString(Predicate);
312 return Stream;
313 }
314
315 void emitTwoAddr(const char *Opcode, const InstARM32Pred *Inst,
316 const Cfg *Func);
317 void emitThreeAddr(const char *Opcode, const InstARM32Pred *Inst,
318 const Cfg *Func, bool SetFlags);
284 319
285 // Instructions of the form x := op(y). 320 // Instructions of the form x := op(y).
286 template <InstARM32::InstKindARM32 K> 321 template <InstARM32::InstKindARM32 K>
287 class InstARM32UnaryopGPR : public InstARM32 { 322 class InstARM32UnaryopGPR : public InstARM32Pred {
288 InstARM32UnaryopGPR() = delete; 323 InstARM32UnaryopGPR() = delete;
289 InstARM32UnaryopGPR(const InstARM32UnaryopGPR &) = delete; 324 InstARM32UnaryopGPR(const InstARM32UnaryopGPR &) = delete;
290 InstARM32UnaryopGPR &operator=(const InstARM32UnaryopGPR &) = delete; 325 InstARM32UnaryopGPR &operator=(const InstARM32UnaryopGPR &) = delete;
291 326
292 public: 327 public:
293 static InstARM32UnaryopGPR *create(Cfg *Func, Variable *Dest, Operand *Src) { 328 static InstARM32UnaryopGPR *create(Cfg *Func, Variable *Dest, Operand *Src,
329 CondARM32::Cond Predicate) {
294 return new (Func->allocate<InstARM32UnaryopGPR>()) 330 return new (Func->allocate<InstARM32UnaryopGPR>())
295 InstARM32UnaryopGPR(Func, Dest, Src); 331 InstARM32UnaryopGPR(Func, Dest, Src, Predicate);
296 } 332 }
297 void emit(const Cfg *Func) const override { 333 void emit(const Cfg *Func) const override {
298 if (!ALLOW_DUMP) 334 if (!ALLOW_DUMP)
299 return; 335 return;
300 Ostream &Str = Func->getContext()->getStrEmit(); 336 Ostream &Str = Func->getContext()->getStrEmit();
301 assert(getSrcSize() == 1); 337 assert(getSrcSize() == 1);
302 Str << "\t" << Opcode << "\t"; 338 Str << "\t" << Opcode << "\t";
303 getDest()->emit(Func); 339 getDest()->emit(Func);
304 Str << ", "; 340 Str << ", ";
305 getSrc(0)->emit(Func); 341 getSrc(0)->emit(Func);
306 } 342 }
307 void emitIAS(const Cfg *Func) const override { 343 void emitIAS(const Cfg *Func) const override {
308 (void)Func; 344 (void)Func;
309 llvm_unreachable("Not yet implemented"); 345 llvm_unreachable("Not yet implemented");
310 } 346 }
311 void dump(const Cfg *Func) const override { 347 void dump(const Cfg *Func) const override {
312 if (!ALLOW_DUMP) 348 if (!ALLOW_DUMP)
313 return; 349 return;
314 Ostream &Str = Func->getContext()->getStrDump(); 350 Ostream &Str = Func->getContext()->getStrDump();
315 dumpDest(Func); 351 dumpDest(Func);
316 Str << " = " << Opcode << "." << getDest()->getType() << " "; 352 Str << " = ";
353 dumpOpcodePred(Str, Opcode, getDest()->getType());
354 Str << " ";
317 dumpSources(Func); 355 dumpSources(Func);
318 } 356 }
319 static bool classof(const Inst *Inst) { return isClassof(Inst, K); } 357 static bool classof(const Inst *Inst) { return isClassof(Inst, K); }
320 358
321 private: 359 private:
322 InstARM32UnaryopGPR(Cfg *Func, Variable *Dest, Operand *Src) 360 InstARM32UnaryopGPR(Cfg *Func, Variable *Dest, Operand *Src,
323 : InstARM32(Func, K, 1, Dest) { 361 CondARM32::Cond Predicate)
362 : InstARM32Pred(Func, K, 1, Dest, Predicate) {
324 addSource(Src); 363 addSource(Src);
325 } 364 }
326 ~InstARM32UnaryopGPR() override {} 365 ~InstARM32UnaryopGPR() override {}
327 static const char *Opcode; 366 static const char *Opcode;
328 }; 367 };
329 368
330 // Instructions of the form x := x op y. 369 // Instructions of the form x := x op y.
331 template <InstARM32::InstKindARM32 K> 370 template <InstARM32::InstKindARM32 K>
332 class InstARM32TwoAddrGPR : public InstARM32 { 371 class InstARM32TwoAddrGPR : public InstARM32Pred {
333 InstARM32TwoAddrGPR() = delete; 372 InstARM32TwoAddrGPR() = delete;
334 InstARM32TwoAddrGPR(const InstARM32TwoAddrGPR &) = delete; 373 InstARM32TwoAddrGPR(const InstARM32TwoAddrGPR &) = delete;
335 InstARM32TwoAddrGPR &operator=(const InstARM32TwoAddrGPR &) = delete; 374 InstARM32TwoAddrGPR &operator=(const InstARM32TwoAddrGPR &) = delete;
336 375
337 public: 376 public:
338 // Dest must be a register. 377 // Dest must be a register.
339 static InstARM32TwoAddrGPR *create(Cfg *Func, Variable *Dest, Operand *Src) { 378 static InstARM32TwoAddrGPR *create(Cfg *Func, Variable *Dest, Operand *Src,
379 CondARM32::Cond Predicate) {
340 return new (Func->allocate<InstARM32TwoAddrGPR>()) 380 return new (Func->allocate<InstARM32TwoAddrGPR>())
341 InstARM32TwoAddrGPR(Func, Dest, Src); 381 InstARM32TwoAddrGPR(Func, Dest, Src, Predicate);
342 } 382 }
343 void emit(const Cfg *Func) const override { 383 void emit(const Cfg *Func) const override {
344 if (!ALLOW_DUMP) 384 if (!ALLOW_DUMP)
345 return; 385 return;
346 emitTwoAddr(Opcode, this, Func); 386 emitTwoAddr(Opcode, this, Func);
347 } 387 }
348 void emitIAS(const Cfg *Func) const override { 388 void emitIAS(const Cfg *Func) const override {
349 (void)Func; 389 (void)Func;
350 llvm::report_fatal_error("Not yet implemented"); 390 llvm::report_fatal_error("Not yet implemented");
351 } 391 }
352 void dump(const Cfg *Func) const override { 392 void dump(const Cfg *Func) const override {
353 if (!ALLOW_DUMP) 393 if (!ALLOW_DUMP)
354 return; 394 return;
355 Ostream &Str = Func->getContext()->getStrDump(); 395 Ostream &Str = Func->getContext()->getStrDump();
356 dumpDest(Func); 396 dumpDest(Func);
357 Str << " = " << Opcode << "." << getDest()->getType() << " "; 397 Str << " = ";
398 dumpOpcodePred(Str, Opcode, getDest()->getType());
399 Str << " ";
358 dumpSources(Func); 400 dumpSources(Func);
359 } 401 }
360 static bool classof(const Inst *Inst) { return isClassof(Inst, K); } 402 static bool classof(const Inst *Inst) { return isClassof(Inst, K); }
361 403
362 private: 404 private:
363 InstARM32TwoAddrGPR(Cfg *Func, Variable *Dest, Operand *Src) 405 InstARM32TwoAddrGPR(Cfg *Func, Variable *Dest, Operand *Src,
364 : InstARM32(Func, K, 2, Dest) { 406 CondARM32::Cond Predicate)
407 : InstARM32Pred(Func, K, 2, Dest, Predicate) {
365 addSource(Dest); 408 addSource(Dest);
366 addSource(Src); 409 addSource(Src);
367 } 410 }
368 ~InstARM32TwoAddrGPR() override {} 411 ~InstARM32TwoAddrGPR() override {}
369 static const char *Opcode; 412 static const char *Opcode;
370 }; 413 };
371 414
372 // Base class for assignment instructions. 415 // Base class for assignment instructions.
373 // These can be tested for redundancy (and elided if redundant). 416 // These can be tested for redundancy (and elided if redundant).
374 template <InstARM32::InstKindARM32 K> 417 template <InstARM32::InstKindARM32 K>
375 class InstARM32Movlike : public InstARM32 { 418 class InstARM32Movlike : public InstARM32Pred {
376 InstARM32Movlike() = delete; 419 InstARM32Movlike() = delete;
377 InstARM32Movlike(const InstARM32Movlike &) = delete; 420 InstARM32Movlike(const InstARM32Movlike &) = delete;
378 InstARM32Movlike &operator=(const InstARM32Movlike &) = delete; 421 InstARM32Movlike &operator=(const InstARM32Movlike &) = delete;
379 422
380 public: 423 public:
381 static InstARM32Movlike *create(Cfg *Func, Variable *Dest, Operand *Source) { 424 static InstARM32Movlike *create(Cfg *Func, Variable *Dest, Operand *Source,
425 CondARM32::Cond Predicate) {
382 return new (Func->allocate<InstARM32Movlike>()) 426 return new (Func->allocate<InstARM32Movlike>())
383 InstARM32Movlike(Func, Dest, Source); 427 InstARM32Movlike(Func, Dest, Source, Predicate);
384 } 428 }
385 bool isRedundantAssign() const override { 429 bool isRedundantAssign() const override {
386 return checkForRedundantAssign(getDest(), getSrc(0)); 430 return checkForRedundantAssign(getDest(), getSrc(0));
387 } 431 }
388 bool isSimpleAssign() const override { return true; } 432 bool isSimpleAssign() const override { return true; }
389 void emit(const Cfg *Func) const override; 433 void emit(const Cfg *Func) const override;
390 void emitIAS(const Cfg *Func) const override; 434 void emitIAS(const Cfg *Func) const override;
391 void dump(const Cfg *Func) const override { 435 void dump(const Cfg *Func) const override {
392 if (!ALLOW_DUMP) 436 if (!ALLOW_DUMP)
393 return; 437 return;
394 Ostream &Str = Func->getContext()->getStrDump(); 438 Ostream &Str = Func->getContext()->getStrDump();
395 Str << Opcode << "." << getDest()->getType() << " "; 439 dumpOpcodePred(Str, Opcode, getDest()->getType());
440 Str << " ";
396 dumpDest(Func); 441 dumpDest(Func);
397 Str << ", "; 442 Str << ", ";
398 dumpSources(Func); 443 dumpSources(Func);
399 } 444 }
400 static bool classof(const Inst *Inst) { return isClassof(Inst, K); } 445 static bool classof(const Inst *Inst) { return isClassof(Inst, K); }
401 446
402 private: 447 private:
403 InstARM32Movlike(Cfg *Func, Variable *Dest, Operand *Source) 448 InstARM32Movlike(Cfg *Func, Variable *Dest, Operand *Source,
404 : InstARM32(Func, K, 1, Dest) { 449 CondARM32::Cond Predicate)
450 : InstARM32Pred(Func, K, 1, Dest, Predicate) {
405 addSource(Source); 451 addSource(Source);
406 } 452 }
407 ~InstARM32Movlike() override {} 453 ~InstARM32Movlike() override {}
408 454
409 static const char *Opcode; 455 static const char *Opcode;
410 }; 456 };
411 457
412 // Instructions of the form x := y op z. May have the side-effect of setting 458 // Instructions of the form x := y op z. May have the side-effect of setting
413 // status flags. 459 // status flags.
414 template <InstARM32::InstKindARM32 K> 460 template <InstARM32::InstKindARM32 K>
415 class InstARM32ThreeAddrGPR : public InstARM32 { 461 class InstARM32ThreeAddrGPR : public InstARM32Pred {
416 InstARM32ThreeAddrGPR() = delete; 462 InstARM32ThreeAddrGPR() = delete;
417 InstARM32ThreeAddrGPR(const InstARM32ThreeAddrGPR &) = delete; 463 InstARM32ThreeAddrGPR(const InstARM32ThreeAddrGPR &) = delete;
418 InstARM32ThreeAddrGPR &operator=(const InstARM32ThreeAddrGPR &) = delete; 464 InstARM32ThreeAddrGPR &operator=(const InstARM32ThreeAddrGPR &) = delete;
419 465
420 public: 466 public:
421 // Create an ordinary binary-op instruction like add, and sub. 467 // Create an ordinary binary-op instruction like add, and sub.
422 // Dest and Src1 must be registers. 468 // Dest and Src1 must be registers.
423 static InstARM32ThreeAddrGPR *create(Cfg *Func, Variable *Dest, 469 static InstARM32ThreeAddrGPR *create(Cfg *Func, Variable *Dest,
424 Variable *Src1, Operand *Src2, 470 Variable *Src1, Operand *Src2,
471 CondARM32::Cond Predicate,
425 bool SetFlags = false) { 472 bool SetFlags = false) {
426 return new (Func->allocate<InstARM32ThreeAddrGPR>()) 473 return new (Func->allocate<InstARM32ThreeAddrGPR>())
427 InstARM32ThreeAddrGPR(Func, Dest, Src1, Src2, SetFlags); 474 InstARM32ThreeAddrGPR(Func, Dest, Src1, Src2, Predicate, SetFlags);
428 } 475 }
429 void emit(const Cfg *Func) const override { 476 void emit(const Cfg *Func) const override {
430 if (!ALLOW_DUMP) 477 if (!ALLOW_DUMP)
431 return; 478 return;
432 emitThreeAddr(Opcode, this, Func, SetFlags); 479 emitThreeAddr(Opcode, this, Func, SetFlags);
433 } 480 }
434 void emitIAS(const Cfg *Func) const override { 481 void emitIAS(const Cfg *Func) const override {
435 (void)Func; 482 (void)Func;
436 llvm::report_fatal_error("Not yet implemented"); 483 llvm::report_fatal_error("Not yet implemented");
437 } 484 }
438 void dump(const Cfg *Func) const override { 485 void dump(const Cfg *Func) const override {
439 if (!ALLOW_DUMP) 486 if (!ALLOW_DUMP)
440 return; 487 return;
441 Ostream &Str = Func->getContext()->getStrDump(); 488 Ostream &Str = Func->getContext()->getStrDump();
442 dumpDest(Func); 489 dumpDest(Func);
443 Str << " = " << Opcode << (SetFlags ? "s" : "") << "." 490 Str << " = ";
444 << getDest()->getType() << " "; 491 dumpOpcodePred(Str, Opcode, getDest()->getType());
492 Str << (SetFlags ? ".s " : " ");
445 dumpSources(Func); 493 dumpSources(Func);
446 } 494 }
447 static bool classof(const Inst *Inst) { return isClassof(Inst, K); } 495 static bool classof(const Inst *Inst) { return isClassof(Inst, K); }
448 496
449 private: 497 private:
450 InstARM32ThreeAddrGPR(Cfg *Func, Variable *Dest, Variable *Src1, 498 InstARM32ThreeAddrGPR(Cfg *Func, Variable *Dest, Variable *Src1,
451 Operand *Src2, bool SetFlags) 499 Operand *Src2, CondARM32::Cond Predicate, bool SetFlags)
452 : InstARM32(Func, K, 2, Dest), SetFlags(SetFlags) { 500 : InstARM32Pred(Func, K, 2, Dest, Predicate), SetFlags(SetFlags) {
453 addSource(Src1); 501 addSource(Src1);
454 addSource(Src2); 502 addSource(Src2);
455 } 503 }
456 ~InstARM32ThreeAddrGPR() override {} 504 ~InstARM32ThreeAddrGPR() override {}
457 static const char *Opcode; 505 static const char *Opcode;
458 bool SetFlags; 506 bool SetFlags;
459 }; 507 };
460 508
461 typedef InstARM32ThreeAddrGPR<InstARM32::Adc> InstARM32Adc; 509 typedef InstARM32ThreeAddrGPR<InstARM32::Adc> InstARM32Adc;
462 typedef InstARM32ThreeAddrGPR<InstARM32::Add> InstARM32Add; 510 typedef InstARM32ThreeAddrGPR<InstARM32::Add> InstARM32Add;
463 typedef InstARM32ThreeAddrGPR<InstARM32::And> InstARM32And; 511 typedef InstARM32ThreeAddrGPR<InstARM32::And> InstARM32And;
464 typedef InstARM32ThreeAddrGPR<InstARM32::Eor> InstARM32Eor; 512 typedef InstARM32ThreeAddrGPR<InstARM32::Eor> InstARM32Eor;
513 typedef InstARM32ThreeAddrGPR<InstARM32::Lsl> InstARM32Lsl;
465 typedef InstARM32ThreeAddrGPR<InstARM32::Mul> InstARM32Mul; 514 typedef InstARM32ThreeAddrGPR<InstARM32::Mul> InstARM32Mul;
466 typedef InstARM32ThreeAddrGPR<InstARM32::Orr> InstARM32Orr; 515 typedef InstARM32ThreeAddrGPR<InstARM32::Orr> InstARM32Orr;
467 typedef InstARM32ThreeAddrGPR<InstARM32::Sbc> InstARM32Sbc; 516 typedef InstARM32ThreeAddrGPR<InstARM32::Sbc> InstARM32Sbc;
468 typedef InstARM32ThreeAddrGPR<InstARM32::Sub> InstARM32Sub; 517 typedef InstARM32ThreeAddrGPR<InstARM32::Sub> InstARM32Sub;
469 // Move instruction (variable <- flex). This is more of a pseudo-inst. 518 // Move instruction (variable <- flex). This is more of a pseudo-inst.
470 // If var is a register, then we use "mov". If var is stack, then we use 519 // If var is a register, then we use "mov". If var is stack, then we use
471 // "str" to store to the stack. 520 // "str" to store to the stack.
472 typedef InstARM32Movlike<InstARM32::Mov> InstARM32Mov; 521 typedef InstARM32Movlike<InstARM32::Mov> InstARM32Mov;
473 // MovT leaves the bottom bits alone so dest is also a source. 522 // MovT leaves the bottom bits alone so dest is also a source.
474 // This helps indicate that a previous MovW setting dest is not dead code. 523 // This helps indicate that a previous MovW setting dest is not dead code.
475 typedef InstARM32TwoAddrGPR<InstARM32::Movt> InstARM32Movt; 524 typedef InstARM32TwoAddrGPR<InstARM32::Movt> InstARM32Movt;
476 typedef InstARM32UnaryopGPR<InstARM32::Movw> InstARM32Movw; 525 typedef InstARM32UnaryopGPR<InstARM32::Movw> InstARM32Movw;
477 typedef InstARM32UnaryopGPR<InstARM32::Mvn> InstARM32Mvn; 526 typedef InstARM32UnaryopGPR<InstARM32::Mvn> InstARM32Mvn;
478 527
528 // Direct branch instruction.
529 class InstARM32Br : public InstARM32Pred {
530 InstARM32Br() = delete;
531 InstARM32Br(const InstARM32Br &) = delete;
532 InstARM32Br &operator=(const InstARM32Br &) = delete;
533
534 public:
535 // Create a conditional branch to one of two nodes.
536 static InstARM32Br *create(Cfg *Func, CfgNode *TargetTrue,
537 CfgNode *TargetFalse, CondARM32::Cond Predicate) {
538 assert(Predicate != CondARM32::AL);
539 return new (Func->allocate<InstARM32Br>())
540 InstARM32Br(Func, TargetTrue, TargetFalse, Predicate);
541 }
542 // Create an unconditional branch to a node.
543 static InstARM32Br *create(Cfg *Func, CfgNode *Target) {
544 const CfgNode *NoCondTarget = nullptr;
545 return new (Func->allocate<InstARM32Br>())
546 InstARM32Br(Func, NoCondTarget, Target, CondARM32::AL);
547 }
548 // Create a non-terminator conditional branch to a node, with a
549 // fallthrough to the next instruction in the current node. This is
550 // used for switch lowering.
jvoung (off chromium) 2015/05/21 22:52:03 Almost a clone of the x86-32 version, but took out
551 static InstARM32Br *create(Cfg *Func, CfgNode *Target,
552 CondARM32::Cond Predicate) {
553 assert(Predicate != CondARM32::AL);
554 const CfgNode *NoUncondTarget = nullptr;
555 return new (Func->allocate<InstARM32Br>())
556 InstARM32Br(Func, Target, NoUncondTarget, Predicate);
557 }
558 const CfgNode *getTargetTrue() const { return TargetTrue; }
559 const CfgNode *getTargetFalse() const { return TargetFalse; }
560 bool optimizeBranch(const CfgNode *NextNode);
561 uint32_t getEmitInstCount() const override {
562 uint32_t Sum = 0;
563 if (getTargetTrue())
564 ++Sum;
565 if (getTargetFalse())
566 ++Sum;
567 return Sum;
568 }
569 bool isUnconditionalBranch() const override {
570 return getPredicate() == CondARM32::AL;
571 }
572 bool repointEdge(CfgNode *OldNode, CfgNode *NewNode) override;
573 void emit(const Cfg *Func) const override;
574 void emitIAS(const Cfg *Func) const override;
575 void dump(const Cfg *Func) const override;
576 static bool classof(const Inst *Inst) { return isClassof(Inst, Br); }
577
578 private:
579 InstARM32Br(Cfg *Func, const CfgNode *TargetTrue, const CfgNode *TargetFalse,
580 CondARM32::Cond Predicate);
581 ~InstARM32Br() override {}
582 const CfgNode *TargetTrue;
583 const CfgNode *TargetFalse;
584 };
585
586 // Call instruction (bl/blx). Arguments should have already been pushed.
587 // Technically bl and the register form of blx can be predicated, but we'll
588 // leave that out until needed.
589 class InstARM32Call : public InstARM32 {
590 InstARM32Call() = delete;
591 InstARM32Call(const InstARM32Call &) = delete;
592 InstARM32Call &operator=(const InstARM32Call &) = delete;
593
594 public:
595 static InstARM32Call *create(Cfg *Func, Variable *Dest, Operand *CallTarget) {
596 return new (Func->allocate<InstARM32Call>())
597 InstARM32Call(Func, Dest, CallTarget);
598 }
599 Operand *getCallTarget() const { return getSrc(0); }
600 void emit(const Cfg *Func) const override;
601 void emitIAS(const Cfg *Func) const override;
602 void dump(const Cfg *Func) const override;
603 static bool classof(const Inst *Inst) { return isClassof(Inst, Call); }
604
605 private:
606 InstARM32Call(Cfg *Func, Variable *Dest, Operand *CallTarget);
607 ~InstARM32Call() override {}
608 };
609
610 // Integer compare instruction.
611 class InstARM32Cmp : public InstARM32Pred {
612 InstARM32Cmp() = delete;
613 InstARM32Cmp(const InstARM32Cmp &) = delete;
614 InstARM32Cmp &operator=(const InstARM32Cmp &) = delete;
615
616 public:
617 static InstARM32Cmp *create(Cfg *Func, Variable *Src1, Operand *Src2,
618 CondARM32::Cond Predicate) {
619 return new (Func->allocate<InstARM32Cmp>())
620 InstARM32Cmp(Func, Src1, Src2, Predicate);
621 }
622 void emit(const Cfg *Func) const override;
623 void emitIAS(const Cfg *Func) const override;
624 void dump(const Cfg *Func) const override;
625 static bool classof(const Inst *Inst) { return isClassof(Inst, Cmp); }
626
627 private:
628 InstARM32Cmp(Cfg *Func, Variable *Src1, Operand *Src2,
629 CondARM32::Cond Predicate);
630 ~InstARM32Cmp() override {}
631 };
632
479 // Load instruction. 633 // Load instruction.
480 class InstARM32Ldr : public InstARM32 { 634 class InstARM32Ldr : public InstARM32Pred {
481 InstARM32Ldr() = delete; 635 InstARM32Ldr() = delete;
482 InstARM32Ldr(const InstARM32Ldr &) = delete; 636 InstARM32Ldr(const InstARM32Ldr &) = delete;
483 InstARM32Ldr &operator=(const InstARM32Ldr &) = delete; 637 InstARM32Ldr &operator=(const InstARM32Ldr &) = delete;
484 638
485 public: 639 public:
486 // Dest must be a register. 640 // Dest must be a register.
487 static InstARM32Ldr *create(Cfg *Func, Variable *Dest, OperandARM32Mem *Mem) { 641 static InstARM32Ldr *create(Cfg *Func, Variable *Dest, OperandARM32Mem *Mem,
488 return new (Func->allocate<InstARM32Ldr>()) InstARM32Ldr(Func, Dest, Mem); 642 CondARM32::Cond Predicate) {
643 return new (Func->allocate<InstARM32Ldr>())
644 InstARM32Ldr(Func, Dest, Mem, Predicate);
489 } 645 }
490 void emit(const Cfg *Func) const override; 646 void emit(const Cfg *Func) const override;
491 void emitIAS(const Cfg *Func) const override; 647 void emitIAS(const Cfg *Func) const override;
492 void dump(const Cfg *Func) const override; 648 void dump(const Cfg *Func) const override;
493 static bool classof(const Inst *Inst) { return isClassof(Inst, Ldr); } 649 static bool classof(const Inst *Inst) { return isClassof(Inst, Ldr); }
494 650
495 private: 651 private:
496 InstARM32Ldr(Cfg *Func, Variable *Dest, OperandARM32Mem *Mem); 652 InstARM32Ldr(Cfg *Func, Variable *Dest, OperandARM32Mem *Mem,
653 CondARM32::Cond Predicate);
497 ~InstARM32Ldr() override {} 654 ~InstARM32Ldr() override {}
498 }; 655 };
499 656
500 // Multiply Accumulate: d := x * y + a 657 // Multiply Accumulate: d := x * y + a
501 class InstARM32Mla : public InstARM32 { 658 class InstARM32Mla : public InstARM32Pred {
502 InstARM32Mla() = delete; 659 InstARM32Mla() = delete;
503 InstARM32Mla(const InstARM32Mla &) = delete; 660 InstARM32Mla(const InstARM32Mla &) = delete;
504 InstARM32Mla &operator=(const InstARM32Mla &) = delete; 661 InstARM32Mla &operator=(const InstARM32Mla &) = delete;
505 662
506 public: 663 public:
507 // Everything must be a register. 664 // Everything must be a register.
508 static InstARM32Mla *create(Cfg *Func, Variable *Dest, Variable *Src0, 665 static InstARM32Mla *create(Cfg *Func, Variable *Dest, Variable *Src0,
509 Variable *Src1, Variable *Acc) { 666 Variable *Src1, Variable *Acc,
667 CondARM32::Cond Predicate) {
510 return new (Func->allocate<InstARM32Mla>()) 668 return new (Func->allocate<InstARM32Mla>())
511 InstARM32Mla(Func, Dest, Src0, Src1, Acc); 669 InstARM32Mla(Func, Dest, Src0, Src1, Acc, Predicate);
512 } 670 }
513 void emit(const Cfg *Func) const override; 671 void emit(const Cfg *Func) const override;
514 void emitIAS(const Cfg *Func) const override; 672 void emitIAS(const Cfg *Func) const override;
515 void dump(const Cfg *Func) const override; 673 void dump(const Cfg *Func) const override;
516 static bool classof(const Inst *Inst) { return isClassof(Inst, Mla); } 674 static bool classof(const Inst *Inst) { return isClassof(Inst, Mla); }
517 675
518 private: 676 private:
519 InstARM32Mla(Cfg *Func, Variable *Dest, Variable *Src0, Variable *Src1, 677 InstARM32Mla(Cfg *Func, Variable *Dest, Variable *Src0, Variable *Src1,
520 Variable *Acc); 678 Variable *Acc, CondARM32::Cond Predicate);
521 ~InstARM32Mla() override {} 679 ~InstARM32Mla() override {}
522 }; 680 };
523 681
524 // Ret pseudo-instruction. This is actually a "bx" instruction with 682 // Ret pseudo-instruction. This is actually a "bx" instruction with
525 // an "lr" register operand, but epilogue lowering will search for a Ret 683 // an "lr" register operand, but epilogue lowering will search for a Ret
526 // instead of a generic "bx". This instruction also takes a Source 684 // instead of a generic "bx". This instruction also takes a Source
527 // operand (for non-void returning functions) for liveness analysis, though 685 // operand (for non-void returning functions) for liveness analysis, though
528 // a FakeUse before the ret would do just as well. 686 // a FakeUse before the ret would do just as well.
687 //
688 // NOTE: Even though "bx" can be predicated, for now leave out the predication
689 // since it's not yet known to be useful for Ret. That may complicate finding
690 // the terminator instruction if it's not guaranteed to be executed.
529 class InstARM32Ret : public InstARM32 { 691 class InstARM32Ret : public InstARM32 {
530 InstARM32Ret() = delete; 692 InstARM32Ret() = delete;
531 InstARM32Ret(const InstARM32Ret &) = delete; 693 InstARM32Ret(const InstARM32Ret &) = delete;
532 InstARM32Ret &operator=(const InstARM32Ret &) = delete; 694 InstARM32Ret &operator=(const InstARM32Ret &) = delete;
533 695
534 public: 696 public:
535 static InstARM32Ret *create(Cfg *Func, Variable *LR, 697 static InstARM32Ret *create(Cfg *Func, Variable *LR,
536 Variable *Source = nullptr) { 698 Variable *Source = nullptr) {
537 return new (Func->allocate<InstARM32Ret>()) InstARM32Ret(Func, LR, Source); 699 return new (Func->allocate<InstARM32Ret>()) InstARM32Ret(Func, LR, Source);
538 } 700 }
539 void emit(const Cfg *Func) const override; 701 void emit(const Cfg *Func) const override;
540 void emitIAS(const Cfg *Func) const override; 702 void emitIAS(const Cfg *Func) const override;
541 void dump(const Cfg *Func) const override; 703 void dump(const Cfg *Func) const override;
542 static bool classof(const Inst *Inst) { return isClassof(Inst, Ret); } 704 static bool classof(const Inst *Inst) { return isClassof(Inst, Ret); }
543 705
544 private: 706 private:
545 InstARM32Ret(Cfg *Func, Variable *LR, Variable *Source); 707 InstARM32Ret(Cfg *Func, Variable *LR, Variable *Source);
546 ~InstARM32Ret() override {} 708 ~InstARM32Ret() override {}
547 }; 709 };
548 710
549 // Unsigned Multiply Long: d.lo, d.hi := x * y 711 // Unsigned Multiply Long: d.lo, d.hi := x * y
550 class InstARM32Umull : public InstARM32 { 712 class InstARM32Umull : public InstARM32Pred {
551 InstARM32Umull() = delete; 713 InstARM32Umull() = delete;
552 InstARM32Umull(const InstARM32Umull &) = delete; 714 InstARM32Umull(const InstARM32Umull &) = delete;
553 InstARM32Umull &operator=(const InstARM32Umull &) = delete; 715 InstARM32Umull &operator=(const InstARM32Umull &) = delete;
554 716
555 public: 717 public:
556 // Everything must be a register. 718 // Everything must be a register.
557 static InstARM32Umull *create(Cfg *Func, Variable *DestLo, Variable *DestHi, 719 static InstARM32Umull *create(Cfg *Func, Variable *DestLo, Variable *DestHi,
558 Variable *Src0, Variable *Src1) { 720 Variable *Src0, Variable *Src1,
721 CondARM32::Cond Predicate) {
559 return new (Func->allocate<InstARM32Umull>()) 722 return new (Func->allocate<InstARM32Umull>())
560 InstARM32Umull(Func, DestLo, DestHi, Src0, Src1); 723 InstARM32Umull(Func, DestLo, DestHi, Src0, Src1, Predicate);
561 } 724 }
562 void emit(const Cfg *Func) const override; 725 void emit(const Cfg *Func) const override;
563 void emitIAS(const Cfg *Func) const override; 726 void emitIAS(const Cfg *Func) const override;
564 void dump(const Cfg *Func) const override; 727 void dump(const Cfg *Func) const override;
565 static bool classof(const Inst *Inst) { return isClassof(Inst, Umull); } 728 static bool classof(const Inst *Inst) { return isClassof(Inst, Umull); }
566 729
567 private: 730 private:
568 InstARM32Umull(Cfg *Func, Variable *DestLo, Variable *DestHi, Variable *Src0, 731 InstARM32Umull(Cfg *Func, Variable *DestLo, Variable *DestHi, Variable *Src0,
569 Variable *Src1); 732 Variable *Src1, CondARM32::Cond Predicate);
570 ~InstARM32Umull() override {} 733 ~InstARM32Umull() override {}
571 Variable *DestHi; 734 Variable *DestHi;
572 }; 735 };
573 736
574 // Declare partial template specializations of emit() methods that 737 // Declare partial template specializations of emit() methods that
575 // already have default implementations. Without this, there is the 738 // already have default implementations. Without this, there is the
576 // possibility of ODR violations and link errors. 739 // possibility of ODR violations and link errors.
577 740
578 template <> void InstARM32Movw::emit(const Cfg *Func) const; 741 template <> void InstARM32Movw::emit(const Cfg *Func) const;
579 template <> void InstARM32Movt::emit(const Cfg *Func) const; 742 template <> void InstARM32Movt::emit(const Cfg *Func) const;
580 743
581 } // end of namespace Ice 744 } // end of namespace Ice
582 745
583 #endif // SUBZERO_SRC_ICEINSTARM32_H 746 #endif // SUBZERO_SRC_ICEINSTARM32_H
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698