<div dir="ltr">[adding llvm-dev for wider audience]<br><div><div class="gmail_extra"><br><div class="gmail_quote">On Fri, May 19, 2017 at 12:28 PM, Daniel Berlin <span dir="ltr"><<a href="mailto:dberlin@dberlin.org" target="_blank">dberlin@dberlin.org</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex"><div dir="ltr"><div><br></div><div class="gmail_extra"><br><div class="gmail_quote"><div><div class="m_6005733253632269409gmail-m_-8430389241588207758gmail-h5">On Fri, May 19, 2017 at 11:00 AM, Davide Italiano <span dir="ltr"><<a href="mailto:davide@freebsd.org" target="_blank">davide@freebsd.org</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex"><span>On Fri, May 19, 2017 at 10:00 AM, Sanjay Patel <<a href="mailto:spatel@rotateright.com" target="_blank">spatel@rotateright.com</a>> wrote:<br>
> Is "VRP" (value range propagation) in LLVM-speak "CVP" (correlated value<br>
> propagation)?<br>
><br>
> If so, we have this comment regarding compares:<br>
>   // As a policy choice, we choose not to waste compile time on anything<br>
> where<br>
>   // the comparison is testing local values.<br>
><br>
> Or this for div/rem:<br>
>   // As a policy choice, we choose not<br>
>   // to waste compile time on anything where the operands are local defs.<br>
><br>
> "Local" means in the same basic block from what I can tell by the code here.<br>
><br>
> I think this means that this pass explicitly defers handling simple cases<br>
> like:<br>
> <a href="https://reviews.llvm.org/D33342" rel="noreferrer" target="_blank">https://reviews.llvm.org/D3334<wbr>2</a><br>
> ...to another pass, and currently that pass is InstCombine (because the<br>
> patterns really can't be any simpler than the tests in that patch, right?).<br>
><br>
> I think that improving compile-time should not preclude improving<br>
> InstCombine. We should do both.<br>
><br>
<br>
</span>Just thoughts, feel free to ignore them.<br>
I didn't measure the impact in LLVM, but I'm sure you can do VRP<br>
relatively fast (GCC does that both interprocedurally and<br>
intraprocedurally and their pass is much faster in some cases than<br>
LLVM's), i.e. O(N) in practice, so, maybe we could re-evaluate this<br>
policy?<br></blockquote><div><br></div></div></div><div>Yeah, that's kind of silly, we can do much better.</div><span class="m_6005733253632269409gmail-m_-8430389241588207758gmail-"><div> </div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">
I think replacing a pass in LLVM is not trivial (I'm learning it the<br>
hard way with NewGVN). OTOH, I'm still not entirely convinced<br>
`InstCombine` should handle these cases given it's already a<br>
compile-time sink?<br></blockquote><div><br></div></span></div></div></div></blockquote><div><br></div><div>Correct me where I'm going wrong.<br><br></div><div>1. We got a bug report with a perf regression that was root caused to this IR:<br><br>  define i1 @xor_of_icmps(i64 %a) {<br>    %b = icmp sgt i64 %a, 0<br>    %c = icmp eq i64 %a, 1<br>    %r = xor i1 %b, %c<br>    ret i1 %r<br>  }<br><br></div><div>Because this was a regression, we know that the optimal/canonical IR for this program is:<br><br>  define i1 @xor_of_icmps_canonical(i64 %a) {<br>    %r = icmp sgt i64 %a, 1<br>     ret i1 %r<br>  } <br><br></div><div>2. I saw this as a bug in InstCombine because I think InstCombine's job is to canonicalize simple IR sequences.<br><br></div><div>3. I assumed that matching an (xor icmp, icmp) pattern can be done efficiently in InstCombine. In fact, I knew that we already did that match, so there is zero incremental cost to find this pattern in InstCombine.<br></div><div><br>4. I knew that we already handle related and-of-cmps and or-of-cmps patterns in InstSimplify/InstCombine.<br><br></div><div>5. Based on that, I have proposed a patch that mostly uses existing logic in those passes to solve this bug in the most general way I am aware of. It makes 2 calls to InstSimplify to find a potential fold before morphing/creating new instructions that may get folded by other existing logic.<br><br><br>Questions:<br></div><div>1. Was/is InstCombine intended to handle this transform?<br><br>2. If there is a measurable compile-time cost for this patch, then there must be a structural 
problem with InstSimplify, InstCombine, and/or the pass pipeline?<br></div><div><br>3. Is there another pass that is more suitable to solve this bug than InstCombine? CVP was mentioned as a candidate for enhancement. Any others?<br><br></div><div>4. If there is a more suitable pass, why is it more suitable?<br></div><div><br>5. If we add to that pass or create a new pass that specifically targets logic-of-cmps, can we pull that code out of InstSimplify and InstCombine?<br><br></div><div>6. If compile-time has become unbearable, shouldn't we be moving optimization logic and/or passes from -O1 to -O2 to -O3, etc? My understanding as a compiler consumer has always been that I'll get better performing code the higher up the -O# ladder I climb, but it will cost increasingly more time to compile. Is that not the model for LLVM?<br></div><div><br></div><div><br></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex"><div dir="ltr"><div class="gmail_extra"><div class="gmail_quote"><span class="m_6005733253632269409gmail-m_-8430389241588207758gmail-"></span><div>People seem intent on adding lots and lots of stuff to instcombine because it's easy.</div><div>This makes it harder to ever replace, while simultaneously making it slower.</div><div>It's not hard to see where that path ends up.</div><div>InstCombine does a lot of random crap that isn't even combining or graph rewrite, too, because well second and third order effects are hard.<br></div></div></div></div></blockquote><div><br></div><div>Can you provide some examples? I honestly don't know where I should draw the line. If I've crossed the line, I'd like to fix it. <br></div><div><br><br></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex"><div dir="ltr"><div class="gmail_extra"><div class="gmail_quote"><div></div><div>If we want an optimal graph rewriter, we should actually just build one, with sane rewrite rules, verification that it fixpoints, and some sense of good ordering, etc,  instead of  slowly every adding possible pattern match to InstCombine.<br></div></div></div></div></blockquote><div><br></div><div>Can you recommend papers or projects to look at to get a better understanding of an optimal graph rewriter? LLVM is the only compiler I've worked on from the inside, so I don't really know what else is possible / feasible.<br></div><div><br> </div></div></div></div></div>