<div dir="ltr"><div class="gmail_extra"><div class="gmail_quote">On 5 July 2017 at 14:05, Douglas Gregor via cfe-commits <span dir="ltr"><<a href="mailto:cfe-commits@lists.llvm.org" target="_blank">cfe-commits@lists.llvm.org</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div style="word-wrap:break-word"><br><div><div><div class="h5"><blockquote type="cite"><div>On Jul 5, 2017, at 1:33 PM, Aaron Ballman <<a href="mailto:aaron@aaronballman.com" target="_blank">aaron@aaronballman.com</a>> wrote:</div><br class="m_3834550471837664634Apple-interchange-newline"><div><span 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;float:none;display:inline!important">On Wed, Jul 5, 2017 at 4:20 PM, Douglas Gregor via cfe-commits</span><br 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"><span 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;float:none;display:inline!important"><</span><a href="mailto:cfe-commits@lists.llvm.org" 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" target="_blank">cfe-commits@lists.llvm.org</a><span 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;float:none;display:inline!important">> wrote:</span><br 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"><blockquote type="cite" 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">Author: dgregor<br>Date: Wed Jul  5 13:20:15 2017<br>New Revision: 307197<br><br>URL: <a href="http://llvm.org/viewvc/llvm-project?rev=307197&view=rev" target="_blank">http://llvm.org/viewvc/llvm-<wbr>project?rev=307197&view=rev</a><br>Log:<br>Cope with Range-v3's CONCEPT_REQUIRES idiom<br><br>Modified:<br>   cfe/trunk/lib/Sema/<wbr>SemaTemplate.cpp<br>   cfe/trunk/test/<wbr>SemaTemplate/overload-<wbr>candidates.cpp<br><br>Modified: cfe/trunk/lib/Sema/<wbr>SemaTemplate.cpp<br>URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/lib/Sema/SemaTemplate.cpp?rev=307197&r1=307196&r2=307197&view=diff" target="_blank">http://llvm.org/viewvc/llvm-<wbr>project/cfe/trunk/lib/Sema/<wbr>SemaTemplate.cpp?rev=307197&<wbr>r1=307196&r2=307197&view=diff</a><br>==============================<wbr>==============================<wbr>==================<br>--- cfe/trunk/lib/Sema/<wbr>SemaTemplate.cpp (original)<br>+++ cfe/trunk/lib/Sema/<wbr>SemaTemplate.cpp Wed Jul  5 13:20:15 2017<br>@@ -2831,10 +2831,44 @@ static void collectConjunctionTerms(Expr<br>  Terms.push_back(Clause);<br>}<br><br>+// The ranges-v3 library uses an odd pattern of a top-level "||" with<br>+// a left-hand side that is value-dependent but never true. Identify<br>+// the idiom and ignore that term.<br>+static Expr *lookThroughRangesV3Condition(<wbr>Preprocessor &PP, Expr *Cond) {<br>+  // Top-level '||'.<br>+  auto *BinOp = dyn_cast<BinaryOperator>(Cond-<wbr>>IgnoreParenImpCasts());<br>+  if (!BinOp) return Cond;<br>+<br>+  if (BinOp->getOpcode() != BO_LOr) return Cond;<br>+<br>+  // With an inner '==' that has a literal on the right-hand side.<br>+  Expr *LHS = BinOp->getLHS();<br>+  auto InnerBinOp = dyn_cast<BinaryOperator>(LHS-><wbr>IgnoreParenImpCasts());<br></blockquote><br 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"><span 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;float:none;display:inline!important">auto * please (or better yet, const auto * here and above).</span><br 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"></div></blockquote><div><br></div></div></div>Sure, commit incoming.</div><div><span class=""><br><blockquote type="cite"><div><br 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"><blockquote type="cite" 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">+  if (!InnerBinOp) return Cond;<br>+<br>+  if (InnerBinOp->getOpcode() != BO_EQ ||<br>+      !isa<IntegerLiteral>(<wbr>InnerBinOp->getRHS()))<br>+    return Cond;<br>+<br>+  // If the inner binary operation came from a macro expansion named<br>+  // CONCEPT_REQUIRES or CONCEPT_REQUIRES_, return the right-hand side<br>+  // of the '||', which is the real, user-provided condition.<br>+  auto Loc = InnerBinOp->getExprLoc();<br></blockquote><br 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"><span 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;float:none;display:inline!important">Please do not use auto here.</span><br 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"></div></blockquote><div><br></div></span>I’m fine with spelling it out, although I don’t know what criteria you’re applying here. Presumably it’s okay to use “auto” when the initializer is some form of cast to that type, but you prefer not to use “auto” elsewhere, despite the “Loc” in the variable name and in the initializer?</div><div><span class=""><br><blockquote type="cite"><div><span 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;float:none;display:inline!important">It's unfortunate that we have this special case instead of a more</span><br 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"><span 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;float:none;display:inline!important">general mechanism. While I love Ranges v3, I'm not keen on a compiler</span><br 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"><span 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;float:none;display:inline!important">hack just for it. These "tricks" have a habit of leaking out into</span><br 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"><span 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;float:none;display:inline!important">other user code, which will behave differently than what happens with</span></div></blockquote><blockquote type="cite"><div><span 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;float:none;display:inline!important">Ranges.</span><br 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"></div></blockquote><div><br></div></span><div>It’s possible that we can do something more general here by converting the expression to disjunctive normal form and identifying when some of the top-level terms will never succeed. </div></div></div></blockquote><div><br></div><div>Hmm. Ranges v3 produces things like:</div><div><br></div><div>template<int N = 42, enable_if<N == 43 || some_condition>></div><div><br></div><div>... for which the top-level "N == 43" check can succeed if N is explicitly specified. Perhaps the trick we need here is to notice that N was not, in fact, explicitly specified, and so a requirement on N is not likely to be interesting.</div><div><br></div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div style="word-wrap:break-word"><div><span class="m_3834550471837664634Apple-tab-span" style="white-space:pre-wrap">        </span>- Doug</div><div><div class="h5"><div><br><blockquote type="cite"><div><span 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;float:none;display:inline!important">~Aaron</span><br 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"><br 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"><blockquote type="cite" 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">+  if (!Loc.isMacroID()) return Cond;<br>+<br>+  StringRef MacroName = PP.getImmediateMacroName(Loc);<br>+  if (MacroName == "CONCEPT_REQUIRES" || MacroName == "CONCEPT_REQUIRES_")<br>+    return BinOp->getRHS();<br>+<br>+  return Cond;<br>+}<br>+<br>/// Find the failed subexpression within enable_if, and describe it<br>/// with a string.<br>static std::pair<Expr *, std::string><br>findFailedEnableIfCondition(<wbr>Sema &S, Expr *Cond) {<br>+  Cond = lookThroughRangesV3Condition(<wbr>S.PP, Cond);<br>+<br>  // Separate out all of the terms in a conjunction.<br>  SmallVector<Expr *, 4> Terms;<br>  collectConjunctionTerms(<wbr>Cond, Terms);<br><br>Modified: cfe/trunk/test/SemaTemplate/<wbr>overload-candidates.cpp<br>URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/test/SemaTemplate/overload-candidates.cpp?rev=307197&r1=307196&r2=307197&view=diff" target="_blank">http://llvm.org/viewvc/llvm-<wbr>project/cfe/trunk/test/<wbr>SemaTemplate/overload-<wbr>candidates.cpp?rev=307197&r1=<wbr>307196&r2=307197&view=diff</a><br>==============================<wbr>==============================<wbr>==================<br>--- cfe/trunk/test/SemaTemplate/<wbr>overload-candidates.cpp (original)<br>+++ cfe/trunk/test/SemaTemplate/<wbr>overload-candidates.cpp Wed Jul  5 13:20:15 2017<br>@@ -137,4 +137,30 @@ namespace PR15673 {<br>#endif<br>  void wibble() {}<br>  void wobble() { wibble<int>(); } // expected-error {{no matching function for call to 'wibble'}}<br>+<br>+  template<typename T><br>+  struct some_passing_trait : std::true_type {};<br>+<br>+#if __cplusplus <= 199711L<br>+  // expected-warning@+4 {{default template arguments for a function template are a C++11 extension}}<br>+  // expected-warning@+4 {{default template arguments for a function template are a C++11 extension}}<br>+#endif<br>+  template<typename T,<br>+           int n = 42,<br>+           typename std::enable_if<n == 43 || (some_passing_trait<T>::value && some_trait<T>::value), int>::type = 0><br>+  void almost_rangesv3(); // expected-note{{candidate template ignored: requirement '42 == 43 || (some_passing_trait<int>::<wbr>value && some_trait<int>::value)' was not satisfied}}<br>+  void test_almost_rangesv3() { almost_rangesv3<int>(); } // expected-error{{no matching function for call to 'almost_rangesv3'}}<br>+<br>+  #define CONCEPT_REQUIRES_(...)                               <wbr>         \<br>+    int x = 42,                               <wbr>                          \<br>+    typename std::enable_if<(x == 43) || (__VA_ARGS__)>::type = 0<br>+<br>+#if __cplusplus <= 199711L<br>+  // expected-warning@+4 {{default template arguments for a function template are a C++11 extension}}<br>+  // expected-warning@+4 {{default template arguments for a function template are a C++11 extension}}<br>+#endif<br>+  template<typename T,<br>+           CONCEPT_REQUIRES_(<wbr>some_passing_trait<T>::value && some_trait<T>::value)><br>+  void rangesv3(); // expected-note{{candidate template ignored: requirement 'some_trait<int>::value' was not satisfied [with T = int, x = 42]}}<br>+  void test_rangesv3() { rangesv3<int>(); } // expected-error{{no matching function for call to 'rangesv3'}}<br>}<br><br><br>______________________________<wbr>_________________<br>cfe-commits mailing list<br><a href="mailto:cfe-commits@lists.llvm.org" target="_blank">cfe-commits@lists.llvm.org</a><br><a href="http://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits" target="_blank">http://lists.llvm.org/cgi-bin/<wbr>mailman/listinfo/cfe-commits</a></blockquote></div></blockquote></div><br></div></div></div><br>______________________________<wbr>_________________<br>
cfe-commits mailing list<br>
<a href="mailto:cfe-commits@lists.llvm.org">cfe-commits@lists.llvm.org</a><br>
<a href="http://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits" rel="noreferrer" target="_blank">http://lists.llvm.org/cgi-bin/<wbr>mailman/listinfo/cfe-commits</a><br>
<br></blockquote></div><br></div></div>