<html><head><meta http-equiv="Content-Type" content="text/html; charset=utf-8"></head><body style="word-wrap: break-word; -webkit-nbsp-mode: space; line-break: after-white-space;" class=""><br class=""><div><br class=""><blockquote type="cite" class=""><div class="">On Oct 11, 2017, at 16:02, Richard Smith <<a href="mailto:richard@metafoo.co.uk" class="">richard@metafoo.co.uk</a>> wrote:</div><br class="Apple-interchange-newline"><div class=""><div dir="ltr" style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><div class="gmail_extra"><div class="gmail_quote">On 22 September 2017 at 18:00, Volodymyr Sapsai via cfe-commits<span class="Apple-converted-space"> </span><span dir="ltr" class=""><<a href="mailto:cfe-commits@lists.llvm.org" target="_blank" class="">cfe-commits@lists.llvm.org</a>></span><span class="Apple-converted-space"> </span>wrote:<br class=""><blockquote class="gmail_quote" style="margin: 0px 0px 0px 0.8ex; border-left-width: 1px; border-left-style: solid; border-left-color: rgb(204, 204, 204); padding-left: 1ex;"><div style="word-wrap: break-word; line-break: after-white-space;" class=""><br class=""><div class=""><div class=""><div class="h5"><br class=""><blockquote type="cite" class=""><div class="">On Sep 21, 2017, at 15:17, Richard Smith <<a href="mailto:richard@metafoo.co.uk" target="_blank" class="">richard@metafoo.co.uk</a>> wrote:</div><br class="m_-443097055292461227Apple-interchange-newline"><div class=""><div dir="ltr" style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; word-spacing: 0px;" class=""><div class="gmail_extra"><div class="gmail_quote">On 15 September 2017 at 12:51, Volodymyr Sapsai via cfe-commits<span class="m_-443097055292461227Apple-converted-space"> </span><span dir="ltr" class=""><<a href="mailto:cfe-commits@lists.llvm.org" target="_blank" class="">cfe-commits@<wbr class="">lists.llvm.org</a>></span><span class="m_-443097055292461227Apple-converted-space"> </span>wrote:<br class=""></div><div class="gmail_quote"><blockquote class="gmail_quote" style="margin: 0px 0px 0px 0.8ex; border-left-width: 1px; border-left-style: solid; border-left-color: rgb(204, 204, 204); padding-left: 1ex;">Author: vsapsai<br class="">Date: Fri Sep 15 12:51:42 2017<br class="">New Revision: 313386<br class=""><br class="">URL:<span class="m_-443097055292461227Apple-converted-space"> </span><a href="http://llvm.org/viewvc/llvm-project?rev=313386&view=rev" rel="noreferrer" target="_blank" class="">http://llvm.org/viewvc/<wbr class="">llvm-project?rev=313386&view=<wbr class="">rev</a><br class="">Log:<br class="">[Sema] Error out early for tags defined inside an enumeration.<br class=""><br class="">This fixes PR28903 by avoiding access check for inner enum constant. We<br class="">are performing access check because one enum constant references another<br class="">and because enum is defined in CXXRecordDecl. But access check doesn't<br class="">work because FindDeclaringClass doesn't expect more than one EnumDecl<br class="">and because inner enum has access AS_none due to not being an immediate<br class="">child of a record.<br class=""><br class="">The change detects an enum is defined in wrong place and allows to skip<br class="">parsing its body. Access check is skipped together with body parsing.<br class="">There was no crash in C, added test case to cover the new error.<br class=""><br class=""><a class="">rdar://problem/28530809</a><br class=""><br class="">Reviewers: rnk, doug.gregor, rsmith<br class=""><br class="">Reviewed By: doug.gregor<br class=""><br class="">Subscribers: cfe-commits<br class=""><br class="">Differential Revision:<span class="m_-443097055292461227Apple-converted-space"> </span><a href="https://reviews.llvm.org/D37089" rel="noreferrer" target="_blank" class="">https://reviews.<wbr class="">llvm.org/D37089</a><br class=""><br class=""><br class="">Modified:<br class=""> <span class="m_-443097055292461227Apple-converted-space"> </span>cfe/trunk/include/clang/<wbr class="">Basic/DiagnosticSemaKinds.td<br class=""> <span class="m_-443097055292461227Apple-converted-space"> </span>cfe/trunk/lib/Sema/SemaDecl.<wbr class="">cpp<br class=""> <span class="m_-443097055292461227Apple-converted-space"> </span>cfe/trunk/test/Sema/enum.c<br class=""> <span class="m_-443097055292461227Apple-converted-space"> </span>cfe/trunk/test/SemaCXX/enum.<wbr class="">cpp<br class=""><br class="">Modified: cfe/trunk/include/clang/Basic/<wbr class="">DiagnosticSemaKinds.td<br class="">URL:<span class="m_-443097055292461227Apple-converted-space"> </span><a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/include/clang/Basic/DiagnosticSemaKinds.td?rev=313386&r1=313385&r2=313386&view=diff" rel="noreferrer" target="_blank" class="">http://llvm.org/viewvc/<wbr class="">llvm-project/cfe/trunk/<wbr class="">include/clang/Basic/Diagnostic<wbr class="">SemaKinds.td?rev=313386&r1=<wbr class="">313385&r2=313386&view=diff</a><br class="">==============================<wbr class="">==============================<wbr class="">==================<br class="">--- cfe/trunk/include/clang/Basic/<wbr class="">DiagnosticSemaKinds.td (original)<br class="">+++ cfe/trunk/include/clang/Basic/<wbr class="">DiagnosticSemaKinds.td Fri Sep 15 12:51:42 2017<br class="">@@ -1335,6 +1335,8 @@ def err_type_defined_in_alias_temp<wbr class="">late :<br class=""> "%0 cannot be defined in a type alias template">;<br class=""> def err_type_defined_in_condition : Error<<br class=""> "%0 cannot be defined in a condition">;<br class="">+def err_type_defined_in_enum : Error<<br class="">+ "%0 cannot be defined in an enumeration">;<br class=""><br class=""> def note_pure_virtual_function : Note<<br class=""> "unimplemented pure virtual method %0 in %1">;<br class=""><br class="">Modified: cfe/trunk/lib/Sema/SemaDecl.cp<wbr class="">p<br class="">URL:<span class="m_-443097055292461227Apple-converted-space"> </span><a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/lib/Sema/SemaDecl.cpp?rev=313386&r1=313385&r2=313386&view=diff" rel="noreferrer" target="_blank" class="">http://llvm.org/viewvc/<wbr class="">llvm-project/cfe/trunk/lib/<wbr class="">Sema/SemaDecl.cpp?rev=313386&<wbr class="">r1=313385&r2=313386&view=diff</a><br class="">==============================<wbr class="">==============================<wbr class="">==================<br class="">--- cfe/trunk/lib/Sema/SemaDecl.cp<wbr class="">p (original)<br class="">+++ cfe/trunk/lib/Sema/SemaDecl.cp<wbr class="">p Fri Sep 15 12:51:42 2017<br class="">@@ -13928,6 +13928,12 @@ CreateNewDecl:<br class=""> Invalid = true;<br class=""> }<br class=""><br class="">+ if (!Invalid && TUK == TUK_Definition && DC->getDeclKind() == Decl::Enum) {<br class="">+ Diag(New->getLocation(), diag::err_type_defined_in_enum<wbr class="">)<br class="">+ << Context.getTagDeclType(New);<br class="">+ Invalid = true;<br class="">+ }<br class=""></blockquote><div class=""><br class=""></div><div class="">This looks like the wrong fix. As noted elsewhere, this is wrong in C. And in C++, the relevant context is a type-specifier, which should be rejected due to the check 7 lines above.</div><div class=""><br class=""></div><div class="">It looks like the actual bug is that we don't consider the type within a C99 compound literal to be a type-specifier. The fact that the context is an enumeration is irrelevant.</div></div></div></div></div></blockquote><br class=""></div></div><div class=""><div class="">At which point can we detect IsTypeSpecifier should be true? Which in turn boils down to DeclSpecContext should be DSC_type_specifier. Currently we have DeclSpecContext DSC_normal because it is a default argument in Parser::<wbr class="">ParseSpecifierQualifierList. Which is called from</div><div class=""><br class=""></div><div class="">#4<span class="m_-443097055292461227Apple-tab-span" style="white-space: pre-wrap;"> </span>clang::Parser::<wbr class="">ParseParenExpression(clang::<wbr class="">Parser::ParenParseOption&, bool, bool, clang::OpaquePtr<clang::<wbr class="">QualType>&, clang::SourceLocation&) at llvm-project/clang/lib/Parse/<wbr class="">ParseExpr.cpp:2375</div></div></div></div></blockquote><div class=""><br class=""></div><div class="">The call to ParseSpecifierQualfiierList here should always pass DSC_type_specifier. We're parsing a type within parentheses (which we've already disambiguated as being a type cast / compound literal rather than an expression), which is the DSC_type_specifier case.</div></div></div></div></div></blockquote>Thanks, Richard, that works. As the result in clang/test/CXX/drs/dr5xx.cpp for</div><div><br class=""></div><div> sizeof(const);</div><div><br class=""></div><div>We change error from "requires a type specifier” to “expected a type” which seems OK to me (we use it for other dr539 cases). Will send patch for code review soon.</div><div><br class=""><blockquote type="cite" class=""><div dir="ltr" style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; word-spacing: 0px; -webkit-text-stroke-width: 0px;" class=""><div class="gmail_extra"><div class="gmail_quote"><blockquote class="gmail_quote" style="margin: 0px 0px 0px 0.8ex; border-left-width: 1px; border-left-style: solid; border-left-color: rgb(204, 204, 204); padding-left: 1ex;"><div style="word-wrap: break-word; line-break: after-white-space;" class=""><div class=""><div class=""><div class="">#5<span class="m_-443097055292461227Apple-tab-span" style="white-space: pre-wrap;"> </span>clang::Parser::<wbr class="">ParseCastExpression(bool, bool, bool&, clang::Parser::TypeCastState, bool) at llvm-project/clang/lib/Parse/<wbr class="">ParseExpr.cpp:768</div><div class="">#6<span class="m_-443097055292461227Apple-tab-span" style="white-space: pre-wrap;"> </span>clang::Parser::<wbr class="">ParseCastExpression(bool, bool, clang::Parser::TypeCastState, bool) at llvm-project/clang/lib/Parse/<wbr class="">ParseExpr.cpp:521</div><div class="">#7<span class="m_-443097055292461227Apple-tab-span" style="white-space: pre-wrap;"> </span>clang::Parser::<wbr class="">ParseConstantExpressionInExprE<wbr class="">valContext(clang::Parser::<wbr class="">TypeCastState) at llvm-project/clang/lib/Parse/<wbr class="">ParseExpr.cpp:201</div><div class=""><br class=""></div><div class="">I have considered using TypeCastState for setting DeclSpecContext but its value is NotTypeCast because Parser::ParseEnumBody calls ParseConstantExpression with default argument. And it looks correct as parsing enum body doesn't imply presence of a type cast.</div><div class=""><br class=""></div><div class="">I was struggling to find a good indication we are parsing type specifier and the best option seems to be ParseCastExpression because it expects a type. But it is too broad and likely to cause false positives. In quick prototype it didn't work so I didn't pursue it further. Do you think it is possible to tell we are in type specifier based on tokens we parsed? Specifically "(" seems to be significant as without it parsing goes in different direction.</div></div><div class=""><div class="h5"><br class=""><blockquote type="cite" class=""><div class=""><div dir="ltr" style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; word-spacing: 0px;" class=""><div class="gmail_extra"><div class="gmail_quote"><div class=""> </div><blockquote class="gmail_quote" style="margin: 0px 0px 0px 0.8ex; border-left-width: 1px; border-left-style: solid; border-left-color: rgb(204, 204, 204); padding-left: 1ex;">+<br class=""> // Maybe add qualifier info.<br class=""> if (SS.isNotEmpty()) {<br class=""> if (SS.isSet()) {<br class=""><br class="">Modified: cfe/trunk/test/Sema/enum.c<br class="">URL:<span class="m_-443097055292461227Apple-converted-space"> </span><a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/test/Sema/enum.c?rev=313386&r1=313385&r2=313386&view=diff" rel="noreferrer" target="_blank" class="">http://llvm.org/viewvc/<wbr class="">llvm-project/cfe/trunk/test/<wbr class="">Sema/enum.c?rev=313386&r1=<wbr class="">313385&r2=313386&view=diff</a><br class="">==============================<wbr class="">==============================<wbr class="">==================<br class="">--- cfe/trunk/test/Sema/enum.c (original)<br class="">+++ cfe/trunk/test/Sema/enum.c Fri Sep 15 12:51:42 2017<br class="">@@ -123,3 +123,14 @@ int NegativeShortTest[NegativeShor<wbr class="">t == -<br class=""> // PR24610<br class=""> enum Color { Red, Green, Blue }; // expected-note{{previous use is here}}<br class=""> typedef struct Color NewColor; // expected-error {{use of 'Color' with tag type that does not match previous declaration}}<br class="">+<br class="">+// PR28903<br class="">+struct PR28903 {<br class="">+ enum {<br class="">+ PR28903_A = (enum { // expected-error-re {{'enum PR28903::(anonymous at {{.*}})' cannot be defined in an enumeration}}<br class="">+ PR28903_B,<br class="">+ PR28903_C = PR28903_B<br class="">+ })0<br class="">+ };<br class="">+ int makeStructNonEmpty;<br class="">+};<br class=""><br class="">Modified: cfe/trunk/test/SemaCXX/enum.cp<wbr class="">p<br class="">URL:<span class="m_-443097055292461227Apple-converted-space"> </span><a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/test/SemaCXX/enum.cpp?rev=313386&r1=313385&r2=313386&view=diff" rel="noreferrer" target="_blank" class="">http://llvm.org/viewvc/<wbr class="">llvm-project/cfe/trunk/test/Se<wbr class="">maCXX/enum.cpp?rev=313386&r1=<wbr class="">313385&r2=313386&view=diff</a><br class="">==============================<wbr class="">==============================<wbr class="">==================<br class="">--- cfe/trunk/test/SemaCXX/enum.cp<wbr class="">p (original)<br class="">+++ cfe/trunk/test/SemaCXX/enum.cp<wbr class="">p Fri Sep 15 12:51:42 2017<br class="">@@ -110,3 +110,13 @@ enum { overflow = 123456 * 234567 };<br class=""> // expected-warning@-2 {{not an integral constant expression}}<br class=""> // expected-note@-3 {{value 28958703552 is outside the range of representable values}}<br class=""> #endif<br class="">+<br class="">+// PR28903<br class="">+struct PR28903 {<br class="">+ enum {<br class="">+ PR28903_A = (enum { // expected-error-re {{'PR28903::(anonymous enum at {{.*}})' cannot be defined in an enumeration}}<br class="">+ PR28903_B,<br class="">+ PR28903_C = PR28903_B<br class="">+ })<br class="">+ };<br class="">+};<br class=""><br class=""><br class="">______________________________<wbr class="">_________________<br class="">cfe-commits mailing list<br class=""><a href="mailto:cfe-commits@lists.llvm.org" target="_blank" class="">cfe-commits@lists.llvm.org</a><br class=""><a href="http://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits" rel="noreferrer" target="_blank" class="">http://lists.llvm.org/cgi-bin/<wbr class="">mailman/listinfo/cfe-commits</a></blockquote></div></div></div></div></blockquote></div></div></div><br class=""></div><br class="">______________________________<wbr class="">_________________<br class="">cfe-commits mailing list<br class=""><a href="mailto:cfe-commits@lists.llvm.org" class="">cfe-commits@lists.llvm.org</a><br class=""><a href="http://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits" rel="noreferrer" target="_blank" class="">http://lists.llvm.org/cgi-bin/<wbr class="">mailman/listinfo/cfe-commits</a></blockquote></div></div></div></blockquote></div><br class=""></body></html>