diff options
author | Amaury Sechet <deadalnix@gmail.com> | 2016-02-18 20:38:32 +0000 |
---|---|---|
committer | Amaury Sechet <deadalnix@gmail.com> | 2016-02-18 20:38:32 +0000 |
commit | 10ae0ad5058f87605399288b1a78932d4a072378 (patch) | |
tree | 4fa3c0ec404c5ea0563f33cf0da2431b23154245 | |
parent | 1dddbb710649c5338f68e64e20be6abe359b3c89 (diff) |
Add support for invoke/landingpad/resume in C API test
Summary: As per title. There was a lot of part missing in the C API, so I had to extend the invoke and landingpad API.
Reviewers: echristo, joker.eph, Wallbraker
Subscribers: llvm-commits
Differential Revision: http://reviews.llvm.org/D17359
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@261254 91177308-0d34-0410-b5e6-96231b3b80d8
-rw-r--r-- | include/llvm-c/Core.h | 52 | ||||
-rw-r--r-- | lib/IR/Core.cpp | 34 | ||||
-rw-r--r-- | test/Bindings/llvm-c/invoke.ll | 82 | ||||
-rw-r--r-- | tools/llvm-c-test/echo.cpp | 36 |
4 files changed, 203 insertions, 1 deletions
diff --git a/include/llvm-c/Core.h b/include/llvm-c/Core.h index c3dacc5b08f..a22e910c455 100644 --- a/include/llvm-c/Core.h +++ b/include/llvm-c/Core.h @@ -1804,6 +1804,13 @@ LLVMValueRef LLVMAddAlias(LLVMModuleRef M, LLVMTypeRef Ty, LLVMValueRef Aliasee, void LLVMDeleteFunction(LLVMValueRef Fn); /** + * Check whether the given function has a personality function. + * + * @see llvm::Function::hasPersonalityFn() + */ +LLVMBool LLVMHasPersonalityFn(LLVMValueRef Fn); + +/** * Obtain the personality function attached to the function. * * @see llvm::Function::getPersonalityFn() @@ -2469,6 +2476,42 @@ LLVMBool LLVMIsTailCall(LLVMValueRef CallInst); void LLVMSetTailCall(LLVMValueRef CallInst, LLVMBool IsTailCall); /** + * Return the normal destination basic block. + * + * This only works on llvm::InvokeInst instructions. + * + * @see llvm::InvokeInst::getNormalDest() + */ +LLVMBasicBlockRef LLVMGetNormalDest(LLVMValueRef InvokeInst); + +/** + * Return the unwind destination basic block. + * + * This only works on llvm::InvokeInst instructions. + * + * @see llvm::InvokeInst::getUnwindDest() + */ +LLVMBasicBlockRef LLVMGetUnwindDest(LLVMValueRef InvokeInst); + +/** + * Set the normal destination basic block. + * + * This only works on llvm::InvokeInst instructions. + * + * @see llvm::InvokeInst::setNormalDest() + */ +void LLVMSetNormalDest(LLVMValueRef InvokeInst, LLVMBasicBlockRef B); + +/** + * Set the unwind destination basic block. + * + * This only works on llvm::InvokeInst instructions. + * + * @see llvm::InvokeInst::setUnwindDest() + */ +void LLVMSetUnwindDest(LLVMValueRef InvokeInst, LLVMBasicBlockRef B); + +/** * @} */ @@ -2706,9 +2749,18 @@ void LLVMAddCase(LLVMValueRef Switch, LLVMValueRef OnVal, /* Add a destination to the indirectbr instruction */ void LLVMAddDestination(LLVMValueRef IndirectBr, LLVMBasicBlockRef Dest); +/* Get the number of clauses on the landingpad instruction */ +unsigned LLVMGetNumClauses(LLVMValueRef LandingPad); + +/* Get the value of the clause at idnex Idx on the landingpad instruction */ +LLVMValueRef LLVMGetClause(LLVMValueRef LandingPad, unsigned Idx); + /* Add a catch or filter clause to the landingpad instruction */ void LLVMAddClause(LLVMValueRef LandingPad, LLVMValueRef ClauseVal); +/* Get the 'cleanup' flag in the landingpad instruction */ +LLVMBool LLVMIsCleanup(LLVMValueRef LandingPad); + /* Set the 'cleanup' flag in the landingpad instruction */ void LLVMSetCleanup(LLVMValueRef LandingPad, LLVMBool Val); diff --git a/lib/IR/Core.cpp b/lib/IR/Core.cpp index 585a24020bc..f6c6d151062 100644 --- a/lib/IR/Core.cpp +++ b/lib/IR/Core.cpp @@ -1701,6 +1701,10 @@ void LLVMDeleteFunction(LLVMValueRef Fn) { unwrap<Function>(Fn)->eraseFromParent(); } +LLVMBool LLVMHasPersonalityFn(LLVMValueRef Fn) { + return unwrap<Function>(Fn)->hasPersonalityFn(); +} + LLVMValueRef LLVMGetPersonalityFn(LLVMValueRef Fn) { return wrap(unwrap<Function>(Fn)->getPersonalityFn()); } @@ -2107,6 +2111,24 @@ void LLVMSetTailCall(LLVMValueRef Call, LLVMBool isTailCall) { unwrap<CallInst>(Call)->setTailCall(isTailCall); } +/*--.. Operations on invoke instructions (only) ............................--*/ + +LLVMBasicBlockRef LLVMGetNormalDest(LLVMValueRef Invoke) { + return wrap(unwrap<InvokeInst>(Invoke)->getNormalDest()); +} + +LLVMBasicBlockRef LLVMGetUnwindDest(LLVMValueRef Invoke) { + return wrap(unwrap<InvokeInst>(Invoke)->getUnwindDest()); +} + +void LLVMSetNormalDest(LLVMValueRef Invoke, LLVMBasicBlockRef B) { + unwrap<InvokeInst>(Invoke)->setNormalDest(unwrap(B)); +} + +void LLVMSetUnwindDest(LLVMValueRef Invoke, LLVMBasicBlockRef B) { + unwrap<InvokeInst>(Invoke)->setUnwindDest(unwrap(B)); +} + /*--.. Operations on terminators ...........................................--*/ unsigned LLVMGetNumSuccessors(LLVMValueRef Term) { @@ -2342,11 +2364,23 @@ void LLVMAddDestination(LLVMValueRef IndirectBr, LLVMBasicBlockRef Dest) { unwrap<IndirectBrInst>(IndirectBr)->addDestination(unwrap(Dest)); } +unsigned LLVMGetNumClauses(LLVMValueRef LandingPad) { + return unwrap<LandingPadInst>(LandingPad)->getNumClauses(); +} + +LLVMValueRef LLVMGetClause(LLVMValueRef LandingPad, unsigned Idx) { + return wrap(unwrap<LandingPadInst>(LandingPad)->getClause(Idx)); +} + void LLVMAddClause(LLVMValueRef LandingPad, LLVMValueRef ClauseVal) { unwrap<LandingPadInst>(LandingPad)-> addClause(cast<Constant>(unwrap(ClauseVal))); } +LLVMBool LLVMIsCleanup(LLVMValueRef LandingPad) { + return unwrap<LandingPadInst>(LandingPad)->isCleanup(); +} + void LLVMSetCleanup(LLVMValueRef LandingPad, LLVMBool Val) { unwrap<LandingPadInst>(LandingPad)->setCleanup(Val); } diff --git a/test/Bindings/llvm-c/invoke.ll b/test/Bindings/llvm-c/invoke.ll new file mode 100644 index 00000000000..22e3c8523ab --- /dev/null +++ b/test/Bindings/llvm-c/invoke.ll @@ -0,0 +1,82 @@ +; RUN: llvm-as < %s | llvm-dis > %t.orig +; RUN: llvm-as < %s | llvm-c-test --echo > %t.echo +; RUN: diff -w %t.orig %t.echo + +%C6object9ClassInfo = type { %C6object9ClassInfo__vtbl*, %C6object9ClassInfo* } +%C6object9ClassInfo__vtbl = type { %C6object9ClassInfo* } +%C6object9Exception__vtbl = type { %C6object9ClassInfo* } +%C6object6Object = type { %C6object6Object__vtbl* } +%C6object6Object__vtbl = type { %C6object9ClassInfo* } +%C6object9Throwable = type { %C6object9Throwable__vtbl* } +%C6object9Throwable__vtbl = type { %C6object9ClassInfo* } + +@C6object9ClassInfo__ClassInfo = linkonce_odr constant %C6object9ClassInfo { %C6object9ClassInfo__vtbl* @C6object9ClassInfo__vtblZ, %C6object9ClassInfo* @C6object8TypeInfo__ClassInfo } +@C6object9ClassInfo__vtblZ = linkonce_odr constant %C6object9ClassInfo__vtbl { %C6object9ClassInfo* @C6object9ClassInfo__ClassInfo } +@C6object8TypeInfo__ClassInfo = linkonce_odr constant %C6object9ClassInfo { %C6object9ClassInfo__vtbl* @C6object9ClassInfo__vtblZ, %C6object9ClassInfo* @C6object6Object__ClassInfo } +@C6object6Object__ClassInfo = linkonce_odr constant %C6object9ClassInfo { %C6object9ClassInfo__vtbl* @C6object9ClassInfo__vtblZ, %C6object9ClassInfo* @C6object6Object__ClassInfo } +@C6object9Throwable__ClassInfo = linkonce_odr constant %C6object9ClassInfo { %C6object9ClassInfo__vtbl* @C6object9ClassInfo__vtblZ, %C6object9ClassInfo* @C6object6Object__ClassInfo } +@C6object9Exception__ClassInfo = linkonce_odr constant %C6object9ClassInfo { %C6object9ClassInfo__vtbl* @C6object9ClassInfo__vtblZ, %C6object9ClassInfo* @C6object9Throwable__ClassInfo } +@C6object9Exception__vtblZ = linkonce_odr constant %C6object9Exception__vtbl { %C6object9ClassInfo* @C6object9Exception__ClassInfo } +@C6object5Error__ClassInfo = linkonce_odr constant %C6object9ClassInfo { %C6object9ClassInfo__vtbl* @C6object9ClassInfo__vtblZ, %C6object9ClassInfo* @C6object9Throwable__ClassInfo } + +define i32 @_D8test01494mainFMZi() personality i32 (i32, i32, i64, i8*, i8*)* @__sd_eh_personality { +body: + %0 = invoke i8* @_d_allocmemory(i64 8) + to label %then unwind label %landingPad + +then: ; preds = %body + %1 = bitcast i8* %0 to i8** + store i8* bitcast (%C6object9Exception__vtbl* @C6object9Exception__vtblZ to i8*), i8** %1, align 8 + %2 = bitcast i8* %0 to %C6object6Object* + invoke void @_D6object6Object6__ctorFMC6object6ObjectZv(%C6object6Object* %2) + to label %then1 unwind label %landingPad + +then1: ; preds = %then + %3 = bitcast i8* %0 to %C6object9Throwable* + invoke void @__sd_eh_throw(%C6object9Throwable* %3) + to label %then2 unwind label %landingPad + +then2: ; preds = %then1 + unreachable + +landingPad: ; preds = %then1, %then, %body + %4 = landingpad { i8*, i32 } + cleanup + catch %C6object9ClassInfo* @C6object5Error__ClassInfo + catch %C6object9ClassInfo* @C6object9Exception__ClassInfo + catch %C6object9ClassInfo* @C6object9Throwable__ClassInfo + %5 = extractvalue { i8*, i32 } %4, 1 + %6 = tail call i32 @llvm.eh.typeid.for(i8* bitcast (%C6object9ClassInfo* @C6object5Error__ClassInfo to i8*)) + %7 = icmp eq i32 %6, %5 + br i1 %7, label %catch, label %unwind3 + +catch: ; preds = %unwind5, %unwind3, %landingPad + %merge = phi i32 [ 23, %landingPad ], [ 19, %unwind3 ], [ 13, %unwind5 ] + ret i32 %merge + +unwind3: ; preds = %landingPad + %8 = tail call i32 @llvm.eh.typeid.for(i8* bitcast (%C6object9ClassInfo* @C6object9Exception__ClassInfo to i8*)) + %9 = icmp eq i32 %8, %5 + br i1 %9, label %catch, label %unwind5 + +unwind5: ; preds = %unwind3 + %10 = tail call i32 @llvm.eh.typeid.for(i8* bitcast (%C6object9ClassInfo* @C6object9Throwable__ClassInfo to i8*)) + %11 = icmp eq i32 %10, %5 + br i1 %11, label %catch, label %unwind7 + +unwind7: ; preds = %unwind5 + resume { i8*, i32 } %4 +} + +declare void @_D6object6Object6__ctorFMC6object6ObjectZv(%C6object6Object*) + +declare i8* @_d_allocmemory(i64) + +declare i32 @__sd_eh_personality(i32, i32, i64, i8*, i8*) + +declare void @__sd_eh_throw(%C6object9Throwable*) + +; Function Attrs: nounwind readnone +declare i32 @llvm.eh.typeid.for(i8*) #0 + +attributes #0 = { nounwind readnone }
\ No newline at end of file diff --git a/tools/llvm-c-test/echo.cpp b/tools/llvm-c-test/echo.cpp index 229530155fc..35c73e22fbc 100644 --- a/tools/llvm-c-test/echo.cpp +++ b/tools/llvm-c-test/echo.cpp @@ -388,8 +388,19 @@ struct FunCloner { } case LLVMSwitch: case LLVMIndirectBr: - case LLVMInvoke: break; + case LLVMInvoke: { + SmallVector<LLVMValueRef, 8> Args; + int ArgCount = LLVMGetNumArgOperands(Src); + for (int i = 0; i < ArgCount; i++) + Args.push_back(CloneValue(LLVMGetOperand(Src, i))); + LLVMValueRef Fn = CloneValue(LLVMGetCalledValue(Src)); + LLVMBasicBlockRef Then = DeclareBB(LLVMGetNormalDest(Src)); + LLVMBasicBlockRef Unwind = DeclareBB(LLVMGetUnwindDest(Src)); + Dst = LLVMBuildInvoke(Builder, Fn, Args.data(), ArgCount, + Then, Unwind, Name); + break; + } case LLVMUnreachable: Dst = LLVMBuildUnreachable(Builder); break; @@ -536,6 +547,20 @@ struct FunCloner { Args.push_back(CloneValue(LLVMGetOperand(Src, i))); LLVMValueRef Fn = CloneValue(LLVMGetCalledValue(Src)); Dst = LLVMBuildCall(Builder, Fn, Args.data(), ArgCount, Name); + LLVMSetTailCall(Dst, LLVMIsTailCall(Src)); + break; + } + case LLVMResume: { + Dst = LLVMBuildResume(Builder, CloneValue(LLVMGetOperand(Src, 0))); + break; + } + case LLVMLandingPad: { + // The landing pad API is a bit screwed up for historical reasons. + Dst = LLVMBuildLandingPad(Builder, CloneType(Src), nullptr, 0, Name); + unsigned NumClauses = LLVMGetNumClauses(Src); + for (unsigned i = 0; i < NumClauses; ++i) + LLVMAddClause(Dst, CloneValue(LLVMGetClause(Src, i))); + LLVMSetCleanup(Dst, LLVMIsCleanup(Src)); break; } case LLVMExtractValue: { @@ -788,6 +813,15 @@ FunClone: LLVMValueRef Fun = LLVMGetNamedFunction(M, Name); if (!Fun) report_fatal_error("Function must have been declared already"); + + if (LLVMHasPersonalityFn(Cur)) { + const char *FName = LLVMGetValueName(LLVMGetPersonalityFn(Cur)); + LLVMValueRef P = LLVMGetNamedFunction(M, FName); + if (!P) + report_fatal_error("Could not find personality function"); + LLVMSetPersonalityFn(Fun, P); + } + FunCloner FC(Cur, Fun); FC.CloneBBs(Cur); |