diff options
author | Steve Naroff <snaroff@apple.com> | 2008-01-09 23:34:55 +0000 |
---|---|---|
committer | Steve Naroff <snaroff@apple.com> | 2008-01-09 23:34:55 +0000 |
commit | ffce4d572df79a95d21138fc28bb78f753fb2653 (patch) | |
tree | c01ac35cc9e6b18358749542065e4001d30cd766 | |
parent | 974c676306758c8c84f5c25e3708186557a678bd (diff) |
Fix Sema::ActOnDeclarator() to call MergeFunctionDecl for function decls that aren't in scope. Since C functions are in a flat namespace, we need to give them special treatment (when compared with variables and typedefs).
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@45789 91177308-0d34-0410-b5e6-96231b3b80d8
-rw-r--r-- | Sema/SemaDecl.cpp | 16 | ||||
-rw-r--r-- | test/Sema/predefined-function.c | 5 |
2 files changed, 13 insertions, 8 deletions
diff --git a/Sema/SemaDecl.cpp b/Sema/SemaDecl.cpp index 7d5fc33161..5e2526e451 100644 --- a/Sema/SemaDecl.cpp +++ b/Sema/SemaDecl.cpp @@ -637,9 +637,6 @@ Sema::ActOnDeclarator(Scope *S, Declarator &D, DeclTy *lastDecl) { // See if this is a redefinition of a variable in the same scope. ScopedDecl *PrevDecl = LookupScopedDecl(II, Decl::IDNS_Ordinary, D.getIdentifierLoc(), S); - if (PrevDecl && !S->isDeclScope(PrevDecl)) - PrevDecl = 0; // If in outer scope, it isn't the same thing. - ScopedDecl *New; bool InvalidDecl = false; @@ -653,8 +650,9 @@ Sema::ActOnDeclarator(Scope *S, Declarator &D, DeclTy *lastDecl) { // Handle attributes prior to checking for duplicates in MergeVarDecl HandleDeclAttributes(NewTD, D.getDeclSpec().getAttributes(), D.getAttributes()); - // Merge the decl with the existing one if appropriate. - if (PrevDecl) { + // Merge the decl with the existing one if appropriate. If the decl is + // in an outer scope, it isn't the same thing. + if (PrevDecl && S->isDeclScope(PrevDecl)) { NewTD = MergeTypeDefDecl(NewTD, PrevDecl); if (NewTD == 0) return 0; } @@ -692,7 +690,8 @@ Sema::ActOnDeclarator(Scope *S, Declarator &D, DeclTy *lastDecl) { // Transfer ownership of DeclSpec attributes to FunctionDecl D.getDeclSpec().clearAttributes(); - // Merge the decl with the existing one if appropriate. + // Merge the decl with the existing one if appropriate. Since C functions + // are in a flat namespace, make sure we consider decls in outer scopes. if (PrevDecl) { NewFD = MergeFunctionDecl(NewFD, PrevDecl); if (NewFD == 0) return 0; @@ -731,8 +730,9 @@ Sema::ActOnDeclarator(Scope *S, Declarator &D, DeclTy *lastDecl) { HandleDeclAttributes(NewVD, D.getDeclSpec().getAttributes(), D.getAttributes()); - // Merge the decl with the existing one if appropriate. - if (PrevDecl) { + // Merge the decl with the existing one if appropriate. If the decl is + // in an outer scope, it isn't the same thing. + if (PrevDecl && S->isDeclScope(PrevDecl)) { NewVD = MergeVarDecl(NewVD, PrevDecl); if (NewVD == 0) return 0; } diff --git a/test/Sema/predefined-function.c b/test/Sema/predefined-function.c index 80b5ab0c6b..d554b8faa8 100644 --- a/test/Sema/predefined-function.c +++ b/test/Sema/predefined-function.c @@ -4,9 +4,14 @@ char *funk(int format); enum Test {A=-1}; char *funk(enum Test x); +int eli(float b); // expected-error {{previous definition is here}} +int b(int c) {return 1;} + int foo(); int foo() { + int eli(int (int)); // expected-error {{redefinition of 'eli'}} + eli(b); return 0; } |