From 2bfeb7133f3fb7c2d98e0b177c3897e4b0c5a822 Mon Sep 17 00:00:00 2001 From: Bruno Cardoso Lopes Date: Thu, 18 Jan 2024 15:34:21 -0800 Subject: [PATCH] [CIR] Add cir.resume op and use it in cir.catch - Add an extra CatchOp region to hold fallback (where EH usually resumes or rethrows as part of try/catch). - Emit `cir.resume` on the fallback region. Incremental step into the next assertion, still missing pieces before adding the first testcase. --- clang/include/clang/CIR/Dialect/IR/CIROps.td | 25 +++++++++ .../include/clang/CIR/Dialect/IR/CIRTypes.td | 2 +- clang/lib/CIR/CodeGen/CIRGenException.cpp | 52 +++++++++++++++++-- clang/lib/CIR/CodeGen/CIRGenFunction.h | 3 +- 4 files changed, 77 insertions(+), 5 deletions(-) diff --git a/clang/include/clang/CIR/Dialect/IR/CIROps.td b/clang/include/clang/CIR/Dialect/IR/CIROps.td index 82a761b73afd..967e50839df4 100644 --- a/clang/include/clang/CIR/Dialect/IR/CIROps.td +++ b/clang/include/clang/CIR/Dialect/IR/CIROps.td @@ -718,6 +718,31 @@ def ContinueOp : CIR_Op<"continue", [Terminator]> { let hasVerifier = 1; } +//===----------------------------------------------------------------------===// +// Resume +//===----------------------------------------------------------------------===// + +def ResumeOp : CIR_Op<"resume", [ReturnLike, Terminator, + ParentOneOf<["CatchOp"]>]> { + let summary = "Resumes execution after not catching exceptions"; + let description = [{ + The `cir.resume` operation terminates a region on `cir.catch`, "resuming" + or continuing the unwind process. The incoming argument is of !cir.eh_info + populated by `cir.try_call` and available in `cir.catch`. + + Examples: + ```mlir + cir.catch %4 { + ... + fallback { cir.resume(%0) }; + } + ``` + }]; + + let arguments = (ins ExceptionInfoPtr:$ptr); + let assemblyFormat = "$ptr attr-dict"; +} + //===----------------------------------------------------------------------===// // ScopeOp //===----------------------------------------------------------------------===// diff --git a/clang/include/clang/CIR/Dialect/IR/CIRTypes.td b/clang/include/clang/CIR/Dialect/IR/CIRTypes.td index 7e85d3ddeff3..341de9406c22 100644 --- a/clang/include/clang/CIR/Dialect/IR/CIRTypes.td +++ b/clang/include/clang/CIR/Dialect/IR/CIRTypes.td @@ -283,7 +283,7 @@ def ExceptionInfoPtr : Type< ]>, "void*">, BuildableType< "mlir::cir::PointerType::get($_builder.getContext()," - "mlir::cir::ExceptionInfo::get($_builder.getContext()))"> { + "mlir::cir::ExceptionInfoType::get($_builder.getContext()))"> { } //===----------------------------------------------------------------------===// diff --git a/clang/lib/CIR/CodeGen/CIRGenException.cpp b/clang/lib/CIR/CodeGen/CIRGenException.cpp index 8a604453870e..0316536e7393 100644 --- a/clang/lib/CIR/CodeGen/CIRGenException.cpp +++ b/clang/lib/CIR/CodeGen/CIRGenException.cpp @@ -251,6 +251,36 @@ void CIRGenFunction::buildAnyExprToExn(const Expr *e, Address addr) { DeactivateCleanupBlock(cleanup, op); } +mlir::Block *CIRGenFunction::getEHResumeBlock(bool isCleanup) { + // Just like some other try/catch related logic: return the basic block + // pointer but only use it to denote we're tracking things, but there + // shouldn't be any changes to that block after work done in this function. + auto catchOp = currExceptionInfo.catchOp; + assert(catchOp.getNumRegions() && "expected at least one region"); + auto &fallbackRegion = catchOp.getRegion(catchOp.getNumRegions() - 1); + + auto *resumeBlock = &fallbackRegion.getBlocks().back(); + if (!resumeBlock->empty()) + return resumeBlock; + + auto ip = getBuilder().saveInsertionPoint(); + getBuilder().setInsertionPointToStart(resumeBlock); + + const EHPersonality &Personality = EHPersonality::get(*this); + + // This can always be a call because we necessarily didn't find + // anything on the EH stack which needs our help. + const char *RethrowName = Personality.CatchallRethrowFn; + if (RethrowName != nullptr && !isCleanup) { + llvm_unreachable("NYI"); + } + + getBuilder().create(catchOp.getLoc(), + currExceptionInfo.exceptionAddr); + getBuilder().restoreInsertionPoint(ip); + return resumeBlock; +} + mlir::LogicalResult CIRGenFunction::buildCXXTryStmt(const CXXTryStmt &S) { const llvm::Triple &T = getTarget().getTriple(); // If we encounter a try statement on in an OpenMP target region offloaded to @@ -288,7 +318,9 @@ mlir::LogicalResult CIRGenFunction::buildCXXTryStmt(const CXXTryStmt &S) { [&](mlir::OpBuilder &b, mlir::Location loc, mlir::OperationState &result) { mlir::OpBuilder::InsertionGuard guard(b); - for (int i = 0, e = numHandlers; i != e; ++i) { + // Once for each handler and one for fallback (which could be a + // resume or rethrow). + for (int i = 0, e = numHandlers + 1; i != e; ++i) { auto *r = result.addRegion(); builder.createBlock(r); } @@ -346,11 +378,25 @@ static void buildCatchDispatchBlock(CIRGenFunction &CGF, // Check for address space mismatch: if (typeValue->getType() != argTy) assert(!UnimplementedFeature::addressSpace()); + bool nextIsEnd = false; // If this is the last handler, we're at the end, and the next // block is the block for the enclosing EH scope. Make sure to call // getEHDispatchBlock for caching it. - if (i + 1 == e) + if (i + 1 == e) { (void)CGF.getEHDispatchBlock(catchScope.getEnclosingEHScope()); + nextIsEnd = true; + + // If the next handler is a catch-all, we're at the end, and the + // next block is that handler. + } else if (catchScope.getHandler(i + 1).isCatchAll()) { + // Block already created when creating CatchOp, just mark this + // is the end. + nextIsEnd = true; + } + + // If the next handler is a catch-all, we're completely done. + if (nextIsEnd) + return; } } @@ -549,7 +595,7 @@ CIRGenFunction::getEHDispatchBlock(EHScopeStack::stable_iterator si) { // The dispatch block for the end of the scope chain is a block that // just resumes unwinding. if (si == EHStack.stable_end()) - llvm_unreachable("NYI"); + return getEHResumeBlock(true); // Otherwise, we should look at the actual scope. EHScope &scope = *EHStack.find(si); diff --git a/clang/lib/CIR/CodeGen/CIRGenFunction.h b/clang/lib/CIR/CodeGen/CIRGenFunction.h index 64f48ba97aeb..02d7b26cd48b 100644 --- a/clang/lib/CIR/CodeGen/CIRGenFunction.h +++ b/clang/lib/CIR/CodeGen/CIRGenFunction.h @@ -1503,8 +1503,9 @@ class CIRGenFunction : public CIRGenTypeCache { bool isConditional() const { return IsConditional; } }; - /// Emits landing pad information for the current EH stack. + /// Emits try/catch information for the current EH stack. mlir::Operation *buildLandingPad(); + mlir::Block *getEHResumeBlock(bool isCleanup); mlir::Block *getEHDispatchBlock(EHScopeStack::stable_iterator scope); mlir::Operation *getInvokeDestImpl();