109467b48Spatrick //===- HexagonInstPrinter.cpp - Convert Hexagon MCInst to assembly syntax -===//
209467b48Spatrick //
309467b48Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
409467b48Spatrick // See https://llvm.org/LICENSE.txt for license information.
509467b48Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
609467b48Spatrick //
709467b48Spatrick //===----------------------------------------------------------------------===//
809467b48Spatrick //
909467b48Spatrick // This class prints an Hexagon MCInst to a .s file.
1009467b48Spatrick //
1109467b48Spatrick //===----------------------------------------------------------------------===//
1209467b48Spatrick 
1309467b48Spatrick #include "HexagonInstPrinter.h"
1409467b48Spatrick #include "MCTargetDesc/HexagonBaseInfo.h"
1509467b48Spatrick #include "MCTargetDesc/HexagonMCInstrInfo.h"
1609467b48Spatrick #include "llvm/MC/MCAsmInfo.h"
1709467b48Spatrick #include "llvm/MC/MCExpr.h"
1809467b48Spatrick #include "llvm/MC/MCInst.h"
1909467b48Spatrick #include "llvm/Support/Debug.h"
2009467b48Spatrick #include "llvm/Support/raw_ostream.h"
2109467b48Spatrick 
2209467b48Spatrick using namespace llvm;
2309467b48Spatrick 
2409467b48Spatrick #define DEBUG_TYPE "asm-printer"
2509467b48Spatrick 
2609467b48Spatrick #define GET_INSTRUCTION_NAME
2709467b48Spatrick #include "HexagonGenAsmWriter.inc"
2809467b48Spatrick 
printRegName(raw_ostream & O,MCRegister Reg) const29*d415bd75Srobert void HexagonInstPrinter::printRegName(raw_ostream &O, MCRegister Reg) const {
30*d415bd75Srobert   O << getRegisterName(Reg);
3109467b48Spatrick }
3209467b48Spatrick 
printInst(const MCInst * MI,uint64_t Address,StringRef Annot,const MCSubtargetInfo & STI,raw_ostream & OS)3309467b48Spatrick void HexagonInstPrinter::printInst(const MCInst *MI, uint64_t Address,
3409467b48Spatrick                                    StringRef Annot, const MCSubtargetInfo &STI,
3509467b48Spatrick                                    raw_ostream &OS) {
3609467b48Spatrick   assert(HexagonMCInstrInfo::isBundle(*MI));
3709467b48Spatrick   assert(HexagonMCInstrInfo::bundleSize(*MI) <= HEXAGON_PACKET_SIZE);
3809467b48Spatrick   assert(HexagonMCInstrInfo::bundleSize(*MI) > 0);
3909467b48Spatrick   HasExtender = false;
4009467b48Spatrick   for (auto const &I : HexagonMCInstrInfo::bundleInstructions(*MI)) {
4109467b48Spatrick     MCInst const &MCI = *I.getInst();
4209467b48Spatrick     if (HexagonMCInstrInfo::isDuplex(MII, MCI)) {
4309467b48Spatrick       printInstruction(MCI.getOperand(1).getInst(), Address, OS);
4409467b48Spatrick       OS << '\v';
4509467b48Spatrick       HasExtender = false;
4609467b48Spatrick       printInstruction(MCI.getOperand(0).getInst(), Address, OS);
4709467b48Spatrick     } else
4809467b48Spatrick       printInstruction(&MCI, Address, OS);
4909467b48Spatrick     HasExtender = HexagonMCInstrInfo::isImmext(MCI);
5009467b48Spatrick     OS << "\n";
5109467b48Spatrick   }
5209467b48Spatrick 
5309467b48Spatrick   bool IsLoop0 = HexagonMCInstrInfo::isInnerLoop(*MI);
5409467b48Spatrick   bool IsLoop1 = HexagonMCInstrInfo::isOuterLoop(*MI);
5509467b48Spatrick   if (IsLoop0) {
5609467b48Spatrick     OS << (IsLoop1 ? " :endloop01" : " :endloop0");
5709467b48Spatrick   } else if (IsLoop1) {
5809467b48Spatrick     OS << " :endloop1";
5909467b48Spatrick   }
6009467b48Spatrick }
6109467b48Spatrick 
printOperand(MCInst const * MI,unsigned OpNo,raw_ostream & O) const6209467b48Spatrick void HexagonInstPrinter::printOperand(MCInst const *MI, unsigned OpNo,
6309467b48Spatrick                                       raw_ostream &O) const {
6409467b48Spatrick   if (HexagonMCInstrInfo::getExtendableOp(MII, *MI) == OpNo &&
6509467b48Spatrick       (HasExtender || HexagonMCInstrInfo::isConstExtended(MII, *MI)))
6609467b48Spatrick     O << "#";
6709467b48Spatrick   MCOperand const &MO = MI->getOperand(OpNo);
6809467b48Spatrick   if (MO.isReg()) {
6909467b48Spatrick     O << getRegisterName(MO.getReg());
7009467b48Spatrick   } else if (MO.isExpr()) {
7109467b48Spatrick     int64_t Value;
7209467b48Spatrick     if (MO.getExpr()->evaluateAsAbsolute(Value))
7309467b48Spatrick       O << formatImm(Value);
7409467b48Spatrick     else
7509467b48Spatrick       O << *MO.getExpr();
7609467b48Spatrick   } else {
7709467b48Spatrick     llvm_unreachable("Unknown operand");
7809467b48Spatrick   }
7909467b48Spatrick }
8009467b48Spatrick 
printBrtarget(MCInst const * MI,unsigned OpNo,raw_ostream & O) const8109467b48Spatrick void HexagonInstPrinter::printBrtarget(MCInst const *MI, unsigned OpNo,
8209467b48Spatrick                                        raw_ostream &O) const {
8309467b48Spatrick   MCOperand const &MO = MI->getOperand(OpNo);
8409467b48Spatrick   assert (MO.isExpr());
8509467b48Spatrick   MCExpr const &Expr = *MO.getExpr();
8609467b48Spatrick   int64_t Value;
8709467b48Spatrick   if (Expr.evaluateAsAbsolute(Value))
8809467b48Spatrick     O << format("0x%" PRIx64, Value);
8909467b48Spatrick   else {
9009467b48Spatrick     if (HasExtender || HexagonMCInstrInfo::isConstExtended(MII, *MI))
9109467b48Spatrick       if (HexagonMCInstrInfo::getExtendableOp(MII, *MI) == OpNo)
9209467b48Spatrick         O << "##";
9309467b48Spatrick     O << Expr;
9409467b48Spatrick   }
9509467b48Spatrick }
96