From: Alexander Musman Date: Thu, 26 Nov 2015 09:34:30 +0000 (+0000) Subject: Fix for merging decls in pragma weak X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=f4db326a17df0b0b5b87ad3e9eee073da0747c33;p=clang Fix for merging decls in pragma weak Calling CheckFunctionDeclaration so that 2 decls for the 'weak' are merged. Differential Revision: http://reviews.llvm.org/D13048 git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@254143 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/Sema/SemaDeclAttr.cpp b/lib/Sema/SemaDeclAttr.cpp index c00914663a..13a11bcac9 100644 --- a/lib/Sema/SemaDeclAttr.cpp +++ b/lib/Sema/SemaDeclAttr.cpp @@ -5435,17 +5435,22 @@ NamedDecl * Sema::DeclClonePragmaWeak(NamedDecl *ND, IdentifierInfo *II, assert(isa(ND) || isa(ND)); NamedDecl *NewD = nullptr; if (FunctionDecl *FD = dyn_cast(ND)) { - FunctionDecl *NewFD; - // FIXME: Missing call to CheckFunctionDeclaration(). // FIXME: Mangling? // FIXME: Is the qualifier info correct? // FIXME: Is the DeclContext correct? - NewFD = FunctionDecl::Create(FD->getASTContext(), FD->getDeclContext(), - Loc, Loc, DeclarationName(II), - FD->getType(), FD->getTypeSourceInfo(), - SC_None, false/*isInlineSpecified*/, - FD->hasPrototype(), - false/*isConstexprSpecified*/); + + LookupResult Previous(*this, II, Loc, LookupOrdinaryName); + LookupParsedName(Previous, TUScope, nullptr, true); + + auto NewFD = FunctionDecl::Create( + FD->getASTContext(), FD->getDeclContext(), Loc, Loc, + DeclarationName(II), FD->getType(), FD->getTypeSourceInfo(), SC_None, + false /*isInlineSpecified*/, FD->hasPrototype(), + false /*isConstexprSpecified*/); + + CheckFunctionDeclaration(TUScope, NewFD, Previous, + false /*IsExplicitSpecialization*/); + NewD = NewFD; if (FD->getQualifier()) diff --git a/test/CodeGen/pragma-weak.c b/test/CodeGen/pragma-weak.c index 36abca5de3..e63277fc23 100644 --- a/test/CodeGen/pragma-weak.c +++ b/test/CodeGen/pragma-weak.c @@ -17,6 +17,7 @@ // CHECK-DAG: @mix2 = weak alias void (), void ()* @__mix2 // CHECK-DAG: @a1 = weak alias void (), void ()* @__a1 // CHECK-DAG: @xxx = weak alias void (), void ()* @__xxx +// CHECK-DAG: @weakfoo = weak alias void {{.*}} @localfoo @@ -173,6 +174,14 @@ label: // CHECK: declare extern_weak i32 @PR16705b() // CHECK: declare extern_weak i32 @PR16705c() +// In this test case, we have a declaration of weakfoo before #pragma weak. +// Test that 2 decls for the weakfoo are merged. +extern void weakfoo(); +void localfoo() { } +#pragma weak weakfoo=localfoo +extern void externmain() { return weakfoo(); } +// CHECK-LABEL: define void @externmain() +// CHECK: call{{.*}}@weakfoo ///////////// TODO: stuff that still doesn't work diff --git a/test/Sema/pragma-weak.c b/test/Sema/pragma-weak.c index c14125eac9..031e78f15f 100644 --- a/test/Sema/pragma-weak.c +++ b/test/Sema/pragma-weak.c @@ -9,3 +9,9 @@ void __a3(void) __attribute((noinline)); #pragma weak a3 = __a3 // expected-note {{previous definition}} void a3(void) __attribute((alias("__a3"))); // expected-error {{redefinition of 'a3'}} void __a3(void) {} + +extern void weak2foo(int); // expected-note {{previous declaration is here}} expected-note {{'weak2foo' declared here}} +void local2foo(double d1, double d2) { } +#pragma weak weak2foo=local2foo // expected-error {{conflicting types for 'weak2foo'}} +extern void extern2main() { return weak2foo(); } // expected-error {{too few arguments to function call, expected 1, have 0}} +