Index: llvm/trunk/lib/Target/PowerPC/PPCInstr64Bit.td =================================================================== --- llvm/trunk/lib/Target/PowerPC/PPCInstr64Bit.td +++ llvm/trunk/lib/Target/PowerPC/PPCInstr64Bit.td @@ -347,12 +347,17 @@ } // hasExtraSrcRegAllocReq = 1 } // hasSideEffects = 0 -let hasSideEffects = 1, isBarrier = 1, usesCustomInserter = 1 in { +// While longjmp is a control-flow barrier (fallthrough isn't allowed), setjmp +// is not. +let hasSideEffects = 1, usesCustomInserter = 1 in { let Defs = [CTR8] in def EH_SjLj_SetJmp64 : Pseudo<(outs gprc:$dst), (ins memr:$buf), "#EH_SJLJ_SETJMP64", [(set i32:$dst, (PPCeh_sjlj_setjmp addr:$buf))]>, Requires<[In64BitMode]>; +} + +let hasSideEffects = 1, isBarrier = 1, usesCustomInserter = 1 in { let isTerminator = 1 in def EH_SjLj_LongJmp64 : Pseudo<(outs), (ins memr:$buf), "#EH_SJLJ_LONGJMP64", Index: llvm/trunk/lib/Target/PowerPC/PPCInstrInfo.td =================================================================== --- llvm/trunk/lib/Target/PowerPC/PPCInstrInfo.td +++ llvm/trunk/lib/Target/PowerPC/PPCInstrInfo.td @@ -1550,12 +1550,17 @@ } -let hasSideEffects = 1, isBarrier = 1, usesCustomInserter = 1 in { +// While longjmp is a control-flow barrier (fallthrough isn't allowed), setjmp +// is not. +let hasSideEffects = 1, usesCustomInserter = 1 in { let Defs = [CTR] in def EH_SjLj_SetJmp32 : Pseudo<(outs gprc:$dst), (ins memr:$buf), "#EH_SJLJ_SETJMP32", [(set i32:$dst, (PPCeh_sjlj_setjmp addr:$buf))]>, Requires<[In32BitMode]>; +} + +let hasSideEffects = 1, isBarrier = 1, usesCustomInserter = 1 in { let isTerminator = 1 in def EH_SjLj_LongJmp32 : Pseudo<(outs), (ins memr:$buf), "#EH_SJLJ_LONGJMP32", Index: llvm/trunk/test/CodeGen/PowerPC/sjlj.ll =================================================================== --- llvm/trunk/test/CodeGen/PowerPC/sjlj.ll +++ llvm/trunk/test/CodeGen/PowerPC/sjlj.ll @@ -1,5 +1,5 @@ -; RUN: llc < %s -mtriple=powerpc64-unknown-linux-gnu -mcpu=pwr7 | FileCheck %s -; RUN: llc < %s -mtriple=powerpc64-unknown-linux-gnu -mcpu=a2 | FileCheck -check-prefix=CHECK-NOAV %s +; RUN: llc < %s -mtriple=powerpc64-unknown-linux-gnu -mcpu=pwr7 -verify-machineinstrs | FileCheck %s +; RUN: llc < %s -mtriple=powerpc64-unknown-linux-gnu -mcpu=a2 -verify-machineinstrs | FileCheck -check-prefix=CHECK-NOAV %s target datalayout = "E-p:64:64:64-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:64:64-f32:32:32-f64:64:64-f128:128:128-v128:128:128-n32:64" target triple = "powerpc64-unknown-linux-gnu" @@ -7,6 +7,7 @@ %struct.__sigset_t = type { [16 x i64] } @env_sigill = internal global [1 x %struct.__jmp_buf_tag] zeroinitializer, align 16 +@cond = external global i8, align 1 define void @foo() #0 { entry: @@ -145,6 +146,24 @@ ; CHECK: blr } +define void @test_sjlj_setjmp() #0 { +entry: + %0 = load i8, i8* @cond, align 1 + %tobool = trunc i8 %0 to i1 + br i1 %tobool, label %return, label %end + +end: + %1 = call i32 @llvm.eh.sjlj.setjmp(i8* bitcast ([1 x %struct.__jmp_buf_tag]* @env_sigill to i8*)) + br label %return + +return: + ret void + +; CHECK-LABEL: test_sjlj_setjmp: +; intrinsic llvm.eh.sjlj.setjmp does not call builtin function _setjmp. +; CHECK-NOT: bl _setjmp +} + declare void @bar(i8*) #3 declare i8* @llvm.frameaddress(i32) #2