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