From c6b9be29f270e6adc0922cb6e9be000f7102b663 Mon Sep 17 00:00:00 2001 From: Faisal Vali Date: Sat, 23 Jul 2016 04:05:19 +0000 Subject: [PATCH] [cxx1z-constexpr-lambda] Make a lambda's closure type eligible as a literal-type in C++1z Additionally, for pre-C++1z, instead of forbidding a lambda's closure type from being a literal type through circumlocutorily setting HasNonLiteralTypeFieldsOrBases falsely to true -- handle lambda's more directly in CXXRecordDecl::isLiteral(). One additional small step towards implementing constexpr-lambdas. Thanks to Richard Smith for his review! https://reviews.llvm.org/D22662 llvm-svn: 276514 --- clang/include/clang/AST/DeclCXX.h | 15 +++++++++------ clang/test/SemaCXX/cxx1z-constexpr-lambdas.cpp | 18 +++++++++++++++--- 2 files changed, 24 insertions(+), 9 deletions(-) diff --git a/clang/include/clang/AST/DeclCXX.h b/clang/include/clang/AST/DeclCXX.h index cd42e05..6a2fcdd 100644 --- a/clang/include/clang/AST/DeclCXX.h +++ b/clang/include/clang/AST/DeclCXX.h @@ -535,11 +535,10 @@ class CXXRecordDecl : public RecordDecl { MethodTyInfo(Info) { IsLambda = true; - // C++11 [expr.prim.lambda]p3: - // This class type is neither an aggregate nor a literal type. + // C++1z [expr.prim.lambda]p4: + // This class type is not an aggregate type. Aggregate = false; PlainOldData = false; - HasNonLiteralTypeFieldsOrBases = true; } /// \brief Whether this lambda is known to be dependent, even if its @@ -1338,11 +1337,15 @@ public: /// /// We resolve DR1361 by ignoring the second bullet. We resolve DR1452 by /// treating types with trivial default constructors as literal types. + /// + /// Only in C++1z and beyond, are lambdas literal types. bool isLiteral() const { return hasTrivialDestructor() && - (isAggregate() || hasConstexprNonCopyMoveConstructor() || - hasTrivialDefaultConstructor()) && - !hasNonLiteralTypeFieldsOrBases(); + (!isLambda() || getASTContext().getLangOpts().CPlusPlus1z) && + !hasNonLiteralTypeFieldsOrBases() && + (isAggregate() || isLambda() || + hasConstexprNonCopyMoveConstructor() || + hasTrivialDefaultConstructor()); } /// \brief If this record is an instantiation of a member class, diff --git a/clang/test/SemaCXX/cxx1z-constexpr-lambdas.cpp b/clang/test/SemaCXX/cxx1z-constexpr-lambdas.cpp index 8e7657f..610d814 100644 --- a/clang/test/SemaCXX/cxx1z-constexpr-lambdas.cpp +++ b/clang/test/SemaCXX/cxx1z-constexpr-lambdas.cpp @@ -1,8 +1,8 @@ // RUN: %clang_cc1 -std=c++1z -verify -fsyntax-only -fblocks %s // RUN: %clang_cc1 -std=c++1z -verify -fsyntax-only -fblocks -fdelayed-template-parsing %s -// RUN: %clang_cc1 -std=c++1z -verify -fsyntax-only -fblocks -fms-extensions %s -// RUN: %clang_cc1 -std=c++1z -verify -fsyntax-only -fblocks -fdelayed-template-parsing -fms-extensions %s +// RUN: %clang_cc1 -std=c++14 -verify -fsyntax-only -fblocks %s -DCPP14_AND_EARLIER +#ifndef CPP14_AND_EARLIER namespace test_constexpr_checking { namespace ns1 { @@ -33,4 +33,16 @@ namespace ns3 { L(3); //expected-note{{non-constexpr function}} } -} // end ns test_constexpr_call \ No newline at end of file +} // end ns test_constexpr_call + +#endif + +namespace test_lambda_is_literal { +#ifdef CPP14_AND_EARLIER +//expected-error@+4{{not a literal type}} +//expected-note@+2{{not an aggregate and has no constexpr constructors}} +#endif +auto L = [] { }; +constexpr int foo(decltype(L) l) { return 0; } + +} \ No newline at end of file -- 2.7.4