<div dir="ltr">Let's consider Sanjoy's earlier example,<div><br></div><div>We start with the following program:<br></div><div><br><div><div>%x = add nuw i32 %m, %n</div><div>%y = zext i32 %x to i64</div><div>%s = lshr i64 %y, 32</div></div><div><br></div><div>If %m and %n are 2**32 - 1, then using your model: %x is entirely poison, %y's top 32-bits would be zero and bottom 32-bits are poison and %s would be enitrely zero.</div><div><br></div><div>This would mean that we couldn't transform the program into:</div><div><br></div><div><div>%m.wide = zext i32 %m to i64</div><div>%n.wide = zext i32 %n to i64</div><div>%z = add nuw i64 %m.wide, %n.wide</div><div>%s = lshr i64 %z, 32</div></div><div><br></div></div><div>Because we would have: %m.wide and %n.wide are 2**32 - 1, %z is 2**33 - 2, %s is 1</div><div><br></div><div>That being said, we don't perform this transform today and I don't see why we would want to.  I'd happily give up this flexibility if it meant that we could keep everything else.</div></div><div class="gmail_extra"><br><div class="gmail_quote">On Sun, Feb 1, 2015 at 2:44 AM, Bruce Hoult <span dir="ltr"><<a href="mailto:bruce@hoult.org" target="_blank">bruce@hoult.org</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div dir="ltr">I don't know how things work at the moment, but it seems to me that you can do lots of sensible things, and avoid lots of silly things, if you keep track of four possible values for each bit:<div><br></div><div>- undef (the default)</div><div>- poison</div><div>- known to be 0</div><div>- known to be 1</div><div><br></div><div>This makes both David's and Chandler's examples work nicely if you assume:</div><div><br></div><div>- ZEXT makes all the new bits known 0</div><div>- SEXT makes all the new bits the same as the high bit</div><div>- AND clears unknown and poison bits to known 0 if the other input is known 0</div><div>- OR sets unknown and poison bits to known 1 if the other input is known 1</div><div><br></div><div>Also things such as ZEXTing a poison i32 to i64 and then right shifting by 32 will result in all known 0 bits.</div><div><br></div></div><div class="gmail_extra"><br><div class="gmail_quote"><div><div class="h5">On Sun, Feb 1, 2015 at 11:19 PM, Chandler Carruth <span dir="ltr"><<a href="mailto:chandlerc@google.com" target="_blank">chandlerc@google.com</a>></span> wrote:<br></div></div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div><div class="h5"><div dir="ltr"><div class="gmail_extra"><div><div><br><div class="gmail_quote">On Sun, Feb 1, 2015 at 1:57 AM, David Majnemer <span dir="ltr"><<a href="mailto:david.majnemer@gmail.com" target="_blank">david.majnemer@gmail.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div><br></div><div class="gmail_quote"><div><div>On Tue, Jan 27, 2015 at 8:58 PM, Sanjoy Das <span dir="ltr"><<a href="mailto:sanjoy@playingwithpointers.com" target="_blank">sanjoy@playingwithpointers.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><span>> Ah, yes.  You are right, we cannot always assume that %y would be zero in<br>
> the second case.<br>
> This wouldn't be the first time we've lost information that we could use to<br>
> optimize a program by transforming it.<br>
><br>
> Do you think this result would be problematic?  It seems consistent with the<br>
> RFC and LLVM's current behavior.<br>
><br>
<br>
</span>The problem is not that we're losing information, the problem is that<br>
we're changing the behavior of a well-defined program.<br>
<br>
I'll try to put the whole argument in one place:<br>
<br>
We start with<br>
<br>
  %x = add nuw i32 %m, %n<br>
  %y = zext i32 %x to i64<br>
  %s = lshr i64 %y, 32<br>
  %addr = gep %some_global, %s<br>
  store i32 42, i32* %addr<br>
<br>
In the above program, for all values of %x, %s is 0.  This means the<br>
program is well-defined when %x is poison (since you don't need to<br>
look at %x to determine the value of %addr, in the same sense as you<br>
don't need to look at X to determine the value of "and X, 0"); and it<br>
stores 42 to &(%some_global)[0].  Specifically, the above program is<br>
well defined for "%m = %n = 2^32-1".<br>
<br>
Now if we do the usual transform of "zext (add nuw X Y)" => "add nuw<br>
(zext X) (zext Y)" then we get<br>
<br>
  %m.wide = zext i32 %m to i64<br>
  %n.wide = zext i32 %n to i64<br>
  %z = add nuw i64 %m.wide, %n.wide<br>
  %s = lshr i64 %y, 32<br>
  %addr = gep %some_global, %s<br>
  store i32 42, i32* %addr<br>
<br>
The new program does *not* have the same behavior as the old program<br>
for "%m = %n = 2^32-1".  We have changed the behavior of a<br>
well-defined program by doing the "zext (add nuw X Y)" => "add nuw<br>
(zext X) (zext Y)" transform.<br></blockquote><div><br></div></div></div><div>After some pondering and combing through LLVM's implementation, I think we must conclude that zexting a value with any poison bits creates poison in every new bit.</div><div><br></div><div>Considering the following program:</div><div><br></div><div>%zext = zext i32 %x to i64</div><div>%icmp = icmp i64 %zext, i64 1</div><div><br></div><div>we'd like to transform this to:</div><div><br></div><div>%icmp = icmp i32 %x, i32 1</div><div><br></div><div>Is it reasonable to say that '%icmp' in the before case is not poison but '%icmp' in the after case is poison?  LLVM assumes it can remove casts with impunity, I think this is a useful property to maintain.</div></div></blockquote></div><br></div></div>FWIW, I agree with your statement.</div><div class="gmail_extra"><br></div><div class="gmail_extra">Here is the line of reasoning that I find troubling.</div><div class="gmail_extra"><br></div><div class="gmail_extra">If we accept the above, we have a surprising result (using small bit-width integers to make it easier to read)</div><div class="gmail_extra"><br></div><div class="gmail_extra">%zext = zext i1 %x to i2</div><div class="gmail_extra">%and = and i2 %zext, 1</div><div class="gmail_extra"><br></div><div class="gmail_extra">We cannot replace %and with %zext because the %and might be removing poison.</div><div class="gmail_extra"><br></div><div class="gmail_extra">Perhaps this restriction is OK though. I just find it somewhat troubling.</div></div>
<br></div></div><span class="">_______________________________________________<br>
LLVM Developers mailing list<br>
<a href="mailto:LLVMdev@cs.uiuc.edu" target="_blank">LLVMdev@cs.uiuc.edu</a>         <a href="http://llvm.cs.uiuc.edu" target="_blank">http://llvm.cs.uiuc.edu</a><br>
<a href="http://lists.cs.uiuc.edu/mailman/listinfo/llvmdev" target="_blank">http://lists.cs.uiuc.edu/mailman/listinfo/llvmdev</a><br>
<br></span></blockquote></div><br></div>
</blockquote></div><br></div>