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 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 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" 590eae32dcSDimitry Andric void CSKYAsmPrinter::EmitToStreamer(MCStreamer &S, const MCInst &Inst) { 600eae32dcSDimitry Andric MCInst CInst; 610eae32dcSDimitry Andric bool Res = compressInst(CInst, Inst, *Subtarget, OutStreamer->getContext()); 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 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 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) { 10804eeddc0SDimitry Andric OutStreamer->emitValueToAlignment(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 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 13081ad6265SDimitry Andric void CSKYAsmPrinter::emitStartOfAsmFile(Module &M) { 13181ad6265SDimitry Andric if (TM.getTargetTriple().isOSBinFormatELF()) 13281ad6265SDimitry Andric emitAttributes(); 13381ad6265SDimitry Andric } 13481ad6265SDimitry Andric 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 143349cc55cSDimitry Andric void CSKYAsmPrinter::emitInstruction(const MachineInstr *MI) { 144*753f127fSDimitry Andric CSKY_MC::verifyInstructionPredicates(MI->getOpcode(), 145*753f127fSDimitry Andric getSubtargetInfo().getFeatureBits()); 146*753f127fSDimitry 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 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 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); 21104eeddc0SDimitry Andric } else { 21204eeddc0SDimitry Andric assert(CCPV->isExtSymbol() && "unrecognized constant pool value"); 21304eeddc0SDimitry Andric StringRef Sym = cast<CSKYConstantPoolSymbol>(CCPV)->getSymbol(); 21404eeddc0SDimitry Andric MCSym = GetExternalSymbolSymbol(Sym); 21504eeddc0SDimitry Andric } 21604eeddc0SDimitry Andric // Create an MCSymbol for the reference. 21704eeddc0SDimitry Andric const MCExpr *Expr = 21804eeddc0SDimitry Andric MCSymbolRefExpr::create(MCSym, MCSymbolRefExpr::VK_None, OutContext); 21904eeddc0SDimitry Andric 22004eeddc0SDimitry Andric if (CCPV->getPCAdjustment()) { 22104eeddc0SDimitry Andric 22204eeddc0SDimitry Andric MCSymbol *PCLabel = OutContext.getOrCreateSymbol( 22304eeddc0SDimitry Andric Twine(MAI->getPrivateGlobalPrefix()) + "PC" + 22404eeddc0SDimitry Andric Twine(getFunctionNumber()) + "_" + Twine(CCPV->getLabelID())); 22504eeddc0SDimitry Andric 22604eeddc0SDimitry Andric const MCExpr *PCRelExpr = MCSymbolRefExpr::create(PCLabel, OutContext); 22704eeddc0SDimitry Andric if (CCPV->mustAddCurrentAddress()) { 22804eeddc0SDimitry Andric // We want "(<expr> - .)", but MC doesn't have a concept of the '.' 22904eeddc0SDimitry Andric // label, so just emit a local label end reference that instead. 23004eeddc0SDimitry Andric MCSymbol *DotSym = OutContext.createTempSymbol(); 23104eeddc0SDimitry Andric OutStreamer->emitLabel(DotSym); 23204eeddc0SDimitry Andric const MCExpr *DotExpr = MCSymbolRefExpr::create(DotSym, OutContext); 23304eeddc0SDimitry Andric PCRelExpr = MCBinaryExpr::createSub(PCRelExpr, DotExpr, OutContext); 23404eeddc0SDimitry Andric } 23504eeddc0SDimitry Andric Expr = MCBinaryExpr::createSub(Expr, PCRelExpr, OutContext); 23604eeddc0SDimitry Andric } 23704eeddc0SDimitry Andric 23804eeddc0SDimitry Andric // Create an MCSymbol for the reference. 23904eeddc0SDimitry Andric Expr = CSKYMCExpr::create(Expr, getModifierVariantKind(CCPV->getModifier()), 24004eeddc0SDimitry Andric OutContext); 24104eeddc0SDimitry Andric 24204eeddc0SDimitry Andric OutStreamer->emitValue(Expr, Size); 24304eeddc0SDimitry Andric } 24404eeddc0SDimitry Andric 24581ad6265SDimitry Andric void CSKYAsmPrinter::emitAttributes() { 24681ad6265SDimitry Andric CSKYTargetStreamer &CTS = 24781ad6265SDimitry Andric static_cast<CSKYTargetStreamer &>(*OutStreamer->getTargetStreamer()); 24881ad6265SDimitry Andric 24981ad6265SDimitry Andric const Triple &TT = TM.getTargetTriple(); 25081ad6265SDimitry Andric StringRef CPU = TM.getTargetCPU(); 25181ad6265SDimitry Andric StringRef FS = TM.getTargetFeatureString(); 25281ad6265SDimitry Andric const CSKYTargetMachine &CTM = static_cast<const CSKYTargetMachine &>(TM); 25381ad6265SDimitry Andric /* TuneCPU doesn't impact emission of ELF attributes, ELF attributes only 25481ad6265SDimitry Andric care about arch related features, so we can set TuneCPU as CPU. */ 25581ad6265SDimitry Andric const CSKYSubtarget STI(TT, CPU, /*TuneCPU=*/CPU, FS, CTM); 25681ad6265SDimitry Andric 25781ad6265SDimitry Andric CTS.emitTargetAttributes(STI); 25881ad6265SDimitry Andric } 25981ad6265SDimitry Andric 26081ad6265SDimitry Andric bool CSKYAsmPrinter::PrintAsmOperand(const MachineInstr *MI, unsigned OpNo, 26181ad6265SDimitry Andric const char *ExtraCode, raw_ostream &OS) { 26281ad6265SDimitry Andric // First try the generic code, which knows about modifiers like 'c' and 'n'. 26381ad6265SDimitry Andric if (!AsmPrinter::PrintAsmOperand(MI, OpNo, ExtraCode, OS)) 26481ad6265SDimitry Andric return false; 26581ad6265SDimitry Andric 26681ad6265SDimitry Andric const MachineOperand &MO = MI->getOperand(OpNo); 26781ad6265SDimitry Andric if (ExtraCode && ExtraCode[0]) { 26881ad6265SDimitry Andric if (ExtraCode[1] != 0) 26981ad6265SDimitry Andric return true; // Unknown modifier. 27081ad6265SDimitry Andric 27181ad6265SDimitry Andric switch (ExtraCode[0]) { 27281ad6265SDimitry Andric default: 27381ad6265SDimitry Andric return true; // Unknown modifier. 27481ad6265SDimitry Andric case 'R': 27581ad6265SDimitry Andric if (MO.getType() == MachineOperand::MO_Register) { 27681ad6265SDimitry Andric OS << CSKYInstPrinter::getRegisterName(MO.getReg() + 1); 27781ad6265SDimitry Andric return false; 27881ad6265SDimitry Andric } 27981ad6265SDimitry Andric } 28081ad6265SDimitry Andric } 28181ad6265SDimitry Andric 28281ad6265SDimitry Andric switch (MO.getType()) { 28381ad6265SDimitry Andric case MachineOperand::MO_Immediate: 28481ad6265SDimitry Andric OS << MO.getImm(); 28581ad6265SDimitry Andric return false; 28681ad6265SDimitry Andric case MachineOperand::MO_Register: 28781ad6265SDimitry Andric if (MO.getReg() == CSKY::C) 28881ad6265SDimitry Andric return false; 28981ad6265SDimitry Andric OS << CSKYInstPrinter::getRegisterName(MO.getReg()); 29081ad6265SDimitry Andric return false; 29181ad6265SDimitry Andric case MachineOperand::MO_GlobalAddress: 29281ad6265SDimitry Andric PrintSymbolOperand(MO, OS); 29381ad6265SDimitry Andric return false; 29481ad6265SDimitry Andric case MachineOperand::MO_BlockAddress: { 29581ad6265SDimitry Andric MCSymbol *Sym = GetBlockAddressSymbol(MO.getBlockAddress()); 29681ad6265SDimitry Andric Sym->print(OS, MAI); 29781ad6265SDimitry Andric return false; 29881ad6265SDimitry Andric } 29981ad6265SDimitry Andric default: 30081ad6265SDimitry Andric break; 30181ad6265SDimitry Andric } 30281ad6265SDimitry Andric 30381ad6265SDimitry Andric return true; 30481ad6265SDimitry Andric } 30581ad6265SDimitry Andric 30681ad6265SDimitry Andric bool CSKYAsmPrinter::PrintAsmMemoryOperand(const MachineInstr *MI, 30781ad6265SDimitry Andric unsigned OpNo, const char *ExtraCode, 30881ad6265SDimitry Andric raw_ostream &OS) { 30981ad6265SDimitry Andric if (!ExtraCode) { 31081ad6265SDimitry Andric const MachineOperand &MO = MI->getOperand(OpNo); 31181ad6265SDimitry Andric // For now, we only support register memory operands in registers and 31281ad6265SDimitry Andric // assume there is no addend 31381ad6265SDimitry Andric if (!MO.isReg()) 31481ad6265SDimitry Andric return true; 31581ad6265SDimitry Andric 31681ad6265SDimitry Andric OS << "(" << CSKYInstPrinter::getRegisterName(MO.getReg()) << ", 0)"; 31781ad6265SDimitry Andric return false; 31881ad6265SDimitry Andric } 31981ad6265SDimitry Andric 32081ad6265SDimitry Andric return AsmPrinter::PrintAsmMemoryOperand(MI, OpNo, ExtraCode, OS); 32181ad6265SDimitry Andric } 32281ad6265SDimitry Andric 323349cc55cSDimitry Andric extern "C" LLVM_EXTERNAL_VISIBILITY void LLVMInitializeCSKYAsmPrinter() { 324349cc55cSDimitry Andric RegisterAsmPrinter<CSKYAsmPrinter> X(getTheCSKYTarget()); 325349cc55cSDimitry Andric } 326