xref: /freebsd-src/contrib/llvm-project/llvm/lib/Target/CSKY/CSKYAsmPrinter.cpp (revision bdd1243df58e60e85101c09001d9812a789b6bc4)
1349cc55cSDimitry Andric //===-- CSKYAsmPrinter.cpp - CSKY LLVM assembly writer --------------------===//
2349cc55cSDimitry Andric //
3349cc55cSDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4349cc55cSDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
5349cc55cSDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6349cc55cSDimitry Andric //
7349cc55cSDimitry Andric //===----------------------------------------------------------------------===//
8349cc55cSDimitry Andric //
9349cc55cSDimitry Andric // This file contains a printer that converts from our internal representation
10349cc55cSDimitry Andric // of machine-dependent LLVM code to the CSKY assembly language.
11349cc55cSDimitry Andric //
12349cc55cSDimitry Andric //===----------------------------------------------------------------------===//
13349cc55cSDimitry Andric #include "CSKYAsmPrinter.h"
14349cc55cSDimitry Andric #include "CSKY.h"
1504eeddc0SDimitry Andric #include "CSKYConstantPoolValue.h"
16349cc55cSDimitry Andric #include "CSKYTargetMachine.h"
17349cc55cSDimitry Andric #include "MCTargetDesc/CSKYInstPrinter.h"
18349cc55cSDimitry Andric #include "MCTargetDesc/CSKYMCExpr.h"
1981ad6265SDimitry Andric #include "MCTargetDesc/CSKYTargetStreamer.h"
20349cc55cSDimitry Andric #include "TargetInfo/CSKYTargetInfo.h"
21349cc55cSDimitry Andric #include "llvm/ADT/Statistic.h"
22349cc55cSDimitry Andric #include "llvm/CodeGen/AsmPrinter.h"
23349cc55cSDimitry Andric #include "llvm/CodeGen/MachineConstantPool.h"
2481ad6265SDimitry Andric #include "llvm/CodeGen/MachineFrameInfo.h"
25349cc55cSDimitry Andric #include "llvm/IR/DataLayout.h"
26349cc55cSDimitry Andric #include "llvm/MC/MCAsmInfo.h"
27349cc55cSDimitry Andric #include "llvm/MC/MCContext.h"
28349cc55cSDimitry Andric #include "llvm/MC/MCInstBuilder.h"
29349cc55cSDimitry Andric #include "llvm/MC/MCStreamer.h"
30349cc55cSDimitry Andric #include "llvm/MC/TargetRegistry.h"
31349cc55cSDimitry Andric 
32349cc55cSDimitry Andric using namespace llvm;
33349cc55cSDimitry Andric 
34349cc55cSDimitry Andric #define DEBUG_TYPE "csky-asm-printer"
35349cc55cSDimitry Andric 
360eae32dcSDimitry Andric STATISTIC(CSKYNumInstrsCompressed,
370eae32dcSDimitry Andric           "Number of C-SKY Compressed instructions emitted");
380eae32dcSDimitry Andric 
CSKYAsmPrinter(llvm::TargetMachine & TM,std::unique_ptr<llvm::MCStreamer> Streamer)39349cc55cSDimitry Andric CSKYAsmPrinter::CSKYAsmPrinter(llvm::TargetMachine &TM,
40349cc55cSDimitry Andric                                std::unique_ptr<llvm::MCStreamer> Streamer)
41349cc55cSDimitry Andric     : AsmPrinter(TM, std::move(Streamer)), MCInstLowering(OutContext, *this) {}
42349cc55cSDimitry Andric 
runOnMachineFunction(MachineFunction & MF)43349cc55cSDimitry Andric bool CSKYAsmPrinter::runOnMachineFunction(MachineFunction &MF) {
4404eeddc0SDimitry Andric   MCP = MF.getConstantPool();
4581ad6265SDimitry Andric   TII = MF.getSubtarget().getInstrInfo();
4681ad6265SDimitry Andric 
4781ad6265SDimitry Andric   // Set the current MCSubtargetInfo to a copy which has the correct
4881ad6265SDimitry Andric   // feature bits for the current MachineFunction
4981ad6265SDimitry Andric   MCSubtargetInfo &NewSTI =
5081ad6265SDimitry Andric       OutStreamer->getContext().getSubtargetCopy(*TM.getMCSubtargetInfo());
5181ad6265SDimitry Andric   NewSTI.setFeatureBits(MF.getSubtarget().getFeatureBits());
5281ad6265SDimitry Andric   Subtarget = &NewSTI;
5381ad6265SDimitry Andric 
54349cc55cSDimitry Andric   return AsmPrinter::runOnMachineFunction(MF);
55349cc55cSDimitry Andric }
56349cc55cSDimitry Andric 
570eae32dcSDimitry Andric #define GEN_COMPRESS_INSTR
580eae32dcSDimitry Andric #include "CSKYGenCompressInstEmitter.inc"
EmitToStreamer(MCStreamer & S,const MCInst & Inst)590eae32dcSDimitry Andric void CSKYAsmPrinter::EmitToStreamer(MCStreamer &S, const MCInst &Inst) {
600eae32dcSDimitry Andric   MCInst CInst;
61*bdd1243dSDimitry Andric   bool Res = compressInst(CInst, Inst, *Subtarget);
620eae32dcSDimitry Andric   if (Res)
630eae32dcSDimitry Andric     ++CSKYNumInstrsCompressed;
640eae32dcSDimitry Andric   AsmPrinter::EmitToStreamer(*OutStreamer, Res ? CInst : Inst);
650eae32dcSDimitry Andric }
660eae32dcSDimitry Andric 
67349cc55cSDimitry Andric // Simple pseudo-instructions have their lowering (with expansion to real
68349cc55cSDimitry Andric // instructions) auto-generated.
69349cc55cSDimitry Andric #include "CSKYGenMCPseudoLowering.inc"
70349cc55cSDimitry Andric 
expandTLSLA(const MachineInstr * MI)7104eeddc0SDimitry Andric void CSKYAsmPrinter::expandTLSLA(const MachineInstr *MI) {
7204eeddc0SDimitry Andric   DebugLoc DL = MI->getDebugLoc();
7304eeddc0SDimitry Andric 
7404eeddc0SDimitry Andric   MCSymbol *PCLabel = OutContext.getOrCreateSymbol(
7504eeddc0SDimitry Andric       Twine(MAI->getPrivateGlobalPrefix()) + "PC" + Twine(getFunctionNumber()) +
7604eeddc0SDimitry Andric       "_" + Twine(MI->getOperand(3).getImm()));
7704eeddc0SDimitry Andric 
7804eeddc0SDimitry Andric   OutStreamer->emitLabel(PCLabel);
7904eeddc0SDimitry Andric 
8004eeddc0SDimitry Andric   auto Instr = BuildMI(*MF, DL, TII->get(CSKY::LRW32))
8104eeddc0SDimitry Andric                    .add(MI->getOperand(0))
8204eeddc0SDimitry Andric                    .add(MI->getOperand(2));
8304eeddc0SDimitry Andric   MCInst LRWInst;
8404eeddc0SDimitry Andric   MCInstLowering.Lower(Instr, LRWInst);
8504eeddc0SDimitry Andric   EmitToStreamer(*OutStreamer, LRWInst);
8604eeddc0SDimitry Andric 
8704eeddc0SDimitry Andric   Instr = BuildMI(*MF, DL, TII->get(CSKY::GRS32))
8804eeddc0SDimitry Andric               .add(MI->getOperand(1))
8904eeddc0SDimitry Andric               .addSym(PCLabel);
9004eeddc0SDimitry Andric   MCInst GRSInst;
9104eeddc0SDimitry Andric   MCInstLowering.Lower(Instr, GRSInst);
9204eeddc0SDimitry Andric   EmitToStreamer(*OutStreamer, GRSInst);
9304eeddc0SDimitry Andric   return;
9404eeddc0SDimitry Andric }
9504eeddc0SDimitry Andric 
emitCustomConstantPool(const MachineInstr * MI)9604eeddc0SDimitry Andric void CSKYAsmPrinter::emitCustomConstantPool(const MachineInstr *MI) {
9704eeddc0SDimitry Andric 
9804eeddc0SDimitry Andric   // This instruction represents a floating constant pool in the function.
9904eeddc0SDimitry Andric   // The first operand is the ID# for this instruction, the second is the
10004eeddc0SDimitry Andric   // index into the MachineConstantPool that this is, the third is the size
10104eeddc0SDimitry Andric   // in bytes of this constant pool entry.
10204eeddc0SDimitry Andric   // The required alignment is specified on the basic block holding this MI.
10304eeddc0SDimitry Andric   unsigned LabelId = (unsigned)MI->getOperand(0).getImm();
10404eeddc0SDimitry Andric   unsigned CPIdx = (unsigned)MI->getOperand(1).getIndex();
10504eeddc0SDimitry Andric 
10604eeddc0SDimitry Andric   // If this is the first entry of the pool, mark it.
10704eeddc0SDimitry Andric   if (!InConstantPool) {
108*bdd1243dSDimitry Andric     OutStreamer->emitValueToAlignment(Align(4));
10904eeddc0SDimitry Andric     InConstantPool = true;
11004eeddc0SDimitry Andric   }
11104eeddc0SDimitry Andric 
11204eeddc0SDimitry Andric   OutStreamer->emitLabel(GetCPISymbol(LabelId));
11304eeddc0SDimitry Andric 
11404eeddc0SDimitry Andric   const MachineConstantPoolEntry &MCPE = MCP->getConstants()[CPIdx];
11504eeddc0SDimitry Andric   if (MCPE.isMachineConstantPoolEntry())
11604eeddc0SDimitry Andric     emitMachineConstantPoolValue(MCPE.Val.MachineCPVal);
11704eeddc0SDimitry Andric   else
11804eeddc0SDimitry Andric     emitGlobalConstant(MF->getDataLayout(), MCPE.Val.ConstVal);
11904eeddc0SDimitry Andric   return;
12004eeddc0SDimitry Andric }
12104eeddc0SDimitry Andric 
emitFunctionBodyEnd()12204eeddc0SDimitry Andric void CSKYAsmPrinter::emitFunctionBodyEnd() {
12304eeddc0SDimitry Andric   // Make sure to terminate any constant pools that were at the end
12404eeddc0SDimitry Andric   // of the function.
12504eeddc0SDimitry Andric   if (!InConstantPool)
12604eeddc0SDimitry Andric     return;
12704eeddc0SDimitry Andric   InConstantPool = false;
12804eeddc0SDimitry Andric }
12904eeddc0SDimitry Andric 
emitStartOfAsmFile(Module & M)13081ad6265SDimitry Andric void CSKYAsmPrinter::emitStartOfAsmFile(Module &M) {
13181ad6265SDimitry Andric   if (TM.getTargetTriple().isOSBinFormatELF())
13281ad6265SDimitry Andric     emitAttributes();
13381ad6265SDimitry Andric }
13481ad6265SDimitry Andric 
emitEndOfAsmFile(Module & M)13581ad6265SDimitry Andric void CSKYAsmPrinter::emitEndOfAsmFile(Module &M) {
13681ad6265SDimitry Andric   CSKYTargetStreamer &CTS =
13781ad6265SDimitry Andric       static_cast<CSKYTargetStreamer &>(*OutStreamer->getTargetStreamer());
13881ad6265SDimitry Andric 
13981ad6265SDimitry Andric   if (TM.getTargetTriple().isOSBinFormatELF())
14081ad6265SDimitry Andric     CTS.finishAttributeSection();
14181ad6265SDimitry Andric }
14281ad6265SDimitry Andric 
emitInstruction(const MachineInstr * MI)143349cc55cSDimitry Andric void CSKYAsmPrinter::emitInstruction(const MachineInstr *MI) {
144753f127fSDimitry Andric   CSKY_MC::verifyInstructionPredicates(MI->getOpcode(),
145753f127fSDimitry Andric                                        getSubtargetInfo().getFeatureBits());
146753f127fSDimitry Andric 
147349cc55cSDimitry Andric   // Do any auto-generated pseudo lowerings.
148349cc55cSDimitry Andric   if (emitPseudoExpansionLowering(*OutStreamer, MI))
149349cc55cSDimitry Andric     return;
150349cc55cSDimitry Andric 
15104eeddc0SDimitry Andric   // If we just ended a constant pool, mark it as such.
15204eeddc0SDimitry Andric   if (InConstantPool && MI->getOpcode() != CSKY::CONSTPOOL_ENTRY) {
15304eeddc0SDimitry Andric     InConstantPool = false;
15404eeddc0SDimitry Andric   }
15504eeddc0SDimitry Andric 
15604eeddc0SDimitry Andric   if (MI->getOpcode() == CSKY::PseudoTLSLA32)
15704eeddc0SDimitry Andric     return expandTLSLA(MI);
15804eeddc0SDimitry Andric 
15904eeddc0SDimitry Andric   if (MI->getOpcode() == CSKY::CONSTPOOL_ENTRY)
16004eeddc0SDimitry Andric     return emitCustomConstantPool(MI);
16104eeddc0SDimitry Andric 
162349cc55cSDimitry Andric   MCInst TmpInst;
163349cc55cSDimitry Andric   MCInstLowering.Lower(MI, TmpInst);
164349cc55cSDimitry Andric   EmitToStreamer(*OutStreamer, TmpInst);
165349cc55cSDimitry Andric }
166349cc55cSDimitry Andric 
16704eeddc0SDimitry Andric // Convert a CSKY-specific constant pool modifier into the associated
16804eeddc0SDimitry Andric // MCSymbolRefExpr variant kind.
16904eeddc0SDimitry Andric static CSKYMCExpr::VariantKind
getModifierVariantKind(CSKYCP::CSKYCPModifier Modifier)17004eeddc0SDimitry Andric getModifierVariantKind(CSKYCP::CSKYCPModifier Modifier) {
17104eeddc0SDimitry Andric   switch (Modifier) {
17204eeddc0SDimitry Andric   case CSKYCP::NO_MOD:
17304eeddc0SDimitry Andric     return CSKYMCExpr::VK_CSKY_None;
17404eeddc0SDimitry Andric   case CSKYCP::ADDR:
17504eeddc0SDimitry Andric     return CSKYMCExpr::VK_CSKY_ADDR;
17604eeddc0SDimitry Andric   case CSKYCP::GOT:
17704eeddc0SDimitry Andric     return CSKYMCExpr::VK_CSKY_GOT;
17804eeddc0SDimitry Andric   case CSKYCP::GOTOFF:
17904eeddc0SDimitry Andric     return CSKYMCExpr::VK_CSKY_GOTOFF;
18004eeddc0SDimitry Andric   case CSKYCP::PLT:
18104eeddc0SDimitry Andric     return CSKYMCExpr::VK_CSKY_PLT;
18204eeddc0SDimitry Andric   case CSKYCP::TLSGD:
18304eeddc0SDimitry Andric     return CSKYMCExpr::VK_CSKY_TLSGD;
18404eeddc0SDimitry Andric   case CSKYCP::TLSLE:
18504eeddc0SDimitry Andric     return CSKYMCExpr::VK_CSKY_TLSLE;
18604eeddc0SDimitry Andric   case CSKYCP::TLSIE:
18704eeddc0SDimitry Andric     return CSKYMCExpr::VK_CSKY_TLSIE;
18804eeddc0SDimitry Andric   }
18904eeddc0SDimitry Andric   llvm_unreachable("Invalid CSKYCPModifier!");
19004eeddc0SDimitry Andric }
19104eeddc0SDimitry Andric 
emitMachineConstantPoolValue(MachineConstantPoolValue * MCPV)19204eeddc0SDimitry Andric void CSKYAsmPrinter::emitMachineConstantPoolValue(
19304eeddc0SDimitry Andric     MachineConstantPoolValue *MCPV) {
19404eeddc0SDimitry Andric   int Size = getDataLayout().getTypeAllocSize(MCPV->getType());
19504eeddc0SDimitry Andric   CSKYConstantPoolValue *CCPV = static_cast<CSKYConstantPoolValue *>(MCPV);
19604eeddc0SDimitry Andric   MCSymbol *MCSym;
19704eeddc0SDimitry Andric 
19804eeddc0SDimitry Andric   if (CCPV->isBlockAddress()) {
19904eeddc0SDimitry Andric     const BlockAddress *BA =
20004eeddc0SDimitry Andric         cast<CSKYConstantPoolConstant>(CCPV)->getBlockAddress();
20104eeddc0SDimitry Andric     MCSym = GetBlockAddressSymbol(BA);
20204eeddc0SDimitry Andric   } else if (CCPV->isGlobalValue()) {
20304eeddc0SDimitry Andric     const GlobalValue *GV = cast<CSKYConstantPoolConstant>(CCPV)->getGV();
20404eeddc0SDimitry Andric     MCSym = getSymbol(GV);
20504eeddc0SDimitry Andric   } else if (CCPV->isMachineBasicBlock()) {
20604eeddc0SDimitry Andric     const MachineBasicBlock *MBB = cast<CSKYConstantPoolMBB>(CCPV)->getMBB();
20704eeddc0SDimitry Andric     MCSym = MBB->getSymbol();
20804eeddc0SDimitry Andric   } else if (CCPV->isJT()) {
20904eeddc0SDimitry Andric     signed JTI = cast<CSKYConstantPoolJT>(CCPV)->getJTI();
21004eeddc0SDimitry Andric     MCSym = GetJTISymbol(JTI);
211*bdd1243dSDimitry Andric   } else if (CCPV->isConstPool()) {
212*bdd1243dSDimitry Andric     const Constant *C = cast<CSKYConstantPoolConstant>(CCPV)->getConstantPool();
213*bdd1243dSDimitry Andric     MCSym = GetCPISymbol(MCP->getConstantPoolIndex(C, Align(4)));
21404eeddc0SDimitry Andric   } else {
21504eeddc0SDimitry Andric     assert(CCPV->isExtSymbol() && "unrecognized constant pool value");
21604eeddc0SDimitry Andric     StringRef Sym = cast<CSKYConstantPoolSymbol>(CCPV)->getSymbol();
21704eeddc0SDimitry Andric     MCSym = GetExternalSymbolSymbol(Sym);
21804eeddc0SDimitry Andric   }
21904eeddc0SDimitry Andric   // Create an MCSymbol for the reference.
22004eeddc0SDimitry Andric   const MCExpr *Expr =
22104eeddc0SDimitry Andric       MCSymbolRefExpr::create(MCSym, MCSymbolRefExpr::VK_None, OutContext);
22204eeddc0SDimitry Andric 
22304eeddc0SDimitry Andric   if (CCPV->getPCAdjustment()) {
22404eeddc0SDimitry Andric 
22504eeddc0SDimitry Andric     MCSymbol *PCLabel = OutContext.getOrCreateSymbol(
22604eeddc0SDimitry Andric         Twine(MAI->getPrivateGlobalPrefix()) + "PC" +
22704eeddc0SDimitry Andric         Twine(getFunctionNumber()) + "_" + Twine(CCPV->getLabelID()));
22804eeddc0SDimitry Andric 
22904eeddc0SDimitry Andric     const MCExpr *PCRelExpr = MCSymbolRefExpr::create(PCLabel, OutContext);
23004eeddc0SDimitry Andric     if (CCPV->mustAddCurrentAddress()) {
23104eeddc0SDimitry Andric       // We want "(<expr> - .)", but MC doesn't have a concept of the '.'
23204eeddc0SDimitry Andric       // label, so just emit a local label end reference that instead.
23304eeddc0SDimitry Andric       MCSymbol *DotSym = OutContext.createTempSymbol();
23404eeddc0SDimitry Andric       OutStreamer->emitLabel(DotSym);
23504eeddc0SDimitry Andric       const MCExpr *DotExpr = MCSymbolRefExpr::create(DotSym, OutContext);
23604eeddc0SDimitry Andric       PCRelExpr = MCBinaryExpr::createSub(PCRelExpr, DotExpr, OutContext);
23704eeddc0SDimitry Andric     }
23804eeddc0SDimitry Andric     Expr = MCBinaryExpr::createSub(Expr, PCRelExpr, OutContext);
23904eeddc0SDimitry Andric   }
24004eeddc0SDimitry Andric 
24104eeddc0SDimitry Andric   // Create an MCSymbol for the reference.
24204eeddc0SDimitry Andric   Expr = CSKYMCExpr::create(Expr, getModifierVariantKind(CCPV->getModifier()),
24304eeddc0SDimitry Andric                             OutContext);
24404eeddc0SDimitry Andric 
24504eeddc0SDimitry Andric   OutStreamer->emitValue(Expr, Size);
24604eeddc0SDimitry Andric }
24704eeddc0SDimitry Andric 
emitAttributes()24881ad6265SDimitry Andric void CSKYAsmPrinter::emitAttributes() {
24981ad6265SDimitry Andric   CSKYTargetStreamer &CTS =
25081ad6265SDimitry Andric       static_cast<CSKYTargetStreamer &>(*OutStreamer->getTargetStreamer());
25181ad6265SDimitry Andric 
25281ad6265SDimitry Andric   const Triple &TT = TM.getTargetTriple();
25381ad6265SDimitry Andric   StringRef CPU = TM.getTargetCPU();
25481ad6265SDimitry Andric   StringRef FS = TM.getTargetFeatureString();
25581ad6265SDimitry Andric   const CSKYTargetMachine &CTM = static_cast<const CSKYTargetMachine &>(TM);
25681ad6265SDimitry Andric   /* TuneCPU doesn't impact emission of ELF attributes, ELF attributes only
25781ad6265SDimitry Andric      care about arch related features, so we can set TuneCPU as CPU.  */
25881ad6265SDimitry Andric   const CSKYSubtarget STI(TT, CPU, /*TuneCPU=*/CPU, FS, CTM);
25981ad6265SDimitry Andric 
26081ad6265SDimitry Andric   CTS.emitTargetAttributes(STI);
26181ad6265SDimitry Andric }
26281ad6265SDimitry Andric 
PrintAsmOperand(const MachineInstr * MI,unsigned OpNo,const char * ExtraCode,raw_ostream & OS)26381ad6265SDimitry Andric bool CSKYAsmPrinter::PrintAsmOperand(const MachineInstr *MI, unsigned OpNo,
26481ad6265SDimitry Andric                                      const char *ExtraCode, raw_ostream &OS) {
26581ad6265SDimitry Andric   // First try the generic code, which knows about modifiers like 'c' and 'n'.
26681ad6265SDimitry Andric   if (!AsmPrinter::PrintAsmOperand(MI, OpNo, ExtraCode, OS))
26781ad6265SDimitry Andric     return false;
26881ad6265SDimitry Andric 
26981ad6265SDimitry Andric   const MachineOperand &MO = MI->getOperand(OpNo);
27081ad6265SDimitry Andric   if (ExtraCode && ExtraCode[0]) {
27181ad6265SDimitry Andric     if (ExtraCode[1] != 0)
27281ad6265SDimitry Andric       return true; // Unknown modifier.
27381ad6265SDimitry Andric 
27481ad6265SDimitry Andric     switch (ExtraCode[0]) {
27581ad6265SDimitry Andric     default:
27681ad6265SDimitry Andric       return true; // Unknown modifier.
27781ad6265SDimitry Andric     case 'R':
27881ad6265SDimitry Andric       if (MO.getType() == MachineOperand::MO_Register) {
27981ad6265SDimitry Andric         OS << CSKYInstPrinter::getRegisterName(MO.getReg() + 1);
28081ad6265SDimitry Andric         return false;
28181ad6265SDimitry Andric       }
28281ad6265SDimitry Andric     }
28381ad6265SDimitry Andric   }
28481ad6265SDimitry Andric 
28581ad6265SDimitry Andric   switch (MO.getType()) {
28681ad6265SDimitry Andric   case MachineOperand::MO_Immediate:
28781ad6265SDimitry Andric     OS << MO.getImm();
28881ad6265SDimitry Andric     return false;
28981ad6265SDimitry Andric   case MachineOperand::MO_Register:
29081ad6265SDimitry Andric     if (MO.getReg() == CSKY::C)
29181ad6265SDimitry Andric       return false;
29281ad6265SDimitry Andric     OS << CSKYInstPrinter::getRegisterName(MO.getReg());
29381ad6265SDimitry Andric     return false;
29481ad6265SDimitry Andric   case MachineOperand::MO_GlobalAddress:
29581ad6265SDimitry Andric     PrintSymbolOperand(MO, OS);
29681ad6265SDimitry Andric     return false;
29781ad6265SDimitry Andric   case MachineOperand::MO_BlockAddress: {
29881ad6265SDimitry Andric     MCSymbol *Sym = GetBlockAddressSymbol(MO.getBlockAddress());
29981ad6265SDimitry Andric     Sym->print(OS, MAI);
30081ad6265SDimitry Andric     return false;
30181ad6265SDimitry Andric   }
30281ad6265SDimitry Andric   default:
30381ad6265SDimitry Andric     break;
30481ad6265SDimitry Andric   }
30581ad6265SDimitry Andric 
30681ad6265SDimitry Andric   return true;
30781ad6265SDimitry Andric }
30881ad6265SDimitry Andric 
PrintAsmMemoryOperand(const MachineInstr * MI,unsigned OpNo,const char * ExtraCode,raw_ostream & OS)30981ad6265SDimitry Andric bool CSKYAsmPrinter::PrintAsmMemoryOperand(const MachineInstr *MI,
31081ad6265SDimitry Andric                                            unsigned OpNo, const char *ExtraCode,
31181ad6265SDimitry Andric                                            raw_ostream &OS) {
31281ad6265SDimitry Andric   if (!ExtraCode) {
31381ad6265SDimitry Andric     const MachineOperand &MO = MI->getOperand(OpNo);
31481ad6265SDimitry Andric     // For now, we only support register memory operands in registers and
31581ad6265SDimitry Andric     // assume there is no addend
31681ad6265SDimitry Andric     if (!MO.isReg())
31781ad6265SDimitry Andric       return true;
31881ad6265SDimitry Andric 
31981ad6265SDimitry Andric     OS << "(" << CSKYInstPrinter::getRegisterName(MO.getReg()) << ", 0)";
32081ad6265SDimitry Andric     return false;
32181ad6265SDimitry Andric   }
32281ad6265SDimitry Andric 
32381ad6265SDimitry Andric   return AsmPrinter::PrintAsmMemoryOperand(MI, OpNo, ExtraCode, OS);
32481ad6265SDimitry Andric }
32581ad6265SDimitry Andric 
LLVMInitializeCSKYAsmPrinter()326349cc55cSDimitry Andric extern "C" LLVM_EXTERNAL_VISIBILITY void LLVMInitializeCSKYAsmPrinter() {
327349cc55cSDimitry Andric   RegisterAsmPrinter<CSKYAsmPrinter> X(getTheCSKYTarget());
328349cc55cSDimitry Andric }
329