96 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			96 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			C++
		
	
	
	
| //===- HexagonInstPrinter.cpp - Convert Hexagon MCInst to assembly syntax -===//
 | |
| //
 | |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
 | |
| // See https://llvm.org/LICENSE.txt for license information.
 | |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| //
 | |
| // This class prints an Hexagon MCInst to a .s file.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #include "HexagonInstPrinter.h"
 | |
| #include "MCTargetDesc/HexagonBaseInfo.h"
 | |
| #include "MCTargetDesc/HexagonMCInstrInfo.h"
 | |
| #include "llvm/MC/MCAsmInfo.h"
 | |
| #include "llvm/MC/MCExpr.h"
 | |
| #include "llvm/MC/MCInst.h"
 | |
| #include "llvm/Support/Debug.h"
 | |
| #include "llvm/Support/raw_ostream.h"
 | |
| 
 | |
| using namespace llvm;
 | |
| 
 | |
| #define DEBUG_TYPE "asm-printer"
 | |
| 
 | |
| #define GET_INSTRUCTION_NAME
 | |
| #include "HexagonGenAsmWriter.inc"
 | |
| 
 | |
| void HexagonInstPrinter::printRegName(raw_ostream &O, unsigned RegNo) const {
 | |
|   O << getRegisterName(RegNo);
 | |
| }
 | |
| 
 | |
| void HexagonInstPrinter::printInst(const MCInst *MI, uint64_t Address,
 | |
|                                    StringRef Annot, const MCSubtargetInfo &STI,
 | |
|                                    raw_ostream &OS) {
 | |
|   assert(HexagonMCInstrInfo::isBundle(*MI));
 | |
|   assert(HexagonMCInstrInfo::bundleSize(*MI) <= HEXAGON_PACKET_SIZE);
 | |
|   assert(HexagonMCInstrInfo::bundleSize(*MI) > 0);
 | |
|   HasExtender = false;
 | |
|   for (auto const &I : HexagonMCInstrInfo::bundleInstructions(*MI)) {
 | |
|     MCInst const &MCI = *I.getInst();
 | |
|     if (HexagonMCInstrInfo::isDuplex(MII, MCI)) {
 | |
|       printInstruction(MCI.getOperand(1).getInst(), Address, OS);
 | |
|       OS << '\v';
 | |
|       HasExtender = false;
 | |
|       printInstruction(MCI.getOperand(0).getInst(), Address, OS);
 | |
|     } else
 | |
|       printInstruction(&MCI, Address, OS);
 | |
|     HasExtender = HexagonMCInstrInfo::isImmext(MCI);
 | |
|     OS << "\n";
 | |
|   }
 | |
| 
 | |
|   bool IsLoop0 = HexagonMCInstrInfo::isInnerLoop(*MI);
 | |
|   bool IsLoop1 = HexagonMCInstrInfo::isOuterLoop(*MI);
 | |
|   if (IsLoop0) {
 | |
|     OS << (IsLoop1 ? " :endloop01" : " :endloop0");
 | |
|   } else if (IsLoop1) {
 | |
|     OS << " :endloop1";
 | |
|   }
 | |
| }
 | |
| 
 | |
| void HexagonInstPrinter::printOperand(MCInst const *MI, unsigned OpNo,
 | |
|                                       raw_ostream &O) const {
 | |
|   if (HexagonMCInstrInfo::getExtendableOp(MII, *MI) == OpNo &&
 | |
|       (HasExtender || HexagonMCInstrInfo::isConstExtended(MII, *MI)))
 | |
|     O << "#";
 | |
|   MCOperand const &MO = MI->getOperand(OpNo);
 | |
|   if (MO.isReg()) {
 | |
|     O << getRegisterName(MO.getReg());
 | |
|   } else if (MO.isExpr()) {
 | |
|     int64_t Value;
 | |
|     if (MO.getExpr()->evaluateAsAbsolute(Value))
 | |
|       O << formatImm(Value);
 | |
|     else
 | |
|       O << *MO.getExpr();
 | |
|   } else {
 | |
|     llvm_unreachable("Unknown operand");
 | |
|   }
 | |
| }
 | |
| 
 | |
| void HexagonInstPrinter::printBrtarget(MCInst const *MI, unsigned OpNo,
 | |
|                                        raw_ostream &O) const {
 | |
|   MCOperand const &MO = MI->getOperand(OpNo);
 | |
|   assert (MO.isExpr());
 | |
|   MCExpr const &Expr = *MO.getExpr();
 | |
|   int64_t Value;
 | |
|   if (Expr.evaluateAsAbsolute(Value))
 | |
|     O << format("0x%" PRIx64, Value);
 | |
|   else {
 | |
|     if (HasExtender || HexagonMCInstrInfo::isConstExtended(MII, *MI))
 | |
|       if (HexagonMCInstrInfo::getExtendableOp(MII, *MI) == OpNo)
 | |
|         O << "##";
 | |
|     O << Expr;
 | |
|   }
 | |
| }
 |