OLD | NEW |
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 336 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
347 return true; | 347 return true; |
348 } | 348 } |
349 } | 349 } |
350 return false; | 350 return false; |
351 } | 351 } |
352 | 352 |
353 OperandARM32FlexFpImm::OperandARM32FlexFpImm(Cfg * /*Func*/, Type Ty, | 353 OperandARM32FlexFpImm::OperandARM32FlexFpImm(Cfg * /*Func*/, Type Ty, |
354 uint32_t ModifiedImm) | 354 uint32_t ModifiedImm) |
355 : OperandARM32Flex(kFlexFpImm, Ty), ModifiedImm(ModifiedImm) {} | 355 : OperandARM32Flex(kFlexFpImm, Ty), ModifiedImm(ModifiedImm) {} |
356 | 356 |
357 bool OperandARM32FlexFpImm::canHoldImm(Operand *C, uint32_t *ModifiedImm) { | 357 bool OperandARM32FlexFpImm::canHoldImm(const Operand *C, |
| 358 uint32_t *ModifiedImm) { |
358 switch (C->getType()) { | 359 switch (C->getType()) { |
359 default: | 360 default: |
360 llvm::report_fatal_error("Unhandled fp constant type."); | 361 llvm::report_fatal_error("Unhandled fp constant type."); |
361 case IceType_f32: { | 362 case IceType_f32: { |
362 // We violate llvm naming conventions a bit here so that the constants are | 363 // We violate llvm naming conventions a bit here so that the constants are |
363 // named after the bit fields they represent. See "A7.5.1 Operation of | 364 // named after the bit fields they represent. See "A7.5.1 Operation of |
364 // modified immediate constants, Floating-point" in the ARM ARM. | 365 // modified immediate constants, Floating-point" in the ARM ARM. |
365 static constexpr uint32_t a = 0x80000000u; | 366 static constexpr uint32_t a = 0x80000000u; |
366 static constexpr uint32_t B = 0x40000000; | 367 static constexpr uint32_t B = 0x40000000; |
367 static constexpr uint32_t bbbbb = 0x3E000000; | 368 static constexpr uint32_t bbbbb = 0x3E000000; |
368 static constexpr uint32_t cdefgh = 0x01F80000; | 369 static constexpr uint32_t cdefgh = 0x01F80000; |
369 static constexpr uint32_t AllowedBits = a | B | bbbbb | cdefgh; | 370 static constexpr uint32_t AllowedBits = a | B | bbbbb | cdefgh; |
370 static_assert(AllowedBits == 0xFFF80000u, | 371 static_assert(AllowedBits == 0xFFF80000u, |
371 "Invalid mask for f32 modified immediates."); | 372 "Invalid mask for f32 modified immediates."); |
372 const float F32 = llvm::cast<ConstantFloat>(C)->getValue(); | 373 const float F32 = llvm::cast<const ConstantFloat>(C)->getValue(); |
373 const uint32_t I32 = Utils::bitCopy<uint32_t>(F32); | 374 const uint32_t I32 = Utils::bitCopy<uint32_t>(F32); |
374 if (I32 & ~AllowedBits) { | 375 if (I32 & ~AllowedBits) { |
375 // constant has disallowed bits. | 376 // constant has disallowed bits. |
376 return false; | 377 return false; |
377 } | 378 } |
378 | 379 |
379 if ((I32 & bbbbb) != bbbbb && (I32 & bbbbb)) { | 380 if ((I32 & bbbbb) != bbbbb && (I32 & bbbbb)) { |
380 // not all bbbbb bits are 0 or 1. | 381 // not all bbbbb bits are 0 or 1. |
381 return false; | 382 return false; |
382 } | 383 } |
383 | 384 |
384 if (((I32 & B) != 0) == ((I32 & bbbbb) != 0)) { | 385 if (((I32 & B) != 0) == ((I32 & bbbbb) != 0)) { |
385 // B ^ b = 0; | 386 // B ^ b = 0; |
386 return false; | 387 return false; |
387 } | 388 } |
388 | 389 |
389 *ModifiedImm = ((I32 & a) ? 0x80 : 0x00) | ((I32 & bbbbb) ? 0x40 : 0x00) | | 390 *ModifiedImm = ((I32 & a) ? 0x80 : 0x00) | ((I32 & bbbbb) ? 0x40 : 0x00) | |
390 ((I32 & cdefgh) >> 19); | 391 ((I32 & cdefgh) >> 19); |
391 return true; | 392 return true; |
392 } | 393 } |
393 case IceType_f64: { | 394 case IceType_f64: { |
394 static constexpr uint32_t a = 0x80000000u; | 395 static constexpr uint32_t a = 0x80000000u; |
395 static constexpr uint32_t B = 0x40000000; | 396 static constexpr uint32_t B = 0x40000000; |
396 static constexpr uint32_t bbbbbbbb = 0x3FC00000; | 397 static constexpr uint32_t bbbbbbbb = 0x3FC00000; |
397 static constexpr uint32_t cdefgh = 0x003F0000; | 398 static constexpr uint32_t cdefgh = 0x003F0000; |
398 static constexpr uint32_t AllowedBits = a | B | bbbbbbbb | cdefgh; | 399 static constexpr uint32_t AllowedBits = a | B | bbbbbbbb | cdefgh; |
399 static_assert(AllowedBits == 0xFFFF0000u, | 400 static_assert(AllowedBits == 0xFFFF0000u, |
400 "Invalid mask for f64 modified immediates."); | 401 "Invalid mask for f64 modified immediates."); |
401 const double F64 = llvm::cast<ConstantDouble>(C)->getValue(); | 402 const double F64 = llvm::cast<const ConstantDouble>(C)->getValue(); |
402 const uint64_t I64 = Utils::bitCopy<uint64_t>(F64); | 403 const uint64_t I64 = Utils::bitCopy<uint64_t>(F64); |
403 if (I64 & 0xFFFFFFFFu) { | 404 if (I64 & 0xFFFFFFFFu) { |
404 // constant has disallowed bits. | 405 // constant has disallowed bits. |
405 return false; | 406 return false; |
406 } | 407 } |
407 const uint32_t I32 = I64 >> 32; | 408 const uint32_t I32 = I64 >> 32; |
408 | 409 |
409 if (I32 & ~AllowedBits) { | 410 if (I32 & ~AllowedBits) { |
410 // constant has disallowed bits. | 411 // constant has disallowed bits. |
411 return false; | 412 return false; |
(...skipping 1265 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1677 // emitted text size. | 1678 // emitted text size. |
1678 if (auto *Asm = Func->getAssembler<ARM32::AssemblerARM32>()) | 1679 if (auto *Asm = Func->getAssembler<ARM32::AssemblerARM32>()) |
1679 Asm->decEmitTextSize(InstSize); | 1680 Asm->decEmitTextSize(InstSize); |
1680 Ostream &Str = Func->getContext()->getStrEmit(); | 1681 Ostream &Str = Func->getContext()->getStrEmit(); |
1681 Str << getName(Func) << ":"; | 1682 Str << getName(Func) << ":"; |
1682 } | 1683 } |
1683 | 1684 |
1684 void InstARM32Label::emitIAS(const Cfg *Func) const { | 1685 void InstARM32Label::emitIAS(const Cfg *Func) const { |
1685 auto *Asm = Func->getAssembler<ARM32::AssemblerARM32>(); | 1686 auto *Asm = Func->getAssembler<ARM32::AssemblerARM32>(); |
1686 Asm->bindLocalLabel(Func, this, Number); | 1687 Asm->bindLocalLabel(Func, this, Number); |
| 1688 if (OffsetReloc != nullptr) { |
| 1689 Asm->bindRelocOffset(OffsetReloc); |
| 1690 } |
1687 if (Asm->needsTextFixup()) | 1691 if (Asm->needsTextFixup()) |
1688 emitUsingTextFixup(Func); | 1692 emitUsingTextFixup(Func); |
1689 } | 1693 } |
1690 | 1694 |
1691 void InstARM32Label::dump(const Cfg *Func) const { | 1695 void InstARM32Label::dump(const Cfg *Func) const { |
1692 if (!BuildDefs::dump()) | 1696 if (!BuildDefs::dump()) |
1693 return; | 1697 return; |
1694 Ostream &Str = Func->getContext()->getStrDump(); | 1698 Ostream &Str = Func->getContext()->getStrDump(); |
1695 Str << getName(Func) << ":"; | 1699 Str << getName(Func) << ":"; |
1696 } | 1700 } |
(...skipping 119 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1816 return; | 1820 return; |
1817 Ostream &Str = Func->getContext()->getStrEmit(); | 1821 Ostream &Str = Func->getContext()->getStrEmit(); |
1818 assert(getSrcSize() == 1); | 1822 assert(getSrcSize() == 1); |
1819 Str << "\t" << Opcode << getPredicate() << "\t"; | 1823 Str << "\t" << Opcode << getPredicate() << "\t"; |
1820 getDest()->emit(Func); | 1824 getDest()->emit(Func); |
1821 Str << ", "; | 1825 Str << ", "; |
1822 auto *Src0 = llvm::cast<Constant>(getSrc(0)); | 1826 auto *Src0 = llvm::cast<Constant>(getSrc(0)); |
1823 if (auto *CR = llvm::dyn_cast<ConstantRelocatable>(Src0)) { | 1827 if (auto *CR = llvm::dyn_cast<ConstantRelocatable>(Src0)) { |
1824 Str << "#:lower16:"; | 1828 Str << "#:lower16:"; |
1825 CR->emitWithoutPrefix(Func->getTarget()); | 1829 CR->emitWithoutPrefix(Func->getTarget()); |
| 1830 if (Func->getContext()->getFlags().getUseNonsfi()) { |
| 1831 Str << " - ."; |
| 1832 } |
1826 } else { | 1833 } else { |
1827 Src0->emit(Func); | 1834 Src0->emit(Func); |
1828 } | 1835 } |
1829 } | 1836 } |
1830 | 1837 |
1831 template <> void InstARM32Movw::emitIAS(const Cfg *Func) const { | 1838 template <> void InstARM32Movw::emitIAS(const Cfg *Func) const { |
1832 assert(getSrcSize() == 1); | 1839 assert(getSrcSize() == 1); |
1833 auto *Asm = Func->getAssembler<ARM32::AssemblerARM32>(); | 1840 auto *Asm = Func->getAssembler<ARM32::AssemblerARM32>(); |
1834 Asm->movw(getDest(), getSrc(0), getPredicate()); | 1841 Asm->movw(getDest(), getSrc(0), getPredicate()); |
1835 if (Asm->needsTextFixup()) | 1842 if (Asm->needsTextFixup()) |
1836 emitUsingTextFixup(Func); | 1843 emitUsingTextFixup(Func); |
1837 } | 1844 } |
1838 | 1845 |
1839 template <> void InstARM32Movt::emit(const Cfg *Func) const { | 1846 template <> void InstARM32Movt::emit(const Cfg *Func) const { |
1840 if (!BuildDefs::dump()) | 1847 if (!BuildDefs::dump()) |
1841 return; | 1848 return; |
1842 Ostream &Str = Func->getContext()->getStrEmit(); | 1849 Ostream &Str = Func->getContext()->getStrEmit(); |
1843 assert(getSrcSize() == 2); | 1850 assert(getSrcSize() == 2); |
1844 Variable *Dest = getDest(); | 1851 Variable *Dest = getDest(); |
1845 auto *Src1 = llvm::cast<Constant>(getSrc(1)); | 1852 auto *Src1 = llvm::cast<Constant>(getSrc(1)); |
1846 Str << "\t" << Opcode << getPredicate() << "\t"; | 1853 Str << "\t" << Opcode << getPredicate() << "\t"; |
1847 Dest->emit(Func); | 1854 Dest->emit(Func); |
1848 Str << ", "; | 1855 Str << ", "; |
1849 if (auto *CR = llvm::dyn_cast<ConstantRelocatable>(Src1)) { | 1856 if (auto *CR = llvm::dyn_cast<ConstantRelocatable>(Src1)) { |
1850 Str << "#:upper16:"; | 1857 Str << "#:upper16:"; |
1851 CR->emitWithoutPrefix(Func->getTarget()); | 1858 CR->emitWithoutPrefix(Func->getTarget()); |
| 1859 if (Func->getContext()->getFlags().getUseNonsfi()) { |
| 1860 Str << " - ."; |
| 1861 } |
1852 } else { | 1862 } else { |
1853 Src1->emit(Func); | 1863 Src1->emit(Func); |
1854 } | 1864 } |
1855 } | 1865 } |
1856 | 1866 |
1857 template <> void InstARM32Movt::emitIAS(const Cfg *Func) const { | 1867 template <> void InstARM32Movt::emitIAS(const Cfg *Func) const { |
1858 assert(getSrcSize() == 2); | 1868 assert(getSrcSize() == 2); |
1859 auto *Asm = Func->getAssembler<ARM32::AssemblerARM32>(); | 1869 auto *Asm = Func->getAssembler<ARM32::AssemblerARM32>(); |
1860 Asm->movt(getDest(), getSrc(1), getPredicate()); | 1870 Asm->movt(getDest(), getSrc(1), getPredicate()); |
1861 if (Asm->needsTextFixup()) | 1871 if (Asm->needsTextFixup()) |
(...skipping 850 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2712 | 2722 |
2713 template class InstARM32FourAddrGPR<InstARM32::Mla>; | 2723 template class InstARM32FourAddrGPR<InstARM32::Mla>; |
2714 template class InstARM32FourAddrGPR<InstARM32::Mls>; | 2724 template class InstARM32FourAddrGPR<InstARM32::Mls>; |
2715 | 2725 |
2716 template class InstARM32CmpLike<InstARM32::Cmn>; | 2726 template class InstARM32CmpLike<InstARM32::Cmn>; |
2717 template class InstARM32CmpLike<InstARM32::Cmp>; | 2727 template class InstARM32CmpLike<InstARM32::Cmp>; |
2718 template class InstARM32CmpLike<InstARM32::Tst>; | 2728 template class InstARM32CmpLike<InstARM32::Tst>; |
2719 | 2729 |
2720 } // end of namespace ARM32 | 2730 } // end of namespace ARM32 |
2721 } // end of namespace Ice | 2731 } // end of namespace Ice |
OLD | NEW |