diff --git a/clang/lib/CodeGen/CodeGenModule.cpp b/clang/lib/CodeGen/CodeGenModule.cpp index a9d67b35641f..04693fd502e0 100644 --- a/clang/lib/CodeGen/CodeGenModule.cpp +++ b/clang/lib/CodeGen/CodeGenModule.cpp @@ -373,8 +373,10 @@ void CodeGenModule::SetInternalFunctionAttributes(const Decl *D, } void CodeGenModule::SetFunctionAttributes(const FunctionDecl *FD, - llvm::Function *F) { - SetLLVMFunctionAttributes(FD, getTypes().getFunctionInfo(FD), F); + llvm::Function *F, + bool IsIncompleteFunction) { + if (!IsIncompleteFunction) + SetLLVMFunctionAttributes(FD, getTypes().getFunctionInfo(FD), F); // Only a few attributes are set on declarations; these may later be // overridden by a definition. @@ -625,18 +627,19 @@ llvm::Constant *CodeGenModule::GetOrCreateLLVMFunction(const char *MangledName, // This function doesn't have a complete type (for example, the return // type is an incomplete struct). Use a fake type instead, and make // sure not to try to set attributes. - bool ShouldSetAttributes = true; + bool IsIncompleteFunction = false; if (!isa(Ty)) { Ty = llvm::FunctionType::get(llvm::Type::VoidTy, std::vector(), false); - ShouldSetAttributes = false; + IsIncompleteFunction = true; } llvm::Function *F = llvm::Function::Create(cast(Ty), llvm::Function::ExternalLinkage, "", &getModule()); F->setName(MangledName); - if (D.getDecl() && ShouldSetAttributes) - SetFunctionAttributes(cast(D.getDecl()), F); + if (D.getDecl()) + SetFunctionAttributes(cast(D.getDecl()), F, + IsIncompleteFunction); Entry = F; return F; } diff --git a/clang/lib/CodeGen/CodeGenModule.h b/clang/lib/CodeGen/CodeGenModule.h index 5c780e755b25..4d50e8946bd6 100644 --- a/clang/lib/CodeGen/CodeGenModule.h +++ b/clang/lib/CodeGen/CodeGenModule.h @@ -397,12 +397,13 @@ private: /// SetFunctionDefinitionAttributes - Set attributes for a global definition. void SetFunctionDefinitionAttributes(const FunctionDecl *D, - llvm::GlobalValue *GV); + llvm::GlobalValue *GV); /// SetFunctionAttributes - Set function attributes for a function /// declaration. void SetFunctionAttributes(const FunctionDecl *FD, - llvm::Function *F); + llvm::Function *F, + bool IsIncompleteFunction); /// EmitGlobal - Emit code for a singal global function or var decl. Forward /// declarations are emitted lazily. diff --git a/clang/test/CodeGen/weak-incomplete.c b/clang/test/CodeGen/weak-incomplete.c new file mode 100644 index 000000000000..a15dbac03cb7 --- /dev/null +++ b/clang/test/CodeGen/weak-incomplete.c @@ -0,0 +1,5 @@ +// RUN: clang-cc -emit-llvm < %s | grep 'extern_weak' | count 1 + +struct S; +void __attribute__((weak)) foo1(struct S); +void (*foo2)(struct S) = foo1;