2019-03-01 14:49:51 +08:00
|
|
|
//===-- SemaCoroutine.cpp - Semantic Analysis for Coroutines --------------===//
|
2015-10-22 14:13:50 +08:00
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2015-10-22 14:13:50 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implements semantic analysis for C++ Coroutines.
|
|
|
|
//
|
2018-06-24 02:01:02 +08:00
|
|
|
// This file contains references to sections of the Coroutines TS, which
|
|
|
|
// can be found at http://wg21.link/coroutines.
|
|
|
|
//
|
2015-10-22 14:13:50 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2017-04-04 03:21:00 +08:00
|
|
|
#include "CoroutineStmtBuilder.h"
|
2018-02-16 04:37:22 +08:00
|
|
|
#include "clang/AST/ASTLambda.h"
|
2015-10-27 14:02:45 +08:00
|
|
|
#include "clang/AST/Decl.h"
|
|
|
|
#include "clang/AST/ExprCXX.h"
|
|
|
|
#include "clang/AST/StmtCXX.h"
|
2019-11-16 08:36:00 +08:00
|
|
|
#include "clang/Basic/Builtins.h"
|
2015-10-27 14:02:45 +08:00
|
|
|
#include "clang/Lex/Preprocessor.h"
|
2015-11-24 10:34:39 +08:00
|
|
|
#include "clang/Sema/Initialization.h"
|
2015-10-27 14:02:45 +08:00
|
|
|
#include "clang/Sema/Overload.h"
|
2018-03-08 06:48:35 +08:00
|
|
|
#include "clang/Sema/ScopeInfo.h"
|
2017-04-04 03:21:00 +08:00
|
|
|
#include "clang/Sema/SemaInternal.h"
|
[Coroutines] Ensure co_await promise.final_suspend() does not throw
Summary:
This patch addresses https://bugs.llvm.org/show_bug.cgi?id=46256
The spec of coroutine requires that the expression co_await promise.final_suspend() shall not be potentially-throwing.
To check this, we recursively look at every call (including Call, MemberCall, OperatorCall and Constructor) in all code
generated by the final suspend, and ensure that the callees are declared with noexcept. We also look at any returned data
type that requires explicit destruction, and check their destructors for noexcept.
This patch does not check declarations with dependent types yet, which will be done in future patches.
Updated all tests to add noexcept to the required functions, and added a dedicated test for this patch.
This patch might start to cause existing codebase fail to compile because most people may not have been strict in tagging
all the related functions noexcept.
Reviewers: lewissbaker, modocache, junparser
Reviewed By: modocache
Subscribers: arphaman, junparser, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D82029
2020-06-16 07:27:41 +08:00
|
|
|
#include "llvm/ADT/SmallSet.h"
|
2017-04-04 03:21:00 +08:00
|
|
|
|
2015-10-22 14:13:50 +08:00
|
|
|
using namespace clang;
|
|
|
|
using namespace sema;
|
|
|
|
|
2017-05-25 22:59:39 +08:00
|
|
|
static LookupResult lookupMember(Sema &S, const char *Name, CXXRecordDecl *RD,
|
|
|
|
SourceLocation Loc, bool &Res) {
|
2017-03-07 07:38:15 +08:00
|
|
|
DeclarationName DN = S.PP.getIdentifierInfo(Name);
|
|
|
|
LookupResult LR(S, DN, Loc, Sema::LookupMemberName);
|
|
|
|
// Suppress diagnostics when a private member is selected. The same warnings
|
|
|
|
// will be produced again when building the call.
|
|
|
|
LR.suppressDiagnostics();
|
2017-05-25 22:59:39 +08:00
|
|
|
Res = S.LookupQualifiedName(LR, RD);
|
|
|
|
return LR;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool lookupMember(Sema &S, const char *Name, CXXRecordDecl *RD,
|
|
|
|
SourceLocation Loc) {
|
|
|
|
bool Res;
|
|
|
|
lookupMember(S, Name, RD, Loc, Res);
|
|
|
|
return Res;
|
2017-03-07 07:38:15 +08:00
|
|
|
}
|
|
|
|
|
2015-10-27 14:02:45 +08:00
|
|
|
/// Look up the std::coroutine_traits<...>::promise_type for the given
|
|
|
|
/// function type.
|
2017-07-10 09:27:22 +08:00
|
|
|
static QualType lookupPromiseType(Sema &S, const FunctionDecl *FD,
|
|
|
|
SourceLocation KwLoc) {
|
|
|
|
const FunctionProtoType *FnType = FD->getType()->castAs<FunctionProtoType>();
|
|
|
|
const SourceLocation FuncLoc = FD->getLocation();
|
2015-10-27 14:02:45 +08:00
|
|
|
|
2021-11-04 11:50:30 +08:00
|
|
|
NamespaceDecl *CoroNamespace = nullptr;
|
|
|
|
ClassTemplateDecl *CoroTraits =
|
|
|
|
S.lookupCoroutineTraits(KwLoc, FuncLoc, CoroNamespace);
|
2015-10-27 14:02:45 +08:00
|
|
|
if (!CoroTraits) {
|
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
|
2017-07-10 09:27:22 +08:00
|
|
|
// Form template argument list for coroutine_traits<R, P1, P2, ...> according
|
|
|
|
// to [dcl.fct.def.coroutine]3
|
2017-03-07 06:52:28 +08:00
|
|
|
TemplateArgumentListInfo Args(KwLoc, KwLoc);
|
2017-07-10 09:27:22 +08:00
|
|
|
auto AddArg = [&](QualType T) {
|
2015-10-27 14:02:45 +08:00
|
|
|
Args.addArgument(TemplateArgumentLoc(
|
2017-03-07 06:52:28 +08:00
|
|
|
TemplateArgument(T), S.Context.getTrivialTypeSourceInfo(T, KwLoc)));
|
2017-07-10 09:27:22 +08:00
|
|
|
};
|
|
|
|
AddArg(FnType->getReturnType());
|
|
|
|
// If the function is a non-static member function, add the type
|
|
|
|
// of the implicit object parameter before the formal parameters.
|
|
|
|
if (auto *MD = dyn_cast<CXXMethodDecl>(FD)) {
|
|
|
|
if (MD->isInstance()) {
|
|
|
|
// [over.match.funcs]4
|
|
|
|
// For non-static member functions, the type of the implicit object
|
|
|
|
// parameter is
|
2017-07-10 10:52:34 +08:00
|
|
|
// -- "lvalue reference to cv X" for functions declared without a
|
|
|
|
// ref-qualifier or with the & ref-qualifier
|
|
|
|
// -- "rvalue reference to cv X" for functions declared with the &&
|
|
|
|
// ref-qualifier
|
2019-10-07 22:25:46 +08:00
|
|
|
QualType T = MD->getThisType()->castAs<PointerType>()->getPointeeType();
|
2017-07-10 09:27:22 +08:00
|
|
|
T = FnType->getRefQualifier() == RQ_RValue
|
|
|
|
? S.Context.getRValueReferenceType(T)
|
|
|
|
: S.Context.getLValueReferenceType(T, /*SpelledAsLValue*/ true);
|
|
|
|
AddArg(T);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (QualType T : FnType->getParamTypes())
|
|
|
|
AddArg(T);
|
2015-10-27 14:02:45 +08:00
|
|
|
|
|
|
|
// Build the template-id.
|
|
|
|
QualType CoroTrait =
|
2017-03-07 06:52:28 +08:00
|
|
|
S.CheckTemplateIdType(TemplateName(CoroTraits), KwLoc, Args);
|
2015-10-27 14:02:45 +08:00
|
|
|
if (CoroTrait.isNull())
|
|
|
|
return QualType();
|
2017-03-07 06:52:28 +08:00
|
|
|
if (S.RequireCompleteType(KwLoc, CoroTrait,
|
2017-03-09 11:09:43 +08:00
|
|
|
diag::err_coroutine_type_missing_specialization))
|
2015-10-27 14:02:45 +08:00
|
|
|
return QualType();
|
|
|
|
|
2017-03-07 06:52:28 +08:00
|
|
|
auto *RD = CoroTrait->getAsCXXRecordDecl();
|
2015-10-27 14:02:45 +08:00
|
|
|
assert(RD && "specialization of class template is not a class?");
|
|
|
|
|
|
|
|
// Look up the ::promise_type member.
|
2017-03-07 06:52:28 +08:00
|
|
|
LookupResult R(S, &S.PP.getIdentifierTable().get("promise_type"), KwLoc,
|
2015-10-27 14:02:45 +08:00
|
|
|
Sema::LookupOrdinaryName);
|
|
|
|
S.LookupQualifiedName(R, RD);
|
|
|
|
auto *Promise = R.getAsSingle<TypeDecl>();
|
|
|
|
if (!Promise) {
|
2017-03-07 06:52:28 +08:00
|
|
|
S.Diag(FuncLoc,
|
|
|
|
diag::err_implied_std_coroutine_traits_promise_type_not_found)
|
2016-10-28 00:28:31 +08:00
|
|
|
<< RD;
|
2015-10-27 14:02:45 +08:00
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
// The promise type is required to be a class type.
|
|
|
|
QualType PromiseType = S.Context.getTypeDeclType(Promise);
|
2017-03-07 06:52:28 +08:00
|
|
|
|
|
|
|
auto buildElaboratedType = [&]() {
|
2021-11-04 11:50:30 +08:00
|
|
|
auto *NNS = NestedNameSpecifier::Create(S.Context, nullptr, CoroNamespace);
|
2015-11-19 10:36:35 +08:00
|
|
|
NNS = NestedNameSpecifier::Create(S.Context, NNS, false,
|
|
|
|
CoroTrait.getTypePtr());
|
2017-03-07 06:52:28 +08:00
|
|
|
return S.Context.getElaboratedType(ETK_None, NNS, PromiseType);
|
|
|
|
};
|
2015-11-19 10:36:35 +08:00
|
|
|
|
2017-03-07 06:52:28 +08:00
|
|
|
if (!PromiseType->getAsCXXRecordDecl()) {
|
|
|
|
S.Diag(FuncLoc,
|
|
|
|
diag::err_implied_std_coroutine_traits_promise_type_not_class)
|
|
|
|
<< buildElaboratedType();
|
2015-10-27 14:02:45 +08:00
|
|
|
return QualType();
|
|
|
|
}
|
2017-03-07 06:52:28 +08:00
|
|
|
if (S.RequireCompleteType(FuncLoc, buildElaboratedType(),
|
|
|
|
diag::err_coroutine_promise_type_incomplete))
|
|
|
|
return QualType();
|
2015-10-27 14:02:45 +08:00
|
|
|
|
|
|
|
return PromiseType;
|
|
|
|
}
|
|
|
|
|
2021-11-04 11:50:30 +08:00
|
|
|
/// Look up the std::coroutine_handle<PromiseType>.
|
2017-03-09 11:09:43 +08:00
|
|
|
static QualType lookupCoroutineHandleType(Sema &S, QualType PromiseType,
|
|
|
|
SourceLocation Loc) {
|
|
|
|
if (PromiseType.isNull())
|
|
|
|
return QualType();
|
|
|
|
|
2021-11-04 11:50:30 +08:00
|
|
|
NamespaceDecl *CoroNamespace = S.getCachedCoroNamespace();
|
|
|
|
assert(CoroNamespace && "Should already be diagnosed");
|
2017-03-09 11:09:43 +08:00
|
|
|
|
|
|
|
LookupResult Result(S, &S.PP.getIdentifierTable().get("coroutine_handle"),
|
|
|
|
Loc, Sema::LookupOrdinaryName);
|
2021-11-04 11:50:30 +08:00
|
|
|
if (!S.LookupQualifiedName(Result, CoroNamespace)) {
|
2017-03-09 11:09:43 +08:00
|
|
|
S.Diag(Loc, diag::err_implied_coroutine_type_not_found)
|
2021-11-04 11:50:30 +08:00
|
|
|
<< "std::coroutine_handle";
|
2017-03-09 11:09:43 +08:00
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
|
|
|
|
ClassTemplateDecl *CoroHandle = Result.getAsSingle<ClassTemplateDecl>();
|
|
|
|
if (!CoroHandle) {
|
|
|
|
Result.suppressDiagnostics();
|
|
|
|
// We found something weird. Complain about the first thing we found.
|
|
|
|
NamedDecl *Found = *Result.begin();
|
|
|
|
S.Diag(Found->getLocation(), diag::err_malformed_std_coroutine_handle);
|
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Form template argument list for coroutine_handle<Promise>.
|
|
|
|
TemplateArgumentListInfo Args(Loc, Loc);
|
|
|
|
Args.addArgument(TemplateArgumentLoc(
|
|
|
|
TemplateArgument(PromiseType),
|
|
|
|
S.Context.getTrivialTypeSourceInfo(PromiseType, Loc)));
|
|
|
|
|
|
|
|
// Build the template-id.
|
|
|
|
QualType CoroHandleType =
|
|
|
|
S.CheckTemplateIdType(TemplateName(CoroHandle), Loc, Args);
|
|
|
|
if (CoroHandleType.isNull())
|
|
|
|
return QualType();
|
|
|
|
if (S.RequireCompleteType(Loc, CoroHandleType,
|
|
|
|
diag::err_coroutine_type_missing_specialization))
|
|
|
|
return QualType();
|
|
|
|
|
|
|
|
return CoroHandleType;
|
|
|
|
}
|
|
|
|
|
2016-10-28 02:43:28 +08:00
|
|
|
static bool isValidCoroutineContext(Sema &S, SourceLocation Loc,
|
|
|
|
StringRef Keyword) {
|
Un-revert "[coroutines][PR40978] Emit error for co_yield within catch block"
Summary:
https://reviews.llvm.org/D59076 added a new coroutine error that
prevented users from using 'co_await' or 'co_yield' within a exception
handler. However, it was reverted in https://reviews.llvm.org/rC356774
because it caused a regression in nested scopes in C++ catch statements,
as documented by https://bugs.llvm.org/show_bug.cgi?id=41171.
The issue was due to an incorrect use of a `clang::ParseScope`. To fix:
1. Add a regression test for catch statement parsing that mimics the bug
report from https://bugs.llvm.org/show_bug.cgi?id=41171.
2. Re-apply the coroutines error patch from
https://reviews.llvm.org/D59076, but this time with the correct
ParseScope behavior.
Reviewers: GorNishanov, tks2103, rsmith, riccibruno, jbulow
Reviewed By: riccibruno
Subscribers: EricWF, jdoerfert, lewissbaker, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D59752
llvm-svn: 356865
2019-03-25 08:53:10 +08:00
|
|
|
// [expr.await]p2 dictates that 'co_await' and 'co_yield' must be used within
|
|
|
|
// a function body.
|
2018-06-24 02:01:02 +08:00
|
|
|
// FIXME: This also covers [expr.await]p2: "An await-expression shall not
|
|
|
|
// appear in a default argument." But the diagnostic QoI here could be
|
|
|
|
// improved to inform the user that default arguments specifically are not
|
|
|
|
// allowed.
|
2015-10-22 14:13:50 +08:00
|
|
|
auto *FD = dyn_cast<FunctionDecl>(S.CurContext);
|
|
|
|
if (!FD) {
|
|
|
|
S.Diag(Loc, isa<ObjCMethodDecl>(S.CurContext)
|
|
|
|
? diag::err_coroutine_objc_method
|
|
|
|
: diag::err_coroutine_outside_function) << Keyword;
|
2016-10-28 02:43:28 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// An enumeration for mapping the diagnostic type to the correct diagnostic
|
|
|
|
// selection index.
|
|
|
|
enum InvalidFuncDiag {
|
|
|
|
DiagCtor = 0,
|
|
|
|
DiagDtor,
|
|
|
|
DiagMain,
|
|
|
|
DiagConstexpr,
|
|
|
|
DiagAutoRet,
|
|
|
|
DiagVarargs,
|
2019-06-14 16:56:20 +08:00
|
|
|
DiagConsteval,
|
2016-10-28 02:43:28 +08:00
|
|
|
};
|
|
|
|
bool Diagnosed = false;
|
|
|
|
auto DiagInvalid = [&](InvalidFuncDiag ID) {
|
|
|
|
S.Diag(Loc, diag::err_coroutine_invalid_func_context) << ID << Keyword;
|
|
|
|
Diagnosed = true;
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
2019-06-19 22:12:19 +08:00
|
|
|
// Diagnose when a constructor, destructor
|
2016-10-28 02:43:28 +08:00
|
|
|
// or the function 'main' are declared as a coroutine.
|
|
|
|
auto *MD = dyn_cast<CXXMethodDecl>(FD);
|
2019-06-19 22:12:19 +08:00
|
|
|
// [class.ctor]p11: "A constructor shall not be a coroutine."
|
2016-10-28 02:43:28 +08:00
|
|
|
if (MD && isa<CXXConstructorDecl>(MD))
|
|
|
|
return DiagInvalid(DiagCtor);
|
2018-06-24 02:01:02 +08:00
|
|
|
// [class.dtor]p17: "A destructor shall not be a coroutine."
|
2016-10-28 02:43:28 +08:00
|
|
|
else if (MD && isa<CXXDestructorDecl>(MD))
|
|
|
|
return DiagInvalid(DiagDtor);
|
2018-06-24 02:01:02 +08:00
|
|
|
// [basic.start.main]p3: "The function main shall not be a coroutine."
|
2016-10-28 02:43:28 +08:00
|
|
|
else if (FD->isMain())
|
|
|
|
return DiagInvalid(DiagMain);
|
|
|
|
|
|
|
|
// Emit a diagnostics for each of the following conditions which is not met.
|
2018-06-24 02:01:02 +08:00
|
|
|
// [expr.const]p2: "An expression e is a core constant expression unless the
|
|
|
|
// evaluation of e [...] would evaluate one of the following expressions:
|
|
|
|
// [...] an await-expression [...] a yield-expression."
|
2016-10-28 02:43:28 +08:00
|
|
|
if (FD->isConstexpr())
|
2019-06-14 16:56:20 +08:00
|
|
|
DiagInvalid(FD->isConsteval() ? DiagConsteval : DiagConstexpr);
|
2018-06-24 02:01:02 +08:00
|
|
|
// [dcl.spec.auto]p15: "A function declared with a return type that uses a
|
|
|
|
// placeholder type shall not be a coroutine."
|
2016-10-28 02:43:28 +08:00
|
|
|
if (FD->getReturnType()->isUndeducedType())
|
|
|
|
DiagInvalid(DiagAutoRet);
|
2021-12-24 11:48:15 +08:00
|
|
|
// [dcl.fct.def.coroutine]p1
|
|
|
|
// The parameter-declaration-clause of the coroutine shall not terminate with
|
|
|
|
// an ellipsis that is not part of a parameter-declaration.
|
2016-10-28 02:43:28 +08:00
|
|
|
if (FD->isVariadic())
|
|
|
|
DiagInvalid(DiagVarargs);
|
|
|
|
|
|
|
|
return !Diagnosed;
|
|
|
|
}
|
|
|
|
|
2017-03-07 07:38:15 +08:00
|
|
|
static ExprResult buildOperatorCoawaitLookupExpr(Sema &SemaRef, Scope *S,
|
|
|
|
SourceLocation Loc) {
|
|
|
|
DeclarationName OpName =
|
|
|
|
SemaRef.Context.DeclarationNames.getCXXOperatorName(OO_Coawait);
|
|
|
|
LookupResult Operators(SemaRef, OpName, SourceLocation(),
|
|
|
|
Sema::LookupOperatorName);
|
|
|
|
SemaRef.LookupName(Operators, S);
|
|
|
|
|
|
|
|
assert(!Operators.isAmbiguous() && "Operator lookup cannot be ambiguous");
|
|
|
|
const auto &Functions = Operators.asUnresolvedSet();
|
|
|
|
bool IsOverloaded =
|
|
|
|
Functions.size() > 1 ||
|
|
|
|
(Functions.size() == 1 && isa<FunctionTemplateDecl>(*Functions.begin()));
|
|
|
|
Expr *CoawaitOp = UnresolvedLookupExpr::Create(
|
|
|
|
SemaRef.Context, /*NamingClass*/ nullptr, NestedNameSpecifierLoc(),
|
|
|
|
DeclarationNameInfo(OpName, Loc), /*RequiresADL*/ true, IsOverloaded,
|
|
|
|
Functions.begin(), Functions.end());
|
|
|
|
assert(CoawaitOp);
|
|
|
|
return CoawaitOp;
|
|
|
|
}
|
2016-10-28 02:43:28 +08:00
|
|
|
|
2017-03-07 07:38:15 +08:00
|
|
|
/// Build a call to 'operator co_await' if there is a suitable operator for
|
|
|
|
/// the given expression.
|
|
|
|
static ExprResult buildOperatorCoawaitCall(Sema &SemaRef, SourceLocation Loc,
|
|
|
|
Expr *E,
|
|
|
|
UnresolvedLookupExpr *Lookup) {
|
|
|
|
UnresolvedSet<16> Functions;
|
|
|
|
Functions.append(Lookup->decls_begin(), Lookup->decls_end());
|
|
|
|
return SemaRef.CreateOverloadedUnaryOp(Loc, UO_Coawait, Functions, E);
|
|
|
|
}
|
2015-10-22 14:13:50 +08:00
|
|
|
|
2017-03-07 07:38:15 +08:00
|
|
|
static ExprResult buildOperatorCoawaitCall(Sema &SemaRef, Scope *S,
|
|
|
|
SourceLocation Loc, Expr *E) {
|
|
|
|
ExprResult R = buildOperatorCoawaitLookupExpr(SemaRef, S, Loc);
|
|
|
|
if (R.isInvalid())
|
|
|
|
return ExprError();
|
|
|
|
return buildOperatorCoawaitCall(SemaRef, Loc, E,
|
|
|
|
cast<UnresolvedLookupExpr>(R.get()));
|
2015-10-22 14:13:50 +08:00
|
|
|
}
|
|
|
|
|
2017-03-09 11:09:43 +08:00
|
|
|
static ExprResult buildCoroutineHandle(Sema &S, QualType PromiseType,
|
|
|
|
SourceLocation Loc) {
|
|
|
|
QualType CoroHandleType = lookupCoroutineHandleType(S, PromiseType, Loc);
|
|
|
|
if (CoroHandleType.isNull())
|
|
|
|
return ExprError();
|
|
|
|
|
|
|
|
DeclContext *LookupCtx = S.computeDeclContext(CoroHandleType);
|
|
|
|
LookupResult Found(S, &S.PP.getIdentifierTable().get("from_address"), Loc,
|
|
|
|
Sema::LookupOrdinaryName);
|
|
|
|
if (!S.LookupQualifiedName(Found, LookupCtx)) {
|
|
|
|
S.Diag(Loc, diag::err_coroutine_handle_missing_member)
|
|
|
|
<< "from_address";
|
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
|
|
|
|
Expr *FramePtr =
|
2021-06-29 00:45:56 +08:00
|
|
|
S.BuildBuiltinCallExpr(Loc, Builtin::BI__builtin_coro_frame, {});
|
2017-03-09 11:09:43 +08:00
|
|
|
|
|
|
|
CXXScopeSpec SS;
|
|
|
|
ExprResult FromAddr =
|
|
|
|
S.BuildDeclarationNameExpr(SS, Found, /*NeedsADL=*/false);
|
|
|
|
if (FromAddr.isInvalid())
|
|
|
|
return ExprError();
|
|
|
|
|
2019-05-08 09:36:36 +08:00
|
|
|
return S.BuildCallExpr(nullptr, FromAddr.get(), Loc, FramePtr, Loc);
|
2017-03-09 11:09:43 +08:00
|
|
|
}
|
2015-10-27 14:02:45 +08:00
|
|
|
|
|
|
|
struct ReadySuspendResumeResult {
|
2017-05-29 02:21:12 +08:00
|
|
|
enum AwaitCallType { ACT_Ready, ACT_Suspend, ACT_Resume };
|
2015-10-27 14:02:45 +08:00
|
|
|
Expr *Results[3];
|
2017-03-11 09:30:17 +08:00
|
|
|
OpaqueValueExpr *OpaqueValue;
|
|
|
|
bool IsInvalid;
|
2015-10-27 14:02:45 +08:00
|
|
|
};
|
|
|
|
|
2015-11-21 06:40:06 +08:00
|
|
|
static ExprResult buildMemberCall(Sema &S, Expr *Base, SourceLocation Loc,
|
2017-03-07 07:38:15 +08:00
|
|
|
StringRef Name, MultiExprArg Args) {
|
2015-11-21 06:40:06 +08:00
|
|
|
DeclarationNameInfo NameInfo(&S.PP.getIdentifierTable().get(Name), Loc);
|
|
|
|
|
|
|
|
// FIXME: Fix BuildMemberReferenceExpr to take a const CXXScopeSpec&.
|
|
|
|
CXXScopeSpec SS;
|
|
|
|
ExprResult Result = S.BuildMemberReferenceExpr(
|
|
|
|
Base, Base->getType(), Loc, /*IsPtr=*/false, SS,
|
|
|
|
SourceLocation(), nullptr, NameInfo, /*TemplateArgs=*/nullptr,
|
|
|
|
/*Scope=*/nullptr);
|
|
|
|
if (Result.isInvalid())
|
|
|
|
return ExprError();
|
|
|
|
|
2018-03-28 04:38:19 +08:00
|
|
|
// We meant exactly what we asked for. No need for typo correction.
|
|
|
|
if (auto *TE = dyn_cast<TypoExpr>(Result.get())) {
|
|
|
|
S.clearDelayedTypo(TE);
|
|
|
|
S.Diag(Loc, diag::err_no_member)
|
|
|
|
<< NameInfo.getName() << Base->getType()->getAsCXXRecordDecl()
|
|
|
|
<< Base->getSourceRange();
|
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
|
2019-05-08 09:36:36 +08:00
|
|
|
return S.BuildCallExpr(nullptr, Result.get(), Loc, Args, Loc, nullptr);
|
2015-11-21 06:40:06 +08:00
|
|
|
}
|
|
|
|
|
2017-08-25 12:46:54 +08:00
|
|
|
// See if return type is coroutine-handle and if so, invoke builtin coro-resume
|
|
|
|
// on its address. This is to enable experimental support for coroutine-handle
|
2018-04-06 23:14:32 +08:00
|
|
|
// returning await_suspend that results in a guaranteed tail call to the target
|
2017-08-25 12:46:54 +08:00
|
|
|
// coroutine.
|
|
|
|
static Expr *maybeTailCall(Sema &S, QualType RetType, Expr *E,
|
2020-11-12 13:17:16 +08:00
|
|
|
SourceLocation Loc) {
|
2017-08-25 12:46:54 +08:00
|
|
|
if (RetType->isReferenceType())
|
|
|
|
return nullptr;
|
|
|
|
Type const *T = RetType.getTypePtr();
|
|
|
|
if (!T->isClassType() && !T->isStructureType())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
// FIXME: Add convertability check to coroutine_handle<>. Possibly via
|
|
|
|
// EvaluateBinaryTypeTrait(BTT_IsConvertible, ...) which is at the moment
|
|
|
|
// a private function in SemaExprCXX.cpp
|
|
|
|
|
|
|
|
ExprResult AddressExpr = buildMemberCall(S, E, Loc, "address", None);
|
|
|
|
if (AddressExpr.isInvalid())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
Expr *JustAddress = AddressExpr.get();
|
2020-07-06 13:39:29 +08:00
|
|
|
|
|
|
|
// Check that the type of AddressExpr is void*
|
|
|
|
if (!JustAddress->getType().getTypePtr()->isVoidPointerType())
|
|
|
|
S.Diag(cast<CallExpr>(JustAddress)->getCalleeDecl()->getLocation(),
|
|
|
|
diag::warn_coroutine_handle_address_invalid_return_type)
|
|
|
|
<< JustAddress->getType();
|
|
|
|
|
[Coroutine][Sema] Cleanup temporaries as early as possible
The original bug was discovered in T75057860. Clang front-end emits an AST that looks like this for an co_await expression:
|- ExprWithCleanups
|- -CoawaitExpr
|- -MaterializeTemporaryExpr ... Awaiter
...
|- -CXXMemberCallExpr ... .await_ready
...
|- -CallExpr ... __builtin_coro_resume
...
|- -CXXMemberCallExpr ... .await_resume
...
ExprWithCleanups is responsible for cleaning up (including calling dtors) for the temporaries generated in the wrapping expression).
In the above structure, the __builtin_coro_resume part (which corresponds to the code for the suspend case in the co_await with symmetric transfer), the pseudocode looks like this:
__builtin_coro_resume(
awaiter.await_suspend(
from_address(
__builtin_coro_frame())).address());
One of the temporaries that's generated as part of this code is the coroutine handle returned from awaiter.await_suspend() call. The call returns a handle which is a prvalue (since it's a returned value on the fly). In order to call the address() method on it, it needs to be converted into an xvalue. Hence a materialized temp is created to hold it. This temp will need to be cleaned up eventually. Now, since all cleanups happen at the end of the entire co_await expression, which is after the <coro.suspend> suspension point, the compiler will think that such a temp needs to live across suspensions, and need to be put on the coroutine frame, even though it's only used temporarily just to call address() method.
Such a phenomena not only unnecessarily increases the frame size, but can lead to ASAN failures, if the coroutine was already destroyed as part of the await_suspend() call. This is because if the coroutine was already destroyed, the frame no longer exists, and one can not store anything into it. But if the temporary object is considered to need to live on the frame, it will be stored into the frame after await_suspend() returns.
A fix attempt was done in https://reviews.llvm.org/D87470. Unfortunately it is incorrect. The reason is that cleanups in Clang works more like linearly than nested. There is one current state indicating whether it needs cleanup, and an ExprWithCleanups resets that state. This means that an ExprWithCleanups must be capable of cleaning up all temporaries created in the wrapping expression, otherwise there will be dangling temporaries cleaned up at the wrong place.
I eventually found a walk-around (https://reviews.llvm.org/D89066) that doesn't break any existing tests while fixing the issue. But it targets the final co_await only. If we ever have a co_await that's not on the final awaiter and the frame gets destroyed after suspend, we are in trouble. Hence we need a proper fix.
This patch is the proper fix. It does the folllowing things to fully resolve the issue:
1. The AST has to be generated in the order according to their nesting relationship. We should not generate AST out of order because then the code generator would incorrectly track the state of temporaries and when a cleanup is needed. So the code in buildCoawaitCalls is reorganized so that we will be generating the AST for each coawait member call in order along with their child AST.
2. await_ready() call is wrapped with an ExprWithCleanups so that temporaries in it gets cleaned up as early as possible to avoid living across suspension.
3. await_suspend() call is wrapped with an ExprWithCleanups if it's not a symmetric transfer. In the case of a symmetric transfer, in order to maintain the musttail call contract, the ExprWithCleanups is wraaped before the resume call.
4. In the end, we mark again that it needs a cleanup, so that the entire CoawaitExpr will be wrapped with a ExprWithCleanups which will clean up the Awaiter object associated with the await expression.
Differential Revision: https://reviews.llvm.org/D90990
2020-11-11 05:02:18 +08:00
|
|
|
// Clean up temporary objects so that they don't live across suspension points
|
|
|
|
// unnecessarily. We choose to clean up before the call to
|
|
|
|
// __builtin_coro_resume so that the cleanup code are not inserted in-between
|
|
|
|
// the resume call and return instruction, which would interfere with the
|
|
|
|
// musttail call contract.
|
|
|
|
JustAddress = S.MaybeCreateExprWithCleanups(JustAddress);
|
2021-06-29 00:45:56 +08:00
|
|
|
return S.BuildBuiltinCallExpr(Loc, Builtin::BI__builtin_coro_resume,
|
|
|
|
JustAddress);
|
2017-08-25 12:46:54 +08:00
|
|
|
}
|
|
|
|
|
2015-10-27 14:02:45 +08:00
|
|
|
/// Build calls to await_ready, await_suspend, and await_resume for a co_await
|
|
|
|
/// expression.
|
[Coroutine][Sema] Cleanup temporaries as early as possible
The original bug was discovered in T75057860. Clang front-end emits an AST that looks like this for an co_await expression:
|- ExprWithCleanups
|- -CoawaitExpr
|- -MaterializeTemporaryExpr ... Awaiter
...
|- -CXXMemberCallExpr ... .await_ready
...
|- -CallExpr ... __builtin_coro_resume
...
|- -CXXMemberCallExpr ... .await_resume
...
ExprWithCleanups is responsible for cleaning up (including calling dtors) for the temporaries generated in the wrapping expression).
In the above structure, the __builtin_coro_resume part (which corresponds to the code for the suspend case in the co_await with symmetric transfer), the pseudocode looks like this:
__builtin_coro_resume(
awaiter.await_suspend(
from_address(
__builtin_coro_frame())).address());
One of the temporaries that's generated as part of this code is the coroutine handle returned from awaiter.await_suspend() call. The call returns a handle which is a prvalue (since it's a returned value on the fly). In order to call the address() method on it, it needs to be converted into an xvalue. Hence a materialized temp is created to hold it. This temp will need to be cleaned up eventually. Now, since all cleanups happen at the end of the entire co_await expression, which is after the <coro.suspend> suspension point, the compiler will think that such a temp needs to live across suspensions, and need to be put on the coroutine frame, even though it's only used temporarily just to call address() method.
Such a phenomena not only unnecessarily increases the frame size, but can lead to ASAN failures, if the coroutine was already destroyed as part of the await_suspend() call. This is because if the coroutine was already destroyed, the frame no longer exists, and one can not store anything into it. But if the temporary object is considered to need to live on the frame, it will be stored into the frame after await_suspend() returns.
A fix attempt was done in https://reviews.llvm.org/D87470. Unfortunately it is incorrect. The reason is that cleanups in Clang works more like linearly than nested. There is one current state indicating whether it needs cleanup, and an ExprWithCleanups resets that state. This means that an ExprWithCleanups must be capable of cleaning up all temporaries created in the wrapping expression, otherwise there will be dangling temporaries cleaned up at the wrong place.
I eventually found a walk-around (https://reviews.llvm.org/D89066) that doesn't break any existing tests while fixing the issue. But it targets the final co_await only. If we ever have a co_await that's not on the final awaiter and the frame gets destroyed after suspend, we are in trouble. Hence we need a proper fix.
This patch is the proper fix. It does the folllowing things to fully resolve the issue:
1. The AST has to be generated in the order according to their nesting relationship. We should not generate AST out of order because then the code generator would incorrectly track the state of temporaries and when a cleanup is needed. So the code in buildCoawaitCalls is reorganized so that we will be generating the AST for each coawait member call in order along with their child AST.
2. await_ready() call is wrapped with an ExprWithCleanups so that temporaries in it gets cleaned up as early as possible to avoid living across suspension.
3. await_suspend() call is wrapped with an ExprWithCleanups if it's not a symmetric transfer. In the case of a symmetric transfer, in order to maintain the musttail call contract, the ExprWithCleanups is wraaped before the resume call.
4. In the end, we mark again that it needs a cleanup, so that the entire CoawaitExpr will be wrapped with a ExprWithCleanups which will clean up the Awaiter object associated with the await expression.
Differential Revision: https://reviews.llvm.org/D90990
2020-11-11 05:02:18 +08:00
|
|
|
/// The generated AST tries to clean up temporary objects as early as
|
|
|
|
/// possible so that they don't live across suspension points if possible.
|
|
|
|
/// Having temporary objects living across suspension points unnecessarily can
|
|
|
|
/// lead to large frame size, and also lead to memory corruptions if the
|
|
|
|
/// coroutine frame is destroyed after coming back from suspension. This is done
|
|
|
|
/// by wrapping both the await_ready call and the await_suspend call with
|
|
|
|
/// ExprWithCleanups. In the end of this function, we also need to explicitly
|
|
|
|
/// set cleanup state so that the CoawaitExpr is also wrapped with an
|
|
|
|
/// ExprWithCleanups to clean up the awaiter associated with the co_await
|
|
|
|
/// expression.
|
2017-03-09 11:09:43 +08:00
|
|
|
static ReadySuspendResumeResult buildCoawaitCalls(Sema &S, VarDecl *CoroPromise,
|
2020-11-12 13:17:16 +08:00
|
|
|
SourceLocation Loc, Expr *E) {
|
2017-03-11 09:30:17 +08:00
|
|
|
OpaqueValueExpr *Operand = new (S.Context)
|
|
|
|
OpaqueValueExpr(Loc, E->getType(), VK_LValue, E->getObjectKind(), E);
|
|
|
|
|
[Coroutine][Sema] Cleanup temporaries as early as possible
The original bug was discovered in T75057860. Clang front-end emits an AST that looks like this for an co_await expression:
|- ExprWithCleanups
|- -CoawaitExpr
|- -MaterializeTemporaryExpr ... Awaiter
...
|- -CXXMemberCallExpr ... .await_ready
...
|- -CallExpr ... __builtin_coro_resume
...
|- -CXXMemberCallExpr ... .await_resume
...
ExprWithCleanups is responsible for cleaning up (including calling dtors) for the temporaries generated in the wrapping expression).
In the above structure, the __builtin_coro_resume part (which corresponds to the code for the suspend case in the co_await with symmetric transfer), the pseudocode looks like this:
__builtin_coro_resume(
awaiter.await_suspend(
from_address(
__builtin_coro_frame())).address());
One of the temporaries that's generated as part of this code is the coroutine handle returned from awaiter.await_suspend() call. The call returns a handle which is a prvalue (since it's a returned value on the fly). In order to call the address() method on it, it needs to be converted into an xvalue. Hence a materialized temp is created to hold it. This temp will need to be cleaned up eventually. Now, since all cleanups happen at the end of the entire co_await expression, which is after the <coro.suspend> suspension point, the compiler will think that such a temp needs to live across suspensions, and need to be put on the coroutine frame, even though it's only used temporarily just to call address() method.
Such a phenomena not only unnecessarily increases the frame size, but can lead to ASAN failures, if the coroutine was already destroyed as part of the await_suspend() call. This is because if the coroutine was already destroyed, the frame no longer exists, and one can not store anything into it. But if the temporary object is considered to need to live on the frame, it will be stored into the frame after await_suspend() returns.
A fix attempt was done in https://reviews.llvm.org/D87470. Unfortunately it is incorrect. The reason is that cleanups in Clang works more like linearly than nested. There is one current state indicating whether it needs cleanup, and an ExprWithCleanups resets that state. This means that an ExprWithCleanups must be capable of cleaning up all temporaries created in the wrapping expression, otherwise there will be dangling temporaries cleaned up at the wrong place.
I eventually found a walk-around (https://reviews.llvm.org/D89066) that doesn't break any existing tests while fixing the issue. But it targets the final co_await only. If we ever have a co_await that's not on the final awaiter and the frame gets destroyed after suspend, we are in trouble. Hence we need a proper fix.
This patch is the proper fix. It does the folllowing things to fully resolve the issue:
1. The AST has to be generated in the order according to their nesting relationship. We should not generate AST out of order because then the code generator would incorrectly track the state of temporaries and when a cleanup is needed. So the code in buildCoawaitCalls is reorganized so that we will be generating the AST for each coawait member call in order along with their child AST.
2. await_ready() call is wrapped with an ExprWithCleanups so that temporaries in it gets cleaned up as early as possible to avoid living across suspension.
3. await_suspend() call is wrapped with an ExprWithCleanups if it's not a symmetric transfer. In the case of a symmetric transfer, in order to maintain the musttail call contract, the ExprWithCleanups is wraaped before the resume call.
4. In the end, we mark again that it needs a cleanup, so that the entire CoawaitExpr will be wrapped with a ExprWithCleanups which will clean up the Awaiter object associated with the await expression.
Differential Revision: https://reviews.llvm.org/D90990
2020-11-11 05:02:18 +08:00
|
|
|
// Assume valid until we see otherwise.
|
|
|
|
// Further operations are responsible for setting IsInalid to true.
|
|
|
|
ReadySuspendResumeResult Calls = {{}, Operand, /*IsInvalid=*/false};
|
2015-10-27 14:02:45 +08:00
|
|
|
|
[Coroutine][Sema] Cleanup temporaries as early as possible
The original bug was discovered in T75057860. Clang front-end emits an AST that looks like this for an co_await expression:
|- ExprWithCleanups
|- -CoawaitExpr
|- -MaterializeTemporaryExpr ... Awaiter
...
|- -CXXMemberCallExpr ... .await_ready
...
|- -CallExpr ... __builtin_coro_resume
...
|- -CXXMemberCallExpr ... .await_resume
...
ExprWithCleanups is responsible for cleaning up (including calling dtors) for the temporaries generated in the wrapping expression).
In the above structure, the __builtin_coro_resume part (which corresponds to the code for the suspend case in the co_await with symmetric transfer), the pseudocode looks like this:
__builtin_coro_resume(
awaiter.await_suspend(
from_address(
__builtin_coro_frame())).address());
One of the temporaries that's generated as part of this code is the coroutine handle returned from awaiter.await_suspend() call. The call returns a handle which is a prvalue (since it's a returned value on the fly). In order to call the address() method on it, it needs to be converted into an xvalue. Hence a materialized temp is created to hold it. This temp will need to be cleaned up eventually. Now, since all cleanups happen at the end of the entire co_await expression, which is after the <coro.suspend> suspension point, the compiler will think that such a temp needs to live across suspensions, and need to be put on the coroutine frame, even though it's only used temporarily just to call address() method.
Such a phenomena not only unnecessarily increases the frame size, but can lead to ASAN failures, if the coroutine was already destroyed as part of the await_suspend() call. This is because if the coroutine was already destroyed, the frame no longer exists, and one can not store anything into it. But if the temporary object is considered to need to live on the frame, it will be stored into the frame after await_suspend() returns.
A fix attempt was done in https://reviews.llvm.org/D87470. Unfortunately it is incorrect. The reason is that cleanups in Clang works more like linearly than nested. There is one current state indicating whether it needs cleanup, and an ExprWithCleanups resets that state. This means that an ExprWithCleanups must be capable of cleaning up all temporaries created in the wrapping expression, otherwise there will be dangling temporaries cleaned up at the wrong place.
I eventually found a walk-around (https://reviews.llvm.org/D89066) that doesn't break any existing tests while fixing the issue. But it targets the final co_await only. If we ever have a co_await that's not on the final awaiter and the frame gets destroyed after suspend, we are in trouble. Hence we need a proper fix.
This patch is the proper fix. It does the folllowing things to fully resolve the issue:
1. The AST has to be generated in the order according to their nesting relationship. We should not generate AST out of order because then the code generator would incorrectly track the state of temporaries and when a cleanup is needed. So the code in buildCoawaitCalls is reorganized so that we will be generating the AST for each coawait member call in order along with their child AST.
2. await_ready() call is wrapped with an ExprWithCleanups so that temporaries in it gets cleaned up as early as possible to avoid living across suspension.
3. await_suspend() call is wrapped with an ExprWithCleanups if it's not a symmetric transfer. In the case of a symmetric transfer, in order to maintain the musttail call contract, the ExprWithCleanups is wraaped before the resume call.
4. In the end, we mark again that it needs a cleanup, so that the entire CoawaitExpr will be wrapped with a ExprWithCleanups which will clean up the Awaiter object associated with the await expression.
Differential Revision: https://reviews.llvm.org/D90990
2020-11-11 05:02:18 +08:00
|
|
|
using ACT = ReadySuspendResumeResult::AwaitCallType;
|
2015-10-27 14:02:45 +08:00
|
|
|
|
[Coroutine][Sema] Cleanup temporaries as early as possible
The original bug was discovered in T75057860. Clang front-end emits an AST that looks like this for an co_await expression:
|- ExprWithCleanups
|- -CoawaitExpr
|- -MaterializeTemporaryExpr ... Awaiter
...
|- -CXXMemberCallExpr ... .await_ready
...
|- -CallExpr ... __builtin_coro_resume
...
|- -CXXMemberCallExpr ... .await_resume
...
ExprWithCleanups is responsible for cleaning up (including calling dtors) for the temporaries generated in the wrapping expression).
In the above structure, the __builtin_coro_resume part (which corresponds to the code for the suspend case in the co_await with symmetric transfer), the pseudocode looks like this:
__builtin_coro_resume(
awaiter.await_suspend(
from_address(
__builtin_coro_frame())).address());
One of the temporaries that's generated as part of this code is the coroutine handle returned from awaiter.await_suspend() call. The call returns a handle which is a prvalue (since it's a returned value on the fly). In order to call the address() method on it, it needs to be converted into an xvalue. Hence a materialized temp is created to hold it. This temp will need to be cleaned up eventually. Now, since all cleanups happen at the end of the entire co_await expression, which is after the <coro.suspend> suspension point, the compiler will think that such a temp needs to live across suspensions, and need to be put on the coroutine frame, even though it's only used temporarily just to call address() method.
Such a phenomena not only unnecessarily increases the frame size, but can lead to ASAN failures, if the coroutine was already destroyed as part of the await_suspend() call. This is because if the coroutine was already destroyed, the frame no longer exists, and one can not store anything into it. But if the temporary object is considered to need to live on the frame, it will be stored into the frame after await_suspend() returns.
A fix attempt was done in https://reviews.llvm.org/D87470. Unfortunately it is incorrect. The reason is that cleanups in Clang works more like linearly than nested. There is one current state indicating whether it needs cleanup, and an ExprWithCleanups resets that state. This means that an ExprWithCleanups must be capable of cleaning up all temporaries created in the wrapping expression, otherwise there will be dangling temporaries cleaned up at the wrong place.
I eventually found a walk-around (https://reviews.llvm.org/D89066) that doesn't break any existing tests while fixing the issue. But it targets the final co_await only. If we ever have a co_await that's not on the final awaiter and the frame gets destroyed after suspend, we are in trouble. Hence we need a proper fix.
This patch is the proper fix. It does the folllowing things to fully resolve the issue:
1. The AST has to be generated in the order according to their nesting relationship. We should not generate AST out of order because then the code generator would incorrectly track the state of temporaries and when a cleanup is needed. So the code in buildCoawaitCalls is reorganized so that we will be generating the AST for each coawait member call in order along with their child AST.
2. await_ready() call is wrapped with an ExprWithCleanups so that temporaries in it gets cleaned up as early as possible to avoid living across suspension.
3. await_suspend() call is wrapped with an ExprWithCleanups if it's not a symmetric transfer. In the case of a symmetric transfer, in order to maintain the musttail call contract, the ExprWithCleanups is wraaped before the resume call.
4. In the end, we mark again that it needs a cleanup, so that the entire CoawaitExpr will be wrapped with a ExprWithCleanups which will clean up the Awaiter object associated with the await expression.
Differential Revision: https://reviews.llvm.org/D90990
2020-11-11 05:02:18 +08:00
|
|
|
auto BuildSubExpr = [&](ACT CallType, StringRef Func,
|
|
|
|
MultiExprArg Arg) -> Expr * {
|
|
|
|
ExprResult Result = buildMemberCall(S, Operand, Loc, Func, Arg);
|
|
|
|
if (Result.isInvalid()) {
|
|
|
|
Calls.IsInvalid = true;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
Calls.Results[CallType] = Result.get();
|
|
|
|
return Result.get();
|
|
|
|
};
|
2017-05-29 02:21:12 +08:00
|
|
|
|
[Coroutine][Sema] Cleanup temporaries as early as possible
The original bug was discovered in T75057860. Clang front-end emits an AST that looks like this for an co_await expression:
|- ExprWithCleanups
|- -CoawaitExpr
|- -MaterializeTemporaryExpr ... Awaiter
...
|- -CXXMemberCallExpr ... .await_ready
...
|- -CallExpr ... __builtin_coro_resume
...
|- -CXXMemberCallExpr ... .await_resume
...
ExprWithCleanups is responsible for cleaning up (including calling dtors) for the temporaries generated in the wrapping expression).
In the above structure, the __builtin_coro_resume part (which corresponds to the code for the suspend case in the co_await with symmetric transfer), the pseudocode looks like this:
__builtin_coro_resume(
awaiter.await_suspend(
from_address(
__builtin_coro_frame())).address());
One of the temporaries that's generated as part of this code is the coroutine handle returned from awaiter.await_suspend() call. The call returns a handle which is a prvalue (since it's a returned value on the fly). In order to call the address() method on it, it needs to be converted into an xvalue. Hence a materialized temp is created to hold it. This temp will need to be cleaned up eventually. Now, since all cleanups happen at the end of the entire co_await expression, which is after the <coro.suspend> suspension point, the compiler will think that such a temp needs to live across suspensions, and need to be put on the coroutine frame, even though it's only used temporarily just to call address() method.
Such a phenomena not only unnecessarily increases the frame size, but can lead to ASAN failures, if the coroutine was already destroyed as part of the await_suspend() call. This is because if the coroutine was already destroyed, the frame no longer exists, and one can not store anything into it. But if the temporary object is considered to need to live on the frame, it will be stored into the frame after await_suspend() returns.
A fix attempt was done in https://reviews.llvm.org/D87470. Unfortunately it is incorrect. The reason is that cleanups in Clang works more like linearly than nested. There is one current state indicating whether it needs cleanup, and an ExprWithCleanups resets that state. This means that an ExprWithCleanups must be capable of cleaning up all temporaries created in the wrapping expression, otherwise there will be dangling temporaries cleaned up at the wrong place.
I eventually found a walk-around (https://reviews.llvm.org/D89066) that doesn't break any existing tests while fixing the issue. But it targets the final co_await only. If we ever have a co_await that's not on the final awaiter and the frame gets destroyed after suspend, we are in trouble. Hence we need a proper fix.
This patch is the proper fix. It does the folllowing things to fully resolve the issue:
1. The AST has to be generated in the order according to their nesting relationship. We should not generate AST out of order because then the code generator would incorrectly track the state of temporaries and when a cleanup is needed. So the code in buildCoawaitCalls is reorganized so that we will be generating the AST for each coawait member call in order along with their child AST.
2. await_ready() call is wrapped with an ExprWithCleanups so that temporaries in it gets cleaned up as early as possible to avoid living across suspension.
3. await_suspend() call is wrapped with an ExprWithCleanups if it's not a symmetric transfer. In the case of a symmetric transfer, in order to maintain the musttail call contract, the ExprWithCleanups is wraaped before the resume call.
4. In the end, we mark again that it needs a cleanup, so that the entire CoawaitExpr will be wrapped with a ExprWithCleanups which will clean up the Awaiter object associated with the await expression.
Differential Revision: https://reviews.llvm.org/D90990
2020-11-11 05:02:18 +08:00
|
|
|
CallExpr *AwaitReady =
|
|
|
|
cast_or_null<CallExpr>(BuildSubExpr(ACT::ACT_Ready, "await_ready", None));
|
|
|
|
if (!AwaitReady)
|
|
|
|
return Calls;
|
2017-05-29 02:21:12 +08:00
|
|
|
if (!AwaitReady->getType()->isDependentType()) {
|
|
|
|
// [expr.await]p3 [...]
|
|
|
|
// — await-ready is the expression e.await_ready(), contextually converted
|
|
|
|
// to bool.
|
|
|
|
ExprResult Conv = S.PerformContextuallyConvertToBool(AwaitReady);
|
|
|
|
if (Conv.isInvalid()) {
|
2018-08-10 05:08:08 +08:00
|
|
|
S.Diag(AwaitReady->getDirectCallee()->getBeginLoc(),
|
2017-05-29 02:21:12 +08:00
|
|
|
diag::note_await_ready_no_bool_conversion);
|
|
|
|
S.Diag(Loc, diag::note_coroutine_promise_call_implicitly_required)
|
|
|
|
<< AwaitReady->getDirectCallee() << E->getSourceRange();
|
|
|
|
Calls.IsInvalid = true;
|
[Coroutine][Sema] Cleanup temporaries as early as possible
The original bug was discovered in T75057860. Clang front-end emits an AST that looks like this for an co_await expression:
|- ExprWithCleanups
|- -CoawaitExpr
|- -MaterializeTemporaryExpr ... Awaiter
...
|- -CXXMemberCallExpr ... .await_ready
...
|- -CallExpr ... __builtin_coro_resume
...
|- -CXXMemberCallExpr ... .await_resume
...
ExprWithCleanups is responsible for cleaning up (including calling dtors) for the temporaries generated in the wrapping expression).
In the above structure, the __builtin_coro_resume part (which corresponds to the code for the suspend case in the co_await with symmetric transfer), the pseudocode looks like this:
__builtin_coro_resume(
awaiter.await_suspend(
from_address(
__builtin_coro_frame())).address());
One of the temporaries that's generated as part of this code is the coroutine handle returned from awaiter.await_suspend() call. The call returns a handle which is a prvalue (since it's a returned value on the fly). In order to call the address() method on it, it needs to be converted into an xvalue. Hence a materialized temp is created to hold it. This temp will need to be cleaned up eventually. Now, since all cleanups happen at the end of the entire co_await expression, which is after the <coro.suspend> suspension point, the compiler will think that such a temp needs to live across suspensions, and need to be put on the coroutine frame, even though it's only used temporarily just to call address() method.
Such a phenomena not only unnecessarily increases the frame size, but can lead to ASAN failures, if the coroutine was already destroyed as part of the await_suspend() call. This is because if the coroutine was already destroyed, the frame no longer exists, and one can not store anything into it. But if the temporary object is considered to need to live on the frame, it will be stored into the frame after await_suspend() returns.
A fix attempt was done in https://reviews.llvm.org/D87470. Unfortunately it is incorrect. The reason is that cleanups in Clang works more like linearly than nested. There is one current state indicating whether it needs cleanup, and an ExprWithCleanups resets that state. This means that an ExprWithCleanups must be capable of cleaning up all temporaries created in the wrapping expression, otherwise there will be dangling temporaries cleaned up at the wrong place.
I eventually found a walk-around (https://reviews.llvm.org/D89066) that doesn't break any existing tests while fixing the issue. But it targets the final co_await only. If we ever have a co_await that's not on the final awaiter and the frame gets destroyed after suspend, we are in trouble. Hence we need a proper fix.
This patch is the proper fix. It does the folllowing things to fully resolve the issue:
1. The AST has to be generated in the order according to their nesting relationship. We should not generate AST out of order because then the code generator would incorrectly track the state of temporaries and when a cleanup is needed. So the code in buildCoawaitCalls is reorganized so that we will be generating the AST for each coawait member call in order along with their child AST.
2. await_ready() call is wrapped with an ExprWithCleanups so that temporaries in it gets cleaned up as early as possible to avoid living across suspension.
3. await_suspend() call is wrapped with an ExprWithCleanups if it's not a symmetric transfer. In the case of a symmetric transfer, in order to maintain the musttail call contract, the ExprWithCleanups is wraaped before the resume call.
4. In the end, we mark again that it needs a cleanup, so that the entire CoawaitExpr will be wrapped with a ExprWithCleanups which will clean up the Awaiter object associated with the await expression.
Differential Revision: https://reviews.llvm.org/D90990
2020-11-11 05:02:18 +08:00
|
|
|
} else
|
|
|
|
Calls.Results[ACT::ACT_Ready] = S.MaybeCreateExprWithCleanups(Conv.get());
|
|
|
|
}
|
|
|
|
|
|
|
|
ExprResult CoroHandleRes =
|
|
|
|
buildCoroutineHandle(S, CoroPromise->getType(), Loc);
|
|
|
|
if (CoroHandleRes.isInvalid()) {
|
|
|
|
Calls.IsInvalid = true;
|
|
|
|
return Calls;
|
2017-05-29 02:21:12 +08:00
|
|
|
}
|
[Coroutine][Sema] Cleanup temporaries as early as possible
The original bug was discovered in T75057860. Clang front-end emits an AST that looks like this for an co_await expression:
|- ExprWithCleanups
|- -CoawaitExpr
|- -MaterializeTemporaryExpr ... Awaiter
...
|- -CXXMemberCallExpr ... .await_ready
...
|- -CallExpr ... __builtin_coro_resume
...
|- -CXXMemberCallExpr ... .await_resume
...
ExprWithCleanups is responsible for cleaning up (including calling dtors) for the temporaries generated in the wrapping expression).
In the above structure, the __builtin_coro_resume part (which corresponds to the code for the suspend case in the co_await with symmetric transfer), the pseudocode looks like this:
__builtin_coro_resume(
awaiter.await_suspend(
from_address(
__builtin_coro_frame())).address());
One of the temporaries that's generated as part of this code is the coroutine handle returned from awaiter.await_suspend() call. The call returns a handle which is a prvalue (since it's a returned value on the fly). In order to call the address() method on it, it needs to be converted into an xvalue. Hence a materialized temp is created to hold it. This temp will need to be cleaned up eventually. Now, since all cleanups happen at the end of the entire co_await expression, which is after the <coro.suspend> suspension point, the compiler will think that such a temp needs to live across suspensions, and need to be put on the coroutine frame, even though it's only used temporarily just to call address() method.
Such a phenomena not only unnecessarily increases the frame size, but can lead to ASAN failures, if the coroutine was already destroyed as part of the await_suspend() call. This is because if the coroutine was already destroyed, the frame no longer exists, and one can not store anything into it. But if the temporary object is considered to need to live on the frame, it will be stored into the frame after await_suspend() returns.
A fix attempt was done in https://reviews.llvm.org/D87470. Unfortunately it is incorrect. The reason is that cleanups in Clang works more like linearly than nested. There is one current state indicating whether it needs cleanup, and an ExprWithCleanups resets that state. This means that an ExprWithCleanups must be capable of cleaning up all temporaries created in the wrapping expression, otherwise there will be dangling temporaries cleaned up at the wrong place.
I eventually found a walk-around (https://reviews.llvm.org/D89066) that doesn't break any existing tests while fixing the issue. But it targets the final co_await only. If we ever have a co_await that's not on the final awaiter and the frame gets destroyed after suspend, we are in trouble. Hence we need a proper fix.
This patch is the proper fix. It does the folllowing things to fully resolve the issue:
1. The AST has to be generated in the order according to their nesting relationship. We should not generate AST out of order because then the code generator would incorrectly track the state of temporaries and when a cleanup is needed. So the code in buildCoawaitCalls is reorganized so that we will be generating the AST for each coawait member call in order along with their child AST.
2. await_ready() call is wrapped with an ExprWithCleanups so that temporaries in it gets cleaned up as early as possible to avoid living across suspension.
3. await_suspend() call is wrapped with an ExprWithCleanups if it's not a symmetric transfer. In the case of a symmetric transfer, in order to maintain the musttail call contract, the ExprWithCleanups is wraaped before the resume call.
4. In the end, we mark again that it needs a cleanup, so that the entire CoawaitExpr will be wrapped with a ExprWithCleanups which will clean up the Awaiter object associated with the await expression.
Differential Revision: https://reviews.llvm.org/D90990
2020-11-11 05:02:18 +08:00
|
|
|
Expr *CoroHandle = CoroHandleRes.get();
|
|
|
|
CallExpr *AwaitSuspend = cast_or_null<CallExpr>(
|
|
|
|
BuildSubExpr(ACT::ACT_Suspend, "await_suspend", CoroHandle));
|
|
|
|
if (!AwaitSuspend)
|
|
|
|
return Calls;
|
2017-05-29 02:21:12 +08:00
|
|
|
if (!AwaitSuspend->getType()->isDependentType()) {
|
|
|
|
// [expr.await]p3 [...]
|
|
|
|
// - await-suspend is the expression e.await_suspend(h), which shall be
|
2020-07-29 10:20:55 +08:00
|
|
|
// a prvalue of type void, bool, or std::coroutine_handle<Z> for some
|
|
|
|
// type Z.
|
2017-06-01 07:41:11 +08:00
|
|
|
QualType RetType = AwaitSuspend->getCallReturnType(S.Context);
|
2017-09-06 03:31:52 +08:00
|
|
|
|
2017-08-25 12:46:54 +08:00
|
|
|
// Experimental support for coroutine_handle returning await_suspend.
|
2020-10-13 03:00:20 +08:00
|
|
|
if (Expr *TailCallSuspend =
|
2020-11-12 13:17:16 +08:00
|
|
|
maybeTailCall(S, RetType, AwaitSuspend, Loc))
|
[Coroutine][Sema] Cleanup temporaries as early as possible
The original bug was discovered in T75057860. Clang front-end emits an AST that looks like this for an co_await expression:
|- ExprWithCleanups
|- -CoawaitExpr
|- -MaterializeTemporaryExpr ... Awaiter
...
|- -CXXMemberCallExpr ... .await_ready
...
|- -CallExpr ... __builtin_coro_resume
...
|- -CXXMemberCallExpr ... .await_resume
...
ExprWithCleanups is responsible for cleaning up (including calling dtors) for the temporaries generated in the wrapping expression).
In the above structure, the __builtin_coro_resume part (which corresponds to the code for the suspend case in the co_await with symmetric transfer), the pseudocode looks like this:
__builtin_coro_resume(
awaiter.await_suspend(
from_address(
__builtin_coro_frame())).address());
One of the temporaries that's generated as part of this code is the coroutine handle returned from awaiter.await_suspend() call. The call returns a handle which is a prvalue (since it's a returned value on the fly). In order to call the address() method on it, it needs to be converted into an xvalue. Hence a materialized temp is created to hold it. This temp will need to be cleaned up eventually. Now, since all cleanups happen at the end of the entire co_await expression, which is after the <coro.suspend> suspension point, the compiler will think that such a temp needs to live across suspensions, and need to be put on the coroutine frame, even though it's only used temporarily just to call address() method.
Such a phenomena not only unnecessarily increases the frame size, but can lead to ASAN failures, if the coroutine was already destroyed as part of the await_suspend() call. This is because if the coroutine was already destroyed, the frame no longer exists, and one can not store anything into it. But if the temporary object is considered to need to live on the frame, it will be stored into the frame after await_suspend() returns.
A fix attempt was done in https://reviews.llvm.org/D87470. Unfortunately it is incorrect. The reason is that cleanups in Clang works more like linearly than nested. There is one current state indicating whether it needs cleanup, and an ExprWithCleanups resets that state. This means that an ExprWithCleanups must be capable of cleaning up all temporaries created in the wrapping expression, otherwise there will be dangling temporaries cleaned up at the wrong place.
I eventually found a walk-around (https://reviews.llvm.org/D89066) that doesn't break any existing tests while fixing the issue. But it targets the final co_await only. If we ever have a co_await that's not on the final awaiter and the frame gets destroyed after suspend, we are in trouble. Hence we need a proper fix.
This patch is the proper fix. It does the folllowing things to fully resolve the issue:
1. The AST has to be generated in the order according to their nesting relationship. We should not generate AST out of order because then the code generator would incorrectly track the state of temporaries and when a cleanup is needed. So the code in buildCoawaitCalls is reorganized so that we will be generating the AST for each coawait member call in order along with their child AST.
2. await_ready() call is wrapped with an ExprWithCleanups so that temporaries in it gets cleaned up as early as possible to avoid living across suspension.
3. await_suspend() call is wrapped with an ExprWithCleanups if it's not a symmetric transfer. In the case of a symmetric transfer, in order to maintain the musttail call contract, the ExprWithCleanups is wraaped before the resume call.
4. In the end, we mark again that it needs a cleanup, so that the entire CoawaitExpr will be wrapped with a ExprWithCleanups which will clean up the Awaiter object associated with the await expression.
Differential Revision: https://reviews.llvm.org/D90990
2020-11-11 05:02:18 +08:00
|
|
|
// Note that we don't wrap the expression with ExprWithCleanups here
|
|
|
|
// because that might interfere with tailcall contract (e.g. inserting
|
|
|
|
// clean up instructions in-between tailcall and return). Instead
|
|
|
|
// ExprWithCleanups is wrapped within maybeTailCall() prior to the resume
|
|
|
|
// call.
|
2017-08-25 12:46:54 +08:00
|
|
|
Calls.Results[ACT::ACT_Suspend] = TailCallSuspend;
|
|
|
|
else {
|
|
|
|
// non-class prvalues always have cv-unqualified types
|
|
|
|
if (RetType->isReferenceType() ||
|
2017-09-06 03:31:52 +08:00
|
|
|
(!RetType->isBooleanType() && !RetType->isVoidType())) {
|
2017-08-25 12:46:54 +08:00
|
|
|
S.Diag(AwaitSuspend->getCalleeDecl()->getLocation(),
|
|
|
|
diag::err_await_suspend_invalid_return_type)
|
|
|
|
<< RetType;
|
|
|
|
S.Diag(Loc, diag::note_coroutine_promise_call_implicitly_required)
|
|
|
|
<< AwaitSuspend->getDirectCallee();
|
|
|
|
Calls.IsInvalid = true;
|
[Coroutine][Sema] Cleanup temporaries as early as possible
The original bug was discovered in T75057860. Clang front-end emits an AST that looks like this for an co_await expression:
|- ExprWithCleanups
|- -CoawaitExpr
|- -MaterializeTemporaryExpr ... Awaiter
...
|- -CXXMemberCallExpr ... .await_ready
...
|- -CallExpr ... __builtin_coro_resume
...
|- -CXXMemberCallExpr ... .await_resume
...
ExprWithCleanups is responsible for cleaning up (including calling dtors) for the temporaries generated in the wrapping expression).
In the above structure, the __builtin_coro_resume part (which corresponds to the code for the suspend case in the co_await with symmetric transfer), the pseudocode looks like this:
__builtin_coro_resume(
awaiter.await_suspend(
from_address(
__builtin_coro_frame())).address());
One of the temporaries that's generated as part of this code is the coroutine handle returned from awaiter.await_suspend() call. The call returns a handle which is a prvalue (since it's a returned value on the fly). In order to call the address() method on it, it needs to be converted into an xvalue. Hence a materialized temp is created to hold it. This temp will need to be cleaned up eventually. Now, since all cleanups happen at the end of the entire co_await expression, which is after the <coro.suspend> suspension point, the compiler will think that such a temp needs to live across suspensions, and need to be put on the coroutine frame, even though it's only used temporarily just to call address() method.
Such a phenomena not only unnecessarily increases the frame size, but can lead to ASAN failures, if the coroutine was already destroyed as part of the await_suspend() call. This is because if the coroutine was already destroyed, the frame no longer exists, and one can not store anything into it. But if the temporary object is considered to need to live on the frame, it will be stored into the frame after await_suspend() returns.
A fix attempt was done in https://reviews.llvm.org/D87470. Unfortunately it is incorrect. The reason is that cleanups in Clang works more like linearly than nested. There is one current state indicating whether it needs cleanup, and an ExprWithCleanups resets that state. This means that an ExprWithCleanups must be capable of cleaning up all temporaries created in the wrapping expression, otherwise there will be dangling temporaries cleaned up at the wrong place.
I eventually found a walk-around (https://reviews.llvm.org/D89066) that doesn't break any existing tests while fixing the issue. But it targets the final co_await only. If we ever have a co_await that's not on the final awaiter and the frame gets destroyed after suspend, we are in trouble. Hence we need a proper fix.
This patch is the proper fix. It does the folllowing things to fully resolve the issue:
1. The AST has to be generated in the order according to their nesting relationship. We should not generate AST out of order because then the code generator would incorrectly track the state of temporaries and when a cleanup is needed. So the code in buildCoawaitCalls is reorganized so that we will be generating the AST for each coawait member call in order along with their child AST.
2. await_ready() call is wrapped with an ExprWithCleanups so that temporaries in it gets cleaned up as early as possible to avoid living across suspension.
3. await_suspend() call is wrapped with an ExprWithCleanups if it's not a symmetric transfer. In the case of a symmetric transfer, in order to maintain the musttail call contract, the ExprWithCleanups is wraaped before the resume call.
4. In the end, we mark again that it needs a cleanup, so that the entire CoawaitExpr will be wrapped with a ExprWithCleanups which will clean up the Awaiter object associated with the await expression.
Differential Revision: https://reviews.llvm.org/D90990
2020-11-11 05:02:18 +08:00
|
|
|
} else
|
|
|
|
Calls.Results[ACT::ACT_Suspend] =
|
|
|
|
S.MaybeCreateExprWithCleanups(AwaitSuspend);
|
2017-05-29 02:21:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
[Coroutine][Sema] Cleanup temporaries as early as possible
The original bug was discovered in T75057860. Clang front-end emits an AST that looks like this for an co_await expression:
|- ExprWithCleanups
|- -CoawaitExpr
|- -MaterializeTemporaryExpr ... Awaiter
...
|- -CXXMemberCallExpr ... .await_ready
...
|- -CallExpr ... __builtin_coro_resume
...
|- -CXXMemberCallExpr ... .await_resume
...
ExprWithCleanups is responsible for cleaning up (including calling dtors) for the temporaries generated in the wrapping expression).
In the above structure, the __builtin_coro_resume part (which corresponds to the code for the suspend case in the co_await with symmetric transfer), the pseudocode looks like this:
__builtin_coro_resume(
awaiter.await_suspend(
from_address(
__builtin_coro_frame())).address());
One of the temporaries that's generated as part of this code is the coroutine handle returned from awaiter.await_suspend() call. The call returns a handle which is a prvalue (since it's a returned value on the fly). In order to call the address() method on it, it needs to be converted into an xvalue. Hence a materialized temp is created to hold it. This temp will need to be cleaned up eventually. Now, since all cleanups happen at the end of the entire co_await expression, which is after the <coro.suspend> suspension point, the compiler will think that such a temp needs to live across suspensions, and need to be put on the coroutine frame, even though it's only used temporarily just to call address() method.
Such a phenomena not only unnecessarily increases the frame size, but can lead to ASAN failures, if the coroutine was already destroyed as part of the await_suspend() call. This is because if the coroutine was already destroyed, the frame no longer exists, and one can not store anything into it. But if the temporary object is considered to need to live on the frame, it will be stored into the frame after await_suspend() returns.
A fix attempt was done in https://reviews.llvm.org/D87470. Unfortunately it is incorrect. The reason is that cleanups in Clang works more like linearly than nested. There is one current state indicating whether it needs cleanup, and an ExprWithCleanups resets that state. This means that an ExprWithCleanups must be capable of cleaning up all temporaries created in the wrapping expression, otherwise there will be dangling temporaries cleaned up at the wrong place.
I eventually found a walk-around (https://reviews.llvm.org/D89066) that doesn't break any existing tests while fixing the issue. But it targets the final co_await only. If we ever have a co_await that's not on the final awaiter and the frame gets destroyed after suspend, we are in trouble. Hence we need a proper fix.
This patch is the proper fix. It does the folllowing things to fully resolve the issue:
1. The AST has to be generated in the order according to their nesting relationship. We should not generate AST out of order because then the code generator would incorrectly track the state of temporaries and when a cleanup is needed. So the code in buildCoawaitCalls is reorganized so that we will be generating the AST for each coawait member call in order along with their child AST.
2. await_ready() call is wrapped with an ExprWithCleanups so that temporaries in it gets cleaned up as early as possible to avoid living across suspension.
3. await_suspend() call is wrapped with an ExprWithCleanups if it's not a symmetric transfer. In the case of a symmetric transfer, in order to maintain the musttail call contract, the ExprWithCleanups is wraaped before the resume call.
4. In the end, we mark again that it needs a cleanup, so that the entire CoawaitExpr will be wrapped with a ExprWithCleanups which will clean up the Awaiter object associated with the await expression.
Differential Revision: https://reviews.llvm.org/D90990
2020-11-11 05:02:18 +08:00
|
|
|
BuildSubExpr(ACT::ACT_Resume, "await_resume", None);
|
|
|
|
|
|
|
|
// Make sure the awaiter object gets a chance to be cleaned up.
|
|
|
|
S.Cleanup.setExprNeedsCleanups(true);
|
|
|
|
|
2015-10-27 14:02:45 +08:00
|
|
|
return Calls;
|
|
|
|
}
|
|
|
|
|
2017-03-07 07:38:15 +08:00
|
|
|
static ExprResult buildPromiseCall(Sema &S, VarDecl *Promise,
|
|
|
|
SourceLocation Loc, StringRef Name,
|
|
|
|
MultiExprArg Args) {
|
|
|
|
|
|
|
|
// Form a reference to the promise.
|
|
|
|
ExprResult PromiseRef = S.BuildDeclRefExpr(
|
|
|
|
Promise, Promise->getType().getNonReferenceType(), VK_LValue, Loc);
|
|
|
|
if (PromiseRef.isInvalid())
|
|
|
|
return ExprError();
|
|
|
|
|
|
|
|
return buildMemberCall(S, PromiseRef.get(), Loc, Name, Args);
|
|
|
|
}
|
|
|
|
|
|
|
|
VarDecl *Sema::buildCoroutinePromise(SourceLocation Loc) {
|
|
|
|
assert(isa<FunctionDecl>(CurContext) && "not in a function scope");
|
|
|
|
auto *FD = cast<FunctionDecl>(CurContext);
|
2017-07-10 09:27:22 +08:00
|
|
|
bool IsThisDependentType = [&] {
|
|
|
|
if (auto *MD = dyn_cast_or_null<CXXMethodDecl>(FD))
|
2019-01-11 09:54:53 +08:00
|
|
|
return MD->isInstance() && MD->getThisType()->isDependentType();
|
2017-07-10 09:27:22 +08:00
|
|
|
else
|
|
|
|
return false;
|
|
|
|
}();
|
|
|
|
|
|
|
|
QualType T = FD->getType()->isDependentType() || IsThisDependentType
|
|
|
|
? Context.DependentTy
|
|
|
|
: lookupPromiseType(*this, FD, Loc);
|
2017-03-07 07:38:15 +08:00
|
|
|
if (T.isNull())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
auto *VD = VarDecl::Create(Context, FD, FD->getLocation(), FD->getLocation(),
|
|
|
|
&PP.getIdentifierTable().get("__promise"), T,
|
2021-01-05 06:17:45 +08:00
|
|
|
Context.getTrivialTypeSourceInfo(T, Loc), SC_None);
|
2020-12-17 03:42:07 +08:00
|
|
|
VD->setImplicit();
|
2017-03-07 07:38:15 +08:00
|
|
|
CheckVariableDeclarationType(VD);
|
|
|
|
if (VD->isInvalidDecl())
|
|
|
|
return nullptr;
|
2018-01-25 06:15:42 +08:00
|
|
|
|
|
|
|
auto *ScopeInfo = getCurFunction();
|
[coroutines] Don't build promise init with no args
Summary:
In the case of a coroutine that takes no arguments,
`Sema::buildCoroutinePromise` constructs a list-initialization
(`clang::InitializationKind::InitKind::IK_DirectList`) of the
promise variable, using a list of empty arguments. So, if one were to
dump the promise `VarDecl` immediately after `Sema::ActOnCoroutineBodyStart`
calls `checkCoroutineContext`, for a coroutine function that takes no
arguments, they'd see the following:
```
VarDecl 0xb514490 <test.cpp:26:3> col:3 __promise '<dependent type>' callinit
`-ParenListExpr 0xb514510 <col:3> 'NULL TYPE'
```
But after this patch, the `ParenListExpr` is no longer constructed, and
the promise variable uses default initialization
(`clang::InitializationKind::InitKind::IK_Default`):
```
VarDecl 0x63100012dae0 <test.cpp:26:3> col:3 __promise '<dependent type>'
```
As far as I know, there's no case in which list-initialization with no
arguments differs from default initialization, but if I'm wrong please
let me know (and I'll add a test case that demonstrates the change --
but as-is I can't think of a functional test case for this). I think both
comply with the wording of C++20 `[dcl.fct.def.coroutine]p5`:
> _promise-constructor-arguments_ is determined as follows: overload
resolution is performed on a promise constructor call created by
assembling an argument list with lvalues `p1 ... pn`. If a viable
constructor is found (12.4.2), then _promise-constructor-arguments_
is `(p1, ... , pn)`, otherwise _promise-constructor-arguments_ is
empty.
Still, I think this patch is an improvement regardless, because it
reduces the size of the AST.
Reviewers: GorNishanov, rsmith, lewissbaker
Subscribers: EricWF, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D70555
2019-11-22 02:51:44 +08:00
|
|
|
|
|
|
|
// Build a list of arguments, based on the coroutine function's arguments,
|
|
|
|
// that if present will be passed to the promise type's constructor.
|
2018-01-25 06:15:42 +08:00
|
|
|
llvm::SmallVector<Expr *, 4> CtorArgExprs;
|
2018-05-29 02:08:47 +08:00
|
|
|
|
|
|
|
// Add implicit object parameter.
|
|
|
|
if (auto *MD = dyn_cast<CXXMethodDecl>(FD)) {
|
|
|
|
if (MD->isInstance() && !isLambdaCallOperator(MD)) {
|
|
|
|
ExprResult ThisExpr = ActOnCXXThis(Loc);
|
|
|
|
if (ThisExpr.isInvalid())
|
|
|
|
return nullptr;
|
|
|
|
ThisExpr = CreateBuiltinUnaryOp(Loc, UO_Deref, ThisExpr.get());
|
|
|
|
if (ThisExpr.isInvalid())
|
|
|
|
return nullptr;
|
|
|
|
CtorArgExprs.push_back(ThisExpr.get());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
[coroutines] Don't build promise init with no args
Summary:
In the case of a coroutine that takes no arguments,
`Sema::buildCoroutinePromise` constructs a list-initialization
(`clang::InitializationKind::InitKind::IK_DirectList`) of the
promise variable, using a list of empty arguments. So, if one were to
dump the promise `VarDecl` immediately after `Sema::ActOnCoroutineBodyStart`
calls `checkCoroutineContext`, for a coroutine function that takes no
arguments, they'd see the following:
```
VarDecl 0xb514490 <test.cpp:26:3> col:3 __promise '<dependent type>' callinit
`-ParenListExpr 0xb514510 <col:3> 'NULL TYPE'
```
But after this patch, the `ParenListExpr` is no longer constructed, and
the promise variable uses default initialization
(`clang::InitializationKind::InitKind::IK_Default`):
```
VarDecl 0x63100012dae0 <test.cpp:26:3> col:3 __promise '<dependent type>'
```
As far as I know, there's no case in which list-initialization with no
arguments differs from default initialization, but if I'm wrong please
let me know (and I'll add a test case that demonstrates the change --
but as-is I can't think of a functional test case for this). I think both
comply with the wording of C++20 `[dcl.fct.def.coroutine]p5`:
> _promise-constructor-arguments_ is determined as follows: overload
resolution is performed on a promise constructor call created by
assembling an argument list with lvalues `p1 ... pn`. If a viable
constructor is found (12.4.2), then _promise-constructor-arguments_
is `(p1, ... , pn)`, otherwise _promise-constructor-arguments_ is
empty.
Still, I think this patch is an improvement regardless, because it
reduces the size of the AST.
Reviewers: GorNishanov, rsmith, lewissbaker
Subscribers: EricWF, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D70555
2019-11-22 02:51:44 +08:00
|
|
|
// Add the coroutine function's parameters.
|
2018-01-25 06:15:42 +08:00
|
|
|
auto &Moves = ScopeInfo->CoroutineParameterMoves;
|
|
|
|
for (auto *PD : FD->parameters()) {
|
|
|
|
if (PD->getType()->isDependentType())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
auto RefExpr = ExprEmpty();
|
|
|
|
auto Move = Moves.find(PD);
|
2018-02-16 04:37:22 +08:00
|
|
|
assert(Move != Moves.end() &&
|
|
|
|
"Coroutine function parameter not inserted into move map");
|
|
|
|
// If a reference to the function parameter exists in the coroutine
|
|
|
|
// frame, use that reference.
|
|
|
|
auto *MoveDecl =
|
|
|
|
cast<VarDecl>(cast<DeclStmt>(Move->second)->getSingleDecl());
|
|
|
|
RefExpr =
|
|
|
|
BuildDeclRefExpr(MoveDecl, MoveDecl->getType().getNonReferenceType(),
|
|
|
|
ExprValueKind::VK_LValue, FD->getLocation());
|
2018-01-25 06:15:42 +08:00
|
|
|
if (RefExpr.isInvalid())
|
|
|
|
return nullptr;
|
|
|
|
CtorArgExprs.push_back(RefExpr.get());
|
|
|
|
}
|
|
|
|
|
[coroutines] Don't build promise init with no args
Summary:
In the case of a coroutine that takes no arguments,
`Sema::buildCoroutinePromise` constructs a list-initialization
(`clang::InitializationKind::InitKind::IK_DirectList`) of the
promise variable, using a list of empty arguments. So, if one were to
dump the promise `VarDecl` immediately after `Sema::ActOnCoroutineBodyStart`
calls `checkCoroutineContext`, for a coroutine function that takes no
arguments, they'd see the following:
```
VarDecl 0xb514490 <test.cpp:26:3> col:3 __promise '<dependent type>' callinit
`-ParenListExpr 0xb514510 <col:3> 'NULL TYPE'
```
But after this patch, the `ParenListExpr` is no longer constructed, and
the promise variable uses default initialization
(`clang::InitializationKind::InitKind::IK_Default`):
```
VarDecl 0x63100012dae0 <test.cpp:26:3> col:3 __promise '<dependent type>'
```
As far as I know, there's no case in which list-initialization with no
arguments differs from default initialization, but if I'm wrong please
let me know (and I'll add a test case that demonstrates the change --
but as-is I can't think of a functional test case for this). I think both
comply with the wording of C++20 `[dcl.fct.def.coroutine]p5`:
> _promise-constructor-arguments_ is determined as follows: overload
resolution is performed on a promise constructor call created by
assembling an argument list with lvalues `p1 ... pn`. If a viable
constructor is found (12.4.2), then _promise-constructor-arguments_
is `(p1, ... , pn)`, otherwise _promise-constructor-arguments_ is
empty.
Still, I think this patch is an improvement regardless, because it
reduces the size of the AST.
Reviewers: GorNishanov, rsmith, lewissbaker
Subscribers: EricWF, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D70555
2019-11-22 02:51:44 +08:00
|
|
|
// If we have a non-zero number of constructor arguments, try to use them.
|
|
|
|
// Otherwise, fall back to the promise type's default constructor.
|
|
|
|
if (!CtorArgExprs.empty()) {
|
|
|
|
// Create an initialization sequence for the promise type using the
|
|
|
|
// constructor arguments, wrapped in a parenthesized list expression.
|
|
|
|
Expr *PLE = ParenListExpr::Create(Context, FD->getLocation(),
|
|
|
|
CtorArgExprs, FD->getLocation());
|
|
|
|
InitializedEntity Entity = InitializedEntity::InitializeVariable(VD);
|
|
|
|
InitializationKind Kind = InitializationKind::CreateForInit(
|
|
|
|
VD->getLocation(), /*DirectInit=*/true, PLE);
|
|
|
|
InitializationSequence InitSeq(*this, Entity, Kind, CtorArgExprs,
|
|
|
|
/*TopLevelOfInitList=*/false,
|
|
|
|
/*TreatUnavailableAsInvalid=*/false);
|
|
|
|
|
2021-12-24 11:48:15 +08:00
|
|
|
// [dcl.fct.def.coroutine]5.7
|
|
|
|
// promise-constructor-arguments is determined as follows: overload
|
|
|
|
// resolution is performed on a promise constructor call created by
|
|
|
|
// assembling an argument list q_1 ... q_n . If a viable constructor is
|
|
|
|
// found ([over.match.viable]), then promise-constructor-arguments is ( q_1
|
|
|
|
// , ..., q_n ), otherwise promise-constructor-arguments is empty.
|
[coroutines] Don't build promise init with no args
Summary:
In the case of a coroutine that takes no arguments,
`Sema::buildCoroutinePromise` constructs a list-initialization
(`clang::InitializationKind::InitKind::IK_DirectList`) of the
promise variable, using a list of empty arguments. So, if one were to
dump the promise `VarDecl` immediately after `Sema::ActOnCoroutineBodyStart`
calls `checkCoroutineContext`, for a coroutine function that takes no
arguments, they'd see the following:
```
VarDecl 0xb514490 <test.cpp:26:3> col:3 __promise '<dependent type>' callinit
`-ParenListExpr 0xb514510 <col:3> 'NULL TYPE'
```
But after this patch, the `ParenListExpr` is no longer constructed, and
the promise variable uses default initialization
(`clang::InitializationKind::InitKind::IK_Default`):
```
VarDecl 0x63100012dae0 <test.cpp:26:3> col:3 __promise '<dependent type>'
```
As far as I know, there's no case in which list-initialization with no
arguments differs from default initialization, but if I'm wrong please
let me know (and I'll add a test case that demonstrates the change --
but as-is I can't think of a functional test case for this). I think both
comply with the wording of C++20 `[dcl.fct.def.coroutine]p5`:
> _promise-constructor-arguments_ is determined as follows: overload
resolution is performed on a promise constructor call created by
assembling an argument list with lvalues `p1 ... pn`. If a viable
constructor is found (12.4.2), then _promise-constructor-arguments_
is `(p1, ... , pn)`, otherwise _promise-constructor-arguments_ is
empty.
Still, I think this patch is an improvement regardless, because it
reduces the size of the AST.
Reviewers: GorNishanov, rsmith, lewissbaker
Subscribers: EricWF, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D70555
2019-11-22 02:51:44 +08:00
|
|
|
if (InitSeq) {
|
|
|
|
ExprResult Result = InitSeq.Perform(*this, Entity, Kind, CtorArgExprs);
|
|
|
|
if (Result.isInvalid()) {
|
|
|
|
VD->setInvalidDecl();
|
|
|
|
} else if (Result.get()) {
|
|
|
|
VD->setInit(MaybeCreateExprWithCleanups(Result.get()));
|
|
|
|
VD->setInitStyle(VarDecl::CallInit);
|
|
|
|
CheckCompleteVariableDeclaration(VD);
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
ActOnUninitializedDecl(VD);
|
2018-01-25 06:15:42 +08:00
|
|
|
} else
|
|
|
|
ActOnUninitializedDecl(VD);
|
|
|
|
|
2017-06-01 03:36:59 +08:00
|
|
|
FD->addDecl(VD);
|
2017-03-07 07:38:15 +08:00
|
|
|
return VD;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Check that this is a context in which a coroutine suspension can appear.
|
|
|
|
static FunctionScopeInfo *checkCoroutineContext(Sema &S, SourceLocation Loc,
|
2017-03-11 10:35:37 +08:00
|
|
|
StringRef Keyword,
|
|
|
|
bool IsImplicit = false) {
|
2017-03-07 07:38:15 +08:00
|
|
|
if (!isValidCoroutineContext(S, Loc, Keyword))
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
assert(isa<FunctionDecl>(S.CurContext) && "not in a function scope");
|
|
|
|
|
|
|
|
auto *ScopeInfo = S.getCurFunction();
|
|
|
|
assert(ScopeInfo && "missing function scope for function");
|
|
|
|
|
2017-03-11 10:35:37 +08:00
|
|
|
if (ScopeInfo->FirstCoroutineStmtLoc.isInvalid() && !IsImplicit)
|
|
|
|
ScopeInfo->setFirstCoroutineStmt(Loc, Keyword);
|
|
|
|
|
2017-03-07 07:38:15 +08:00
|
|
|
if (ScopeInfo->CoroutinePromise)
|
|
|
|
return ScopeInfo;
|
|
|
|
|
2018-01-25 06:15:42 +08:00
|
|
|
if (!S.buildCoroutineParameterMoves(Loc))
|
|
|
|
return nullptr;
|
|
|
|
|
2017-03-07 07:38:15 +08:00
|
|
|
ScopeInfo->CoroutinePromise = S.buildCoroutinePromise(Loc);
|
|
|
|
if (!ScopeInfo->CoroutinePromise)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
return ScopeInfo;
|
|
|
|
}
|
|
|
|
|
[Coroutines] Ensure co_await promise.final_suspend() does not throw
Summary:
This patch addresses https://bugs.llvm.org/show_bug.cgi?id=46256
The spec of coroutine requires that the expression co_await promise.final_suspend() shall not be potentially-throwing.
To check this, we recursively look at every call (including Call, MemberCall, OperatorCall and Constructor) in all code
generated by the final suspend, and ensure that the callees are declared with noexcept. We also look at any returned data
type that requires explicit destruction, and check their destructors for noexcept.
This patch does not check declarations with dependent types yet, which will be done in future patches.
Updated all tests to add noexcept to the required functions, and added a dedicated test for this patch.
This patch might start to cause existing codebase fail to compile because most people may not have been strict in tagging
all the related functions noexcept.
Reviewers: lewissbaker, modocache, junparser
Reviewed By: modocache
Subscribers: arphaman, junparser, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D82029
2020-06-16 07:27:41 +08:00
|
|
|
/// Recursively check \p E and all its children to see if any call target
|
|
|
|
/// (including constructor call) is declared noexcept. Also any value returned
|
|
|
|
/// from the call has a noexcept destructor.
|
|
|
|
static void checkNoThrow(Sema &S, const Stmt *E,
|
|
|
|
llvm::SmallPtrSetImpl<const Decl *> &ThrowingDecls) {
|
|
|
|
auto checkDeclNoexcept = [&](const Decl *D, bool IsDtor = false) {
|
|
|
|
// In the case of dtor, the call to dtor is implicit and hence we should
|
|
|
|
// pass nullptr to canCalleeThrow.
|
|
|
|
if (Sema::canCalleeThrow(S, IsDtor ? nullptr : cast<Expr>(E), D)) {
|
2020-06-24 06:06:18 +08:00
|
|
|
if (const auto *FD = dyn_cast<FunctionDecl>(D)) {
|
|
|
|
// co_await promise.final_suspend() could end up calling
|
|
|
|
// __builtin_coro_resume for symmetric transfer if await_suspend()
|
|
|
|
// returns a handle. In that case, even __builtin_coro_resume is not
|
|
|
|
// declared as noexcept and may throw, it does not throw _into_ the
|
|
|
|
// coroutine that just suspended, but rather throws back out from
|
|
|
|
// whoever called coroutine_handle::resume(), hence we claim that
|
|
|
|
// logically it does not throw.
|
|
|
|
if (FD->getBuiltinID() == Builtin::BI__builtin_coro_resume)
|
|
|
|
return;
|
|
|
|
}
|
[Coroutines] Ensure co_await promise.final_suspend() does not throw
Summary:
This patch addresses https://bugs.llvm.org/show_bug.cgi?id=46256
The spec of coroutine requires that the expression co_await promise.final_suspend() shall not be potentially-throwing.
To check this, we recursively look at every call (including Call, MemberCall, OperatorCall and Constructor) in all code
generated by the final suspend, and ensure that the callees are declared with noexcept. We also look at any returned data
type that requires explicit destruction, and check their destructors for noexcept.
This patch does not check declarations with dependent types yet, which will be done in future patches.
Updated all tests to add noexcept to the required functions, and added a dedicated test for this patch.
This patch might start to cause existing codebase fail to compile because most people may not have been strict in tagging
all the related functions noexcept.
Reviewers: lewissbaker, modocache, junparser
Reviewed By: modocache
Subscribers: arphaman, junparser, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D82029
2020-06-16 07:27:41 +08:00
|
|
|
if (ThrowingDecls.empty()) {
|
2021-12-24 11:48:15 +08:00
|
|
|
// [dcl.fct.def.coroutine]p15
|
2022-01-12 14:10:07 +08:00
|
|
|
// The expression co_await promise.final_suspend() shall not be
|
2021-12-24 11:48:15 +08:00
|
|
|
// potentially-throwing ([except.spec]).
|
|
|
|
//
|
[Coroutines] Ensure co_await promise.final_suspend() does not throw
Summary:
This patch addresses https://bugs.llvm.org/show_bug.cgi?id=46256
The spec of coroutine requires that the expression co_await promise.final_suspend() shall not be potentially-throwing.
To check this, we recursively look at every call (including Call, MemberCall, OperatorCall and Constructor) in all code
generated by the final suspend, and ensure that the callees are declared with noexcept. We also look at any returned data
type that requires explicit destruction, and check their destructors for noexcept.
This patch does not check declarations with dependent types yet, which will be done in future patches.
Updated all tests to add noexcept to the required functions, and added a dedicated test for this patch.
This patch might start to cause existing codebase fail to compile because most people may not have been strict in tagging
all the related functions noexcept.
Reviewers: lewissbaker, modocache, junparser
Reviewed By: modocache
Subscribers: arphaman, junparser, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D82029
2020-06-16 07:27:41 +08:00
|
|
|
// First time seeing an error, emit the error message.
|
|
|
|
S.Diag(cast<FunctionDecl>(S.CurContext)->getLocation(),
|
|
|
|
diag::err_coroutine_promise_final_suspend_requires_nothrow);
|
|
|
|
}
|
|
|
|
ThrowingDecls.insert(D);
|
|
|
|
}
|
|
|
|
};
|
2022-01-14 15:37:01 +08:00
|
|
|
|
|
|
|
if (auto *CE = dyn_cast<CXXConstructExpr>(E)) {
|
|
|
|
CXXConstructorDecl *Ctor = CE->getConstructor();
|
[Coroutines] Ensure co_await promise.final_suspend() does not throw
Summary:
This patch addresses https://bugs.llvm.org/show_bug.cgi?id=46256
The spec of coroutine requires that the expression co_await promise.final_suspend() shall not be potentially-throwing.
To check this, we recursively look at every call (including Call, MemberCall, OperatorCall and Constructor) in all code
generated by the final suspend, and ensure that the callees are declared with noexcept. We also look at any returned data
type that requires explicit destruction, and check their destructors for noexcept.
This patch does not check declarations with dependent types yet, which will be done in future patches.
Updated all tests to add noexcept to the required functions, and added a dedicated test for this patch.
This patch might start to cause existing codebase fail to compile because most people may not have been strict in tagging
all the related functions noexcept.
Reviewers: lewissbaker, modocache, junparser
Reviewed By: modocache
Subscribers: arphaman, junparser, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D82029
2020-06-16 07:27:41 +08:00
|
|
|
checkDeclNoexcept(Ctor);
|
|
|
|
// Check the corresponding destructor of the constructor.
|
2022-01-14 15:37:01 +08:00
|
|
|
checkDeclNoexcept(Ctor->getParent()->getDestructor(), /*IsDtor=*/true);
|
|
|
|
} else if (auto *CE = dyn_cast<CallExpr>(E)) {
|
|
|
|
if (CE->isTypeDependent())
|
|
|
|
return;
|
|
|
|
|
|
|
|
checkDeclNoexcept(CE->getCalleeDecl());
|
|
|
|
QualType ReturnType = CE->getCallReturnType(S.getASTContext());
|
|
|
|
// Check the destructor of the call return type, if any.
|
|
|
|
if (ReturnType.isDestructedType() ==
|
|
|
|
QualType::DestructionKind::DK_cxx_destructor) {
|
|
|
|
const auto *T =
|
|
|
|
cast<RecordType>(ReturnType.getCanonicalType().getTypePtr());
|
2022-02-12 19:18:49 +08:00
|
|
|
checkDeclNoexcept(cast<CXXRecordDecl>(T->getDecl())->getDestructor(),
|
2022-01-14 15:37:01 +08:00
|
|
|
/*IsDtor=*/true);
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
for (const auto *Child : E->children()) {
|
|
|
|
if (!Child)
|
|
|
|
continue;
|
|
|
|
checkNoThrow(S, Child, ThrowingDecls);
|
[Coroutines] Ensure co_await promise.final_suspend() does not throw
Summary:
This patch addresses https://bugs.llvm.org/show_bug.cgi?id=46256
The spec of coroutine requires that the expression co_await promise.final_suspend() shall not be potentially-throwing.
To check this, we recursively look at every call (including Call, MemberCall, OperatorCall and Constructor) in all code
generated by the final suspend, and ensure that the callees are declared with noexcept. We also look at any returned data
type that requires explicit destruction, and check their destructors for noexcept.
This patch does not check declarations with dependent types yet, which will be done in future patches.
Updated all tests to add noexcept to the required functions, and added a dedicated test for this patch.
This patch might start to cause existing codebase fail to compile because most people may not have been strict in tagging
all the related functions noexcept.
Reviewers: lewissbaker, modocache, junparser
Reviewed By: modocache
Subscribers: arphaman, junparser, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D82029
2020-06-16 07:27:41 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-23 05:38:41 +08:00
|
|
|
bool Sema::checkFinalSuspendNoThrow(const Stmt *FinalSuspend) {
|
[Coroutines] Ensure co_await promise.final_suspend() does not throw
Summary:
This patch addresses https://bugs.llvm.org/show_bug.cgi?id=46256
The spec of coroutine requires that the expression co_await promise.final_suspend() shall not be potentially-throwing.
To check this, we recursively look at every call (including Call, MemberCall, OperatorCall and Constructor) in all code
generated by the final suspend, and ensure that the callees are declared with noexcept. We also look at any returned data
type that requires explicit destruction, and check their destructors for noexcept.
This patch does not check declarations with dependent types yet, which will be done in future patches.
Updated all tests to add noexcept to the required functions, and added a dedicated test for this patch.
This patch might start to cause existing codebase fail to compile because most people may not have been strict in tagging
all the related functions noexcept.
Reviewers: lewissbaker, modocache, junparser
Reviewed By: modocache
Subscribers: arphaman, junparser, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D82029
2020-06-16 07:27:41 +08:00
|
|
|
llvm::SmallPtrSet<const Decl *, 4> ThrowingDecls;
|
|
|
|
// We first collect all declarations that should not throw but not declared
|
|
|
|
// with noexcept. We then sort them based on the location before printing.
|
|
|
|
// This is to avoid emitting the same note multiple times on the same
|
|
|
|
// declaration, and also provide a deterministic order for the messages.
|
2020-06-23 05:38:41 +08:00
|
|
|
checkNoThrow(*this, FinalSuspend, ThrowingDecls);
|
[Coroutines] Ensure co_await promise.final_suspend() does not throw
Summary:
This patch addresses https://bugs.llvm.org/show_bug.cgi?id=46256
The spec of coroutine requires that the expression co_await promise.final_suspend() shall not be potentially-throwing.
To check this, we recursively look at every call (including Call, MemberCall, OperatorCall and Constructor) in all code
generated by the final suspend, and ensure that the callees are declared with noexcept. We also look at any returned data
type that requires explicit destruction, and check their destructors for noexcept.
This patch does not check declarations with dependent types yet, which will be done in future patches.
Updated all tests to add noexcept to the required functions, and added a dedicated test for this patch.
This patch might start to cause existing codebase fail to compile because most people may not have been strict in tagging
all the related functions noexcept.
Reviewers: lewissbaker, modocache, junparser
Reviewed By: modocache
Subscribers: arphaman, junparser, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D82029
2020-06-16 07:27:41 +08:00
|
|
|
auto SortedDecls = llvm::SmallVector<const Decl *, 4>{ThrowingDecls.begin(),
|
|
|
|
ThrowingDecls.end()};
|
|
|
|
sort(SortedDecls, [](const Decl *A, const Decl *B) {
|
|
|
|
return A->getEndLoc() < B->getEndLoc();
|
|
|
|
});
|
|
|
|
for (const auto *D : SortedDecls) {
|
2020-06-23 05:38:41 +08:00
|
|
|
Diag(D->getEndLoc(), diag::note_coroutine_function_declare_noexcept);
|
[Coroutines] Ensure co_await promise.final_suspend() does not throw
Summary:
This patch addresses https://bugs.llvm.org/show_bug.cgi?id=46256
The spec of coroutine requires that the expression co_await promise.final_suspend() shall not be potentially-throwing.
To check this, we recursively look at every call (including Call, MemberCall, OperatorCall and Constructor) in all code
generated by the final suspend, and ensure that the callees are declared with noexcept. We also look at any returned data
type that requires explicit destruction, and check their destructors for noexcept.
This patch does not check declarations with dependent types yet, which will be done in future patches.
Updated all tests to add noexcept to the required functions, and added a dedicated test for this patch.
This patch might start to cause existing codebase fail to compile because most people may not have been strict in tagging
all the related functions noexcept.
Reviewers: lewissbaker, modocache, junparser
Reviewed By: modocache
Subscribers: arphaman, junparser, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D82029
2020-06-16 07:27:41 +08:00
|
|
|
}
|
|
|
|
return ThrowingDecls.empty();
|
|
|
|
}
|
|
|
|
|
2017-06-14 11:24:55 +08:00
|
|
|
bool Sema::ActOnCoroutineBodyStart(Scope *SC, SourceLocation KWLoc,
|
|
|
|
StringRef Keyword) {
|
|
|
|
if (!checkCoroutineContext(*this, KWLoc, Keyword))
|
2017-03-07 07:38:15 +08:00
|
|
|
return false;
|
2017-06-14 11:24:55 +08:00
|
|
|
auto *ScopeInfo = getCurFunction();
|
2017-03-07 07:38:15 +08:00
|
|
|
assert(ScopeInfo->CoroutinePromise);
|
|
|
|
|
|
|
|
// If we have existing coroutine statements then we have already built
|
|
|
|
// the initial and final suspend points.
|
|
|
|
if (!ScopeInfo->NeedsCoroutineSuspends)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
ScopeInfo->setNeedsCoroutineSuspends(false);
|
|
|
|
|
2017-06-14 11:24:55 +08:00
|
|
|
auto *Fn = cast<FunctionDecl>(CurContext);
|
2017-03-07 07:38:15 +08:00
|
|
|
SourceLocation Loc = Fn->getLocation();
|
|
|
|
// Build the initial suspend point
|
|
|
|
auto buildSuspends = [&](StringRef Name) mutable -> StmtResult {
|
|
|
|
ExprResult Suspend =
|
2017-06-14 11:24:55 +08:00
|
|
|
buildPromiseCall(*this, ScopeInfo->CoroutinePromise, Loc, Name, None);
|
2017-03-07 07:38:15 +08:00
|
|
|
if (Suspend.isInvalid())
|
|
|
|
return StmtError();
|
2017-06-14 11:24:55 +08:00
|
|
|
Suspend = buildOperatorCoawaitCall(*this, SC, Loc, Suspend.get());
|
2017-03-07 07:38:15 +08:00
|
|
|
if (Suspend.isInvalid())
|
|
|
|
return StmtError();
|
2017-06-14 11:24:55 +08:00
|
|
|
Suspend = BuildResolvedCoawaitExpr(Loc, Suspend.get(),
|
|
|
|
/*IsImplicit*/ true);
|
2019-01-05 00:58:14 +08:00
|
|
|
Suspend = ActOnFinishFullExpr(Suspend.get(), /*DiscardedValue*/ false);
|
2017-03-07 07:38:15 +08:00
|
|
|
if (Suspend.isInvalid()) {
|
2017-06-14 11:24:55 +08:00
|
|
|
Diag(Loc, diag::note_coroutine_promise_suspend_implicitly_required)
|
2017-03-07 07:38:15 +08:00
|
|
|
<< ((Name == "initial_suspend") ? 0 : 1);
|
2017-06-14 11:24:55 +08:00
|
|
|
Diag(KWLoc, diag::note_declared_coroutine_here) << Keyword;
|
2017-03-07 07:38:15 +08:00
|
|
|
return StmtError();
|
|
|
|
}
|
|
|
|
return cast<Stmt>(Suspend.get());
|
|
|
|
};
|
|
|
|
|
|
|
|
StmtResult InitSuspend = buildSuspends("initial_suspend");
|
|
|
|
if (InitSuspend.isInvalid())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
StmtResult FinalSuspend = buildSuspends("final_suspend");
|
2020-06-23 05:38:41 +08:00
|
|
|
if (FinalSuspend.isInvalid() || !checkFinalSuspendNoThrow(FinalSuspend.get()))
|
2017-03-07 07:38:15 +08:00
|
|
|
return true;
|
|
|
|
|
|
|
|
ScopeInfo->setCoroutineSuspends(InitSuspend.get(), FinalSuspend.get());
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
Un-revert "[coroutines][PR40978] Emit error for co_yield within catch block"
Summary:
https://reviews.llvm.org/D59076 added a new coroutine error that
prevented users from using 'co_await' or 'co_yield' within a exception
handler. However, it was reverted in https://reviews.llvm.org/rC356774
because it caused a regression in nested scopes in C++ catch statements,
as documented by https://bugs.llvm.org/show_bug.cgi?id=41171.
The issue was due to an incorrect use of a `clang::ParseScope`. To fix:
1. Add a regression test for catch statement parsing that mimics the bug
report from https://bugs.llvm.org/show_bug.cgi?id=41171.
2. Re-apply the coroutines error patch from
https://reviews.llvm.org/D59076, but this time with the correct
ParseScope behavior.
Reviewers: GorNishanov, tks2103, rsmith, riccibruno, jbulow
Reviewed By: riccibruno
Subscribers: EricWF, jdoerfert, lewissbaker, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D59752
llvm-svn: 356865
2019-03-25 08:53:10 +08:00
|
|
|
// Recursively walks up the scope hierarchy until either a 'catch' or a function
|
|
|
|
// scope is found, whichever comes first.
|
|
|
|
static bool isWithinCatchScope(Scope *S) {
|
|
|
|
// 'co_await' and 'co_yield' keywords are disallowed within catch blocks, but
|
|
|
|
// lambdas that use 'co_await' are allowed. The loop below ends when a
|
|
|
|
// function scope is found in order to ensure the following behavior:
|
|
|
|
//
|
|
|
|
// void foo() { // <- function scope
|
|
|
|
// try { //
|
|
|
|
// co_await x; // <- 'co_await' is OK within a function scope
|
|
|
|
// } catch { // <- catch scope
|
|
|
|
// co_await x; // <- 'co_await' is not OK within a catch scope
|
|
|
|
// []() { // <- function scope
|
|
|
|
// co_await x; // <- 'co_await' is OK within a function scope
|
|
|
|
// }();
|
|
|
|
// }
|
|
|
|
// }
|
2022-04-16 11:27:18 +08:00
|
|
|
while (S && !S->isFunctionScope()) {
|
|
|
|
if (S->isCatchScope())
|
Un-revert "[coroutines][PR40978] Emit error for co_yield within catch block"
Summary:
https://reviews.llvm.org/D59076 added a new coroutine error that
prevented users from using 'co_await' or 'co_yield' within a exception
handler. However, it was reverted in https://reviews.llvm.org/rC356774
because it caused a regression in nested scopes in C++ catch statements,
as documented by https://bugs.llvm.org/show_bug.cgi?id=41171.
The issue was due to an incorrect use of a `clang::ParseScope`. To fix:
1. Add a regression test for catch statement parsing that mimics the bug
report from https://bugs.llvm.org/show_bug.cgi?id=41171.
2. Re-apply the coroutines error patch from
https://reviews.llvm.org/D59076, but this time with the correct
ParseScope behavior.
Reviewers: GorNishanov, tks2103, rsmith, riccibruno, jbulow
Reviewed By: riccibruno
Subscribers: EricWF, jdoerfert, lewissbaker, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D59752
llvm-svn: 356865
2019-03-25 08:53:10 +08:00
|
|
|
return true;
|
|
|
|
S = S->getParent();
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// [expr.await]p2, emphasis added: "An await-expression shall appear only in
|
|
|
|
// a *potentially evaluated* expression within the compound-statement of a
|
|
|
|
// function-body *outside of a handler* [...] A context within a function
|
|
|
|
// where an await-expression can appear is called a suspension context of the
|
|
|
|
// function."
|
|
|
|
static void checkSuspensionContext(Sema &S, SourceLocation Loc,
|
|
|
|
StringRef Keyword) {
|
|
|
|
// First emphasis of [expr.await]p2: must be a potentially evaluated context.
|
|
|
|
// That is, 'co_await' and 'co_yield' cannot appear in subexpressions of
|
|
|
|
// \c sizeof.
|
|
|
|
if (S.isUnevaluatedContext())
|
|
|
|
S.Diag(Loc, diag::err_coroutine_unevaluated_context) << Keyword;
|
|
|
|
|
|
|
|
// Second emphasis of [expr.await]p2: must be outside of an exception handler.
|
|
|
|
if (isWithinCatchScope(S.getCurScope()))
|
|
|
|
S.Diag(Loc, diag::err_coroutine_within_handler) << Keyword;
|
|
|
|
}
|
|
|
|
|
2015-10-27 14:02:45 +08:00
|
|
|
ExprResult Sema::ActOnCoawaitExpr(Scope *S, SourceLocation Loc, Expr *E) {
|
2017-06-14 11:24:55 +08:00
|
|
|
if (!ActOnCoroutineBodyStart(S, Loc, "co_await")) {
|
2016-09-30 05:47:39 +08:00
|
|
|
CorrectDelayedTyposInExpr(E);
|
|
|
|
return ExprError();
|
|
|
|
}
|
2017-03-07 07:38:15 +08:00
|
|
|
|
Un-revert "[coroutines][PR40978] Emit error for co_yield within catch block"
Summary:
https://reviews.llvm.org/D59076 added a new coroutine error that
prevented users from using 'co_await' or 'co_yield' within a exception
handler. However, it was reverted in https://reviews.llvm.org/rC356774
because it caused a regression in nested scopes in C++ catch statements,
as documented by https://bugs.llvm.org/show_bug.cgi?id=41171.
The issue was due to an incorrect use of a `clang::ParseScope`. To fix:
1. Add a regression test for catch statement parsing that mimics the bug
report from https://bugs.llvm.org/show_bug.cgi?id=41171.
2. Re-apply the coroutines error patch from
https://reviews.llvm.org/D59076, but this time with the correct
ParseScope behavior.
Reviewers: GorNishanov, tks2103, rsmith, riccibruno, jbulow
Reviewed By: riccibruno
Subscribers: EricWF, jdoerfert, lewissbaker, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D59752
llvm-svn: 356865
2019-03-25 08:53:10 +08:00
|
|
|
checkSuspensionContext(*this, Loc, "co_await");
|
|
|
|
|
2022-01-29 08:50:48 +08:00
|
|
|
if (E->hasPlaceholderType()) {
|
2015-11-21 06:57:24 +08:00
|
|
|
ExprResult R = CheckPlaceholderExpr(E);
|
|
|
|
if (R.isInvalid()) return ExprError();
|
|
|
|
E = R.get();
|
|
|
|
}
|
2017-03-07 07:38:15 +08:00
|
|
|
ExprResult Lookup = buildOperatorCoawaitLookupExpr(*this, S, Loc);
|
|
|
|
if (Lookup.isInvalid())
|
|
|
|
return ExprError();
|
|
|
|
return BuildUnresolvedCoawaitExpr(Loc, E,
|
|
|
|
cast<UnresolvedLookupExpr>(Lookup.get()));
|
|
|
|
}
|
|
|
|
|
|
|
|
ExprResult Sema::BuildUnresolvedCoawaitExpr(SourceLocation Loc, Expr *E,
|
2017-03-11 10:35:37 +08:00
|
|
|
UnresolvedLookupExpr *Lookup) {
|
2017-03-07 07:38:15 +08:00
|
|
|
auto *FSI = checkCoroutineContext(*this, Loc, "co_await");
|
|
|
|
if (!FSI)
|
|
|
|
return ExprError();
|
2015-11-21 06:57:24 +08:00
|
|
|
|
2022-01-29 08:50:48 +08:00
|
|
|
if (E->hasPlaceholderType()) {
|
2017-03-07 07:38:15 +08:00
|
|
|
ExprResult R = CheckPlaceholderExpr(E);
|
|
|
|
if (R.isInvalid())
|
|
|
|
return ExprError();
|
|
|
|
E = R.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
auto *Promise = FSI->CoroutinePromise;
|
|
|
|
if (Promise->getType()->isDependentType()) {
|
|
|
|
Expr *Res =
|
|
|
|
new (Context) DependentCoawaitExpr(Loc, Context.DependentTy, E, Lookup);
|
|
|
|
return Res;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto *RD = Promise->getType()->getAsCXXRecordDecl();
|
|
|
|
if (lookupMember(*this, "await_transform", RD, Loc)) {
|
|
|
|
ExprResult R = buildPromiseCall(*this, Promise, Loc, "await_transform", E);
|
|
|
|
if (R.isInvalid()) {
|
|
|
|
Diag(Loc,
|
|
|
|
diag::note_coroutine_promise_implicit_await_transform_required_here)
|
|
|
|
<< E->getSourceRange();
|
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
E = R.get();
|
|
|
|
}
|
|
|
|
ExprResult Awaitable = buildOperatorCoawaitCall(*this, Loc, E, Lookup);
|
2015-10-27 14:02:45 +08:00
|
|
|
if (Awaitable.isInvalid())
|
|
|
|
return ExprError();
|
2016-09-30 05:47:39 +08:00
|
|
|
|
2017-03-07 07:38:15 +08:00
|
|
|
return BuildResolvedCoawaitExpr(Loc, Awaitable.get());
|
2015-10-27 14:02:45 +08:00
|
|
|
}
|
2017-03-07 07:38:15 +08:00
|
|
|
|
|
|
|
ExprResult Sema::BuildResolvedCoawaitExpr(SourceLocation Loc, Expr *E,
|
|
|
|
bool IsImplicit) {
|
2017-03-11 10:35:37 +08:00
|
|
|
auto *Coroutine = checkCoroutineContext(*this, Loc, "co_await", IsImplicit);
|
2015-11-20 10:54:01 +08:00
|
|
|
if (!Coroutine)
|
|
|
|
return ExprError();
|
2015-10-27 14:02:45 +08:00
|
|
|
|
2022-01-29 08:50:48 +08:00
|
|
|
if (E->hasPlaceholderType()) {
|
2015-10-27 14:02:45 +08:00
|
|
|
ExprResult R = CheckPlaceholderExpr(E);
|
|
|
|
if (R.isInvalid()) return ExprError();
|
|
|
|
E = R.get();
|
|
|
|
}
|
|
|
|
|
2015-11-21 06:57:24 +08:00
|
|
|
if (E->getType()->isDependentType()) {
|
2017-03-07 07:38:15 +08:00
|
|
|
Expr *Res = new (Context)
|
|
|
|
CoawaitExpr(Loc, Context.DependentTy, E, IsImplicit);
|
2015-11-21 06:57:24 +08:00
|
|
|
return Res;
|
|
|
|
}
|
|
|
|
|
2015-11-22 11:13:02 +08:00
|
|
|
// If the expression is a temporary, materialize it as an lvalue so that we
|
|
|
|
// can use it multiple times.
|
2021-04-19 04:24:39 +08:00
|
|
|
if (E->isPRValue())
|
2016-06-22 04:29:17 +08:00
|
|
|
E = CreateMaterializeTemporaryExpr(E->getType(), E, true);
|
2015-10-27 14:02:45 +08:00
|
|
|
|
2018-03-27 11:15:46 +08:00
|
|
|
// The location of the `co_await` token cannot be used when constructing
|
|
|
|
// the member call expressions since it's before the location of `Expr`, which
|
|
|
|
// is used as the start of the member call expression.
|
|
|
|
SourceLocation CallLoc = E->getExprLoc();
|
|
|
|
|
2015-10-27 14:02:45 +08:00
|
|
|
// Build the await_ready, await_suspend, await_resume calls.
|
2020-10-13 03:00:20 +08:00
|
|
|
ReadySuspendResumeResult RSS = buildCoawaitCalls(
|
2020-11-12 13:17:16 +08:00
|
|
|
*this, Coroutine->CoroutinePromise, CallLoc, E);
|
2015-10-27 14:02:45 +08:00
|
|
|
if (RSS.IsInvalid)
|
|
|
|
return ExprError();
|
|
|
|
|
2017-03-11 09:30:17 +08:00
|
|
|
Expr *Res =
|
|
|
|
new (Context) CoawaitExpr(Loc, E, RSS.Results[0], RSS.Results[1],
|
|
|
|
RSS.Results[2], RSS.OpaqueValue, IsImplicit);
|
2017-03-11 10:35:37 +08:00
|
|
|
|
2015-10-22 14:13:50 +08:00
|
|
|
return Res;
|
|
|
|
}
|
|
|
|
|
2015-10-27 14:02:45 +08:00
|
|
|
ExprResult Sema::ActOnCoyieldExpr(Scope *S, SourceLocation Loc, Expr *E) {
|
2017-06-14 11:24:55 +08:00
|
|
|
if (!ActOnCoroutineBodyStart(S, Loc, "co_yield")) {
|
2016-09-30 05:47:39 +08:00
|
|
|
CorrectDelayedTyposInExpr(E);
|
2015-11-21 06:40:06 +08:00
|
|
|
return ExprError();
|
2016-09-30 05:47:39 +08:00
|
|
|
}
|
2015-11-21 06:40:06 +08:00
|
|
|
|
Un-revert "[coroutines][PR40978] Emit error for co_yield within catch block"
Summary:
https://reviews.llvm.org/D59076 added a new coroutine error that
prevented users from using 'co_await' or 'co_yield' within a exception
handler. However, it was reverted in https://reviews.llvm.org/rC356774
because it caused a regression in nested scopes in C++ catch statements,
as documented by https://bugs.llvm.org/show_bug.cgi?id=41171.
The issue was due to an incorrect use of a `clang::ParseScope`. To fix:
1. Add a regression test for catch statement parsing that mimics the bug
report from https://bugs.llvm.org/show_bug.cgi?id=41171.
2. Re-apply the coroutines error patch from
https://reviews.llvm.org/D59076, but this time with the correct
ParseScope behavior.
Reviewers: GorNishanov, tks2103, rsmith, riccibruno, jbulow
Reviewed By: riccibruno
Subscribers: EricWF, jdoerfert, lewissbaker, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D59752
llvm-svn: 356865
2019-03-25 08:53:10 +08:00
|
|
|
checkSuspensionContext(*this, Loc, "co_yield");
|
|
|
|
|
2015-11-21 06:40:06 +08:00
|
|
|
// Build yield_value call.
|
2017-03-07 07:38:15 +08:00
|
|
|
ExprResult Awaitable = buildPromiseCall(
|
|
|
|
*this, getCurFunction()->CoroutinePromise, Loc, "yield_value", E);
|
2015-10-27 14:02:45 +08:00
|
|
|
if (Awaitable.isInvalid())
|
|
|
|
return ExprError();
|
2015-11-21 06:40:06 +08:00
|
|
|
|
|
|
|
// Build 'operator co_await' call.
|
|
|
|
Awaitable = buildOperatorCoawaitCall(*this, S, Loc, Awaitable.get());
|
|
|
|
if (Awaitable.isInvalid())
|
|
|
|
return ExprError();
|
|
|
|
|
2015-10-27 14:02:45 +08:00
|
|
|
return BuildCoyieldExpr(Loc, Awaitable.get());
|
|
|
|
}
|
|
|
|
ExprResult Sema::BuildCoyieldExpr(SourceLocation Loc, Expr *E) {
|
|
|
|
auto *Coroutine = checkCoroutineContext(*this, Loc, "co_yield");
|
2015-11-20 10:54:01 +08:00
|
|
|
if (!Coroutine)
|
|
|
|
return ExprError();
|
2015-10-22 14:13:50 +08:00
|
|
|
|
2022-01-29 08:50:48 +08:00
|
|
|
if (E->hasPlaceholderType()) {
|
2015-11-21 06:57:24 +08:00
|
|
|
ExprResult R = CheckPlaceholderExpr(E);
|
|
|
|
if (R.isInvalid()) return ExprError();
|
|
|
|
E = R.get();
|
|
|
|
}
|
|
|
|
|
2015-11-22 10:57:17 +08:00
|
|
|
if (E->getType()->isDependentType()) {
|
|
|
|
Expr *Res = new (Context) CoyieldExpr(Loc, Context.DependentTy, E);
|
|
|
|
return Res;
|
|
|
|
}
|
|
|
|
|
2015-11-22 11:13:02 +08:00
|
|
|
// If the expression is a temporary, materialize it as an lvalue so that we
|
|
|
|
// can use it multiple times.
|
2021-04-19 04:24:39 +08:00
|
|
|
if (E->isPRValue())
|
2016-06-22 04:29:17 +08:00
|
|
|
E = CreateMaterializeTemporaryExpr(E->getType(), E, true);
|
2015-11-22 10:57:17 +08:00
|
|
|
|
|
|
|
// Build the await_ready, await_suspend, await_resume calls.
|
2020-10-13 03:00:20 +08:00
|
|
|
ReadySuspendResumeResult RSS = buildCoawaitCalls(
|
2020-11-12 13:17:16 +08:00
|
|
|
*this, Coroutine->CoroutinePromise, Loc, E);
|
2015-11-22 10:57:17 +08:00
|
|
|
if (RSS.IsInvalid)
|
|
|
|
return ExprError();
|
|
|
|
|
2017-06-14 11:24:55 +08:00
|
|
|
Expr *Res =
|
|
|
|
new (Context) CoyieldExpr(Loc, E, RSS.Results[0], RSS.Results[1],
|
|
|
|
RSS.Results[2], RSS.OpaqueValue);
|
2017-03-11 10:35:37 +08:00
|
|
|
|
2015-10-22 14:13:50 +08:00
|
|
|
return Res;
|
|
|
|
}
|
|
|
|
|
2017-03-07 07:38:15 +08:00
|
|
|
StmtResult Sema::ActOnCoreturnStmt(Scope *S, SourceLocation Loc, Expr *E) {
|
2017-06-14 11:24:55 +08:00
|
|
|
if (!ActOnCoroutineBodyStart(S, Loc, "co_return")) {
|
2016-09-30 05:47:39 +08:00
|
|
|
CorrectDelayedTyposInExpr(E);
|
|
|
|
return StmtError();
|
|
|
|
}
|
2015-10-27 14:02:45 +08:00
|
|
|
return BuildCoreturnStmt(Loc, E);
|
|
|
|
}
|
2016-10-04 08:31:16 +08:00
|
|
|
|
2017-03-07 07:38:15 +08:00
|
|
|
StmtResult Sema::BuildCoreturnStmt(SourceLocation Loc, Expr *E,
|
|
|
|
bool IsImplicit) {
|
2017-03-11 10:35:37 +08:00
|
|
|
auto *FSI = checkCoroutineContext(*this, Loc, "co_return", IsImplicit);
|
2017-03-07 07:38:15 +08:00
|
|
|
if (!FSI)
|
2015-11-22 15:33:28 +08:00
|
|
|
return StmtError();
|
|
|
|
|
2022-01-29 08:50:48 +08:00
|
|
|
if (E && E->hasPlaceholderType() &&
|
|
|
|
!E->hasPlaceholderType(BuiltinType::Overload)) {
|
2015-11-21 06:57:24 +08:00
|
|
|
ExprResult R = CheckPlaceholderExpr(E);
|
|
|
|
if (R.isInvalid()) return StmtError();
|
|
|
|
E = R.get();
|
|
|
|
}
|
|
|
|
|
2017-03-07 07:38:15 +08:00
|
|
|
VarDecl *Promise = FSI->CoroutinePromise;
|
2015-11-22 15:05:16 +08:00
|
|
|
ExprResult PC;
|
2016-10-07 05:23:38 +08:00
|
|
|
if (E && (isa<InitListExpr>(E) || !E->getType()->isVoidType())) {
|
2021-07-14 07:55:34 +08:00
|
|
|
getNamedReturnInfo(E, SimplerImplicitMoveMode::ForceOn);
|
2017-03-07 07:38:15 +08:00
|
|
|
PC = buildPromiseCall(*this, Promise, Loc, "return_value", E);
|
2015-11-22 15:05:16 +08:00
|
|
|
} else {
|
|
|
|
E = MakeFullDiscardedValueExpr(E).get();
|
2017-03-07 07:38:15 +08:00
|
|
|
PC = buildPromiseCall(*this, Promise, Loc, "return_void", None);
|
2015-11-22 15:05:16 +08:00
|
|
|
}
|
|
|
|
if (PC.isInvalid())
|
|
|
|
return StmtError();
|
|
|
|
|
2019-01-05 00:58:14 +08:00
|
|
|
Expr *PCE = ActOnFinishFullExpr(PC.get(), /*DiscardedValue*/ false).get();
|
2015-11-22 15:05:16 +08:00
|
|
|
|
2017-03-07 07:38:15 +08:00
|
|
|
Stmt *Res = new (Context) CoreturnStmt(Loc, E, PCE, IsImplicit);
|
2015-10-22 14:13:50 +08:00
|
|
|
return Res;
|
|
|
|
}
|
|
|
|
|
2017-04-18 11:12:48 +08:00
|
|
|
/// Look up the std::nothrow object.
|
|
|
|
static Expr *buildStdNoThrowDeclRef(Sema &S, SourceLocation Loc) {
|
|
|
|
NamespaceDecl *Std = S.getStdNamespace();
|
|
|
|
assert(Std && "Should already be diagnosed");
|
|
|
|
|
|
|
|
LookupResult Result(S, &S.PP.getIdentifierTable().get("nothrow"), Loc,
|
|
|
|
Sema::LookupOrdinaryName);
|
|
|
|
if (!S.LookupQualifiedName(Result, Std)) {
|
2021-12-24 11:48:15 +08:00
|
|
|
// <coroutine> is not requred to include <new>, so we couldn't omit
|
|
|
|
// the check here.
|
2017-04-18 11:12:48 +08:00
|
|
|
S.Diag(Loc, diag::err_implicit_coroutine_std_nothrow_type_not_found);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto *VD = Result.getAsSingle<VarDecl>();
|
|
|
|
if (!VD) {
|
|
|
|
Result.suppressDiagnostics();
|
|
|
|
// We found something weird. Complain about the first thing we found.
|
|
|
|
NamedDecl *Found = *Result.begin();
|
|
|
|
S.Diag(Found->getLocation(), diag::err_malformed_std_nothrow);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
ExprResult DR = S.BuildDeclRefExpr(VD, VD->getType(), VK_LValue, Loc);
|
|
|
|
if (DR.isInvalid())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
return DR.get();
|
|
|
|
}
|
|
|
|
|
2016-10-28 00:28:31 +08:00
|
|
|
// Find an appropriate delete for the promise.
|
|
|
|
static FunctionDecl *findDeleteForPromise(Sema &S, SourceLocation Loc,
|
|
|
|
QualType PromiseType) {
|
|
|
|
FunctionDecl *OperatorDelete = nullptr;
|
|
|
|
|
|
|
|
DeclarationName DeleteName =
|
|
|
|
S.Context.DeclarationNames.getCXXOperatorName(OO_Delete);
|
|
|
|
|
|
|
|
auto *PointeeRD = PromiseType->getAsCXXRecordDecl();
|
|
|
|
assert(PointeeRD && "PromiseType must be a CxxRecordDecl type");
|
|
|
|
|
2021-12-24 11:48:15 +08:00
|
|
|
// [dcl.fct.def.coroutine]p12
|
|
|
|
// The deallocation function's name is looked up by searching for it in the
|
|
|
|
// scope of the promise type. If nothing is found, a search is performed in
|
|
|
|
// the global scope.
|
2016-10-28 00:28:31 +08:00
|
|
|
if (S.FindDeallocationFunction(Loc, PointeeRD, DeleteName, OperatorDelete))
|
|
|
|
return nullptr;
|
|
|
|
|
2021-12-24 11:48:15 +08:00
|
|
|
// FIXME: We didn't implement following selection:
|
|
|
|
// [dcl.fct.def.coroutine]p12
|
|
|
|
// If both a usual deallocation function with only a pointer parameter and a
|
|
|
|
// usual deallocation function with both a pointer parameter and a size
|
|
|
|
// parameter are found, then the selected deallocation function shall be the
|
|
|
|
// one with two parameters. Otherwise, the selected deallocation function
|
|
|
|
// shall be the function with one parameter.
|
|
|
|
|
2016-10-28 00:28:31 +08:00
|
|
|
if (!OperatorDelete) {
|
|
|
|
// Look for a global declaration.
|
|
|
|
const bool CanProvideSize = S.isCompleteType(Loc, PromiseType);
|
|
|
|
const bool Overaligned = false;
|
|
|
|
OperatorDelete = S.FindUsualDeallocationFunction(Loc, CanProvideSize,
|
|
|
|
Overaligned, DeleteName);
|
|
|
|
}
|
|
|
|
S.MarkFunctionReferenced(Loc, OperatorDelete);
|
|
|
|
return OperatorDelete;
|
|
|
|
}
|
|
|
|
|
2017-02-13 13:05:02 +08:00
|
|
|
|
2015-11-24 10:34:39 +08:00
|
|
|
void Sema::CheckCompletedCoroutineBody(FunctionDecl *FD, Stmt *&Body) {
|
2015-10-22 14:13:50 +08:00
|
|
|
FunctionScopeInfo *Fn = getCurFunction();
|
2017-05-25 10:16:53 +08:00
|
|
|
assert(Fn && Fn->isCoroutine() && "not a coroutine");
|
2017-03-09 11:09:43 +08:00
|
|
|
if (!Body) {
|
|
|
|
assert(FD->isInvalidDecl() &&
|
|
|
|
"a null body is only allowed for invalid declarations");
|
|
|
|
return;
|
|
|
|
}
|
2017-05-25 10:16:53 +08:00
|
|
|
// We have a function that uses coroutine keywords, but we failed to build
|
|
|
|
// the promise type.
|
|
|
|
if (!Fn->CoroutinePromise)
|
|
|
|
return FD->setInvalidDecl();
|
2017-03-09 11:09:43 +08:00
|
|
|
|
|
|
|
if (isa<CoroutineBodyStmt>(Body)) {
|
2017-05-24 22:34:19 +08:00
|
|
|
// Nothing todo. the body is already a transformed coroutine body statement.
|
2017-03-09 11:09:43 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-02-08 11:42:38 +08:00
|
|
|
// The always_inline attribute doesn't reliably apply to a coroutine,
|
|
|
|
// because the coroutine will be split into pieces and some pieces
|
|
|
|
// might be called indirectly, as in a virtual call. Even the ramp
|
|
|
|
// function cannot be inlined at -O0, due to pipeline ordering
|
|
|
|
// problems (see https://llvm.org/PR53413). Tell the user about it.
|
|
|
|
if (FD->hasAttr<AlwaysInlineAttr>())
|
|
|
|
Diag(FD->getLocation(), diag::warn_always_inline_coroutine);
|
|
|
|
|
2021-12-24 11:48:15 +08:00
|
|
|
// [stmt.return.coroutine]p1:
|
|
|
|
// A coroutine shall not enclose a return statement ([stmt.return]).
|
2015-10-27 14:02:45 +08:00
|
|
|
if (Fn->FirstReturnLoc.isValid()) {
|
2017-03-11 10:35:37 +08:00
|
|
|
assert(Fn->FirstCoroutineStmtLoc.isValid() &&
|
|
|
|
"first coroutine location not set");
|
2015-10-27 14:02:45 +08:00
|
|
|
Diag(Fn->FirstReturnLoc, diag::err_return_in_coroutine);
|
2017-03-11 10:35:37 +08:00
|
|
|
Diag(Fn->FirstCoroutineStmtLoc, diag::note_declared_coroutine_here)
|
|
|
|
<< Fn->getFirstCoroutineStmtKeyword();
|
2015-10-22 14:13:50 +08:00
|
|
|
}
|
2017-04-04 03:21:00 +08:00
|
|
|
CoroutineStmtBuilder Builder(*this, *FD, *Fn, Body);
|
|
|
|
if (Builder.isInvalid() || !Builder.buildStatements())
|
2017-02-13 13:05:02 +08:00
|
|
|
return FD->setInvalidDecl();
|
2015-10-22 14:13:50 +08:00
|
|
|
|
2017-02-13 13:05:02 +08:00
|
|
|
// Build body for the coroutine wrapper statement.
|
|
|
|
Body = CoroutineBodyStmt::Create(Context, Builder);
|
|
|
|
}
|
2015-11-24 10:34:39 +08:00
|
|
|
|
2017-04-04 03:21:00 +08:00
|
|
|
CoroutineStmtBuilder::CoroutineStmtBuilder(Sema &S, FunctionDecl &FD,
|
|
|
|
sema::FunctionScopeInfo &Fn,
|
|
|
|
Stmt *Body)
|
|
|
|
: S(S), FD(FD), Fn(Fn), Loc(FD.getLocation()),
|
|
|
|
IsPromiseDependentType(
|
|
|
|
!Fn.CoroutinePromise ||
|
|
|
|
Fn.CoroutinePromise->getType()->isDependentType()) {
|
|
|
|
this->Body = Body;
|
2018-01-25 06:15:42 +08:00
|
|
|
|
|
|
|
for (auto KV : Fn.CoroutineParameterMoves)
|
|
|
|
this->ParamMovesVector.push_back(KV.second);
|
|
|
|
this->ParamMoves = this->ParamMovesVector;
|
|
|
|
|
2017-04-04 03:21:00 +08:00
|
|
|
if (!IsPromiseDependentType) {
|
|
|
|
PromiseRecordDecl = Fn.CoroutinePromise->getType()->getAsCXXRecordDecl();
|
|
|
|
assert(PromiseRecordDecl && "Type should have already been checked");
|
|
|
|
}
|
|
|
|
this->IsValid = makePromiseStmt() && makeInitialAndFinalSuspend();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CoroutineStmtBuilder::buildStatements() {
|
|
|
|
assert(this->IsValid && "coroutine already invalid");
|
2018-01-25 06:15:42 +08:00
|
|
|
this->IsValid = makeReturnObject();
|
2017-04-04 03:21:00 +08:00
|
|
|
if (this->IsValid && !IsPromiseDependentType)
|
|
|
|
buildDependentStatements();
|
|
|
|
return this->IsValid;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CoroutineStmtBuilder::buildDependentStatements() {
|
|
|
|
assert(this->IsValid && "coroutine already invalid");
|
|
|
|
assert(!this->IsPromiseDependentType &&
|
|
|
|
"coroutine cannot have a dependent promise type");
|
|
|
|
this->IsValid = makeOnException() && makeOnFallthrough() &&
|
2017-05-23 04:22:23 +08:00
|
|
|
makeGroDeclAndReturnStmt() && makeReturnOnAllocFailure() &&
|
|
|
|
makeNewAndDeleteExpr();
|
2017-04-04 03:21:00 +08:00
|
|
|
return this->IsValid;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CoroutineStmtBuilder::makePromiseStmt() {
|
2015-11-24 10:34:39 +08:00
|
|
|
// Form a declaration statement for the promise declaration, so that AST
|
|
|
|
// visitors can more easily find it.
|
|
|
|
StmtResult PromiseStmt =
|
2017-02-13 13:05:02 +08:00
|
|
|
S.ActOnDeclStmt(S.ConvertDeclToDeclGroup(Fn.CoroutinePromise), Loc, Loc);
|
2015-11-24 10:34:39 +08:00
|
|
|
if (PromiseStmt.isInvalid())
|
2017-02-13 13:05:02 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
this->Promise = PromiseStmt.get();
|
|
|
|
return true;
|
|
|
|
}
|
2015-11-24 10:34:39 +08:00
|
|
|
|
2017-04-04 03:21:00 +08:00
|
|
|
bool CoroutineStmtBuilder::makeInitialAndFinalSuspend() {
|
2017-03-07 07:38:15 +08:00
|
|
|
if (Fn.hasInvalidCoroutineSuspends())
|
2017-02-13 13:05:02 +08:00
|
|
|
return false;
|
2017-03-07 07:38:15 +08:00
|
|
|
this->InitialSuspend = cast<Expr>(Fn.CoroutineSuspends.first);
|
|
|
|
this->FinalSuspend = cast<Expr>(Fn.CoroutineSuspends.second);
|
2017-02-13 13:05:02 +08:00
|
|
|
return true;
|
|
|
|
}
|
2015-11-24 10:34:39 +08:00
|
|
|
|
2017-03-28 07:36:59 +08:00
|
|
|
static bool diagReturnOnAllocFailure(Sema &S, Expr *E,
|
|
|
|
CXXRecordDecl *PromiseRecordDecl,
|
|
|
|
FunctionScopeInfo &Fn) {
|
|
|
|
auto Loc = E->getExprLoc();
|
|
|
|
if (auto *DeclRef = dyn_cast_or_null<DeclRefExpr>(E)) {
|
|
|
|
auto *Decl = DeclRef->getDecl();
|
|
|
|
if (CXXMethodDecl *Method = dyn_cast_or_null<CXXMethodDecl>(Decl)) {
|
|
|
|
if (Method->isStatic())
|
|
|
|
return true;
|
|
|
|
else
|
|
|
|
Loc = Decl->getLocation();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
S.Diag(
|
|
|
|
Loc,
|
|
|
|
diag::err_coroutine_promise_get_return_object_on_allocation_failure)
|
|
|
|
<< PromiseRecordDecl;
|
|
|
|
S.Diag(Fn.FirstCoroutineStmtLoc, diag::note_declared_coroutine_here)
|
|
|
|
<< Fn.getFirstCoroutineStmtKeyword();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-04-04 03:21:00 +08:00
|
|
|
bool CoroutineStmtBuilder::makeReturnOnAllocFailure() {
|
|
|
|
assert(!IsPromiseDependentType &&
|
|
|
|
"cannot make statement while the promise type is dependent");
|
2017-03-28 07:36:59 +08:00
|
|
|
|
2021-12-24 11:48:15 +08:00
|
|
|
// [dcl.fct.def.coroutine]p10
|
2022-01-12 14:10:07 +08:00
|
|
|
// If a search for the name get_return_object_on_allocation_failure in
|
2021-12-24 11:48:15 +08:00
|
|
|
// the scope of the promise type ([class.member.lookup]) finds any
|
|
|
|
// declarations, then the result of a call to an allocation function used to
|
|
|
|
// obtain storage for the coroutine state is assumed to return nullptr if it
|
|
|
|
// fails to obtain storage, ... If the allocation function returns nullptr,
|
|
|
|
// ... and the return value is obtained by a call to
|
2022-01-12 14:10:07 +08:00
|
|
|
// T::get_return_object_on_allocation_failure(), where T is the
|
2021-12-24 11:48:15 +08:00
|
|
|
// promise type.
|
2017-03-28 07:36:59 +08:00
|
|
|
DeclarationName DN =
|
|
|
|
S.PP.getIdentifierInfo("get_return_object_on_allocation_failure");
|
|
|
|
LookupResult Found(S, DN, Loc, Sema::LookupMemberName);
|
2017-04-04 03:21:00 +08:00
|
|
|
if (!S.LookupQualifiedName(Found, PromiseRecordDecl))
|
|
|
|
return true;
|
2017-03-28 07:36:59 +08:00
|
|
|
|
|
|
|
CXXScopeSpec SS;
|
|
|
|
ExprResult DeclNameExpr =
|
|
|
|
S.BuildDeclarationNameExpr(SS, Found, /*NeedsADL=*/false);
|
2017-04-04 03:21:00 +08:00
|
|
|
if (DeclNameExpr.isInvalid())
|
|
|
|
return false;
|
2017-03-28 07:36:59 +08:00
|
|
|
|
|
|
|
if (!diagReturnOnAllocFailure(S, DeclNameExpr.get(), PromiseRecordDecl, Fn))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
ExprResult ReturnObjectOnAllocationFailure =
|
2019-05-08 09:36:36 +08:00
|
|
|
S.BuildCallExpr(nullptr, DeclNameExpr.get(), Loc, {}, Loc);
|
2017-04-04 03:21:00 +08:00
|
|
|
if (ReturnObjectOnAllocationFailure.isInvalid())
|
|
|
|
return false;
|
2017-03-28 07:36:59 +08:00
|
|
|
|
2017-03-28 10:51:45 +08:00
|
|
|
StmtResult ReturnStmt =
|
|
|
|
S.BuildReturnStmt(Loc, ReturnObjectOnAllocationFailure.get());
|
2017-05-23 04:22:23 +08:00
|
|
|
if (ReturnStmt.isInvalid()) {
|
2017-05-25 22:59:39 +08:00
|
|
|
S.Diag(Found.getFoundDecl()->getLocation(), diag::note_member_declared_here)
|
|
|
|
<< DN;
|
2017-05-23 04:22:23 +08:00
|
|
|
S.Diag(Fn.FirstCoroutineStmtLoc, diag::note_declared_coroutine_here)
|
|
|
|
<< Fn.getFirstCoroutineStmtKeyword();
|
2017-04-04 03:21:00 +08:00
|
|
|
return false;
|
2017-05-23 04:22:23 +08:00
|
|
|
}
|
2017-03-28 07:36:59 +08:00
|
|
|
|
|
|
|
this->ReturnStmtOnAllocFailure = ReturnStmt.get();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-04-04 03:21:00 +08:00
|
|
|
bool CoroutineStmtBuilder::makeNewAndDeleteExpr() {
|
2017-03-07 07:38:15 +08:00
|
|
|
// Form and check allocation and deallocation calls.
|
2017-04-04 03:21:00 +08:00
|
|
|
assert(!IsPromiseDependentType &&
|
|
|
|
"cannot make statement while the promise type is dependent");
|
2017-03-07 07:38:15 +08:00
|
|
|
QualType PromiseType = Fn.CoroutinePromise->getType();
|
|
|
|
|
|
|
|
if (S.RequireCompleteType(Loc, PromiseType, diag::err_incomplete_type))
|
2017-02-13 13:05:02 +08:00
|
|
|
return false;
|
2015-11-24 10:34:39 +08:00
|
|
|
|
2017-04-18 11:12:48 +08:00
|
|
|
const bool RequiresNoThrowAlloc = ReturnStmtOnAllocFailure != nullptr;
|
|
|
|
|
2021-12-24 11:48:15 +08:00
|
|
|
// According to [dcl.fct.def.coroutine]p9, Lookup allocation functions using a
|
|
|
|
// parameter list composed of the requested size of the coroutine state being
|
|
|
|
// allocated, followed by the coroutine function's arguments. If a matching
|
|
|
|
// allocation function exists, use it. Otherwise, use an allocation function
|
|
|
|
// that just takes the requested size.
|
2017-02-13 13:05:02 +08:00
|
|
|
|
2017-03-07 07:38:15 +08:00
|
|
|
FunctionDecl *OperatorNew = nullptr;
|
|
|
|
FunctionDecl *OperatorDelete = nullptr;
|
|
|
|
FunctionDecl *UnusedResult = nullptr;
|
|
|
|
bool PassAlignment = false;
|
2017-04-18 13:08:08 +08:00
|
|
|
SmallVector<Expr *, 1> PlacementArgs;
|
2017-03-07 07:38:15 +08:00
|
|
|
|
2021-12-24 11:48:15 +08:00
|
|
|
// [dcl.fct.def.coroutine]p9
|
|
|
|
// An implementation may need to allocate additional storage for a
|
|
|
|
// coroutine.
|
|
|
|
// This storage is known as the coroutine state and is obtained by calling a
|
|
|
|
// non-array allocation function ([basic.stc.dynamic.allocation]). The
|
|
|
|
// allocation function's name is looked up by searching for it in the scope of
|
|
|
|
// the promise type.
|
|
|
|
// - If any declarations are found, overload resolution is performed on a
|
|
|
|
// function call created by assembling an argument list. The first argument is
|
|
|
|
// the amount of space requested, and has type std::size_t. The
|
|
|
|
// lvalues p1 ... pn are the succeeding arguments.
|
2018-02-16 04:37:22 +08:00
|
|
|
//
|
|
|
|
// ...where "p1 ... pn" are defined earlier as:
|
|
|
|
//
|
2021-12-24 11:48:15 +08:00
|
|
|
// [dcl.fct.def.coroutine]p3
|
|
|
|
// The promise type of a coroutine is `std::coroutine_traits<R, P1, ...,
|
|
|
|
// Pn>`
|
|
|
|
// , where R is the return type of the function, and `P1, ..., Pn` are the
|
|
|
|
// sequence of types of the non-object function parameters, preceded by the
|
|
|
|
// type of the object parameter ([dcl.fct]) if the coroutine is a non-static
|
|
|
|
// member function. [dcl.fct.def.coroutine]p4 In the following, p_i is an
|
|
|
|
// lvalue of type P_i, where p1 denotes the object parameter and p_i+1 denotes
|
|
|
|
// the i-th non-object function parameter for a non-static member function,
|
|
|
|
// and p_i denotes the i-th function parameter otherwise. For a non-static
|
|
|
|
// member function, q_1 is an lvalue that denotes *this; any other q_i is an
|
|
|
|
// lvalue that denotes the parameter copy corresponding to p_i.
|
2018-02-16 04:37:22 +08:00
|
|
|
if (auto *MD = dyn_cast<CXXMethodDecl>(&FD)) {
|
|
|
|
if (MD->isInstance() && !isLambdaCallOperator(MD)) {
|
|
|
|
ExprResult ThisExpr = S.ActOnCXXThis(Loc);
|
|
|
|
if (ThisExpr.isInvalid())
|
|
|
|
return false;
|
|
|
|
ThisExpr = S.CreateBuiltinUnaryOp(Loc, UO_Deref, ThisExpr.get());
|
|
|
|
if (ThisExpr.isInvalid())
|
|
|
|
return false;
|
|
|
|
PlacementArgs.push_back(ThisExpr.get());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (auto *PD : FD.parameters()) {
|
|
|
|
if (PD->getType()->isDependentType())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Build a reference to the parameter.
|
|
|
|
auto PDLoc = PD->getLocation();
|
|
|
|
ExprResult PDRefExpr =
|
|
|
|
S.BuildDeclRefExpr(PD, PD->getOriginalType().getNonReferenceType(),
|
|
|
|
ExprValueKind::VK_LValue, PDLoc);
|
|
|
|
if (PDRefExpr.isInvalid())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
PlacementArgs.push_back(PDRefExpr.get());
|
|
|
|
}
|
[Coroutines] Find custom allocators in class scope
Summary:
https://reviews.llvm.org/rL325291 implemented Coroutines TS N4723
section [dcl.fct.def.coroutine]/7, but it performed lookup of allocator
functions within both the global and class scope, whereas the specified
behavior is to perform lookup for custom allocators within just the
class scope.
To fix, add parameters to the `Sema::FindAllocationFunctions` function
such that it can be used to lookup allocators in global scope,
class scope, or both (instead of just being able to look up in just global
scope or in both global and class scope). Then, use those parameters
from within the coroutine Sema.
This incorrect behavior had the unfortunate side-effect of causing the
bug https://bugs.llvm.org/show_bug.cgi?id=36578 (or at least the reports
of that bug in C++ programs). That bug would occur for any C++ user with
a coroutine frame that took a single pointer argument, since it would
then find the global placement form `operator new`, described in the
C++ standard 18.6.1.3.1. This patch prevents Clang from generating code
that triggers the LLVM assert described in that bug report.
Test Plan: `check-clang`
Reviewers: GorNishanov, eric_niebler, lewissbaker
Reviewed By: GorNishanov
Subscribers: EricWF, cfe-commits
Differential Revision: https://reviews.llvm.org/D44552
llvm-svn: 328949
2018-04-02 06:59:22 +08:00
|
|
|
S.FindAllocationFunctions(Loc, SourceRange(), /*NewScope*/ Sema::AFS_Class,
|
|
|
|
/*DeleteScope*/ Sema::AFS_Both, PromiseType,
|
2017-04-18 11:12:48 +08:00
|
|
|
/*isArray*/ false, PassAlignment, PlacementArgs,
|
2018-02-16 04:37:22 +08:00
|
|
|
OperatorNew, UnusedResult, /*Diagnose*/ false);
|
|
|
|
|
2021-12-24 11:48:15 +08:00
|
|
|
// [dcl.fct.def.coroutine]p9
|
|
|
|
// If no viable function is found ([over.match.viable]), overload resolution
|
|
|
|
// is performed again on a function call created by passing just the amount of
|
|
|
|
// space required as an argument of type std::size_t.
|
2018-02-16 04:37:22 +08:00
|
|
|
if (!OperatorNew && !PlacementArgs.empty()) {
|
|
|
|
PlacementArgs.clear();
|
[Coroutines] Find custom allocators in class scope
Summary:
https://reviews.llvm.org/rL325291 implemented Coroutines TS N4723
section [dcl.fct.def.coroutine]/7, but it performed lookup of allocator
functions within both the global and class scope, whereas the specified
behavior is to perform lookup for custom allocators within just the
class scope.
To fix, add parameters to the `Sema::FindAllocationFunctions` function
such that it can be used to lookup allocators in global scope,
class scope, or both (instead of just being able to look up in just global
scope or in both global and class scope). Then, use those parameters
from within the coroutine Sema.
This incorrect behavior had the unfortunate side-effect of causing the
bug https://bugs.llvm.org/show_bug.cgi?id=36578 (or at least the reports
of that bug in C++ programs). That bug would occur for any C++ user with
a coroutine frame that took a single pointer argument, since it would
then find the global placement form `operator new`, described in the
C++ standard 18.6.1.3.1. This patch prevents Clang from generating code
that triggers the LLVM assert described in that bug report.
Test Plan: `check-clang`
Reviewers: GorNishanov, eric_niebler, lewissbaker
Reviewed By: GorNishanov
Subscribers: EricWF, cfe-commits
Differential Revision: https://reviews.llvm.org/D44552
llvm-svn: 328949
2018-04-02 06:59:22 +08:00
|
|
|
S.FindAllocationFunctions(Loc, SourceRange(), /*NewScope*/ Sema::AFS_Class,
|
|
|
|
/*DeleteScope*/ Sema::AFS_Both, PromiseType,
|
|
|
|
/*isArray*/ false, PassAlignment, PlacementArgs,
|
|
|
|
OperatorNew, UnusedResult, /*Diagnose*/ false);
|
|
|
|
}
|
|
|
|
|
2021-12-24 11:48:15 +08:00
|
|
|
// [dcl.fct.def.coroutine]p9
|
|
|
|
// The allocation function's name is looked up by searching for it in the
|
|
|
|
// scope of the promise type.
|
|
|
|
// - If any declarations are found, ...
|
|
|
|
// - Otherwise, a search is performed in the global scope.
|
[Coroutines] Find custom allocators in class scope
Summary:
https://reviews.llvm.org/rL325291 implemented Coroutines TS N4723
section [dcl.fct.def.coroutine]/7, but it performed lookup of allocator
functions within both the global and class scope, whereas the specified
behavior is to perform lookup for custom allocators within just the
class scope.
To fix, add parameters to the `Sema::FindAllocationFunctions` function
such that it can be used to lookup allocators in global scope,
class scope, or both (instead of just being able to look up in just global
scope or in both global and class scope). Then, use those parameters
from within the coroutine Sema.
This incorrect behavior had the unfortunate side-effect of causing the
bug https://bugs.llvm.org/show_bug.cgi?id=36578 (or at least the reports
of that bug in C++ programs). That bug would occur for any C++ user with
a coroutine frame that took a single pointer argument, since it would
then find the global placement form `operator new`, described in the
C++ standard 18.6.1.3.1. This patch prevents Clang from generating code
that triggers the LLVM assert described in that bug report.
Test Plan: `check-clang`
Reviewers: GorNishanov, eric_niebler, lewissbaker
Reviewed By: GorNishanov
Subscribers: EricWF, cfe-commits
Differential Revision: https://reviews.llvm.org/D44552
llvm-svn: 328949
2018-04-02 06:59:22 +08:00
|
|
|
if (!OperatorNew) {
|
|
|
|
S.FindAllocationFunctions(Loc, SourceRange(), /*NewScope*/ Sema::AFS_Global,
|
|
|
|
/*DeleteScope*/ Sema::AFS_Both, PromiseType,
|
|
|
|
/*isArray*/ false, PassAlignment, PlacementArgs,
|
|
|
|
OperatorNew, UnusedResult);
|
2018-02-16 04:37:22 +08:00
|
|
|
}
|
2017-04-18 11:12:48 +08:00
|
|
|
|
|
|
|
bool IsGlobalOverload =
|
|
|
|
OperatorNew && !isa<CXXRecordDecl>(OperatorNew->getDeclContext());
|
|
|
|
// If we didn't find a class-local new declaration and non-throwing new
|
|
|
|
// was is required then we need to lookup the non-throwing global operator
|
|
|
|
// instead.
|
|
|
|
if (RequiresNoThrowAlloc && (!OperatorNew || IsGlobalOverload)) {
|
|
|
|
auto *StdNoThrow = buildStdNoThrowDeclRef(S, Loc);
|
|
|
|
if (!StdNoThrow)
|
|
|
|
return false;
|
2017-04-18 13:08:08 +08:00
|
|
|
PlacementArgs = {StdNoThrow};
|
2017-04-18 11:12:48 +08:00
|
|
|
OperatorNew = nullptr;
|
[Coroutines] Find custom allocators in class scope
Summary:
https://reviews.llvm.org/rL325291 implemented Coroutines TS N4723
section [dcl.fct.def.coroutine]/7, but it performed lookup of allocator
functions within both the global and class scope, whereas the specified
behavior is to perform lookup for custom allocators within just the
class scope.
To fix, add parameters to the `Sema::FindAllocationFunctions` function
such that it can be used to lookup allocators in global scope,
class scope, or both (instead of just being able to look up in just global
scope or in both global and class scope). Then, use those parameters
from within the coroutine Sema.
This incorrect behavior had the unfortunate side-effect of causing the
bug https://bugs.llvm.org/show_bug.cgi?id=36578 (or at least the reports
of that bug in C++ programs). That bug would occur for any C++ user with
a coroutine frame that took a single pointer argument, since it would
then find the global placement form `operator new`, described in the
C++ standard 18.6.1.3.1. This patch prevents Clang from generating code
that triggers the LLVM assert described in that bug report.
Test Plan: `check-clang`
Reviewers: GorNishanov, eric_niebler, lewissbaker
Reviewed By: GorNishanov
Subscribers: EricWF, cfe-commits
Differential Revision: https://reviews.llvm.org/D44552
llvm-svn: 328949
2018-04-02 06:59:22 +08:00
|
|
|
S.FindAllocationFunctions(Loc, SourceRange(), /*NewScope*/ Sema::AFS_Both,
|
|
|
|
/*DeleteScope*/ Sema::AFS_Both, PromiseType,
|
2017-04-18 11:12:48 +08:00
|
|
|
/*isArray*/ false, PassAlignment, PlacementArgs,
|
|
|
|
OperatorNew, UnusedResult);
|
|
|
|
}
|
2017-03-07 07:38:15 +08:00
|
|
|
|
2018-02-16 04:37:22 +08:00
|
|
|
if (!OperatorNew)
|
|
|
|
return false;
|
2017-04-18 13:30:39 +08:00
|
|
|
|
|
|
|
if (RequiresNoThrowAlloc) {
|
2020-01-09 20:05:30 +08:00
|
|
|
const auto *FT = OperatorNew->getType()->castAs<FunctionProtoType>();
|
2018-05-03 11:58:32 +08:00
|
|
|
if (!FT->isNothrow(/*ResultIfDependent*/ false)) {
|
2017-04-18 11:12:48 +08:00
|
|
|
S.Diag(OperatorNew->getLocation(),
|
|
|
|
diag::err_coroutine_promise_new_requires_nothrow)
|
|
|
|
<< OperatorNew;
|
|
|
|
S.Diag(Loc, diag::note_coroutine_promise_call_implicitly_required)
|
|
|
|
<< OperatorNew;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2017-03-07 07:38:15 +08:00
|
|
|
|
2021-12-24 11:48:15 +08:00
|
|
|
if ((OperatorDelete = findDeleteForPromise(S, Loc, PromiseType)) == nullptr) {
|
|
|
|
// FIXME: We should add an error here. According to:
|
|
|
|
// [dcl.fct.def.coroutine]p12
|
|
|
|
// If no usual deallocation function is found, the program is ill-formed.
|
2017-03-07 07:38:15 +08:00
|
|
|
return false;
|
2021-12-24 11:48:15 +08:00
|
|
|
}
|
2017-03-07 07:38:15 +08:00
|
|
|
|
|
|
|
Expr *FramePtr =
|
2021-06-29 00:45:56 +08:00
|
|
|
S.BuildBuiltinCallExpr(Loc, Builtin::BI__builtin_coro_frame, {});
|
2017-03-07 07:38:15 +08:00
|
|
|
|
|
|
|
Expr *FrameSize =
|
2021-06-29 00:45:56 +08:00
|
|
|
S.BuildBuiltinCallExpr(Loc, Builtin::BI__builtin_coro_size, {});
|
2017-03-07 07:38:15 +08:00
|
|
|
|
|
|
|
// Make new call.
|
|
|
|
|
|
|
|
ExprResult NewRef =
|
|
|
|
S.BuildDeclRefExpr(OperatorNew, OperatorNew->getType(), VK_LValue, Loc);
|
|
|
|
if (NewRef.isInvalid())
|
|
|
|
return false;
|
|
|
|
|
2017-04-18 13:08:08 +08:00
|
|
|
SmallVector<Expr *, 2> NewArgs(1, FrameSize);
|
2022-03-18 08:15:43 +08:00
|
|
|
llvm::append_range(NewArgs, PlacementArgs);
|
2017-04-18 11:12:48 +08:00
|
|
|
|
2017-03-07 07:38:15 +08:00
|
|
|
ExprResult NewExpr =
|
2019-05-08 09:36:36 +08:00
|
|
|
S.BuildCallExpr(S.getCurScope(), NewRef.get(), Loc, NewArgs, Loc);
|
2019-01-05 00:58:14 +08:00
|
|
|
NewExpr = S.ActOnFinishFullExpr(NewExpr.get(), /*DiscardedValue*/ false);
|
2017-03-07 07:38:15 +08:00
|
|
|
if (NewExpr.isInvalid())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Make delete call.
|
|
|
|
|
|
|
|
QualType OpDeleteQualType = OperatorDelete->getType();
|
|
|
|
|
|
|
|
ExprResult DeleteRef =
|
|
|
|
S.BuildDeclRefExpr(OperatorDelete, OpDeleteQualType, VK_LValue, Loc);
|
|
|
|
if (DeleteRef.isInvalid())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
Expr *CoroFree =
|
2021-06-29 00:45:56 +08:00
|
|
|
S.BuildBuiltinCallExpr(Loc, Builtin::BI__builtin_coro_free, {FramePtr});
|
2017-03-07 07:38:15 +08:00
|
|
|
|
|
|
|
SmallVector<Expr *, 2> DeleteArgs{CoroFree};
|
|
|
|
|
2021-12-24 11:48:15 +08:00
|
|
|
// [dcl.fct.def.coroutine]p12
|
|
|
|
// The selected deallocation function shall be called with the address of
|
|
|
|
// the block of storage to be reclaimed as its first argument. If a
|
|
|
|
// deallocation function with a parameter of type std::size_t is
|
|
|
|
// used, the size of the block is passed as the corresponding argument.
|
2017-03-07 07:38:15 +08:00
|
|
|
const auto *OpDeleteType =
|
2020-01-09 20:05:30 +08:00
|
|
|
OpDeleteQualType.getTypePtr()->castAs<FunctionProtoType>();
|
2017-03-07 07:38:15 +08:00
|
|
|
if (OpDeleteType->getNumParams() > 1)
|
|
|
|
DeleteArgs.push_back(FrameSize);
|
|
|
|
|
|
|
|
ExprResult DeleteExpr =
|
2019-05-08 09:36:36 +08:00
|
|
|
S.BuildCallExpr(S.getCurScope(), DeleteRef.get(), Loc, DeleteArgs, Loc);
|
2019-01-05 00:58:14 +08:00
|
|
|
DeleteExpr =
|
|
|
|
S.ActOnFinishFullExpr(DeleteExpr.get(), /*DiscardedValue*/ false);
|
2017-03-07 07:38:15 +08:00
|
|
|
if (DeleteExpr.isInvalid())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
this->Allocate = NewExpr.get();
|
|
|
|
this->Deallocate = DeleteExpr.get();
|
|
|
|
|
|
|
|
return true;
|
2017-02-13 13:05:02 +08:00
|
|
|
}
|
|
|
|
|
2017-04-04 03:21:00 +08:00
|
|
|
bool CoroutineStmtBuilder::makeOnFallthrough() {
|
|
|
|
assert(!IsPromiseDependentType &&
|
|
|
|
"cannot make statement while the promise type is dependent");
|
2017-02-13 13:05:02 +08:00
|
|
|
|
2021-12-24 13:33:52 +08:00
|
|
|
// [dcl.fct.def.coroutine]/p6
|
2022-01-12 14:10:07 +08:00
|
|
|
// If searches for the names return_void and return_value in the scope of
|
2021-12-24 13:33:52 +08:00
|
|
|
// the promise type each find any declarations, the program is ill-formed.
|
2022-01-12 14:10:07 +08:00
|
|
|
// [Note 1: If return_void is found, flowing off the end of a coroutine is
|
|
|
|
// equivalent to a co_return with no operand. Otherwise, flowing off the end
|
2021-12-24 13:33:52 +08:00
|
|
|
// of a coroutine results in undefined behavior ([stmt.return.coroutine]). —
|
|
|
|
// end note]
|
2017-05-25 22:59:39 +08:00
|
|
|
bool HasRVoid, HasRValue;
|
|
|
|
LookupResult LRVoid =
|
|
|
|
lookupMember(S, "return_void", PromiseRecordDecl, Loc, HasRVoid);
|
|
|
|
LookupResult LRValue =
|
|
|
|
lookupMember(S, "return_value", PromiseRecordDecl, Loc, HasRValue);
|
2017-02-13 13:05:02 +08:00
|
|
|
|
|
|
|
StmtResult Fallthrough;
|
|
|
|
if (HasRVoid && HasRValue) {
|
|
|
|
// FIXME Improve this diagnostic
|
2017-05-25 22:59:39 +08:00
|
|
|
S.Diag(FD.getLocation(),
|
|
|
|
diag::err_coroutine_promise_incompatible_return_functions)
|
|
|
|
<< PromiseRecordDecl;
|
|
|
|
S.Diag(LRVoid.getRepresentativeDecl()->getLocation(),
|
|
|
|
diag::note_member_first_declared_here)
|
|
|
|
<< LRVoid.getLookupName();
|
|
|
|
S.Diag(LRValue.getRepresentativeDecl()->getLocation(),
|
|
|
|
diag::note_member_first_declared_here)
|
|
|
|
<< LRValue.getLookupName();
|
|
|
|
return false;
|
|
|
|
} else if (!HasRVoid && !HasRValue) {
|
2021-12-24 13:33:52 +08:00
|
|
|
// We need to set 'Fallthrough'. Otherwise the other analysis part might
|
|
|
|
// think the coroutine has defined a return_value method. So it might emit
|
|
|
|
// **false** positive warning. e.g.,
|
|
|
|
//
|
|
|
|
// promise_without_return_func foo() {
|
|
|
|
// co_await something();
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// Then AnalysisBasedWarning would emit a warning about `foo()` lacking a
|
|
|
|
// co_return statements, which isn't correct.
|
|
|
|
Fallthrough = S.ActOnNullStmt(PromiseRecordDecl->getLocation());
|
|
|
|
if (Fallthrough.isInvalid())
|
|
|
|
return false;
|
2017-02-13 13:05:02 +08:00
|
|
|
} else if (HasRVoid) {
|
2017-03-07 07:38:15 +08:00
|
|
|
Fallthrough = S.BuildCoreturnStmt(FD.getLocation(), nullptr,
|
|
|
|
/*IsImplicit*/false);
|
2017-02-13 13:05:02 +08:00
|
|
|
Fallthrough = S.ActOnFinishFullStmt(Fallthrough.get());
|
|
|
|
if (Fallthrough.isInvalid())
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
this->OnFallthrough = Fallthrough.get();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-04-04 03:21:00 +08:00
|
|
|
bool CoroutineStmtBuilder::makeOnException() {
|
2017-03-23 08:33:33 +08:00
|
|
|
// Try to form 'p.unhandled_exception();'
|
2017-04-04 03:21:00 +08:00
|
|
|
assert(!IsPromiseDependentType &&
|
|
|
|
"cannot make statement while the promise type is dependent");
|
2017-02-13 13:05:02 +08:00
|
|
|
|
2017-03-23 08:33:33 +08:00
|
|
|
const bool RequireUnhandledException = S.getLangOpts().CXXExceptions;
|
|
|
|
|
|
|
|
if (!lookupMember(S, "unhandled_exception", PromiseRecordDecl, Loc)) {
|
|
|
|
auto DiagID =
|
|
|
|
RequireUnhandledException
|
|
|
|
? diag::err_coroutine_promise_unhandled_exception_required
|
|
|
|
: diag::
|
|
|
|
warn_coroutine_promise_unhandled_exception_required_with_exceptions;
|
|
|
|
S.Diag(Loc, DiagID) << PromiseRecordDecl;
|
2017-05-24 22:34:19 +08:00
|
|
|
S.Diag(PromiseRecordDecl->getLocation(), diag::note_defined_here)
|
|
|
|
<< PromiseRecordDecl;
|
2017-03-23 08:33:33 +08:00
|
|
|
return !RequireUnhandledException;
|
|
|
|
}
|
|
|
|
|
2017-02-13 13:05:02 +08:00
|
|
|
// If exceptions are disabled, don't try to build OnException.
|
|
|
|
if (!S.getLangOpts().CXXExceptions)
|
|
|
|
return true;
|
|
|
|
|
2017-03-23 08:33:33 +08:00
|
|
|
ExprResult UnhandledException = buildPromiseCall(S, Fn.CoroutinePromise, Loc,
|
|
|
|
"unhandled_exception", None);
|
2019-01-05 00:58:14 +08:00
|
|
|
UnhandledException = S.ActOnFinishFullExpr(UnhandledException.get(), Loc,
|
|
|
|
/*DiscardedValue*/ false);
|
2017-03-23 08:33:33 +08:00
|
|
|
if (UnhandledException.isInvalid())
|
|
|
|
return false;
|
2015-11-24 10:34:39 +08:00
|
|
|
|
2017-05-23 06:33:17 +08:00
|
|
|
// Since the body of the coroutine will be wrapped in try-catch, it will
|
|
|
|
// be incompatible with SEH __try if present in a function.
|
|
|
|
if (!S.getLangOpts().Borland && Fn.FirstSEHTryLoc.isValid()) {
|
|
|
|
S.Diag(Fn.FirstSEHTryLoc, diag::err_seh_in_a_coroutine_with_cxx_exceptions);
|
|
|
|
S.Diag(Fn.FirstCoroutineStmtLoc, diag::note_declared_coroutine_here)
|
|
|
|
<< Fn.getFirstCoroutineStmtKeyword();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-03-23 08:33:33 +08:00
|
|
|
this->OnException = UnhandledException.get();
|
2017-02-13 13:05:02 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-04-04 03:21:00 +08:00
|
|
|
bool CoroutineStmtBuilder::makeReturnObject() {
|
2021-12-24 11:48:15 +08:00
|
|
|
// [dcl.fct.def.coroutine]p7
|
2022-01-12 14:10:07 +08:00
|
|
|
// The expression promise.get_return_object() is used to initialize the
|
2021-12-24 11:48:15 +08:00
|
|
|
// returned reference or prvalue result object of a call to a coroutine.
|
2015-11-24 10:34:39 +08:00
|
|
|
ExprResult ReturnObject =
|
2017-03-07 07:38:15 +08:00
|
|
|
buildPromiseCall(S, Fn.CoroutinePromise, Loc, "get_return_object", None);
|
2015-11-24 10:34:39 +08:00
|
|
|
if (ReturnObject.isInvalid())
|
2017-02-13 13:05:02 +08:00
|
|
|
return false;
|
2017-05-23 04:22:23 +08:00
|
|
|
|
|
|
|
this->ReturnValue = ReturnObject.get();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void noteMemberDeclaredHere(Sema &S, Expr *E, FunctionScopeInfo &Fn) {
|
|
|
|
if (auto *MbrRef = dyn_cast<CXXMemberCallExpr>(E)) {
|
|
|
|
auto *MethodDecl = MbrRef->getMethodDecl();
|
2017-05-25 22:59:39 +08:00
|
|
|
S.Diag(MethodDecl->getLocation(), diag::note_member_declared_here)
|
|
|
|
<< MethodDecl;
|
2017-05-23 04:22:23 +08:00
|
|
|
}
|
|
|
|
S.Diag(Fn.FirstCoroutineStmtLoc, diag::note_declared_coroutine_here)
|
|
|
|
<< Fn.getFirstCoroutineStmtKeyword();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CoroutineStmtBuilder::makeGroDeclAndReturnStmt() {
|
|
|
|
assert(!IsPromiseDependentType &&
|
|
|
|
"cannot make statement while the promise type is dependent");
|
|
|
|
assert(this->ReturnValue && "ReturnValue must be already formed");
|
|
|
|
|
|
|
|
QualType const GroType = this->ReturnValue->getType();
|
|
|
|
assert(!GroType->isDependentType() &&
|
|
|
|
"get_return_object type must no longer be dependent");
|
|
|
|
|
|
|
|
QualType const FnRetType = FD.getReturnType();
|
|
|
|
assert(!FnRetType->isDependentType() &&
|
|
|
|
"get_return_object type must no longer be dependent");
|
|
|
|
|
|
|
|
if (FnRetType->isVoidType()) {
|
2019-01-05 00:58:14 +08:00
|
|
|
ExprResult Res =
|
|
|
|
S.ActOnFinishFullExpr(this->ReturnValue, Loc, /*DiscardedValue*/ false);
|
2017-05-23 04:22:23 +08:00
|
|
|
if (Res.isInvalid())
|
2017-02-13 13:05:02 +08:00
|
|
|
return false;
|
2017-05-23 04:22:23 +08:00
|
|
|
|
|
|
|
return true;
|
2015-11-24 10:34:39 +08:00
|
|
|
}
|
2017-05-23 04:22:23 +08:00
|
|
|
|
|
|
|
if (GroType->isVoidType()) {
|
|
|
|
// Trigger a nice error message.
|
|
|
|
InitializedEntity Entity =
|
2021-09-15 07:46:30 +08:00
|
|
|
InitializedEntity::InitializeResult(Loc, FnRetType);
|
2021-03-19 10:32:06 +08:00
|
|
|
S.PerformCopyInitialization(Entity, SourceLocation(), ReturnValue);
|
2017-05-23 04:22:23 +08:00
|
|
|
noteMemberDeclaredHere(S, ReturnValue, Fn);
|
2017-02-13 13:05:02 +08:00
|
|
|
return false;
|
2017-05-23 04:22:23 +08:00
|
|
|
}
|
2015-11-24 10:34:39 +08:00
|
|
|
|
2022-01-10 19:18:33 +08:00
|
|
|
StmtResult ReturnStmt = S.BuildReturnStmt(Loc, ReturnValue);
|
2017-05-23 04:22:23 +08:00
|
|
|
if (ReturnStmt.isInvalid()) {
|
|
|
|
noteMemberDeclaredHere(S, ReturnValue, Fn);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
this->ReturnStmt = ReturnStmt.get();
|
2017-02-13 13:05:02 +08:00
|
|
|
return true;
|
|
|
|
}
|
2015-11-24 10:34:39 +08:00
|
|
|
|
2017-05-25 04:09:14 +08:00
|
|
|
// Create a static_cast\<T&&>(expr).
|
|
|
|
static Expr *castForMoving(Sema &S, Expr *E, QualType T = QualType()) {
|
|
|
|
if (T.isNull())
|
|
|
|
T = E->getType();
|
|
|
|
QualType TargetType = S.BuildReferenceType(
|
|
|
|
T, /*SpelledAsLValue*/ false, SourceLocation(), DeclarationName());
|
2018-08-10 05:08:08 +08:00
|
|
|
SourceLocation ExprLoc = E->getBeginLoc();
|
2017-05-25 04:09:14 +08:00
|
|
|
TypeSourceInfo *TargetLoc =
|
|
|
|
S.Context.getTrivialTypeSourceInfo(TargetType, ExprLoc);
|
|
|
|
|
|
|
|
return S
|
|
|
|
.BuildCXXNamedCast(ExprLoc, tok::kw_static_cast, TargetLoc, E,
|
|
|
|
SourceRange(ExprLoc, ExprLoc), E->getSourceRange())
|
|
|
|
.get();
|
|
|
|
}
|
|
|
|
|
2018-05-09 09:00:01 +08:00
|
|
|
/// Build a variable declaration for move parameter.
|
2017-05-25 04:09:14 +08:00
|
|
|
static VarDecl *buildVarDecl(Sema &S, SourceLocation Loc, QualType Type,
|
2017-06-03 08:22:18 +08:00
|
|
|
IdentifierInfo *II) {
|
2017-05-25 04:09:14 +08:00
|
|
|
TypeSourceInfo *TInfo = S.Context.getTrivialTypeSourceInfo(Type, Loc);
|
2018-01-25 06:15:42 +08:00
|
|
|
VarDecl *Decl = VarDecl::Create(S.Context, S.CurContext, Loc, Loc, II, Type,
|
2021-01-05 06:17:45 +08:00
|
|
|
TInfo, SC_None);
|
2017-05-25 04:09:14 +08:00
|
|
|
Decl->setImplicit();
|
|
|
|
return Decl;
|
|
|
|
}
|
|
|
|
|
2018-01-25 06:15:42 +08:00
|
|
|
// Build statements that move coroutine function parameters to the coroutine
|
|
|
|
// frame, and store them on the function scope info.
|
|
|
|
bool Sema::buildCoroutineParameterMoves(SourceLocation Loc) {
|
|
|
|
assert(isa<FunctionDecl>(CurContext) && "not in a function scope");
|
|
|
|
auto *FD = cast<FunctionDecl>(CurContext);
|
|
|
|
|
|
|
|
auto *ScopeInfo = getCurFunction();
|
2019-11-23 00:25:19 +08:00
|
|
|
if (!ScopeInfo->CoroutineParameterMoves.empty())
|
|
|
|
return false;
|
2018-01-25 06:15:42 +08:00
|
|
|
|
2021-12-24 11:48:15 +08:00
|
|
|
// [dcl.fct.def.coroutine]p13
|
|
|
|
// When a coroutine is invoked, after initializing its parameters
|
|
|
|
// ([expr.call]), a copy is created for each coroutine parameter. For a
|
|
|
|
// parameter of type cv T, the copy is a variable of type cv T with
|
|
|
|
// automatic storage duration that is direct-initialized from an xvalue of
|
|
|
|
// type T referring to the parameter.
|
2018-01-25 06:15:42 +08:00
|
|
|
for (auto *PD : FD->parameters()) {
|
|
|
|
if (PD->getType()->isDependentType())
|
2017-05-25 04:09:14 +08:00
|
|
|
continue;
|
|
|
|
|
2018-02-16 04:37:22 +08:00
|
|
|
ExprResult PDRefExpr =
|
|
|
|
BuildDeclRefExpr(PD, PD->getType().getNonReferenceType(),
|
|
|
|
ExprValueKind::VK_LValue, Loc); // FIXME: scope?
|
|
|
|
if (PDRefExpr.isInvalid())
|
|
|
|
return false;
|
2017-05-25 04:09:14 +08:00
|
|
|
|
2018-02-16 04:37:22 +08:00
|
|
|
Expr *CExpr = nullptr;
|
|
|
|
if (PD->getType()->getAsCXXRecordDecl() ||
|
|
|
|
PD->getType()->isRValueReferenceType())
|
|
|
|
CExpr = castForMoving(*this, PDRefExpr.get());
|
|
|
|
else
|
|
|
|
CExpr = PDRefExpr.get();
|
2021-12-24 11:48:15 +08:00
|
|
|
// [dcl.fct.def.coroutine]p13
|
|
|
|
// The initialization and destruction of each parameter copy occurs in the
|
|
|
|
// context of the called coroutine.
|
2018-02-16 04:37:22 +08:00
|
|
|
auto D = buildVarDecl(*this, Loc, PD->getType(), PD->getIdentifier());
|
|
|
|
AddInitializerToDecl(D, CExpr, /*DirectInit=*/true);
|
2017-05-25 04:09:14 +08:00
|
|
|
|
2018-02-16 04:37:22 +08:00
|
|
|
// Convert decl to a statement.
|
|
|
|
StmtResult Stmt = ActOnDeclStmt(ConvertDeclToDeclGroup(D), Loc, Loc);
|
|
|
|
if (Stmt.isInvalid())
|
|
|
|
return false;
|
2017-05-25 04:09:14 +08:00
|
|
|
|
2018-02-16 04:37:22 +08:00
|
|
|
ScopeInfo->CoroutineParameterMoves.insert(std::make_pair(PD, Stmt.get()));
|
2017-05-25 04:09:14 +08:00
|
|
|
}
|
2017-02-13 13:05:02 +08:00
|
|
|
return true;
|
2015-10-22 14:13:50 +08:00
|
|
|
}
|
2017-03-07 07:38:15 +08:00
|
|
|
|
|
|
|
StmtResult Sema::BuildCoroutineBodyStmt(CoroutineBodyStmt::CtorArgs Args) {
|
|
|
|
CoroutineBodyStmt *Res = CoroutineBodyStmt::Create(Context, Args);
|
|
|
|
if (!Res)
|
|
|
|
return StmtError();
|
|
|
|
return Res;
|
|
|
|
}
|
2018-07-15 02:21:44 +08:00
|
|
|
|
|
|
|
ClassTemplateDecl *Sema::lookupCoroutineTraits(SourceLocation KwLoc,
|
2021-11-04 11:50:30 +08:00
|
|
|
SourceLocation FuncLoc,
|
|
|
|
NamespaceDecl *&Namespace) {
|
2018-07-15 02:21:44 +08:00
|
|
|
if (!StdCoroutineTraitsCache) {
|
2021-12-21 22:45:25 +08:00
|
|
|
// Because coroutines moved from std::experimental in the TS to std in
|
|
|
|
// C++20, we look in both places to give users time to transition their
|
|
|
|
// TS-specific code to C++20. Diagnostics are given when the TS usage is
|
|
|
|
// discovered.
|
|
|
|
// TODO: Become stricter when <experimental/coroutine> is removed.
|
|
|
|
|
|
|
|
auto const &TraitIdent = PP.getIdentifierTable().get("coroutine_traits");
|
|
|
|
|
|
|
|
NamespaceDecl *StdSpace = getStdNamespace();
|
|
|
|
LookupResult ResStd(*this, &TraitIdent, FuncLoc, LookupOrdinaryName);
|
|
|
|
bool InStd = StdSpace && LookupQualifiedName(ResStd, StdSpace);
|
|
|
|
|
|
|
|
NamespaceDecl *ExpSpace = lookupStdExperimentalNamespace();
|
|
|
|
LookupResult ResExp(*this, &TraitIdent, FuncLoc, LookupOrdinaryName);
|
|
|
|
bool InExp = ExpSpace && LookupQualifiedName(ResExp, ExpSpace);
|
|
|
|
|
|
|
|
if (!InStd && !InExp) {
|
|
|
|
// The goggles, they found nothing!
|
|
|
|
Diag(KwLoc, diag::err_implied_coroutine_type_not_found)
|
|
|
|
<< "std::coroutine_traits";
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Prefer ::std to std::experimental.
|
|
|
|
auto &Result = InStd ? ResStd : ResExp;
|
|
|
|
CoroTraitsNamespaceCache = InStd ? StdSpace : ExpSpace;
|
|
|
|
|
|
|
|
// coroutine_traits is required to be a class template.
|
2021-12-26 02:29:58 +08:00
|
|
|
StdCoroutineTraitsCache = Result.getAsSingle<ClassTemplateDecl>();
|
|
|
|
if (!StdCoroutineTraitsCache) {
|
2021-11-04 11:50:30 +08:00
|
|
|
Result.suppressDiagnostics();
|
|
|
|
NamedDecl *Found = *Result.begin();
|
|
|
|
Diag(Found->getLocation(), diag::err_malformed_std_coroutine_traits);
|
|
|
|
return nullptr;
|
|
|
|
}
|
2021-12-26 02:29:58 +08:00
|
|
|
|
|
|
|
if (InExp) {
|
|
|
|
// Found in std::experimental
|
|
|
|
Diag(KwLoc, diag::warn_deprecated_coroutine_namespace)
|
|
|
|
<< "coroutine_traits";
|
|
|
|
ResExp.suppressDiagnostics();
|
|
|
|
auto *Found = *ResExp.begin();
|
|
|
|
Diag(Found->getLocation(), diag::note_entity_declared_at) << Found;
|
|
|
|
|
2021-12-26 02:30:42 +08:00
|
|
|
if (InStd &&
|
|
|
|
StdCoroutineTraitsCache != ResExp.getAsSingle<ClassTemplateDecl>()) {
|
|
|
|
// Also found something different in std
|
2021-12-26 02:29:58 +08:00
|
|
|
Diag(KwLoc,
|
|
|
|
diag::err_mixed_use_std_and_experimental_namespace_for_coroutine);
|
|
|
|
Diag(StdCoroutineTraitsCache->getLocation(),
|
|
|
|
diag::note_entity_declared_at)
|
|
|
|
<< StdCoroutineTraitsCache;
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
}
|
2018-07-15 02:21:44 +08:00
|
|
|
}
|
2021-11-04 11:50:30 +08:00
|
|
|
Namespace = CoroTraitsNamespaceCache;
|
2018-07-15 02:21:44 +08:00
|
|
|
return StdCoroutineTraitsCache;
|
|
|
|
}
|