From b51325dbdb10f0b4ab2b9c5ec4a979dffc246794 Mon Sep 17 00:00:00 2001 From: Sanjoy Das Date: Fri, 11 Mar 2016 19:08:34 +0000 Subject: 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 --- .../test/Transforms/Inline/deoptimize-intrinsic.ll | 90 ++++++++++++++++++++++ llvm/test/Verifier/deoptimize-intrinsic.ll | 42 ++++++++++ 2 files changed, 132 insertions(+) create mode 100644 llvm/test/Transforms/Inline/deoptimize-intrinsic.ll create mode 100644 llvm/test/Verifier/deoptimize-intrinsic.ll (limited to 'llvm/test') 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 +} -- cgit v1.2.3