106f32e7eSjoerg //===- HexagonMCInstLower.cpp - Convert Hexagon MachineInstr to an MCInst -===//
206f32e7eSjoerg //
306f32e7eSjoerg // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
406f32e7eSjoerg // See https://llvm.org/LICENSE.txt for license information.
506f32e7eSjoerg // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
606f32e7eSjoerg //
706f32e7eSjoerg //===----------------------------------------------------------------------===//
806f32e7eSjoerg //
906f32e7eSjoerg // This file contains code to lower Hexagon MachineInstrs to their corresponding
1006f32e7eSjoerg // MCInst records.
1106f32e7eSjoerg //
1206f32e7eSjoerg //===----------------------------------------------------------------------===//
1306f32e7eSjoerg 
1406f32e7eSjoerg #include "Hexagon.h"
1506f32e7eSjoerg #include "HexagonAsmPrinter.h"
1606f32e7eSjoerg #include "MCTargetDesc/HexagonMCExpr.h"
1706f32e7eSjoerg #include "MCTargetDesc/HexagonMCInstrInfo.h"
1806f32e7eSjoerg #include "MCTargetDesc/HexagonMCTargetDesc.h"
1906f32e7eSjoerg #include "llvm/ADT/APFloat.h"
2006f32e7eSjoerg #include "llvm/ADT/APInt.h"
2106f32e7eSjoerg #include "llvm/CodeGen/MachineBasicBlock.h"
2206f32e7eSjoerg #include "llvm/CodeGen/MachineInstr.h"
2306f32e7eSjoerg #include "llvm/CodeGen/MachineOperand.h"
2406f32e7eSjoerg #include "llvm/IR/Constants.h"
2506f32e7eSjoerg #include "llvm/MC/MCContext.h"
2606f32e7eSjoerg #include "llvm/MC/MCExpr.h"
2706f32e7eSjoerg #include "llvm/MC/MCInst.h"
2806f32e7eSjoerg #include "llvm/Support/ErrorHandling.h"
2906f32e7eSjoerg #include "llvm/Support/raw_ostream.h"
3006f32e7eSjoerg #include <cassert>
3106f32e7eSjoerg 
3206f32e7eSjoerg using namespace llvm;
3306f32e7eSjoerg 
3406f32e7eSjoerg namespace llvm {
3506f32e7eSjoerg 
3606f32e7eSjoerg void HexagonLowerToMC(const MCInstrInfo &MCII, const MachineInstr *MI,
3706f32e7eSjoerg                       MCInst &MCB, HexagonAsmPrinter &AP);
3806f32e7eSjoerg 
3906f32e7eSjoerg } // end namespace llvm
4006f32e7eSjoerg 
GetSymbolRef(const MachineOperand & MO,const MCSymbol * Symbol,HexagonAsmPrinter & Printer,bool MustExtend)4106f32e7eSjoerg static MCOperand GetSymbolRef(const MachineOperand &MO, const MCSymbol *Symbol,
4206f32e7eSjoerg                               HexagonAsmPrinter &Printer, bool MustExtend) {
4306f32e7eSjoerg   MCContext &MC = Printer.OutContext;
4406f32e7eSjoerg   const MCExpr *ME;
4506f32e7eSjoerg 
4606f32e7eSjoerg   // Populate the relocation type based on Hexagon target flags
4706f32e7eSjoerg   // set on an operand
4806f32e7eSjoerg   MCSymbolRefExpr::VariantKind RelocationType;
4906f32e7eSjoerg   switch (MO.getTargetFlags() & ~HexagonII::HMOTF_ConstExtended) {
5006f32e7eSjoerg   default:
5106f32e7eSjoerg     RelocationType = MCSymbolRefExpr::VK_None;
5206f32e7eSjoerg     break;
5306f32e7eSjoerg   case HexagonII::MO_PCREL:
54*da58b97aSjoerg     RelocationType = MCSymbolRefExpr::VK_PCREL;
5506f32e7eSjoerg     break;
5606f32e7eSjoerg   case HexagonII::MO_GOT:
5706f32e7eSjoerg     RelocationType = MCSymbolRefExpr::VK_GOT;
5806f32e7eSjoerg     break;
5906f32e7eSjoerg   case HexagonII::MO_LO16:
6006f32e7eSjoerg     RelocationType = MCSymbolRefExpr::VK_Hexagon_LO16;
6106f32e7eSjoerg     break;
6206f32e7eSjoerg   case HexagonII::MO_HI16:
6306f32e7eSjoerg     RelocationType = MCSymbolRefExpr::VK_Hexagon_HI16;
6406f32e7eSjoerg     break;
6506f32e7eSjoerg   case HexagonII::MO_GPREL:
6606f32e7eSjoerg     RelocationType = MCSymbolRefExpr::VK_Hexagon_GPREL;
6706f32e7eSjoerg     break;
6806f32e7eSjoerg   case HexagonII::MO_GDGOT:
6906f32e7eSjoerg     RelocationType = MCSymbolRefExpr::VK_Hexagon_GD_GOT;
7006f32e7eSjoerg     break;
7106f32e7eSjoerg   case HexagonII::MO_GDPLT:
7206f32e7eSjoerg     RelocationType = MCSymbolRefExpr::VK_Hexagon_GD_PLT;
7306f32e7eSjoerg     break;
7406f32e7eSjoerg   case HexagonII::MO_IE:
7506f32e7eSjoerg     RelocationType = MCSymbolRefExpr::VK_Hexagon_IE;
7606f32e7eSjoerg     break;
7706f32e7eSjoerg   case HexagonII::MO_IEGOT:
7806f32e7eSjoerg     RelocationType = MCSymbolRefExpr::VK_Hexagon_IE_GOT;
7906f32e7eSjoerg     break;
8006f32e7eSjoerg   case HexagonII::MO_TPREL:
8106f32e7eSjoerg     RelocationType = MCSymbolRefExpr::VK_TPREL;
8206f32e7eSjoerg     break;
8306f32e7eSjoerg   }
8406f32e7eSjoerg 
8506f32e7eSjoerg   ME = MCSymbolRefExpr::create(Symbol, RelocationType, MC);
8606f32e7eSjoerg 
8706f32e7eSjoerg   if (!MO.isJTI() && MO.getOffset())
8806f32e7eSjoerg     ME = MCBinaryExpr::createAdd(ME, MCConstantExpr::create(MO.getOffset(), MC),
8906f32e7eSjoerg                                  MC);
9006f32e7eSjoerg 
9106f32e7eSjoerg   ME = HexagonMCExpr::create(ME, MC);
9206f32e7eSjoerg   HexagonMCInstrInfo::setMustExtend(*ME, MustExtend);
9306f32e7eSjoerg   return MCOperand::createExpr(ME);
9406f32e7eSjoerg }
9506f32e7eSjoerg 
9606f32e7eSjoerg // Create an MCInst from a MachineInstr
HexagonLowerToMC(const MCInstrInfo & MCII,const MachineInstr * MI,MCInst & MCB,HexagonAsmPrinter & AP)9706f32e7eSjoerg void llvm::HexagonLowerToMC(const MCInstrInfo &MCII, const MachineInstr *MI,
9806f32e7eSjoerg                             MCInst &MCB, HexagonAsmPrinter &AP) {
9906f32e7eSjoerg   if (MI->getOpcode() == Hexagon::ENDLOOP0) {
10006f32e7eSjoerg     HexagonMCInstrInfo::setInnerLoop(MCB);
10106f32e7eSjoerg     return;
10206f32e7eSjoerg   }
10306f32e7eSjoerg   if (MI->getOpcode() == Hexagon::ENDLOOP1) {
10406f32e7eSjoerg     HexagonMCInstrInfo::setOuterLoop(MCB);
10506f32e7eSjoerg     return;
10606f32e7eSjoerg   }
107*da58b97aSjoerg   MCInst *MCI = AP.OutContext.createMCInst();
10806f32e7eSjoerg   MCI->setOpcode(MI->getOpcode());
10906f32e7eSjoerg   assert(MCI->getOpcode() == static_cast<unsigned>(MI->getOpcode()) &&
11006f32e7eSjoerg          "MCI opcode should have been set on construction");
11106f32e7eSjoerg 
11206f32e7eSjoerg   for (unsigned i = 0, e = MI->getNumOperands(); i < e; i++) {
11306f32e7eSjoerg     const MachineOperand &MO = MI->getOperand(i);
11406f32e7eSjoerg     MCOperand MCO;
11506f32e7eSjoerg     bool MustExtend = MO.getTargetFlags() & HexagonII::HMOTF_ConstExtended;
11606f32e7eSjoerg 
11706f32e7eSjoerg     switch (MO.getType()) {
11806f32e7eSjoerg     default:
11906f32e7eSjoerg       MI->print(errs());
12006f32e7eSjoerg       llvm_unreachable("unknown operand type");
12106f32e7eSjoerg     case MachineOperand::MO_RegisterMask:
12206f32e7eSjoerg       continue;
12306f32e7eSjoerg     case MachineOperand::MO_Register:
12406f32e7eSjoerg       // Ignore all implicit register operands.
12506f32e7eSjoerg       if (MO.isImplicit())
12606f32e7eSjoerg         continue;
12706f32e7eSjoerg       MCO = MCOperand::createReg(MO.getReg());
12806f32e7eSjoerg       break;
12906f32e7eSjoerg     case MachineOperand::MO_FPImmediate: {
13006f32e7eSjoerg       APFloat Val = MO.getFPImm()->getValueAPF();
13106f32e7eSjoerg       // FP immediates are used only when setting GPRs, so they may be dealt
13206f32e7eSjoerg       // with like regular immediates from this point on.
13306f32e7eSjoerg       auto Expr = HexagonMCExpr::create(
13406f32e7eSjoerg           MCConstantExpr::create(*Val.bitcastToAPInt().getRawData(),
13506f32e7eSjoerg                                  AP.OutContext),
13606f32e7eSjoerg           AP.OutContext);
13706f32e7eSjoerg       HexagonMCInstrInfo::setMustExtend(*Expr, MustExtend);
13806f32e7eSjoerg       MCO = MCOperand::createExpr(Expr);
13906f32e7eSjoerg       break;
14006f32e7eSjoerg     }
14106f32e7eSjoerg     case MachineOperand::MO_Immediate: {
14206f32e7eSjoerg       auto Expr = HexagonMCExpr::create(
14306f32e7eSjoerg           MCConstantExpr::create(MO.getImm(), AP.OutContext), AP.OutContext);
14406f32e7eSjoerg       HexagonMCInstrInfo::setMustExtend(*Expr, MustExtend);
14506f32e7eSjoerg       MCO = MCOperand::createExpr(Expr);
14606f32e7eSjoerg       break;
14706f32e7eSjoerg     }
14806f32e7eSjoerg     case MachineOperand::MO_MachineBasicBlock: {
14906f32e7eSjoerg       MCExpr const *Expr = MCSymbolRefExpr::create(MO.getMBB()->getSymbol(),
15006f32e7eSjoerg                                                    AP.OutContext);
15106f32e7eSjoerg       Expr = HexagonMCExpr::create(Expr, AP.OutContext);
15206f32e7eSjoerg       HexagonMCInstrInfo::setMustExtend(*Expr, MustExtend);
15306f32e7eSjoerg       MCO = MCOperand::createExpr(Expr);
15406f32e7eSjoerg       break;
15506f32e7eSjoerg     }
15606f32e7eSjoerg     case MachineOperand::MO_GlobalAddress:
15706f32e7eSjoerg       MCO = GetSymbolRef(MO, AP.getSymbol(MO.getGlobal()), AP, MustExtend);
15806f32e7eSjoerg       break;
15906f32e7eSjoerg     case MachineOperand::MO_ExternalSymbol:
16006f32e7eSjoerg       MCO = GetSymbolRef(MO, AP.GetExternalSymbolSymbol(MO.getSymbolName()),
16106f32e7eSjoerg                          AP, MustExtend);
16206f32e7eSjoerg       break;
16306f32e7eSjoerg     case MachineOperand::MO_JumpTableIndex:
16406f32e7eSjoerg       MCO = GetSymbolRef(MO, AP.GetJTISymbol(MO.getIndex()), AP, MustExtend);
16506f32e7eSjoerg       break;
16606f32e7eSjoerg     case MachineOperand::MO_ConstantPoolIndex:
16706f32e7eSjoerg       MCO = GetSymbolRef(MO, AP.GetCPISymbol(MO.getIndex()), AP, MustExtend);
16806f32e7eSjoerg       break;
16906f32e7eSjoerg     case MachineOperand::MO_BlockAddress:
17006f32e7eSjoerg       MCO = GetSymbolRef(MO, AP.GetBlockAddressSymbol(MO.getBlockAddress()), AP,
17106f32e7eSjoerg                          MustExtend);
17206f32e7eSjoerg       break;
17306f32e7eSjoerg     }
17406f32e7eSjoerg 
17506f32e7eSjoerg     MCI->addOperand(MCO);
17606f32e7eSjoerg   }
17706f32e7eSjoerg   AP.HexagonProcessInstruction(*MCI, *MI);
17806f32e7eSjoerg   HexagonMCInstrInfo::extendIfNeeded(AP.OutContext, MCII, MCB, *MCI);
17906f32e7eSjoerg   MCB.addOperand(MCOperand::createInst(MCI));
18006f32e7eSjoerg }
181