forked from OSchip/llvm-project
Don't emit prologues or epilogues for naked functions (PR18791, PR20028)
For naked functions with parameters, Clang would still emit stores in the prologue that would clobber the stack, because LLVM doesn't set up a stack frame. (This shows up in -O0 compiles, because the stores are optimized away otherwise.) For example: __attribute__((naked)) int f(int x) { asm("movl $42, %eax"); asm("retl"); } Would result in: _Z1fi: movl 12(%esp), %eax movl %eax, (%esp) <--- Oops. movl $42, %eax retl Differential Revision: http://reviews.llvm.org/D5183 llvm-svn: 217198
This commit is contained in:
parent
8d082d187e
commit
d71907dd07
|
@ -1462,6 +1462,10 @@ static bool shouldAddNonNullAttr(const Decl *FD, const ParmVarDecl *PVD) {
|
||||||
void CodeGenFunction::EmitFunctionProlog(const CGFunctionInfo &FI,
|
void CodeGenFunction::EmitFunctionProlog(const CGFunctionInfo &FI,
|
||||||
llvm::Function *Fn,
|
llvm::Function *Fn,
|
||||||
const FunctionArgList &Args) {
|
const FunctionArgList &Args) {
|
||||||
|
if (CurCodeDecl && CurCodeDecl->hasAttr<NakedAttr>())
|
||||||
|
// Naked functions don't have prologues.
|
||||||
|
return;
|
||||||
|
|
||||||
// If this is an implicit-return-zero function, go ahead and
|
// If this is an implicit-return-zero function, go ahead and
|
||||||
// initialize the return value. TODO: it might be nice to have
|
// initialize the return value. TODO: it might be nice to have
|
||||||
// a more general mechanism for this that didn't require synthesized
|
// a more general mechanism for this that didn't require synthesized
|
||||||
|
@ -1985,6 +1989,12 @@ static llvm::StoreInst *findDominatingStoreToReturnValue(CodeGenFunction &CGF) {
|
||||||
void CodeGenFunction::EmitFunctionEpilog(const CGFunctionInfo &FI,
|
void CodeGenFunction::EmitFunctionEpilog(const CGFunctionInfo &FI,
|
||||||
bool EmitRetDbgLoc,
|
bool EmitRetDbgLoc,
|
||||||
SourceLocation EndLoc) {
|
SourceLocation EndLoc) {
|
||||||
|
if (CurCodeDecl && CurCodeDecl->hasAttr<NakedAttr>()) {
|
||||||
|
// Naked functions don't have epilogues.
|
||||||
|
Builder.CreateUnreachable();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Functions with no result always return void.
|
// Functions with no result always return void.
|
||||||
if (!ReturnValue) {
|
if (!ReturnValue) {
|
||||||
Builder.CreateRetVoid();
|
Builder.CreateRetVoid();
|
||||||
|
|
|
@ -15,4 +15,12 @@ void t1()
|
||||||
__attribute((naked, always_inline)) void t2() {
|
__attribute((naked, always_inline)) void t2() {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Make sure not to generate prolog or epilog for naked functions.
|
||||||
|
__attribute((naked)) void t3(int x) {
|
||||||
|
// CHECK: define void @t3(i32)
|
||||||
|
// CHECK-NOT: alloca
|
||||||
|
// CHECK-NOT: store
|
||||||
|
// CHECK: unreachable
|
||||||
|
}
|
||||||
|
|
||||||
// CHECK: attributes [[NAKED]] = { naked noinline nounwind{{.*}} }
|
// CHECK: attributes [[NAKED]] = { naked noinline nounwind{{.*}} }
|
||||||
|
|
Loading…
Reference in New Issue