forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			223 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			223 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
| //===- ARMInstructionSelector.cpp ----------------------------*- C++ -*-==//
 | |
| //
 | |
| //                     The LLVM Compiler Infrastructure
 | |
| //
 | |
| // This file is distributed under the University of Illinois Open Source
 | |
| // License. See LICENSE.TXT for details.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| /// \file
 | |
| /// This file implements the targeting of the InstructionSelector class for ARM.
 | |
| /// \todo This should be generated by TableGen.
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #include "ARMInstructionSelector.h"
 | |
| #include "ARMRegisterBankInfo.h"
 | |
| #include "ARMSubtarget.h"
 | |
| #include "ARMTargetMachine.h"
 | |
| #include "llvm/CodeGen/MachineRegisterInfo.h"
 | |
| #include "llvm/Support/Debug.h"
 | |
| 
 | |
| #define DEBUG_TYPE "arm-isel"
 | |
| 
 | |
| using namespace llvm;
 | |
| 
 | |
| #ifndef LLVM_BUILD_GLOBAL_ISEL
 | |
| #error "You shouldn't build this"
 | |
| #endif
 | |
| 
 | |
| ARMInstructionSelector::ARMInstructionSelector(const ARMSubtarget &STI,
 | |
|                                                const ARMRegisterBankInfo &RBI)
 | |
|     : InstructionSelector(), TII(*STI.getInstrInfo()),
 | |
|       TRI(*STI.getRegisterInfo()), RBI(RBI) {}
 | |
| 
 | |
