rjmccall added a reviewer: varungandhi-apple.
rjmccall added a comment.

CC'ing Varun Gandhi.

Is `musttail` actually supported generically on all LLVM backends, or does this 
need a target restriction?

You should structure this code so it's easy to add exceptions for certain 
calling conventions that can support tail calls with weaker restrictions 
(principally, callee-pop conventions).  Mostly that probably means checking the 
calling convention first, or extracting the type restriction checks into a 
different function that you can skip.  For example, I believe x86's `fastcall` 
convention can logically support any combination of prototypes as `musttail` as 
long as the return types are vaguely compatible.



================
Comment at: clang/lib/CodeGen/CGCall.cpp:5318-5319
+      EHCleanupScope *Cleanup = dyn_cast<EHCleanupScope>(&*it);
+      assert(Cleanup && Cleanup->getCleanup()->isCallLifetimeEnd() &&
+             "found unexpected cleanup generating musttail exit");
+    }
----------------
rsmith wrote:
> Given the potential for mismatch between the JumpDiagnostics checks and this 
> one, especially as new more exotic kinds of cleanup are added, I wonder if we 
> should use an `ErrorUnsupported` here instead of an `assert`.
> 
> I strongly suspect we can still reach the problematic case here for a tail 
> call in a statement expression. I don't think it's feasible to check for all 
> the ways that an arbitrary expression context can have pending cleanups, 
> which we'd need in order to produce precise `Sema` diagnostics for that, so 
> either we handle that here or we blanket reject all `musttail` returns in 
> statement expressions. I think either approach is probably acceptable.
Yes, I think ErrorUnsupported is a much better idea.


================
Comment at: clang/lib/Sema/SemaStmt.cpp:631
+    // Caller is an Obj-C block decl: ^(void) { /* ... */ }
+    assert(dyn_cast<BlockDecl>(CurContext) && "unexpected decl context");
+    Diag(St->getBeginLoc(), diag::err_musttail_from_block_forbidden) << &MTA;
----------------
rsmith wrote:
> There are a couple of other contexts that can include a return statement: the 
> caller could also be an `ObjCMethodDecl` (an Objective-C method) or a 
> `CapturedDecl` (the body of a `#pragma omp` parallel region). I'd probably 
> use a specific diagnostic ("cannot be used from a block" / "cannot be used 
> from an Objective-C function") for the block and ObjCMethod case, and a 
> nonsepcific-but-correct "cannot be used from this context" for anything else.
Blocks ought to be extremely straightforward to support.  Just validate that 
the tail call is to a block pointer and then compare the underlying function 
types line up in the same way.  You will need to be able to verify that there 
isn't a non-trivial conversion on the return types, even if the return type 
isn't known at this point in the function, but that's a problem in C++ as well 
due to lambdas and `auto` deduced return types.

Also, you can use `isa<...>` for checks like this instead of `dyn_cast<...>`.


Repository:
  rG LLVM Github Monorepo

CHANGES SINCE LAST ACTION
  https://reviews.llvm.org/D99517/new/

https://reviews.llvm.org/D99517

_______________________________________________
cfe-commits mailing list
cfe-commits@lists.llvm.org
https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits

Reply via email to