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

Side by Side Diff: src/IceInstARM32.cpp

Issue 1359193003: Subzero. Enables (most) crosstests for ARM32. (Closed) Base URL: https://chromium.googlesource.com/native_client/pnacl-subzero.git@master
Patch Set: Fixes the broken lit tests. Created 5 years, 2 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.cpp - ARM32 instruction implementation ----===// 1 //===- subzero/src/IceInstARM32.cpp - ARM32 instruction implementation ----===//
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 15 matching lines...) Expand all
26 namespace Ice { 26 namespace Ice {
27 27
28 namespace { 28 namespace {
29 29
30 const struct TypeARM32Attributes_ { 30 const struct TypeARM32Attributes_ {
31 const char *WidthString; // b, h, <blank>, or d 31 const char *WidthString; // b, h, <blank>, or d
32 const char *VecWidthString; // i8, i16, i32, f32, f64 32 const char *VecWidthString; // i8, i16, i32, f32, f64
33 int8_t SExtAddrOffsetBits; 33 int8_t SExtAddrOffsetBits;
34 int8_t ZExtAddrOffsetBits; 34 int8_t ZExtAddrOffsetBits;
35 } TypeARM32Attributes[] = { 35 } TypeARM32Attributes[] = {
36 #define X(tag, elementty, int_width, vec_width, sbits, ubits) \ 36 #define X(tag, elementty, int_width, vec_width, sbits, ubits, rraddr) \
37 { int_width, vec_width, sbits, ubits } \ 37 { int_width, vec_width, sbits, ubits } \
38 , 38 ,
39 ICETYPEARM32_TABLE 39 ICETYPEARM32_TABLE
40 #undef X 40 #undef X
41 }; 41 };
42 42
43 const struct InstARM32ShiftAttributes_ { 43 const struct InstARM32ShiftAttributes_ {
44 const char *EmitString; 44 const char *EmitString;
45 } InstARM32ShiftAttributes[] = { 45 } InstARM32ShiftAttributes[] = {
46 #define X(tag, emit) \ 46 #define X(tag, emit) \
(...skipping 157 matching lines...) Expand 10 before | Expand all | Expand 10 after
204 Vars[1] = Index; 204 Vars[1] = Index;
205 } 205 }
206 206
207 bool OperandARM32Mem::canHoldOffset(Type Ty, bool SignExt, int32_t Offset) { 207 bool OperandARM32Mem::canHoldOffset(Type Ty, bool SignExt, int32_t Offset) {
208 int32_t Bits = SignExt ? TypeARM32Attributes[Ty].SExtAddrOffsetBits 208 int32_t Bits = SignExt ? TypeARM32Attributes[Ty].SExtAddrOffsetBits
209 : TypeARM32Attributes[Ty].ZExtAddrOffsetBits; 209 : TypeARM32Attributes[Ty].ZExtAddrOffsetBits;
210 if (Bits == 0) 210 if (Bits == 0)
211 return Offset == 0; 211 return Offset == 0;
212 // Note that encodings for offsets are sign-magnitude for ARM, so we check 212 // Note that encodings for offsets are sign-magnitude for ARM, so we check
213 // with IsAbsoluteUint(). 213 // with IsAbsoluteUint().
214 if (isScalarFloatingType(Ty)) 214 /*if (isScalarFloatingType(Ty))
Jim Stichnoth 2015/09/24 05:57:13 Explain or delete comment?
John 2015/09/24 21:28:39 Done.
215 return Utils::IsAligned(Offset, 4) && Utils::IsAbsoluteUint(Bits, Offset); 215 return Utils::IsAligned(Offset, 4) && Utils::IsAbsoluteUint(Bits, Offset);*/
216 return Utils::IsAbsoluteUint(Bits, Offset); 216 return Utils::IsAbsoluteUint(Bits, Offset);
217 } 217 }
218 218
219 OperandARM32FlexImm::OperandARM32FlexImm(Cfg * /* Func */, Type Ty, 219 OperandARM32FlexImm::OperandARM32FlexImm(Cfg * /* Func */, Type Ty,
220 uint32_t Imm, uint32_t RotateAmt) 220 uint32_t Imm, uint32_t RotateAmt)
221 : OperandARM32Flex(kFlexImm, Ty), Imm(Imm), RotateAmt(RotateAmt) { 221 : OperandARM32Flex(kFlexImm, Ty), Imm(Imm), RotateAmt(RotateAmt) {
222 NumVars = 0; 222 NumVars = 0;
223 Vars = nullptr; 223 Vars = nullptr;
224 } 224 }
225 225
(...skipping 159 matching lines...) Expand 10 before | Expand all | Expand 10 after
385 InstARM32Vcmp::InstARM32Vcmp(Cfg *Func, Variable *Src0, Variable *Src1, 385 InstARM32Vcmp::InstARM32Vcmp(Cfg *Func, Variable *Src0, Variable *Src1,
386 CondARM32::Cond Predicate) 386 CondARM32::Cond Predicate)
387 : InstARM32Pred(Func, InstARM32::Vcmp, 2, nullptr, Predicate) { 387 : InstARM32Pred(Func, InstARM32::Vcmp, 2, nullptr, Predicate) {
388 addSource(Src0); 388 addSource(Src0);
389 addSource(Src1); 389 addSource(Src1);
390 } 390 }
391 391
392 InstARM32Vmrs::InstARM32Vmrs(Cfg *Func, CondARM32::Cond Predicate) 392 InstARM32Vmrs::InstARM32Vmrs(Cfg *Func, CondARM32::Cond Predicate)
393 : InstARM32Pred(Func, InstARM32::Vmrs, 0, nullptr, Predicate) {} 393 : InstARM32Pred(Func, InstARM32::Vmrs, 0, nullptr, Predicate) {}
394 394
395 InstARM32Vabs::InstARM32Vabs(Cfg *Func, Variable *Dest, Variable *Src,
396 CondARM32::Cond Predicate)
397 : InstARM32Pred(Func, InstARM32::Vabs, 1, Dest, Predicate) {
398 addSource(Src);
399 }
395 // ======================== Dump routines ======================== // 400 // ======================== Dump routines ======================== //
396 401
397 // Two-addr ops 402 // Two-addr ops
398 template <> const char *InstARM32Movt::Opcode = "movt"; 403 template <> const char *InstARM32Movt::Opcode = "movt";
399 // Unary ops 404 // Unary ops
400 template <> const char *InstARM32Movw::Opcode = "movw"; 405 template <> const char *InstARM32Movw::Opcode = "movw";
401 template <> const char *InstARM32Clz::Opcode = "clz"; 406 template <> const char *InstARM32Clz::Opcode = "clz";
402 template <> const char *InstARM32Mvn::Opcode = "mvn"; 407 template <> const char *InstARM32Mvn::Opcode = "mvn";
403 template <> const char *InstARM32Rbit::Opcode = "rbit"; 408 template <> const char *InstARM32Rbit::Opcode = "rbit";
404 template <> const char *InstARM32Rev::Opcode = "rev"; 409 template <> const char *InstARM32Rev::Opcode = "rev";
405 template <> const char *InstARM32Sxt::Opcode = "sxt"; // still requires b/h 410 template <> const char *InstARM32Sxt::Opcode = "sxt"; // still requires b/h
406 template <> const char *InstARM32Uxt::Opcode = "uxt"; // still requires b/h 411 template <> const char *InstARM32Uxt::Opcode = "uxt"; // still requires b/h
407 // FP 412 // FP
408 template <> const char *InstARM32Vsqrt::Opcode = "vsqrt"; 413 template <> const char *InstARM32Vsqrt::Opcode = "vsqrt";
409 // Mov-like ops 414 // Mov-like ops
410 template <> const char *InstARM32Ldr::Opcode = "ldr"; 415 template <> const char *InstARM32Ldr::Opcode = "ldr";
411 template <> const char *InstARM32Mov::Opcode = "mov";
412 // FP
413 template <> const char *InstARM32Vldr::Opcode = "vldr";
414 // Three-addr ops 416 // Three-addr ops
415 template <> const char *InstARM32Adc::Opcode = "adc"; 417 template <> const char *InstARM32Adc::Opcode = "adc";
416 template <> const char *InstARM32Add::Opcode = "add"; 418 template <> const char *InstARM32Add::Opcode = "add";
417 template <> const char *InstARM32And::Opcode = "and"; 419 template <> const char *InstARM32And::Opcode = "and";
418 template <> const char *InstARM32Asr::Opcode = "asr"; 420 template <> const char *InstARM32Asr::Opcode = "asr";
419 template <> const char *InstARM32Bic::Opcode = "bic"; 421 template <> const char *InstARM32Bic::Opcode = "bic";
420 template <> const char *InstARM32Eor::Opcode = "eor"; 422 template <> const char *InstARM32Eor::Opcode = "eor";
421 template <> const char *InstARM32Lsl::Opcode = "lsl"; 423 template <> const char *InstARM32Lsl::Opcode = "lsl";
422 template <> const char *InstARM32Lsr::Opcode = "lsr"; 424 template <> const char *InstARM32Lsr::Opcode = "lsr";
423 template <> const char *InstARM32Mul::Opcode = "mul"; 425 template <> const char *InstARM32Mul::Opcode = "mul";
(...skipping 16 matching lines...) Expand all
440 template <> const char *InstARM32Tst::Opcode = "tst"; 442 template <> const char *InstARM32Tst::Opcode = "tst";
441 443
442 void InstARM32::dump(const Cfg *Func) const { 444 void InstARM32::dump(const Cfg *Func) const {
443 if (!BuildDefs::dump()) 445 if (!BuildDefs::dump())
444 return; 446 return;
445 Ostream &Str = Func->getContext()->getStrDump(); 447 Ostream &Str = Func->getContext()->getStrDump();
446 Str << "[ARM32] "; 448 Str << "[ARM32] ";
447 Inst::dump(Func); 449 Inst::dump(Func);
448 } 450 }
449 451
450 template <> void InstARM32Mov::emit(const Cfg *Func) const { 452 void InstARM32Mov::emitMultiDestSingleSource(const Cfg *Func) const {
451 if (!BuildDefs::dump()) 453 if (!BuildDefs::dump())
452 return; 454 return;
453 Ostream &Str = Func->getContext()->getStrEmit(); 455 Ostream &Str = Func->getContext()->getStrEmit();
454 assert(getSrcSize() == 1); 456 auto *Dest = llvm::cast<Variable64On32>(getDest());
455 Variable *Dest = getDest(); 457 Operand *Src = getSrc(0);
456 if (Dest->hasReg()) { 458
457 IceString ActualOpcode = Opcode; 459 assert(Dest->getType() == IceType_i64);
458 Operand *Src0 = getSrc(0); 460 assert(Dest->getHi()->hasReg());
459 if (const auto *Src0V = llvm::dyn_cast<Variable>(Src0)) { 461 assert(Dest->getLo()->hasReg());
460 if (!Src0V->hasReg()) { 462 assert(!llvm::isa<OperandARM32Mem>(Src));
461 // Always use the whole stack slot. A 32-bit load has a larger range of 463
462 // offsets than 16-bit, etc. 464 Str << "\t"
463 ActualOpcode = IceString("ldr"); 465 << "vmov" << getPredicate() << "\t";
464 } 466 Dest->getLo()->emit(Func);
465 } else { 467 Str << ", ";
466 if (llvm::isa<OperandARM32Mem>(Src0)) 468 Dest->getHi()->emit(Func);
467 ActualOpcode = IceString("ldr") + getWidthString(Dest->getType()); 469 Str << ", ";
468 } 470 Src->emit(Func);
469 Str << "\t" << ActualOpcode << getPredicate() << "\t";
470 getDest()->emit(Func);
471 Str << ", ";
472 getSrc(0)->emit(Func);
473 } else {
474 Variable *Src0 = llvm::cast<Variable>(getSrc(0));
475 assert(Src0->hasReg());
476 Str << "\t"
477 << "str" << getPredicate() << "\t";
478 Src0->emit(Func);
479 Str << ", ";
480 Dest->emit(Func);
481 }
482 } 471 }
483 472
484 template <> void InstARM32Mov::emitIAS(const Cfg *Func) const { 473 void InstARM32Mov::emitSingleDestMultiSource(const Cfg *Func) const {
485 assert(getSrcSize() == 1);
486 (void)Func;
487 llvm_unreachable("Not yet implemented");
488 }
489
490 template <> void InstARM32Vldr::emit(const Cfg *Func) const {
491 if (!BuildDefs::dump()) 474 if (!BuildDefs::dump())
492 return; 475 return;
493 Ostream &Str = Func->getContext()->getStrEmit(); 476 Ostream &Str = Func->getContext()->getStrEmit();
494 assert(getSrcSize() == 1); 477 Variable *Dest = getDest();
495 assert(getDest()->hasReg()); 478 auto *Src = llvm::cast<Variable64On32>(getSrc(0));
496 Str << "\t" << Opcode << getPredicate() << "\t";
497 getDest()->emit(Func);
498 Str << ", ";
499 getSrc(0)->emit(Func);
500 }
501 479
502 template <> void InstARM32Vldr::emitIAS(const Cfg *Func) const { 480 assert(Src->getType() == IceType_i64);
503 assert(getSrcSize() == 1); 481 assert(Src->getHi()->hasReg());
504 (void)Func; 482 assert(Src->getLo()->hasReg());
505 llvm_unreachable("Not yet implemented"); 483 assert(Dest->hasReg());
506 }
507
508 void InstARM32Vmov::emitMultiDestSingleSource(const Cfg *Func) const {
509 if (!BuildDefs::dump())
510 return;
511 Ostream &Str = Func->getContext()->getStrEmit();
512 Variable *Dest0 = getDest();
513 Operand *Src0 = getSrc(0);
514
515 assert(Dest0->hasReg());
516 assert(Dest1->hasReg());
517 assert(!llvm::isa<OperandARM32Mem>(Src0));
518 484
519 Str << "\t" 485 Str << "\t"
520 << "vmov" << getPredicate() << "\t"; 486 << "vmov" << getPredicate() << "\t";
521 Dest0->emit(Func); 487 Dest->emit(Func);
522 Str << ", "; 488 Str << ", ";
523 Dest1->emit(Func); 489 Src->getLo()->emit(Func);
524 Str << ", "; 490 Str << ", ";
525 Src0->emit(Func); 491 Src->getHi()->emit(Func);
526 }
527
528 void InstARM32Vmov::emitSingleDestMultiSource(const Cfg *Func) const {
529 if (!BuildDefs::dump())
530 return;
531 Ostream &Str = Func->getContext()->getStrEmit();
532 Variable *Dest0 = getDest();
533 Operand *Src0 = getSrc(0);
534 Operand *Src1 = getSrc(1);
535
536 assert(Dest0->hasReg());
537 assert(!llvm::isa<OperandARM32Mem>(Src0));
538 assert(!llvm::isa<OperandARM32Mem>(Src1));
539
540 Str << "\t"
541 << "vmov" << getPredicate() << "\t";
542 Dest0->emit(Func);
543 Str << ", ";
544 Src0->emit(Func);
545 Str << ", ";
546 Src1->emit(Func);
547 } 492 }
548 493
549 namespace { 494 namespace {
495
550 bool isVariableWithoutRegister(const Operand *Op) { 496 bool isVariableWithoutRegister(const Operand *Op) {
551 if (const auto *OpV = llvm::dyn_cast<const Variable>(Op)) { 497 if (const auto *OpV = llvm::dyn_cast<const Variable>(Op)) {
552 return !OpV->hasReg(); 498 return !OpV->hasReg();
553 } 499 }
554 return false; 500 return false;
555 } 501 }
556
557 bool isMemoryAccess(Operand *Op) { 502 bool isMemoryAccess(Operand *Op) {
558 return isVariableWithoutRegister(Op) || llvm::isa<OperandARM32Mem>(Op); 503 return isVariableWithoutRegister(Op) || llvm::isa<OperandARM32Mem>(Op);
559 } 504 }
560 505
561 bool isMoveBetweenCoreAndVFPRegisters(Variable *Dest, Operand *Src) { 506 bool isMoveBetweenCoreAndVFPRegisters(Variable *Dest, Operand *Src) {
562 const Type DestTy = Dest->getType(); 507 const Type DestTy = Dest->getType();
563 const Type SrcTy = Src->getType(); 508 const Type SrcTy = Src->getType();
564 assert(!(isScalarIntegerType(DestTy) && isScalarIntegerType(SrcTy)) && 509 return !isVectorType(DestTy) && !isVectorType(SrcTy) &&
565 "At most one of vmov's operands can be a core register."); 510 (isScalarIntegerType(DestTy) == isScalarFloatingType(SrcTy));
566 return isScalarIntegerType(DestTy) || isScalarIntegerType(SrcTy);
567 } 511 }
512
568 } // end of anonymous namespace 513 } // end of anonymous namespace
569 514
570 void InstARM32Vmov::emitSingleDestSingleSource(const Cfg *Func) const { 515 void InstARM32Mov::emitSingleDestSingleSource(const Cfg *Func) const {
571 if (!BuildDefs::dump()) 516 if (!BuildDefs::dump())
572 return; 517 return;
573 Ostream &Str = Func->getContext()->getStrEmit(); 518 Ostream &Str = Func->getContext()->getStrEmit();
574 Variable *Dest = getDest(); 519 Variable *Dest = getDest();
520
575 if (Dest->hasReg()) { 521 if (Dest->hasReg()) {
522 Type DestTy = Dest->getType();
576 Operand *Src0 = getSrc(0); 523 Operand *Src0 = getSrc(0);
577 const char *ActualOpcode = isMemoryAccess(Src0) ? "vldr" : "vmov"; 524 const bool DestIsVector = isVectorType(DestTy);
525 const bool DestIsScalarFP = isScalarFloatingType(Dest->getType());
526 const bool CoreVFPMove = isMoveBetweenCoreAndVFPRegisters(Dest, Src0);
527 const char *LoadOpcode =
528 DestIsVector ? "vld1" : (DestIsScalarFP ? "vldr" : "ldr");
529 const char *RegMovOpcode =
530 (DestIsVector || DestIsScalarFP || CoreVFPMove) ? "vmov" : "mov";
531 const char *ActualOpcode = isMemoryAccess(Src0) ? LoadOpcode : RegMovOpcode;
578 // when vmov{c}'ing, we need to emit a width string. Otherwise, the 532 // when vmov{c}'ing, we need to emit a width string. Otherwise, the
579 // assembler might be tempted to assume we want a vector vmov{c}, and that 533 // assembler might be tempted to assume we want a vector vmov{c}, and that
580 // is disallowed because ARM. 534 // is disallowed because ARM.
535 const char *NoWidthString = "";
581 const char *WidthString = 536 const char *WidthString =
582 (isMemoryAccess(Src0) || isMoveBetweenCoreAndVFPRegisters(Dest, Src0)) 537 isMemoryAccess(Src0)
583 ? "" 538 ? (DestIsVector ? ".64" : NoWidthString)
584 : getVecWidthString(Src0->getType()); 539 : (!CoreVFPMove ? getVecWidthString(DestTy) : NoWidthString);
540
585 Str << "\t" << ActualOpcode << getPredicate() << WidthString << "\t"; 541 Str << "\t" << ActualOpcode << getPredicate() << WidthString << "\t";
586 Dest->emit(Func); 542 Dest->emit(Func);
587 Str << ", "; 543 Str << ", ";
588 Src0->emit(Func); 544 Src0->emit(Func);
589 } else { 545 } else {
590 Variable *Src0 = llvm::cast<Variable>(getSrc(0)); 546 Variable *Src0 = llvm::cast<Variable>(getSrc(0));
591 assert(Src0->hasReg()); 547 assert(Src0->hasReg());
592 Str << "\t" 548 const char *ActualOpcode =
593 "vstr" << getPredicate() << "\t"; 549 isVectorType(Src0->getType())
550 ? "vst1"
551 : (isScalarFloatingType(Src0->getType()) ? "vstr" : "str");
552 const char *NoWidthString = "";
553 const char *WidthString =
554 isVectorType(Src0->getType()) ? ".64" : NoWidthString;
555 Str << "\t" << ActualOpcode << getPredicate() << WidthString << "\t";
594 Src0->emit(Func); 556 Src0->emit(Func);
595 Str << ", "; 557 Str << ", ";
596 Dest->emit(Func); 558 Dest->emit(Func);
597 } 559 }
598 } 560 }
599 561
600 void InstARM32Vmov::emit(const Cfg *Func) const { 562 void InstARM32Mov::emit(const Cfg *Func) const {
601 if (!BuildDefs::dump()) 563 if (!BuildDefs::dump())
602 return; 564 return;
603 assert(isMultiDest() + isMultiSource() <= 1 && "Invalid vmov type."); 565 assert(!(isMultiDest() && isMultiSource()) && "Invalid vmov type.");
604 if (isMultiDest()) { 566 if (isMultiDest()) {
605 emitMultiDestSingleSource(Func); 567 emitMultiDestSingleSource(Func);
606 return; 568 return;
607 } 569 }
608 570
609 if (isMultiSource()) { 571 if (isMultiSource()) {
610 emitSingleDestMultiSource(Func); 572 emitSingleDestMultiSource(Func);
611 return; 573 return;
612 } 574 }
613 575
614 emitSingleDestSingleSource(Func); 576 emitSingleDestSingleSource(Func);
615 } 577 }
616 578
617 void InstARM32Vmov::emitIAS(const Cfg *Func) const { 579 void InstARM32Mov::emitIAS(const Cfg *Func) const {
618 assert(getSrcSize() == 1); 580 assert(getSrcSize() == 1);
619 (void)Func; 581 (void)Func;
620 llvm_unreachable("Not yet implemented"); 582 llvm_unreachable("Not yet implemented");
621 } 583 }
622 584
623 void InstARM32Vmov::dump(const Cfg *Func) const { 585 void InstARM32Mov::dump(const Cfg *Func) const {
624 if (!BuildDefs::dump()) 586 if (!BuildDefs::dump())
625 return; 587 return;
588 assert(getSrcSize() == 1);
626 Ostream &Str = Func->getContext()->getStrDump(); 589 Ostream &Str = Func->getContext()->getStrDump();
627 dumpOpcodePred(Str, "vmov", getDest()->getType()); 590 Variable *Dest = getDest();
591 if (auto *Dest64 = llvm::dyn_cast<Variable64On32>(Dest)) {
592 Dest64->getLo()->dump(Func);
593 Str << ", ";
594 Dest64->getHi()->dump(Func);
595 } else {
596 Dest->dump(Func);
597 }
598
599 dumpOpcodePred(Str, " = mov", getDest()->getType());
628 Str << " "; 600 Str << " ";
629 dumpDest(Func); 601
630 Str << ", "; 602 Operand *Src = getSrc(0);
631 dumpSources(Func); 603 if (auto *Src64 = llvm::dyn_cast<Variable64On32>(Src)) {
604 Src64->getLo()->dump(Func);
605 Str << ", ";
606 Src64->getHi()->dump(Func);
607 } else {
608 Src->dump(Func);
609 }
632 } 610 }
633 611
634 void InstARM32Br::emit(const Cfg *Func) const { 612 void InstARM32Br::emit(const Cfg *Func) const {
635 if (!BuildDefs::dump()) 613 if (!BuildDefs::dump())
636 return; 614 return;
637 Ostream &Str = Func->getContext()->getStrEmit(); 615 Ostream &Str = Func->getContext()->getStrEmit();
638 Str << "\t" 616 Str << "\t"
639 << "b" << getPredicate() << "\t"; 617 << "b" << getPredicate() << "\t";
640 if (Label) { 618 if (Label) {
641 Str << Label->getName(Func); 619 Str << Label->getName(Func);
(...skipping 99 matching lines...) Expand 10 before | Expand all | Expand 10 after
741 Ostream &Str = Func->getContext()->getStrDump(); 719 Ostream &Str = Func->getContext()->getStrDump();
742 Str << getName(Func) << ":"; 720 Str << getName(Func) << ":";
743 } 721 }
744 722
745 template <> void InstARM32Ldr::emit(const Cfg *Func) const { 723 template <> void InstARM32Ldr::emit(const Cfg *Func) const {
746 if (!BuildDefs::dump()) 724 if (!BuildDefs::dump())
747 return; 725 return;
748 Ostream &Str = Func->getContext()->getStrEmit(); 726 Ostream &Str = Func->getContext()->getStrEmit();
749 assert(getSrcSize() == 1); 727 assert(getSrcSize() == 1);
750 assert(getDest()->hasReg()); 728 assert(getDest()->hasReg());
751 Type Ty = getSrc(0)->getType(); 729 Variable *Dest = getDest();
752 Str << "\t" << Opcode << getWidthString(Ty) << getPredicate() << "\t"; 730 Type DestTy = Dest->getType();
731 const bool DestIsVector = isVectorType(DestTy);
732 const bool DestIsScalarFloat = isScalarFloatingType(DestTy);
733 const char *ActualOpcode =
734 DestIsVector ? "vld1" : (DestIsScalarFloat ? "vldr" : "ldr");
735 const char *VectorMarker = DestIsVector ? ".64" : "";
736 const char *WidthString = DestIsVector ? "" : getWidthString(DestTy);
737 Str << "\t" << ActualOpcode << WidthString << getPredicate() << VectorMarker
738 << "\t";
753 getDest()->emit(Func); 739 getDest()->emit(Func);
754 Str << ", "; 740 Str << ", ";
755 getSrc(0)->emit(Func); 741 getSrc(0)->emit(Func);
756 } 742 }
757 743
758 template <> void InstARM32Ldr::emitIAS(const Cfg *Func) const { 744 template <> void InstARM32Ldr::emitIAS(const Cfg *Func) const {
759 assert(getSrcSize() == 1); 745 assert(getSrcSize() == 1);
760 (void)Func; 746 (void)Func;
761 llvm_unreachable("Not yet implemented"); 747 llvm_unreachable("Not yet implemented");
762 } 748 }
(...skipping 29 matching lines...) Expand all
792 Str << "#:upper16:"; 778 Str << "#:upper16:";
793 CR->emitWithoutPrefix(Func->getTarget()); 779 CR->emitWithoutPrefix(Func->getTarget());
794 } else { 780 } else {
795 Src1->emit(Func); 781 Src1->emit(Func);
796 } 782 }
797 } 783 }
798 784
799 void InstARM32Pop::emit(const Cfg *Func) const { 785 void InstARM32Pop::emit(const Cfg *Func) const {
800 if (!BuildDefs::dump()) 786 if (!BuildDefs::dump())
801 return; 787 return;
802 assert(Dests.size() > 0); 788 SizeT IntegerCount = 0;
789 for (const Operand *Op : Dests) {
790 if (isScalarIntegerType(Op->getType())) {
791 ++IntegerCount;
792 }
793 }
803 Ostream &Str = Func->getContext()->getStrEmit(); 794 Ostream &Str = Func->getContext()->getStrEmit();
795 if (IntegerCount == 0) {
796 Str << "\t@ empty pop";
797 return;
798 }
804 Str << "\t" 799 Str << "\t"
805 << "pop" 800 << "pop"
806 << "\t{"; 801 << "\t{";
807 for (SizeT I = 0; I < Dests.size(); ++I) { 802 bool PrintComma = false;
808 if (I > 0) 803 for (const Operand *Op : Dests) {
809 Str << ", "; 804 if (isScalarIntegerType(Op->getType())) {
810 Dests[I]->emit(Func); 805 if (PrintComma)
806 Str << ", ";
807 Op->emit(Func);
808 PrintComma = true;
809 }
811 } 810 }
812 Str << "}"; 811 Str << "}";
813 } 812 }
814 813
815 void InstARM32Pop::emitIAS(const Cfg *Func) const { 814 void InstARM32Pop::emitIAS(const Cfg *Func) const {
816 (void)Func; 815 (void)Func;
817 llvm_unreachable("Not yet implemented"); 816 llvm_unreachable("Not yet implemented");
818 } 817 }
819 818
820 void InstARM32Pop::dump(const Cfg *Func) const { 819 void InstARM32Pop::dump(const Cfg *Func) const {
(...skipping 38 matching lines...) Expand 10 before | Expand all | Expand 10 after
859 getDest()->dump(Func); 858 getDest()->dump(Func);
860 Str << " = sub.i32 "; 859 Str << " = sub.i32 ";
861 getSrc(0)->dump(Func); 860 getSrc(0)->dump(Func);
862 Str << ", " << Amount << " ; "; 861 Str << ", " << Amount << " ; ";
863 getSrc(1)->dump(Func); 862 getSrc(1)->dump(Func);
864 } 863 }
865 864
866 void InstARM32Push::emit(const Cfg *Func) const { 865 void InstARM32Push::emit(const Cfg *Func) const {
867 if (!BuildDefs::dump()) 866 if (!BuildDefs::dump())
868 return; 867 return;
869 assert(getSrcSize() > 0); 868 SizeT IntegerCount = 0;
869 for (SizeT i = 0; i < getSrcSize(); ++i) {
870 if (isScalarIntegerType(getSrc(i)->getType())) {
871 ++IntegerCount;
872 }
873 }
870 Ostream &Str = Func->getContext()->getStrEmit(); 874 Ostream &Str = Func->getContext()->getStrEmit();
875 if (IntegerCount == 0) {
876 Str << "\t"
877 << "@empty push";
878 return;
879 }
871 Str << "\t" 880 Str << "\t"
872 << "push" 881 << "push"
873 << "\t{"; 882 << "\t{";
874 emitSources(Func); 883 bool PrintComma = false;
884 for (SizeT i = 0; i < getSrcSize(); ++i) {
885 Operand *Op = getSrc(i);
886 if (isScalarIntegerType(Op->getType())) {
887 if (PrintComma)
888 Str << ", ";
889 Op->emit(Func);
890 PrintComma = true;
891 }
892 }
875 Str << "}"; 893 Str << "}";
876 } 894 }
877 895
878 void InstARM32Push::emitIAS(const Cfg *Func) const { 896 void InstARM32Push::emitIAS(const Cfg *Func) const {
879 (void)Func; 897 (void)Func;
880 llvm_unreachable("Not yet implemented"); 898 llvm_unreachable("Not yet implemented");
881 } 899 }
882 900
883 void InstARM32Push::dump(const Cfg *Func) const { 901 void InstARM32Push::dump(const Cfg *Func) const {
884 if (!BuildDefs::dump()) 902 if (!BuildDefs::dump())
(...skipping 31 matching lines...) Expand 10 before | Expand all | Expand 10 after
916 Str << "ret." << Ty << " "; 934 Str << "ret." << Ty << " ";
917 dumpSources(Func); 935 dumpSources(Func);
918 } 936 }
919 937
920 void InstARM32Str::emit(const Cfg *Func) const { 938 void InstARM32Str::emit(const Cfg *Func) const {
921 if (!BuildDefs::dump()) 939 if (!BuildDefs::dump())
922 return; 940 return;
923 Ostream &Str = Func->getContext()->getStrEmit(); 941 Ostream &Str = Func->getContext()->getStrEmit();
924 assert(getSrcSize() == 2); 942 assert(getSrcSize() == 2);
925 Type Ty = getSrc(0)->getType(); 943 Type Ty = getSrc(0)->getType();
926 const char *Opcode = isScalarFloatingType(Ty) ? "vstr" : "str"; 944 const bool IsVectorStore = isVectorType(Ty);
927 Str << "\t" << Opcode << getWidthString(Ty) << getPredicate() << "\t"; 945 const char *Opcode =
946 IsVectorStore ? "vst1" : (isScalarFloatingType(Ty) ? "vstr" : "str");
947 const char *VecEltWidthString = IsVectorStore ? ".64" : "";
948 Str << "\t" << Opcode << getWidthString(Ty) << getPredicate()
949 << VecEltWidthString << "\t";
928 getSrc(0)->emit(Func); 950 getSrc(0)->emit(Func);
929 Str << ", "; 951 Str << ", ";
930 getSrc(1)->emit(Func); 952 getSrc(1)->emit(Func);
931 } 953 }
932 954
933 void InstARM32Str::emitIAS(const Cfg *Func) const { 955 void InstARM32Str::emitIAS(const Cfg *Func) const {
934 assert(getSrcSize() == 2); 956 assert(getSrcSize() == 2);
935 (void)Func; 957 (void)Func;
936 llvm_unreachable("Not yet implemented"); 958 llvm_unreachable("Not yet implemented");
937 } 959 }
(...skipping 174 matching lines...) Expand 10 before | Expand all | Expand 10 after
1112 } 1134 }
1113 1135
1114 void InstARM32Vmrs::dump(const Cfg *Func) const { 1136 void InstARM32Vmrs::dump(const Cfg *Func) const {
1115 if (!BuildDefs::dump()) 1137 if (!BuildDefs::dump())
1116 return; 1138 return;
1117 Ostream &Str = Func->getContext()->getStrDump(); 1139 Ostream &Str = Func->getContext()->getStrDump();
1118 Str << "APSR{n,z,v,c} = vmrs" << getPredicate() << "\t" 1140 Str << "APSR{n,z,v,c} = vmrs" << getPredicate() << "\t"
1119 "FPSCR{n,z,c,v}"; 1141 "FPSCR{n,z,c,v}";
1120 } 1142 }
1121 1143
1144 void InstARM32Vabs::emit(const Cfg *Func) const {
1145 if (!BuildDefs::dump())
1146 return;
1147 Ostream &Str = Func->getContext()->getStrEmit();
1148 assert(getSrcSize() == 1);
1149 Str << "\t"
1150 "vabs" << getPredicate() << getVecWidthString(getSrc(0)->getType())
1151 << "\t";
1152 getDest()->emit(Func);
1153 Str << ", ";
1154 getSrc(0)->emit(Func);
1155 }
1156
1157 void InstARM32Vabs::emitIAS(const Cfg *Func) const {
1158 assert(getSrcSize() == 1);
1159 (void)Func;
1160 llvm_unreachable("Not yet implemented");
1161 }
1162
1163 void InstARM32Vabs::dump(const Cfg *Func) const {
1164 if (!BuildDefs::dump())
1165 return;
1166 Ostream &Str = Func->getContext()->getStrDump();
1167 dumpDest(Func);
1168 Str << " = vabs" << getPredicate() << getVecWidthString(getSrc(0)->getType());
1169 }
1170
1122 void OperandARM32Mem::emit(const Cfg *Func) const { 1171 void OperandARM32Mem::emit(const Cfg *Func) const {
1123 if (!BuildDefs::dump()) 1172 if (!BuildDefs::dump())
1124 return; 1173 return;
1125 Ostream &Str = Func->getContext()->getStrEmit(); 1174 Ostream &Str = Func->getContext()->getStrEmit();
1126 Str << "["; 1175 Str << "[";
1127 getBase()->emit(Func); 1176 getBase()->emit(Func);
1128 switch (getAddrMode()) { 1177 switch (getAddrMode()) {
1129 case PostIndex: 1178 case PostIndex:
1130 case NegPostIndex: 1179 case NegPostIndex:
1131 Str << "], "; 1180 Str << "]";
1132 break; 1181 break;
1133 default: 1182 default:
1134 Str << ", ";
1135 break; 1183 break;
1136 } 1184 }
1137 if (isRegReg()) { 1185 if (isRegReg()) {
1186 Str << ", ";
1138 if (isNegAddrMode()) { 1187 if (isNegAddrMode()) {
1139 Str << "-"; 1188 Str << "-";
1140 } 1189 }
1141 getIndex()->emit(Func); 1190 getIndex()->emit(Func);
1142 if (getShiftOp() != kNoShift) { 1191 if (getShiftOp() != kNoShift) {
1143 Str << ", " << InstARM32ShiftAttributes[getShiftOp()].EmitString << " #" 1192 Str << ", " << InstARM32ShiftAttributes[getShiftOp()].EmitString << " #"
1144 << getShiftAmt(); 1193 << getShiftAmt();
1145 } 1194 }
1146 } else { 1195 } else {
1147 getOffset()->emit(Func); 1196 ConstantInteger32 *Offset = getOffset();
1197 if (Offset && Offset->getValue() != 0) {
1198 Str << ", ";
1199 Offset->emit(Func);
1200 }
1148 } 1201 }
1149 switch (getAddrMode()) { 1202 switch (getAddrMode()) {
1150 case Offset: 1203 case Offset:
1151 case NegOffset: 1204 case NegOffset:
1152 Str << "]"; 1205 Str << "]";
1153 break; 1206 break;
1154 case PreIndex: 1207 case PreIndex:
1155 case NegPreIndex: 1208 case NegPreIndex:
1156 Str << "]!"; 1209 Str << "]!";
1157 break; 1210 break;
(...skipping 70 matching lines...) Expand 10 before | Expand all | Expand 10 after
1228 if (getShiftOp() != kNoShift) { 1281 if (getShiftOp() != kNoShift) {
1229 Str << ", " << InstARM32ShiftAttributes[getShiftOp()].EmitString << " "; 1282 Str << ", " << InstARM32ShiftAttributes[getShiftOp()].EmitString << " ";
1230 if (Func) 1283 if (Func)
1231 getShiftAmt()->dump(Func); 1284 getShiftAmt()->dump(Func);
1232 else 1285 else
1233 getShiftAmt()->dump(Str); 1286 getShiftAmt()->dump(Str);
1234 } 1287 }
1235 } 1288 }
1236 1289
1237 } // end of namespace Ice 1290 } // end of namespace Ice
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698