Index: clang/docs/ReleaseNotes.rst =================================================================== --- clang/docs/ReleaseNotes.rst +++ clang/docs/ReleaseNotes.rst @@ -97,6 +97,8 @@ - Clang will now print more information about failed static assertions. In particular, simple static assertion expressions are evaluated to their compile-time value and printed out if the assertion fails. +- Clang will now reject the GNU extension address of label in coroutines explicitly. + This fixes `Issue 56436 `_. Non-comprehensive list of changes in this release ------------------------------------------------- Index: clang/include/clang/Basic/DiagnosticSemaKinds.td =================================================================== --- clang/include/clang/Basic/DiagnosticSemaKinds.td +++ clang/include/clang/Basic/DiagnosticSemaKinds.td @@ -11260,6 +11260,9 @@ def err_coroutine_unusable_new : Error< "'operator new' provided by %0 is not usable with the function signature of %1" >; +def err_coro_invalid_addr_of_label : Error< + "the GNU address of label extension is not allowed in coroutines." +>; } // end of coroutines issue category let CategoryName = "Documentation Issue" in { Index: clang/include/clang/Sema/ScopeInfo.h =================================================================== --- clang/include/clang/Sema/ScopeInfo.h +++ clang/include/clang/Sema/ScopeInfo.h @@ -233,6 +233,9 @@ /// modified in the function. llvm::SmallPtrSet ModifiedNonNullParams; + /// The set of GNU address of label extension "&&label". + llvm::SmallVector AddrLabels; + public: /// Represents a simple identification of a weak object. /// Index: clang/lib/Sema/ScopeInfo.cpp =================================================================== --- clang/lib/Sema/ScopeInfo.cpp +++ clang/lib/Sema/ScopeInfo.cpp @@ -56,6 +56,7 @@ ModifiedNonNullParams.clear(); Blocks.clear(); ByrefBlockVars.clear(); + AddrLabels.clear(); } static const NamedDecl *getBestPropertyDecl(const ObjCPropertyRefExpr *PropE) { Index: clang/lib/Sema/SemaCoroutine.cpp =================================================================== --- clang/lib/Sema/SemaCoroutine.cpp +++ clang/lib/Sema/SemaCoroutine.cpp @@ -1103,6 +1103,12 @@ Diag(Fn->FirstCoroutineStmtLoc, diag::note_declared_coroutine_here) << Fn->getFirstCoroutineStmtKeyword(); } + + // Corotuines will get splitted into pieces. The GNU address of label + // extension wouldn't be meaningful in coroutines. + for (AddrLabelExpr *ALE : Fn->AddrLabels) + Diag(ALE->getBeginLoc(), diag::err_coro_invalid_addr_of_label); + CoroutineStmtBuilder Builder(*this, *FD, *Fn, Body); if (Builder.isInvalid() || !Builder.buildStatements()) return FD->setInvalidDecl(); Index: clang/lib/Sema/SemaExpr.cpp =================================================================== --- clang/lib/Sema/SemaExpr.cpp +++ clang/lib/Sema/SemaExpr.cpp @@ -15840,8 +15840,13 @@ LabelDecl *TheDecl) { TheDecl->markUsed(Context); // Create the AST node. The address of a label always has type 'void*'. - return new (Context) AddrLabelExpr(OpLoc, LabLoc, TheDecl, - Context.getPointerType(Context.VoidTy)); + auto *Res = new (Context) AddrLabelExpr( + OpLoc, LabLoc, TheDecl, Context.getPointerType(Context.VoidTy)); + + if (getCurFunction()) + getCurFunction()->AddrLabels.push_back(Res); + + return Res; } void Sema::ActOnStartStmtExpr() { Index: clang/test/SemaCXX/addr-label-in-coroutines.cpp =================================================================== --- /dev/null +++ clang/test/SemaCXX/addr-label-in-coroutines.cpp @@ -0,0 +1,72 @@ +// RUN: %clang_cc1 -std=c++20 -fsyntax-only -verify %s + +#include "Inputs/std-coroutine.h" + +struct Allocator {}; + +struct resumable { + struct promise_type { + resumable get_return_object() { return {}; } + auto initial_suspend() { return std::suspend_always(); } + auto final_suspend() noexcept { return std::suspend_always(); } + void unhandled_exception() {} + void return_void(){}; + }; +}; + +resumable f1(int &out, int *inst) { + static void* dispatch_table[] = {&&inc, // expected-error {{the GNU address of label extension is not allowed in coroutines.}} + &&suspend, // expected-error {{the GNU address of label extension is not allowed in coroutines.}} + &&stop}; // expected-error {{the GNU address of label extension is not allowed in coroutines.}} + #define DISPATCH() goto *dispatch_table[*inst++] +inc: + out++; + DISPATCH(); + +suspend: + co_await std::suspend_always{}; + DISPATCH(); + +stop: + co_return; +} + +resumable f2(int &out, int *inst) { + void* dispatch_table[] = {nullptr, nullptr, nullptr}; + dispatch_table[0] = &&inc; // expected-error {{the GNU address of label extension is not allowed in coroutines.}} + dispatch_table[1] = &&suspend; // expected-error {{the GNU address of label extension is not allowed in coroutines.}} + dispatch_table[2] = &&stop; // expected-error {{the GNU address of label extension is not allowed in coroutines.}} + #define DISPATCH() goto *dispatch_table[*inst++] +inc: + out++; + DISPATCH(); + +suspend: + co_await std::suspend_always{}; + DISPATCH(); + +stop: + co_return; +} + +resumable f3(int &out, int *inst) { + void* dispatch_table[] = {nullptr, nullptr, nullptr}; + [&]() -> resumable { + dispatch_table[0] = &&inc; // expected-error {{the GNU address of label extension is not allowed in coroutines.}} + dispatch_table[1] = &&suspend; // expected-error {{the GNU address of label extension is not allowed in coroutines.}} + dispatch_table[2] = &&stop; // expected-error {{the GNU address of label extension is not allowed in coroutines.}} + #define DISPATCH() goto *dispatch_table[*inst++] + inc: + out++; + DISPATCH(); + + suspend: + co_await std::suspend_always{}; + DISPATCH(); + + stop: + co_return; + }(); + + co_return; +}