available_externall linkage is not local, this was confusing the codegenerator,
authorTorok Edwin <edwintorok@gmail.com>
Sat, 23 May 2009 14:06:57 +0000 (14:06 +0000)
committerTorok Edwin <edwintorok@gmail.com>
Sat, 23 May 2009 14:06:57 +0000 (14:06 +0000)
and it wasn't generating calls through @PLT for these functions.
hasLocalLinkage() is now false for available_externally,
I attempted to fix the inliner and dce to handle available_externally properly.
It passed make check.

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@72328 91177308-0d34-0410-b5e6-96231b3b80d8

include/llvm/GlobalValue.h
lib/Transforms/IPO/GlobalDCE.cpp
lib/Transforms/IPO/Inliner.cpp
lib/Transforms/Utils/BasicInliner.cpp
lib/Transforms/Utils/InlineCost.cpp
test/CodeGen/X86/2009-05-23-available_externally.ll [new file with mode: 0644]

index 50c613ef6a20a2f808dd0eca49140b501d52c601..3b7f67d5d03668991043c98636ff278111b6afa9 100644 (file)
@@ -123,8 +123,7 @@ public:
   bool hasInternalLinkage() const { return Linkage == InternalLinkage; }
   bool hasPrivateLinkage() const { return Linkage == PrivateLinkage; }
   bool hasLocalLinkage() const {
-    return Linkage == InternalLinkage || Linkage == PrivateLinkage ||
-           Linkage == AvailableExternallyLinkage;
+    return Linkage == InternalLinkage || Linkage == PrivateLinkage;
   }
   bool hasDLLImportLinkage() const { return Linkage == DLLImportLinkage; }
   bool hasDLLExportLinkage() const { return Linkage == DLLExportLinkage; }
index 4f97ae175b13a342e261f8c123c4ace7d64f0a20..db378b0d0b286e9908b9f889678c2d02eff99356 100644 (file)
@@ -64,7 +64,7 @@ bool GlobalDCE::runOnModule(Module &M) {
     Changed |= RemoveUnusedGlobalValue(*I);
     // Functions with external linkage are needed if they have a body
     if (!I->hasLocalLinkage() && !I->hasLinkOnceLinkage() &&
-        !I->isDeclaration())
+        !I->isDeclaration() && !I->hasAvailableExternallyLinkage())
       GlobalIsNeeded(I);
   }
 
@@ -74,7 +74,7 @@ bool GlobalDCE::runOnModule(Module &M) {
     // Externally visible & appending globals are needed, if they have an
     // initializer.
     if (!I->hasLocalLinkage() && !I->hasLinkOnceLinkage() &&
-        !I->isDeclaration())
+        !I->isDeclaration() && !I->hasAvailableExternallyLinkage())
       GlobalIsNeeded(I);
   }
 
index b589792022a709780be0e39f84768f908692cfa8..b382837289bda1a2fab561edbd437e6cd6e1f93c 100644 (file)
@@ -69,7 +69,8 @@ bool Inliner::InlineCallIfPossible(CallSite CS, CallGraph &CG,
 
   // If we inlined the last possible call site to the function, delete the
   // function body now.
-  if (Callee->use_empty() && Callee->hasLocalLinkage() &&
+  if (Callee->use_empty() && (Callee->hasLocalLinkage() ||
+                              Callee->hasAvailableExternallyLinkage()) &&
       !SCCFunctions.count(Callee)) {
     DOUT << "    -> Deleting dead function: " << Callee->getName() << "\n";
     CallGraphNode *CalleeNode = CG[Callee];
index ef37614997cfd01cab8b26668b088df1d5cc7e0e..1650cfa3065330cbc1aa97f1ee905253a6b970ae 100644 (file)
@@ -130,7 +130,8 @@ void BasicInlinerImpl::inlineFunctions() {
         
         // Inline
         if (InlineFunction(CS, NULL, TD)) {
-          if (Callee->use_empty() && Callee->hasLocalLinkage())
+          if (Callee->use_empty() && (Callee->hasLocalLinkage() ||
+                                      Callee->hasAvailableExternallyLinkage()))
             DeadFunctions.insert(Callee);
           Changed = true;
           CallSites.erase(CallSites.begin() + index);
index c9eb0ea1c907cb7be304df49e16597ab2e31d8f7..87aff01a58576804615c0f4f7faa0c3083934bf7 100644 (file)
@@ -189,7 +189,8 @@ InlineCost InlineCostAnalyzer::getInlineCost(CallSite CS,
   // If there is only one call of the function, and it has internal linkage,
   // make it almost guaranteed to be inlined.
   //
-  if (Callee->hasLocalLinkage() && Callee->hasOneUse())
+  if ((Callee->hasLocalLinkage() || Callee->hasAvailableExternallyLinkage()) && 
+      Callee->hasOneUse())
     InlineCost -= 15000;
   
   // If this function uses the coldcc calling convention, prefer not to inline
diff --git a/test/CodeGen/X86/2009-05-23-available_externally.ll b/test/CodeGen/X86/2009-05-23-available_externally.ll
new file mode 100644 (file)
index 0000000..f4881ba
--- /dev/null
@@ -0,0 +1,19 @@
+; RUN: llvm-as < %s | llc -relocation-model=pic | grep atoi | grep PLT
+; PR4253
+target datalayout = "e-p:64:64:64-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:64:64-f32:32:32-f64:64:64-v64:64:64-v128:128:128-a0:0:64-f80:128:128"
+target triple = "x86_64-unknown-linux-gnu"
+
+define i32 @foo(i8* %x) nounwind readonly {
+entry:
+       %call = tail call fastcc i32 @atoi(i8* %x) nounwind readonly            ; <i32> [#uses=1]
+       ret i32 %call
+}
+
+define available_externally fastcc i32 @atoi(i8* %__nptr) nounwind readonly {
+entry:
+       %call = tail call i64 @strtol(i8* nocapture %__nptr, i8** null, i32 10) nounwind readonly               ; <i64> [#uses=1]
+       %conv = trunc i64 %call to i32          ; <i32> [#uses=1]
+       ret i32 %conv
+}
+
+declare i64 @strtol(i8*, i8** nocapture, i32) nounwind