Make IRGen compatible with declaring a function with incomplete

return/argument types.  (The generated IR isn't ideal, but we can't 
really do better in general.)

llvm-svn: 66132
This commit is contained in:
Eli Friedman 2009-03-05 04:18:07 +00:00
parent 9ce427b21f
commit cecc21d2f6
2 changed files with 16 additions and 5 deletions

View File

@ -779,13 +779,24 @@ void CodeGenModule::EmitGlobalVarDefinition(const VarDecl *D) {
llvm::GlobalValue *
CodeGenModule::EmitForwardFunctionDefinition(const FunctionDecl *D,
const llvm::Type *Ty) {
if (!Ty)
bool DoSetAttributes = true;
if (!Ty) {
Ty = getTypes().ConvertType(D->getType());
if (!isa<llvm::FunctionType>(Ty)) {
// 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.
Ty = llvm::FunctionType::get(llvm::Type::VoidTy,
std::vector<const llvm::Type*>(), false);
DoSetAttributes = false;
}
}
llvm::Function *F = llvm::Function::Create(cast<llvm::FunctionType>(Ty),
llvm::Function::ExternalLinkage,
getMangledName(D),
&getModule());
SetFunctionAttributes(D, F);
if (DoSetAttributes)
SetFunctionAttributes(D, F);
return F;
}

View File

@ -1,9 +1,9 @@
// RUN: clang -emit-llvm %s -o - | not grep opaque
enum teste1 (*test1)(void);
struct tests2 (*test2)();
enum teste1 test1f(void), (*test1)(void) = test1f;
struct tests2 test2f(), (*test2)() = test2f;
struct tests3;
void (*test3)(struct tests3);
void test3f(struct tests3), (*test3)(struct tests3) = test3f;
enum teste1 { TEST1 };
struct tests2 { int x,y,z,a,b,c,d,e,f,g; };
struct tests3 { float x; };