diff --git a/llvm/lib/Target/PowerPC/PPCISelLowering.cpp b/llvm/lib/Target/PowerPC/PPCISelLowering.cpp --- a/llvm/lib/Target/PowerPC/PPCISelLowering.cpp +++ b/llvm/lib/Target/PowerPC/PPCISelLowering.cpp @@ -5083,6 +5083,17 @@ return PPCISD::CALL; } + +static bool isValidAIXExternalSymSDNode(StringRef SymName) { + return StringSwitch(SymName) + .Cases("memcpy", "memmove", "memset", true) + .Cases("__moddi3", "__divdi3", "__umoddi3", "__udivdi3", true) + .Cases("__floatdidf", "__fixdfdi", "__fixunsdfdi", "__floatdisf", + "__floatundidf", "__floatundisf", true) + .Cases("ceil", "floor", "round", true) + .Default(false); +} + static SDValue transformCallee(const SDValue &Callee, SelectionDAG &DAG, const SDLoc &dl, const PPCSubtarget &Subtarget) { if (!Subtarget.usesFunctionDescriptors() && !Subtarget.isELFv2ABI()) @@ -5100,42 +5111,68 @@ bool UsePlt = Subtarget.is32BitELFABI() && !isLocalCallee(); + const auto replaceCalleeWithAIXFuncEntryPoint = + [&](StringRef FuncName, bool IsDeclaration, + const XCOFF::StorageClass &SC) { + auto &Context = DAG.getMachineFunction().getMMI().getContext(); + + MCSymbolXCOFF *S = cast( + Context.getOrCreateSymbol(Twine(".") + Twine(FuncName))); + + if (IsDeclaration && !S->hasContainingCsect()) { + // On AIX, an undefined symbol needs to be associated with a + // MCSectionXCOFF to get the correct storage mapping class. + // In this case, XCOFF::XMC_PR. + MCSectionXCOFF *Sec = Context.getXCOFFSection( + S->getName(), XCOFF::XMC_PR, XCOFF::XTY_ER, SC, + SectionKind::getMetadata()); + S->setContainingCsect(Sec); + } + + // Replace the callee SDNode with the MCSymbolSDNode. + EVT PtrVT = + DAG.getTargetLoweringInfo().getPointerTy(DAG.getDataLayout()); + return DAG.getMCSymbol(S, PtrVT); + }; + if (isFunctionGlobalAddress(Callee)) { const GlobalAddressSDNode *G = cast(Callee); + const GlobalObject *GO = cast(G->getGlobal()); + if (!Subtarget.isAIXABI()) - return DAG.getTargetGlobalAddress(G->getGlobal(), dl, - Callee.getValueType(), 0, + return DAG.getTargetGlobalAddress(GO, dl, Callee.getValueType(), 0, UsePlt ? PPCII::MO_PLT : 0); // On AIX, direct function calls reference the symbol for the function's // entry point, which is named by prepending a "." before the function's // C-linkage name. - auto &Context = DAG.getMachineFunction().getMMI().getContext(); - - const GlobalObject *GO = cast(G->getGlobal()); - MCSymbolXCOFF *S = cast( - Context.getOrCreateSymbol(Twine(".") + Twine(GO->getName()))); - - if (GO && GO->isDeclaration() && !S->hasContainingCsect()) { - // On AIX, an undefined symbol needs to be associated with a - // MCSectionXCOFF to get the correct storage mapping class. - // In this case, XCOFF::XMC_PR. - const XCOFF::StorageClass SC = - TargetLoweringObjectFileXCOFF::getStorageClassForGlobal(GO); - MCSectionXCOFF *Sec = - Context.getXCOFFSection(S->getName(), XCOFF::XMC_PR, XCOFF::XTY_ER, - SC, SectionKind::getMetadata()); - S->setContainingCsect(Sec); + assert(GO && + "The GlobalAddressSDNode does not have a global object associated."); + const XCOFF::StorageClass SC = + TargetLoweringObjectFileXCOFF::getStorageClassForGlobal(GO); + return replaceCalleeWithAIXFuncEntryPoint(GO->getName(), + GO->isDeclaration(), SC); + } + + if (ExternalSymbolSDNode *S = dyn_cast(Callee)) { + const char *SymName = S->getSymbol(); + if (!Subtarget.isAIXABI()) + return DAG.getTargetExternalSymbol(SymName, Callee.getValueType(), + UsePlt ? PPCII::MO_PLT : 0); + + assert(SymName && + "The ExternalSymbolSDNode does not have a MCSymbol associated."); + // TODO: Remove this when the support for ExternalSymbolSDNode is complete. + if (isValidAIXExternalSymSDNode(SymName)) { + // On AIX, direct function calls reference the symbol for the function's + // entry point, which is named by prepending a "." before the function's + // C-linkage name. + return replaceCalleeWithAIXFuncEntryPoint(SymName, true, XCOFF::C_EXT); + } else { + report_fatal_error("Unexpected ExternalSymbolSDNode: " + Twine(SymName)); } - - EVT PtrVT = DAG.getTargetLoweringInfo().getPointerTy(DAG.getDataLayout()); - return DAG.getMCSymbol(S, PtrVT); } - if (ExternalSymbolSDNode *S = dyn_cast(Callee)) - return DAG.getTargetExternalSymbol(S->getSymbol(), Callee.getValueType(), - UsePlt ? PPCII::MO_PLT : 0); - // No transformation needed. assert(Callee.getNode() && "What no callee?"); return Callee; diff --git a/llvm/test/CodeGen/PowerPC/aix-external-sym-sdnode-lowering.ll b/llvm/test/CodeGen/PowerPC/aix-external-sym-sdnode-lowering.ll new file mode 100644 --- /dev/null +++ b/llvm/test/CodeGen/PowerPC/aix-external-sym-sdnode-lowering.ll @@ -0,0 +1,40 @@ +; RUN: llc -verify-machineinstrs -mtriple powerpc-ibm-aix-xcoff -mcpu=pwr7 \ +; RUN: -mattr=-altivec -stop-after=machine-cp < %s | FileCheck \ +; RUN: --check-prefix=32BIT %s + +; RUN: llc -verify-machineinstrs -mtriple powerpc64-ibm-aix-xcoff -mcpu=pwr7 \ +; RUN: -mattr=-altivec -stop-after=machine-cp < %s | FileCheck \ +; RUN: --check-prefix=64BIT %s + +define void @call_memcpy(i8* %p, i8* %q, i32 %n) { +entry: + call void @llvm.memcpy.p0i8.p0i8.i32(i8* %p, i8* %q, i32 %n, i1 false) + ret void +} + +declare void @llvm.memcpy.p0i8.p0i8.i32(i8* nocapture writeonly, i8* nocapture readonly, i32, i1) + +; 32BIT: BL_NOP +; 64BIT: BL8_NOP + +define void @call_memmove(i8* %p, i8* %q, i32 %n) { +entry: + call void @llvm.memmove.p0i8.p0i8.i32(i8* %p, i8* %q, i32 %n, i1 false) + ret void +} + +declare void @llvm.memmove.p0i8.p0i8.i32(i8* nocapture, i8* nocapture readonly, i32, i1) + +; 32BIT: BL_NOP +; 64BIT: BL8_NOP + +define void @call_memset(i8* %p, i8 %q, i32 %n) #0 { +entry: + call void @llvm.memset.p0i8.i32(i8* %p, i8 %q, i32 %n, i1 false) + ret void +} + +declare void @llvm.memset.p0i8.i32(i8* nocapture, i8, i32, i1) + +; 32BIT: BL_NOP +; 64BIT: BL8_NOP