<html><body><div style="font-family: arial, helvetica, sans-serif; font-size: 12pt; color: #000000"><div><br></div><div><br></div><hr id="zwchr" data-marker="__DIVIDER__"><div data-marker="__HEADERS__"><blockquote style="border-left:2px solid #1010FF;margin-left:5px;padding-left:5px;color:#000;font-weight:normal;font-style:normal;text-decoration:none;font-family:Helvetica,Arial,sans-serif;font-size:12pt;"><b>From: </b>"-" <liangchenblue@gmail.com><br><b>To: </b>"Remi Forax" <forax@univ-mlv.fr><br><b>Cc: </b>"valhalla-dev" <valhalla-dev@openjdk.org><br><b>Sent: </b>Saturday, October 21, 2023 11:10:07 AM<br><b>Subject: </b>Re: Ensuring atomicity of long values<br></blockquote></div><div data-marker="__QUOTED_TEXT__"><blockquote style="border-left:2px solid #1010FF;margin-left:5px;padding-left:5px;color:#000;font-weight:normal;font-style:normal;text-decoration:none;font-family:Helvetica,Arial,sans-serif;font-size:12pt;"><div dir="ltr"><div>Thank you for your quick response!</div><br><div class="gmail_quote"><div dir="ltr" class="gmail_attr">On Sat, Oct 21, 2023 at 4:46 PM Remi Forax <<a href="mailto:forax@univ-mlv.fr" target="_blank">forax@univ-mlv.fr</a>> wrote:<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><div style="font-family:arial,helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)"><br><div><div>By default, a null restricted value class does not allow tearing by default so wrapping a long should be enough (I hope).<br></div></div></div></div></blockquote><div>News of relief.</div></div></div></blockquote><div><br></div><div>:)<br data-mce-bogus="1"></div><div><br data-mce-bogus="1"></div><blockquote style="border-left:2px solid #1010FF;margin-left:5px;padding-left:5px;color:#000;font-weight:normal;font-style:normal;text-decoration:none;font-family:Helvetica,Arial,sans-serif;font-size:12pt;"><div dir="ltr"><div class="gmail_quote"><br><div>On a side note, if a non-atomic class allows tearing, then Long and Double wrappers cannot be migrated to be non-atomic (as you imply in your recent "Not yet again!" against LooselyConsistentValue interface):</div><div>Compare a non-final long field and a non-final Long field. The long field can be read with tearing, but Long, an object reference, cannot. Say we write 1L<<32 and 1 to the field from 2 threads racily, then long read can yield 0, 1, 1L<<32, 1L<<32 | 1, but Long read can only yield null, 1, 1L<<32 (as the final long field is published and visible when the Long object is visible)</div></div></div></blockquote><div><br></div><div>Long will never tear, but Long! (note, there is a bang here) will tear. Long! and long have exactly the same semantics with respect to tearing. <br data-mce-bogus="1"></div><div>From the spec POV, if you want flattenes on heap you need both a value class with an implicit constructor AND using ! at use site (where the field is declared). <br data-mce-bogus="1"></div><div>After, a specific VM can do heroics and discover that the actual CPU supports atomic 128 bits read/write using vector instructions and decide to store a Long (no bang) as a 128 bits values, 64 for the value and 1 for null or not. But this is a tradeoff, using more memory may backfire.<br data-mce-bogus="1"></div><div><br data-mce-bogus="1"></div><blockquote style="border-left:2px solid #1010FF;margin-left:5px;padding-left:5px;color:#000;font-weight:normal;font-style:normal;text-decoration:none;font-family:Helvetica,Arial,sans-serif;font-size:12pt;"><div dir="ltr"><div class="gmail_quote"><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex"><div><div style="font-family:arial,helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)"><div><br><div> class LongWrapper {<br></div><div> long value;<br></div><div> implicit LongWrapper();</div><div> LongWrapper(long value) { this.value = value; }<br></div><div> }</div><br><div>class Foo {<br></div><div> LongWrapper! w; // this is a non-tearing long<br></div><div>}<br></div><br><div>From the spec POV, there is no garantee that an instance will not be a pointer instead of the direct value but given that Java only works on platforms (even 32 bits) that are able to write 64 bits atomically (otherwise you can not implement the VarHandle API), practically, it should be ok.</div></div></div></div></blockquote></div></div></blockquote><div><br></div><blockquote style="border-left:2px solid #1010FF;margin-left:5px;padding-left:5px;color:#000;font-weight:normal;font-style:normal;text-decoration:none;font-family:Helvetica,Arial,sans-serif;font-size:12pt;"><div dir="ltr"><div class="gmail_quote"><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex"><div><div style="font-family:arial,helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)"><div><div><br></div></div></div></div></blockquote><div>Agreed. Even if VarHandle and 64-bit atomic read/writes aren't used, JVM can always fall back to traditional final field handling and heap object layouts, which is still functionally correct.</div></div></div></blockquote><div><br></div><div>Yes.<br data-mce-bogus="1"></div><div>In Java, a developer does not control where his code run. So the semantics has to be the same for a wide range of hardware.<br data-mce-bogus="1"></div><div><br data-mce-bogus="1"></div><div>regards,</div><div>Rémi<br data-mce-bogus="1"></div><div><br data-mce-bogus="1"></div></div></div></body></html>