[PATCH] D83681: [clang] Provide a more specific diagnostic for a misplaced lambda capture-default.
Bruno Ricci via Phabricator via cfe-commits
cfe-commits at lists.llvm.org
Wed Jul 15 06:29:44 PDT 2020
riccibruno updated this revision to Diff 278164.
riccibruno marked 3 inline comments as done.
riccibruno added a comment.
Add some tests showing that a by-ref capture is still parsed properly, including some tests with a capture containing a pack expansion.
Repository:
rG LLVM Github Monorepo
CHANGES SINCE LAST ACTION
https://reviews.llvm.org/D83681/new/
https://reviews.llvm.org/D83681
Files:
clang/include/clang/Basic/DiagnosticParseKinds.td
clang/lib/Parse/ParseExprCXX.cpp
clang/test/Parser/lambda-misplaced-capture-default.cpp
Index: clang/test/Parser/lambda-misplaced-capture-default.cpp
===================================================================
--- /dev/null
+++ clang/test/Parser/lambda-misplaced-capture-default.cpp
@@ -0,0 +1,38 @@
+// RUN: %clang_cc1 -std=c++20 -Wno-unused-value -fsyntax-only -verify %s
+
+namespace misplaced_capture_default {
+void Test() {
+ int i = 0;
+ [&, i, &] {}; // expected-error {{expected variable name or 'this' in lambda capture list}}
+ [&, i, = ] {}; // expected-error {{expected variable name or 'this' in lambda capture list}}
+ [=, &i, &] {}; // expected-error {{expected variable name or 'this' in lambda capture list}}
+ [=, &i, = ] {}; // expected-error {{expected variable name or 'this' in lambda capture list}}
+
+ [i, &] {}; // expected-error {{capture default must be first}}
+ [i, = ] {}; // expected-error {{capture default must be first}}
+ [i, = x] {}; // expected-error {{expected variable name or 'this' in lambda capture list}}
+ [=, &i] {}; // ok
+ [&, &i] {}; // expected-error {{'&' cannot precede a capture when the capture default is '&'}}
+ [&x = i] {}; // ok
+ [=, &x = i] {}; // ok
+ [x = &i] {}; // ok
+ [=, &x = &i] {}; // expected-error {{non-const lvalue reference to type 'int *' cannot bind to a temporary of type 'int *'}}
+ [&, this] {}; // expected-error {{'this' cannot be captured in this context}}
+
+ [i, &, x = 2] {}; // expected-error {{capture default must be first}}
+ [i, =, x = 2] {}; // expected-error {{capture default must be first}}
+}
+} // namespace misplaced_capture_default
+
+namespace misplaced_capture_default_pack {
+template <typename... Args> void Test(Args... args) {
+ [&, args...] {}; // ok
+ [args..., &] {}; // expected-error {{capture default must be first}}
+ [=, &args...] {}; // ok
+ [&, ... xs = &args] {}; // ok
+ [&, ... xs = &] {}; // expected-error {{expected expression}}
+ [... xs = &] {}; // expected-error {{expected expression}}
+ [... xs = &args, = ] {}; // expected-error {{capture default must be first}}
+ [... xs = &args, &] {}; // expected-error {{capture default must be first}}
+}
+} // namespace misplaced_capture_default_pack
Index: clang/lib/Parse/ParseExprCXX.cpp
===================================================================
--- clang/lib/Parse/ParseExprCXX.cpp
+++ clang/lib/Parse/ParseExprCXX.cpp
@@ -926,6 +926,15 @@
} else if (Tok.is(tok::kw_this)) {
Kind = LCK_This;
Loc = ConsumeToken();
+ } else if (Tok.isOneOf(tok::amp, tok::equal) &&
+ NextToken().isOneOf(tok::comma, tok::r_square) &&
+ Intro.Default == LCD_None) {
+ // We have a lone "&" or "=" which is either a misplaced capture-default
+ // or the start of a capture (in the "&" case) with the rest of the
+ // capture missing. Both are an error but a misplaced capture-default
+ // is more likely if we don't already have a capture default.
+ return Invalid(
+ [&] { Diag(Tok.getLocation(), diag::err_capture_default_first); });
} else {
TryConsumeToken(tok::ellipsis, EllipsisLocs[0]);
Index: clang/include/clang/Basic/DiagnosticParseKinds.td
===================================================================
--- clang/include/clang/Basic/DiagnosticParseKinds.td
+++ clang/include/clang/Basic/DiagnosticParseKinds.td
@@ -934,6 +934,8 @@
"the name of the capture">;
def err_lambda_capture_multiple_ellipses : Error<
"multiple ellipses in pack capture">;
+def err_capture_default_first : Error<
+ "capture default must be first">;
// C++17 lambda expressions
def err_expected_star_this_capture : Error<
"expected 'this' following '*' in lambda capture list">;
-------------- next part --------------
A non-text attachment was scrubbed...
Name: D83681.278164.patch
Type: text/x-patch
Size: 3736 bytes
Desc: not available
URL: <http://lists.llvm.org/pipermail/cfe-commits/attachments/20200715/80208836/attachment.bin>
More information about the cfe-commits
mailing list