diff options
author | Nick Lewycky <nicholas@mxc.ca> | 2011-07-18 05:26:13 +0000 |
---|---|---|
committer | Nick Lewycky <nicholas@mxc.ca> | 2011-07-18 05:26:13 +0000 |
commit | 26da4ddfa6d1269731adc1432bf10fb22dde9ba7 (patch) | |
tree | 48d7ffbab42e2d4b3a86513c7f58c67ca5c5ea3e /clang/lib/AST | |
parent | 14b3b4df6ff599c665ffb625920e93c0cbae663a (diff) | |
download | bcm5719-llvm-26da4ddfa6d1269731adc1432bf10fb22dde9ba7.tar.gz bcm5719-llvm-26da4ddfa6d1269731adc1432bf10fb22dde9ba7.zip |
In C99, emit an inline function when encountering an extern redeclaration.
Fixes PR10233!
llvm-svn: 135377
Diffstat (limited to 'clang/lib/AST')
-rw-r--r-- | clang/lib/AST/ASTContext.cpp | 2 | ||||
-rw-r--r-- | clang/lib/AST/Decl.cpp | 26 |
2 files changed, 27 insertions, 1 deletions
diff --git a/clang/lib/AST/ASTContext.cpp b/clang/lib/AST/ASTContext.cpp index 6eada6e22f4..9f3be94a958 100644 --- a/clang/lib/AST/ASTContext.cpp +++ b/clang/lib/AST/ASTContext.cpp @@ -6405,7 +6405,7 @@ bool ASTContext::DeclMustBeEmitted(const Decl *D) { if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) { // Forward declarations aren't required. if (!FD->doesThisDeclarationHaveABody()) - return false; + return FD->doesDeclarationForceExternallyVisibleDefinition(); // Constructors and destructors are required. if (FD->hasAttr<ConstructorAttr>() || FD->hasAttr<DestructorAttr>()) diff --git a/clang/lib/AST/Decl.cpp b/clang/lib/AST/Decl.cpp index 4c323da7eee..d0c285186b0 100644 --- a/clang/lib/AST/Decl.cpp +++ b/clang/lib/AST/Decl.cpp @@ -1762,6 +1762,32 @@ bool FunctionDecl::isInlined() const { return false; } +/// \brief For a function declaration in C or C++, determine whether this +/// declaration causes the definition to be externally visible. +/// +/// Determines whether this is the first non-inline redeclaration of an inline +/// function in a language where "inline" does not normally require an +/// externally visible definition. +bool FunctionDecl::doesDeclarationForceExternallyVisibleDefinition() const { + assert(!doesThisDeclarationHaveABody() && + "Must have a declaration without a body."); + + ASTContext &Context = getASTContext(); + + // In C99 mode, a function may have an inline definition (causing it to + // be deferred) then redeclared later. As a special case, "extern inline" + // is not required to produce an external symbol. + if (Context.getLangOptions().GNUInline || !Context.getLangOptions().C99 || + Context.getLangOptions().CPlusPlus) + return false; + if (getLinkage() != ExternalLinkage || isInlineSpecified()) + return false; + const FunctionDecl *InlineDefinition = 0; + if (hasBody(InlineDefinition)) + return InlineDefinition->isInlineDefinitionExternallyVisible(); + return false; +} + /// \brief For an inline function definition in C or C++, determine whether the /// definition will be externally visible. /// |