From: John McCall Date: Sat, 15 Aug 2009 02:09:25 +0000 (+0000) Subject: Disable all recognition of main() in -ffreestanding. Addresses bug #4720. X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=07a5c22bb6fb0674c95205ae189365bf8e1b695e;p=clang Disable all recognition of main() in -ffreestanding. Addresses bug #4720. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@79070 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/include/clang/AST/Decl.h b/include/clang/AST/Decl.h index 697c561cb7..d603e677b2 100644 --- a/include/clang/AST/Decl.h +++ b/include/clang/AST/Decl.h @@ -850,7 +850,7 @@ public: /// \brief Determines whether this is a function "main", which is /// the entry point into an executable program. - bool isMain() const; + bool isMain(ASTContext &Context) const; /// \brief Determines whether this function is a function with /// external, C linkage. diff --git a/lib/AST/Decl.cpp b/lib/AST/Decl.cpp index 6d2f2301ba..c91cb1dba2 100644 --- a/lib/AST/Decl.cpp +++ b/lib/AST/Decl.cpp @@ -414,8 +414,9 @@ void FunctionDecl::setBody(Stmt *B) { EndRangeLoc = B->getLocEnd(); } -bool FunctionDecl::isMain() const { - return getDeclContext()->getLookupContext()->isTranslationUnit() && +bool FunctionDecl::isMain(ASTContext &Context) const { + return !Context.getLangOptions().Freestanding && + getDeclContext()->getLookupContext()->isTranslationUnit() && getIdentifier() && getIdentifier()->isStr("main"); } diff --git a/lib/CodeGen/Mangle.cpp b/lib/CodeGen/Mangle.cpp index 341e230e90..71b84e7001 100644 --- a/lib/CodeGen/Mangle.cpp +++ b/lib/CodeGen/Mangle.cpp @@ -91,7 +91,7 @@ bool CXXNameMangler::mangleFunctionDecl(const FunctionDecl *FD) { // name mangling (always). if (!FD->hasAttr()) { // C functions are not mangled, and "main" is never mangled. - if (!Context.getLangOptions().CPlusPlus || FD->isMain()) + if (!Context.getLangOptions().CPlusPlus || FD->isMain(Context)) return false; // No mangling in an "implicit extern C" header. diff --git a/lib/Sema/SemaDecl.cpp b/lib/Sema/SemaDecl.cpp index ab5578b378..d75f322aa5 100644 --- a/lib/Sema/SemaDecl.cpp +++ b/lib/Sema/SemaDecl.cpp @@ -2766,7 +2766,7 @@ void Sema::CheckFunctionDeclaration(FunctionDecl *NewFD, NamedDecl *&PrevDecl, return NewFD->setInvalidDecl(); } - if (NewFD->isMain()) CheckMain(NewFD); + if (NewFD->isMain(Context)) CheckMain(NewFD); // Semantic checking for this function declaration (in isolation). if (getLangOptions().CPlusPlus) { @@ -3542,7 +3542,7 @@ Sema::DeclPtrTy Sema::ActOnStartOfFunctionDef(Scope *FnBodyScope, DeclPtrTy D) { // definition itself provides a prototype. The aim is to detect // global functions that fail to be declared in header files. if (!FD->isInvalidDecl() && FD->isGlobal() && !isa(FD) && - !FD->isMain()) { + !FD->isMain(Context)) { bool MissingPrototype = true; for (const FunctionDecl *Prev = FD->getPreviousDeclaration(); Prev; Prev = Prev->getPreviousDeclaration()) { @@ -3608,7 +3608,7 @@ Sema::DeclPtrTy Sema::ActOnFinishFunctionBody(DeclPtrTy D, StmtArg BodyArg, Stmt *Body = BodyArg.takeAs(); if (FunctionDecl *FD = dyn_cast_or_null(dcl)) { FD->setBody(Body); - if (FD->isMain()) + if (FD->isMain(Context)) // C and C++ allow for main to automagically return 0. // Implements C++ [basic.start.main]p5 and C99 5.1.2.2.3. FD->setHasImplicitReturnZero(true); diff --git a/test/Sema/freemain.c b/test/Sema/freemain.c new file mode 100644 index 0000000000..a2364df259 --- /dev/null +++ b/test/Sema/freemain.c @@ -0,0 +1,9 @@ +// RUN: clang-cc -fsyntax-only -verify -ffreestanding %s + +// Tests that -ffreestanding disables all special treatment of main(). + +void* allocate(long size); + +void* main(void* context, long size) { + if (context) return allocate(size); +} // expected-warning {{control may reach end of non-void function}}