| static bool selectCopy(MachineInstr &I, const TargetInstrInfo &TII,
 | |
|                        MachineRegisterInfo &MRI, const TargetRegisterInfo &TRI,
 | |
|                        const RegisterBankInfo &RBI) {
 | |
|   unsigned DstReg = I.getOperand(0).getReg();
 | |
|   if (TargetRegisterInfo::isPhysicalRegister(DstReg))
 | |
|     return true;
 | |
| 
 | |
|   const RegisterBank *RegBank = RBI.getRegBank(DstReg, MRI, TRI);
 | |
|   (void)RegBank;
 | |
|   assert(RegBank && "Can't get reg bank for virtual register");
 | |
| 
 | |
|   const unsigned DstSize = MRI.getType(DstReg).getSizeInBits();
 | |
|   (void)DstSize;
 | |
|   unsigned SrcReg = I.getOperand(1).getReg();
 | |
|   const unsigned SrcSize = RBI.getSizeInBits(SrcReg, MRI, TRI);
 | |
|   (void)SrcSize;
 | |
|   assert((DstSize == SrcSize ||
 | |
|           // Copies are a means to setup initial types, the number of
 | |
|           // bits may not exactly match.
 | |
|           (TargetRegisterInfo::isPhysicalRegister(SrcReg) &&
 | |
|            DstSize <= SrcSize)) &&
 | |
|          "Copy with different width?!");
 | |
| 
 | |
|   assert((RegBank->getID() == ARM::GPRRegBankID ||
 | |
|           RegBank->getID() == ARM::FPRRegBankID) &&
 | |
|          "Unsupported reg bank");
 | |
| 
 | |
|   const TargetRegisterClass *RC = &ARM::GPRRegClass;
 | |
| 
 | |
|   if (RegBank->getID() == ARM::FPRRegBankID) {
 | |
|     assert(DstSize == 32 && "Only 32-bit FP values are supported");
 | |
|     RC = &ARM::SPRRegClass;
 | |
|   }
 | |
| 
 | |
|   // No need to constrain SrcReg. It will get constrained when
 | |
|   // we hit another of its uses or its defs.
 | |
|   // Copies do not have constraints.
 | |
|   if (!RBI.constrainGenericRegister(DstReg, *RC, MRI)) {
 | |
|     DEBUG(dbgs() << "Failed to constrain " << TII.getName(I.getOpcode())
 | |
|                  << " operand\n");
 | |
|     return false;
 | |
|   }
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| /// Select the opcode for simple extensions (that translate to a single SXT/UXT
 | |
| /// instruction). Extension operations more complicated than that should not
 | |
| /// invoke this.
 | |
| static unsigned selectSimpleExtOpc(unsigned Opc, unsigned Size) {
 | |
|   using namespace TargetOpcode;
 | |
| 
 | |
|   assert((Size == 8 || Size == 16) && "Unsupported size");
 | |
| 
 | |
|   if (Opc == G_SEXT)
 | |
|     return Size == 8 ? ARM::SXTB : ARM::SXTH;
 | |
| 
 | |
|   if (Opc == G_ZEXT)
 | |
|     return Size == 8 ? ARM::UXTB : ARM::UXTH;
 | |
| 
 | |
|   llvm_unreachable("Unsupported opcode");
 | |
| }
 | |
| 
 | |
| /// Select the opcode for simple loads. For types smaller than 32 bits, the
 | |
| /// value will be zero extended.
 | |
| static unsigned selectLoadOpCode(unsigned Size) {
 | |
|   switch (Size) {
 | |
|   case 1:
 | |
|   case 8:
 | |
|     return ARM::LDRBi12;
 | |
|   case 16:
 | |
|     return ARM::LDRH;
 | |
|   case 32:
 | |
|     return ARM::LDRi12;
 | |
|   }
 | |
| 
 | |
|   llvm_unreachable("Unsupported size");
 | |
| }
 | |
| 
 | |
| bool ARMInstructionSelector::select(MachineInstr &I) const {
 | |
|   assert(I.getParent() && "Instruction should be in a basic block!");
 | |
|   assert(I.getParent()->getParent() && "Instruction should be in a function!");
 | |
| 
 | |
|   auto &MBB = *I.getParent();
 | |
|   auto &MF = *MBB.getParent();
 | |
|   auto &MRI = MF.getRegInfo();
 | |
| 
 | |
|   if (!isPreISelGenericOpcode(I.getOpcode())) {
 | |
|     if (I.isCopy())
 | |
|       return selectCopy(I, TII, MRI, TRI, RBI);
 | |
| 
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|   MachineInstrBuilder MIB{MF, I};
 | |
|   bool isSExt = false;
 | |
| 
 | |
|   using namespace TargetOpcode;
 | |
|   switch (I.getOpcode()) {
 | |
|   case G_SEXT:
 | |
|     isSExt = true;
 | |
|     LLVM_FALLTHROUGH;
 | |
|   case G_ZEXT: {
 | |
|     LLT DstTy = MRI.getType(I.getOperand(0).getReg());
 | |
|     // FIXME: Smaller destination sizes coming soon!
 | |
|     if (DstTy.getSizeInBits() != 32) {
 | |
|       DEBUG(dbgs() << "Unsupported destination size for extension");
 | |
|       return false;
 | |
|     }
 | |
| 
 | |
|     LLT SrcTy = MRI.getType(I.getOperand(1).getReg());
 | |
|     unsigned SrcSize = SrcTy.getSizeInBits();
 | |
|     switch (SrcSize) {
 | |
|     case 1: {
 | |
|       // ZExt boils down to & 0x1; for SExt we also subtract that from 0
 | |
|       I.setDesc(TII.get(ARM::ANDri));
 | |
|       MIB.addImm(1).add(predOps(ARMCC::AL)).add(condCodeOp());
 | |
| 
 | |
|       if (isSExt) {
 | |
|         unsigned SExtResult = I.getOperand(0).getReg();
 | |
| 
 | |
|         // Use a new virtual register for the result of the AND
 | |
|         unsigned AndResult = MRI.createVirtualRegister(&ARM::GPRRegClass);
 | |
|         I.getOperand(0).setReg(AndResult);
 | |
| 
 | |
|         auto InsertBefore = std::next(I.getIterator());
 | |
|         auto SubI =
 | |
|             BuildMI(MBB, InsertBefore, I.getDebugLoc(), TII.get(ARM::RSBri))
 | |
|                 .addDef(SExtResult)
 | |
|                 .addUse(AndResult)
 | |
|                 .addImm(0)
 | |
|                 .add(predOps(ARMCC::AL))
 | |
|                 .add(condCodeOp());
 | |
|         if (!constrainSelectedInstRegOperands(*SubI, TII, TRI, RBI))
 | |
|           return false;
 | |
|       }
 | |
|       break;
 | |
|     }
 | |
|     case 8:
 | |
|     case 16: {
 | |
|       unsigned NewOpc = selectSimpleExtOpc(I.getOpcode(), SrcSize);
 | |
|       I.setDesc(TII.get(NewOpc));
 | |
|       MIB.addImm(0).add(predOps(ARMCC::AL));
 | |
|       break;
 | |
|     }
 | |
|     default:
 | |
|       DEBUG(dbgs() << "Unsupported source size for extension");
 | |
|       return false;
 | |
|     }
 | |
|     break;
 | |
|   }
 | |
|   case G_ADD:
 | |
|     I.setDesc(TII.get(ARM::ADDrr));
 | |
|     MIB.add(predOps(ARMCC::AL)).add(condCodeOp());
 | |
|     break;
 | |
|   case G_FADD:
 | |
|     if (!TII.getSubtarget().hasVFP2() ||
 | |
|         TII.getSubtarget().useNEONForSinglePrecisionFP())
 | |
|       return false;
 | |
|     I.setDesc(TII.get(ARM::VADDS));
 | |
|     MIB.add(predOps(ARMCC::AL));
 | |
|     break;
 | |
|   case G_FRAME_INDEX:
 | |
|     // Add 0 to the given frame index and hope it will eventually be folded into
 | |
|     // the user(s).
 | |
|     I.setDesc(TII.get(ARM::ADDri));
 | |
|     MIB.addImm(0).add(predOps(ARMCC::AL)).add(condCodeOp());
 | |
|     break;
 | |
|   case G_LOAD: {
 | |
|     LLT ValTy = MRI.getType(I.getOperand(0).getReg());
 | |
|     const auto ValSize = ValTy.getSizeInBits();
 | |
| 
 | |
|     if (ValSize != 32 && ValSize != 16 && ValSize != 8 && ValSize != 1)
 | |
|       return false;
 | |
| 
 | |
|     const auto NewOpc = selectLoadOpCode(ValSize);
 | |
|     I.setDesc(TII.get(NewOpc));
 | |
| 
 | |
|     if (NewOpc == ARM::LDRH)
 | |
|       // LDRH has a funny addressing mode (there's already a FIXME for it).
 | |
|       MIB.addReg(0);
 | |
|     MIB.addImm(0).add(predOps(ARMCC::AL));
 | |
|     break;
 | |
|   }
 | |
|   default:
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   return constrainSelectedInstRegOperands(I, TII, TRI, RBI);
 | |
| }
 |