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" 19*81ad6265SDimitry 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" 24*81ad6265SDimitry 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(); 45*81ad6265SDimitry Andric TII = MF.getSubtarget().getInstrInfo(); 46*81ad6265SDimitry Andric 47*81ad6265SDimitry Andric // Set the current MCSubtargetInfo to a copy which has the correct 48*81ad6265SDimitry Andric // feature bits for the current MachineFunction 49*81ad6265SDimitry Andric MCSubtargetInfo &NewSTI = 50*81ad6265SDimitry Andric OutStreamer->getContext().getSubtargetCopy(*TM.getMCSubtargetInfo()); 51*81ad6265SDimitry Andric NewSTI.setFeatureBits(MF.getSubtarget().getFeatureBits()); 52*81ad6265SDimitry Andric Subtarget = &NewSTI; 53*81ad6265SDimitry 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 130*81ad6265SDimitry Andric void CSKYAsmPrinter::emitStartOfAsmFile(Module &M) { 131*81ad6265SDimitry Andric if (TM.getTargetTriple().isOSBinFormatELF()) 132*81ad6265SDimitry Andric emitAttributes(); 133*81ad6265SDimitry Andric } 134*81ad6265SDimitry Andric 135*81ad6265SDimitry Andric void CSKYAsmPrinter::emitEndOfAsmFile(Module &M) { 136*81ad6265SDimitry Andric CSKYTargetStreamer &CTS = 137*81ad6265SDimitry Andric static_cast<CSKYTargetStreamer &>(*OutStreamer->getTargetStreamer()); 138*81ad6265SDimitry Andric 139*81ad6265SDimitry Andric if (TM.getTargetTriple().isOSBinFormatELF()) 140*81ad6265SDimitry Andric CTS.finishAttributeSection(); 141*81ad6265SDimitry Andric } 142*81ad6265SDimitry Andric 143349cc55cSDimitry Andric void CSKYAsmPrinter::emitInstruction(const MachineInstr *MI) { 144349cc55cSDimitry Andric // Do any auto-generated pseudo lowerings. 145349cc55cSDimitry Andric if (emitPseudoExpansionLowering(*OutStreamer, MI)) 146349cc55cSDimitry Andric return; 147349cc55cSDimitry Andric 14804eeddc0SDimitry Andric // If we just ended a constant pool, mark it as such. 14904eeddc0SDimitry Andric if (InConstantPool && MI->getOpcode() != CSKY::CONSTPOOL_ENTRY) { 15004eeddc0SDimitry Andric InConstantPool = false; 15104eeddc0SDimitry Andric } 15204eeddc0SDimitry Andric 15304eeddc0SDimitry Andric if (MI->getOpcode() == CSKY::PseudoTLSLA32) 15404eeddc0SDimitry Andric return expandTLSLA(MI); 15504eeddc0SDimitry Andric 15604eeddc0SDimitry Andric if (MI->getOpcode() == CSKY::CONSTPOOL_ENTRY) 15704eeddc0SDimitry Andric return emitCustomConstantPool(MI); 15804eeddc0SDimitry Andric 159349cc55cSDimitry Andric MCInst TmpInst; 160349cc55cSDimitry Andric MCInstLowering.Lower(MI, TmpInst); 161349cc55cSDimitry Andric EmitToStreamer(*OutStreamer, TmpInst); 162349cc55cSDimitry Andric } 163349cc55cSDimitry Andric 16404eeddc0SDimitry Andric // Convert a CSKY-specific constant pool modifier into the associated 16504eeddc0SDimitry Andric // MCSymbolRefExpr variant kind. 16604eeddc0SDimitry Andric static CSKYMCExpr::VariantKind 16704eeddc0SDimitry Andric getModifierVariantKind(CSKYCP::CSKYCPModifier Modifier) { 16804eeddc0SDimitry Andric switch (Modifier) { 16904eeddc0SDimitry Andric case CSKYCP::NO_MOD: 17004eeddc0SDimitry Andric return CSKYMCExpr::VK_CSKY_None; 17104eeddc0SDimitry Andric case CSKYCP::ADDR: 17204eeddc0SDimitry Andric return CSKYMCExpr::VK_CSKY_ADDR; 17304eeddc0SDimitry Andric case CSKYCP::GOT: 17404eeddc0SDimitry Andric return CSKYMCExpr::VK_CSKY_GOT; 17504eeddc0SDimitry Andric case CSKYCP::GOTOFF: 17604eeddc0SDimitry Andric return CSKYMCExpr::VK_CSKY_GOTOFF; 17704eeddc0SDimitry Andric case CSKYCP::PLT: 17804eeddc0SDimitry Andric return CSKYMCExpr::VK_CSKY_PLT; 17904eeddc0SDimitry Andric case CSKYCP::TLSGD: 18004eeddc0SDimitry Andric return CSKYMCExpr::VK_CSKY_TLSGD; 18104eeddc0SDimitry Andric case CSKYCP::TLSLE: 18204eeddc0SDimitry Andric return CSKYMCExpr::VK_CSKY_TLSLE; 18304eeddc0SDimitry Andric case CSKYCP::TLSIE: 18404eeddc0SDimitry Andric return CSKYMCExpr::VK_CSKY_TLSIE; 18504eeddc0SDimitry Andric } 18604eeddc0SDimitry Andric llvm_unreachable("Invalid CSKYCPModifier!"); 18704eeddc0SDimitry Andric } 18804eeddc0SDimitry Andric 18904eeddc0SDimitry Andric void CSKYAsmPrinter::emitMachineConstantPoolValue( 19004eeddc0SDimitry Andric MachineConstantPoolValue *MCPV) { 19104eeddc0SDimitry Andric int Size = getDataLayout().getTypeAllocSize(MCPV->getType()); 19204eeddc0SDimitry Andric CSKYConstantPoolValue *CCPV = static_cast<CSKYConstantPoolValue *>(MCPV); 19304eeddc0SDimitry Andric MCSymbol *MCSym; 19404eeddc0SDimitry Andric 19504eeddc0SDimitry Andric if (CCPV->isBlockAddress()) { 19604eeddc0SDimitry Andric const BlockAddress *BA = 19704eeddc0SDimitry Andric cast<CSKYConstantPoolConstant>(CCPV)->getBlockAddress(); 19804eeddc0SDimitry Andric MCSym = GetBlockAddressSymbol(BA); 19904eeddc0SDimitry Andric } else if (CCPV->isGlobalValue()) { 20004eeddc0SDimitry Andric const GlobalValue *GV = cast<CSKYConstantPoolConstant>(CCPV)->getGV(); 20104eeddc0SDimitry Andric MCSym = getSymbol(GV); 20204eeddc0SDimitry Andric } else if (CCPV->isMachineBasicBlock()) { 20304eeddc0SDimitry Andric const MachineBasicBlock *MBB = cast<CSKYConstantPoolMBB>(CCPV)->getMBB(); 20404eeddc0SDimitry Andric MCSym = MBB->getSymbol(); 20504eeddc0SDimitry Andric } else if (CCPV->isJT()) { 20604eeddc0SDimitry Andric signed JTI = cast<CSKYConstantPoolJT>(CCPV)->getJTI(); 20704eeddc0SDimitry Andric MCSym = GetJTISymbol(JTI); 20804eeddc0SDimitry Andric } else { 20904eeddc0SDimitry Andric assert(CCPV->isExtSymbol() && "unrecognized constant pool value"); 21004eeddc0SDimitry Andric StringRef Sym = cast<CSKYConstantPoolSymbol>(CCPV)->getSymbol(); 21104eeddc0SDimitry Andric MCSym = GetExternalSymbolSymbol(Sym); 21204eeddc0SDimitry Andric } 21304eeddc0SDimitry Andric // Create an MCSymbol for the reference. 21404eeddc0SDimitry Andric const MCExpr *Expr = 21504eeddc0SDimitry Andric MCSymbolRefExpr::create(MCSym, MCSymbolRefExpr::VK_None, OutContext); 21604eeddc0SDimitry Andric 21704eeddc0SDimitry Andric if (CCPV->getPCAdjustment()) { 21804eeddc0SDimitry Andric 21904eeddc0SDimitry Andric MCSymbol *PCLabel = OutContext.getOrCreateSymbol( 22004eeddc0SDimitry Andric Twine(MAI->getPrivateGlobalPrefix()) + "PC" + 22104eeddc0SDimitry Andric Twine(getFunctionNumber()) + "_" + Twine(CCPV->getLabelID())); 22204eeddc0SDimitry Andric 22304eeddc0SDimitry Andric const MCExpr *PCRelExpr = MCSymbolRefExpr::create(PCLabel, OutContext); 22404eeddc0SDimitry Andric if (CCPV->mustAddCurrentAddress()) { 22504eeddc0SDimitry Andric // We want "(<expr> - .)", but MC doesn't have a concept of the '.' 22604eeddc0SDimitry Andric // label, so just emit a local label end reference that instead. 22704eeddc0SDimitry Andric MCSymbol *DotSym = OutContext.createTempSymbol(); 22804eeddc0SDimitry Andric OutStreamer->emitLabel(DotSym); 22904eeddc0SDimitry Andric const MCExpr *DotExpr = MCSymbolRefExpr::create(DotSym, OutContext); 23004eeddc0SDimitry Andric PCRelExpr = MCBinaryExpr::createSub(PCRelExpr, DotExpr, OutContext); 23104eeddc0SDimitry Andric } 23204eeddc0SDimitry Andric Expr = MCBinaryExpr::createSub(Expr, PCRelExpr, OutContext); 23304eeddc0SDimitry Andric } 23404eeddc0SDimitry Andric 23504eeddc0SDimitry Andric // Create an MCSymbol for the reference. 23604eeddc0SDimitry Andric Expr = CSKYMCExpr::create(Expr, getModifierVariantKind(CCPV->getModifier()), 23704eeddc0SDimitry Andric OutContext); 23804eeddc0SDimitry Andric 23904eeddc0SDimitry Andric OutStreamer->emitValue(Expr, Size); 24004eeddc0SDimitry Andric } 24104eeddc0SDimitry Andric 242*81ad6265SDimitry Andric void CSKYAsmPrinter::emitAttributes() { 243*81ad6265SDimitry Andric CSKYTargetStreamer &CTS = 244*81ad6265SDimitry Andric static_cast<CSKYTargetStreamer &>(*OutStreamer->getTargetStreamer()); 245*81ad6265SDimitry Andric 246*81ad6265SDimitry Andric const Triple &TT = TM.getTargetTriple(); 247*81ad6265SDimitry Andric StringRef CPU = TM.getTargetCPU(); 248*81ad6265SDimitry Andric StringRef FS = TM.getTargetFeatureString(); 249*81ad6265SDimitry Andric const CSKYTargetMachine &CTM = static_cast<const CSKYTargetMachine &>(TM); 250*81ad6265SDimitry Andric /* TuneCPU doesn't impact emission of ELF attributes, ELF attributes only 251*81ad6265SDimitry Andric care about arch related features, so we can set TuneCPU as CPU. */ 252*81ad6265SDimitry Andric const CSKYSubtarget STI(TT, CPU, /*TuneCPU=*/CPU, FS, CTM); 253*81ad6265SDimitry Andric 254*81ad6265SDimitry Andric CTS.emitTargetAttributes(STI); 255*81ad6265SDimitry Andric } 256*81ad6265SDimitry Andric 257*81ad6265SDimitry Andric bool CSKYAsmPrinter::PrintAsmOperand(const MachineInstr *MI, unsigned OpNo, 258*81ad6265SDimitry Andric const char *ExtraCode, raw_ostream &OS) { 259*81ad6265SDimitry Andric // First try the generic code, which knows about modifiers like 'c' and 'n'. 260*81ad6265SDimitry Andric if (!AsmPrinter::PrintAsmOperand(MI, OpNo, ExtraCode, OS)) 261*81ad6265SDimitry Andric return false; 262*81ad6265SDimitry Andric 263*81ad6265SDimitry Andric const MachineOperand &MO = MI->getOperand(OpNo); 264*81ad6265SDimitry Andric if (ExtraCode && ExtraCode[0]) { 265*81ad6265SDimitry Andric if (ExtraCode[1] != 0) 266*81ad6265SDimitry Andric return true; // Unknown modifier. 267*81ad6265SDimitry Andric 268*81ad6265SDimitry Andric switch (ExtraCode[0]) { 269*81ad6265SDimitry Andric default: 270*81ad6265SDimitry Andric return true; // Unknown modifier. 271*81ad6265SDimitry Andric case 'R': 272*81ad6265SDimitry Andric if (MO.getType() == MachineOperand::MO_Register) { 273*81ad6265SDimitry Andric OS << CSKYInstPrinter::getRegisterName(MO.getReg() + 1); 274*81ad6265SDimitry Andric return false; 275*81ad6265SDimitry Andric } 276*81ad6265SDimitry Andric } 277*81ad6265SDimitry Andric } 278*81ad6265SDimitry Andric 279*81ad6265SDimitry Andric switch (MO.getType()) { 280*81ad6265SDimitry Andric case MachineOperand::MO_Immediate: 281*81ad6265SDimitry Andric OS << MO.getImm(); 282*81ad6265SDimitry Andric return false; 283*81ad6265SDimitry Andric case MachineOperand::MO_Register: 284*81ad6265SDimitry Andric if (MO.getReg() == CSKY::C) 285*81ad6265SDimitry Andric return false; 286*81ad6265SDimitry Andric OS << CSKYInstPrinter::getRegisterName(MO.getReg()); 287*81ad6265SDimitry Andric return false; 288*81ad6265SDimitry Andric case MachineOperand::MO_GlobalAddress: 289*81ad6265SDimitry Andric PrintSymbolOperand(MO, OS); 290*81ad6265SDimitry Andric return false; 291*81ad6265SDimitry Andric case MachineOperand::MO_BlockAddress: { 292*81ad6265SDimitry Andric MCSymbol *Sym = GetBlockAddressSymbol(MO.getBlockAddress()); 293*81ad6265SDimitry Andric Sym->print(OS, MAI); 294*81ad6265SDimitry Andric return false; 295*81ad6265SDimitry Andric } 296*81ad6265SDimitry Andric default: 297*81ad6265SDimitry Andric break; 298*81ad6265SDimitry Andric } 299*81ad6265SDimitry Andric 300*81ad6265SDimitry Andric return true; 301*81ad6265SDimitry Andric } 302*81ad6265SDimitry Andric 303*81ad6265SDimitry Andric bool CSKYAsmPrinter::PrintAsmMemoryOperand(const MachineInstr *MI, 304*81ad6265SDimitry Andric unsigned OpNo, const char *ExtraCode, 305*81ad6265SDimitry Andric raw_ostream &OS) { 306*81ad6265SDimitry Andric if (!ExtraCode) { 307*81ad6265SDimitry Andric const MachineOperand &MO = MI->getOperand(OpNo); 308*81ad6265SDimitry Andric // For now, we only support register memory operands in registers and 309*81ad6265SDimitry Andric // assume there is no addend 310*81ad6265SDimitry Andric if (!MO.isReg()) 311*81ad6265SDimitry Andric return true; 312*81ad6265SDimitry Andric 313*81ad6265SDimitry Andric OS << "(" << CSKYInstPrinter::getRegisterName(MO.getReg()) << ", 0)"; 314*81ad6265SDimitry Andric return false; 315*81ad6265SDimitry Andric } 316*81ad6265SDimitry Andric 317*81ad6265SDimitry Andric return AsmPrinter::PrintAsmMemoryOperand(MI, OpNo, ExtraCode, OS); 318*81ad6265SDimitry Andric } 319*81ad6265SDimitry Andric 320349cc55cSDimitry Andric extern "C" LLVM_EXTERNAL_VISIBILITY void LLVMInitializeCSKYAsmPrinter() { 321349cc55cSDimitry Andric RegisterAsmPrinter<CSKYAsmPrinter> X(getTheCSKYTarget()); 322349cc55cSDimitry Andric } 323