diff options
| author | Sanjoy Das <sanjoy@playingwithpointers.com> | 2016-03-11 19:08:34 +0000 |
|---|---|---|
| committer | Sanjoy Das <sanjoy@playingwithpointers.com> | 2016-03-11 19:08:34 +0000 |
| commit | b51325dbdb10f0b4ab2b9c5ec4a979dffc246794 (patch) | |
| tree | 5ee98ad08e88068ab4ad4151d67d74bb2ddea6ea /llvm/test | |
| parent | ebce18b6fc00b21826d1077399976d2447798ba7 (diff) | |
| download | bcm5719-llvm-b51325dbdb10f0b4ab2b9c5ec4a979dffc246794.tar.gz bcm5719-llvm-b51325dbdb10f0b4ab2b9c5ec4a979dffc246794.zip | |
Introduce @llvm.experimental.deoptimize
Summary:
This intrinsic, together with deoptimization operand bundles, allow
frontends to express transfer of control and frame-local state from
one (typically more specialized, hence faster) version of a function
into another (typically more generic, hence slower) version.
In languages with a fully integrated managed runtime this intrinsic can
be used to implement "uncommon trap" like functionality. In unmanaged
languages like C and C++, this intrinsic can be used to represent the
slow paths of specialized functions.
Note: this change does not address how `@llvm.experimental_deoptimize`
is lowered. That will be done in a later change.
Reviewers: chandlerc, rnk, atrick, reames
Subscribers: llvm-commits, kmod, mjacob, maksfb, mcrosier, JosephTremoulet
Differential Revision: http://reviews.llvm.org/D17732
llvm-svn: 263281
Diffstat (limited to 'llvm/test')
| -rw-r--r-- | llvm/test/Transforms/Inline/deoptimize-intrinsic.ll | 90 | ||||
| -rw-r--r-- | llvm/test/Verifier/deoptimize-intrinsic.ll | 42 |
2 files changed, 132 insertions, 0 deletions
diff --git a/llvm/test/Transforms/Inline/deoptimize-intrinsic.ll b/llvm/test/Transforms/Inline/deoptimize-intrinsic.ll new file mode 100644 index 00000000000..84e54a09e8a --- /dev/null +++ b/llvm/test/Transforms/Inline/deoptimize-intrinsic.ll @@ -0,0 +1,90 @@ +; RUN: opt -S -always-inline < %s | FileCheck %s + +declare i8 @llvm.experimental.deoptimize.i8(...) + +define i8 @callee(i1* %c) alwaysinline { + %c0 = load volatile i1, i1* %c + br i1 %c0, label %left, label %right + +left: + %c1 = load volatile i1, i1* %c + br i1 %c1, label %lleft, label %lright + +lleft: + %v0 = call i8(...) @llvm.experimental.deoptimize.i8(i32 1) [ "deopt"(i32 1) ] + ret i8 %v0 + +lright: + ret i8 10 + +right: + %c2 = load volatile i1, i1* %c + br i1 %c2, label %rleft, label %rright + +rleft: + %v1 = call i8(...) @llvm.experimental.deoptimize.i8(i32 1, i32 300, float 500.0, <2 x i32*> undef) [ "deopt"(i32 1) ] + ret i8 %v1 + +rright: + %v2 = call i8(...) @llvm.experimental.deoptimize.i8() [ "deopt"(i32 1) ] + ret i8 %v2 +} + +define void @caller_0(i1* %c, i8* %ptr) { +; CHECK-LABEL: @caller_0( +entry: + %v = call i8 @callee(i1* %c) [ "deopt"(i32 2) ] + store i8 %v, i8* %ptr + ret void + +; CHECK: lleft.i: +; CHECK-NEXT: call void (...) @llvm.experimental.deoptimize.isVoid(i32 1) [ "deopt"(i32 2, i32 1) ] +; CHECK-NEXT: ret void + +; CHECK: rleft.i: +; CHECK-NEXT: call void (...) @llvm.experimental.deoptimize.isVoid(i32 1, i32 300, float 5.000000e+02, <2 x i32*> undef) [ "deopt"(i32 2, i32 1) ] +; CHECK-NEXT: ret void + +; CHECK: rright.i: +; CHECK-NEXT: call void (...) @llvm.experimental.deoptimize.isVoid() [ "deopt"(i32 2, i32 1) ] +; CHECK-NEXT: ret void + +; CHECK: callee.exit: +; CHECK-NEXT: store i8 10, i8* %ptr +; CHECK-NEXT: ret void + +} + +define i32 @caller_1(i1* %c, i8* %ptr) personality i8 3 { +; CHECK-LABEL: @caller_1( +entry: + %v = invoke i8 @callee(i1* %c) [ "deopt"(i32 3) ] to label %normal + unwind label %unwind + +; CHECK: lleft.i: +; CHECK-NEXT: %0 = call i32 (...) @llvm.experimental.deoptimize.i32(i32 1) [ "deopt"(i32 3, i32 1) ] +; CHECK-NEXT: ret i32 %0 + +; CHECK: rleft.i: +; CHECK-NEXT: %1 = call i32 (...) @llvm.experimental.deoptimize.i32(i32 1, i32 300, float 5.000000e+02, <2 x i32*> undef) [ "deopt"(i32 3, i32 1) ] +; CHECK-NEXT: ret i32 %1 + +; CHECK: rright.i: +; CHECK-NEXT: %2 = call i32 (...) @llvm.experimental.deoptimize.i32() [ "deopt"(i32 3, i32 1) ] +; CHECK-NEXT: ret i32 %2 + +; CHECK: callee.exit: +; CHECK-NEXT: br label %normal + +; CHECK: normal: +; CHECK-NEXT: store i8 10, i8* %ptr +; CHECK-NEXT: ret i32 42 + +unwind: + %lp = landingpad i32 cleanup + ret i32 43 + +normal: + store i8 %v, i8* %ptr + ret i32 42 +} diff --git a/llvm/test/Verifier/deoptimize-intrinsic.ll b/llvm/test/Verifier/deoptimize-intrinsic.ll new file mode 100644 index 00000000000..81e4d3a8981 --- /dev/null +++ b/llvm/test/Verifier/deoptimize-intrinsic.ll @@ -0,0 +1,42 @@ +; RUN: not opt -verify < %s 2>&1 | FileCheck %s + +declare i8 @llvm.experimental.deoptimize.i8(...) +declare void @llvm.experimental.deoptimize.isVoid(...) + +declare void @unknown() + +define void @f_notail() { +entry: + call void(...) @llvm.experimental.deoptimize.isVoid(i32 0) [ "deopt"() ] +; CHECK: calls to experimental_deoptimize must be followed by a return + call void @unknown() + ret void +} + +define void @f_nodeopt() { +entry: + call void(...) @llvm.experimental.deoptimize.isVoid() +; CHECK: experimental_deoptimize must have exactly one "deopt" operand bundle + ret void +} + +define void @f_invoke() personality i8 3 { +entry: + invoke void(...) @llvm.experimental.deoptimize.isVoid(i32 0, float 0.0) to label %ok unwind label %not_ok +; CHECK: experimental_deoptimize cannot be invoked + +ok: + ret void + +not_ok: + %0 = landingpad { i8*, i32 } + filter [0 x i8*] zeroinitializer + ret void +} + +define i8 @f_incorrect_return() { +entry: + %val = call i8(...) @llvm.experimental.deoptimize.i8() [ "deopt"() ] +; CHECK: calls to experimental_deoptimize must be followed by a return of the value computed by experimental_deoptimize + ret i8 0 +} |

