<div dir="ltr"><div class="gmail_extra"><div class="gmail_quote">2014-02-03 Andrei Alexandrescu <span dir="ltr"><<a href="mailto:SeeWebsiteForEmail@erdani.org" target="_blank">SeeWebsiteForEmail@erdani.org</a>></span>:<blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-color:rgb(204,204,204);border-left-style:solid;padding-left:1ex">
Here are a few questions and comments:<br></blockquote><div><br></div><div>First of all, thanks for your reviewing! </div><div><br></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-color:rgb(204,204,204);border-left-style:solid;padding-left:1ex">
* Why is there an immutable postblit? I wonder if there are cases in which that's needed. Most of the time it's fine to just memcpy() the data - it's immutable anyway so nothing will change! In fact there is one use case for immutable postblit, and that's refcounting (when copying data the reference count must be incremented). But that would be increment via a pointer, which is not allowed in your proposal. (I think it is ok for the postblit proposal to not cover refcounting - we can make it built in the language.)<br>
</blockquote><div><br></div><div>immutable postblit "this(this) immutable" is necessary to support immutable(T) to immutable(T) copy.</div><div>(It's impossible by "this(this)".)</div><div><br></div>
<div><div>In immutable postblit body, modifying referenced data is not allowed. But, rebinding references is still allowed</div><div>because first field assignment is treated as field initializing.</div><div><br></div><div>
struct S {</div><div> int[] arr;</div><div> this(this) immutable {</div><div> //arr[0] = 10; // NG, but...</div><div> arr = [1,2,3]; // OK so this is 'arr' field initializing.</div><div> }</div>
<div>}</div></div><div> </div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-color:rgb(204,204,204);border-left-style:solid;padding-left:1ex">
* The inout case looks good, but again we need to have some good examples on when it would be necessary.<br></blockquote><div><br></div><div>If mutable psotblit and immutabe postblit have same code, you can merge two into one inout postblit.</div>
<div><br></div><div>struct S {</div><div> int[] arr;</div><div> this(this) { arr = arr.dup; }</div><div> this(this) immutable { arr = arr.idup; }</div><div><br></div><div> // you can merge above two postblits into:</div>
<div> this(this) inout { arr = arr.dup; }</div><div> // arr.dup makes unique data, so implicitly conversion to inout(int[]) is allowed.</div><div>}</div><div><br></div><div> </div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-color:rgb(204,204,204);border-left-style:solid;padding-left:1ex">
* What happens if an object defines both the mutable and the inout version of postblit? I assume the mutable has priority when duplicating mutable object. The DIP should clarify that.<br></blockquote><div><br></div><div>
It is clarified in "Overloading of qualified postblits" section.</div><div><br></div><div>> If immutable postblit is defined, it is alwasy used for the copies:</div><div>> - immutable to const</div><div>> - immutable to immutable</div>
<div>></div><div>> These priority order is defined based on the following rule:</div><div>> - If source is mutable or immutable, most specialized postblits (mutable/immutable postblit) will be used, if they exists.</div>
<div><br></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-color:rgb(204,204,204);border-left-style:solid;padding-left:1ex">
* Const postblit is spelled "this(this) const" but that's misleading because it's really "this(whatever1 this) whatever2" with whatever1 and whatever2 being arbitrary qualifiers. We should probably find a more descriptive syntax for that. "this(this) auto" comes to mind. We can even add a contextual keyword such that the compiler recognizes "this(this) unique" without otherwise giving "unique" any special meaning.<br>
</blockquote><div><br></div><div>I have deliberately avoided "contextual keyword" because it does not currently exist in D grammar.</div><div>But unfortunately, using existing keyword for the new meaning seems to cause just confusion.</div>
<div><br></div><div>I still don't have a strong opinion for the syntax ...</div><div> </div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-color:rgb(204,204,204);border-left-style:solid;padding-left:1ex">
* My main beef is with this constructor. I'll refer to it henceforth as "the unique constructor".<br>
<br>
* Qualifiers obey a hierarchy, i.e. const(T) is a supertype of both T and immutable(T) as shown below with ASCII art.<br>
<br>
const(T)<br>
/\<br>
/ \<br>
T immutable(T)<br>
<br>
That imparts structure over the mixed-qualifier constructors. However, that structure is missing from the unique constructor; all mixed-qualifier constructors are heaped into one. This makes the rules forced for certain constructors (e.g. constructing a const(T) from a T should be much less restricted than constructing a T from a const(T)). The proposed unique constructor is not sensitive to such distinctions.<br>
</blockquote><div><br></div><div>??? An unique constructor is always less specialized than any other constructors (mutable, immutable, and inout).</div><div>Specialization order is:</div><div><br></div><div> mutable == immutable > inout > unique postblit</div>
<div><br></div><div>So, by overloading postblits, you can control copy cost as you expect.</div><div>For example:</div><div><br></div><div>struct Array(T) {</div><div> T[] data;</div><div> this(this) unique { data = data.dup; }</div>
<div>}</div><div><br></div><div>Array!T allows copying object between arbitrary qualifiers. But it also needs data duplication.</div><div><br></div><div>If you want to share data in possible case, overload postblit as follows.</div>
<div><br></div><div>struct Array(T) {</div><div> T[] data;</div><div> this(this) inout { /* nothing to do.*/; }</div><div> this(this) unique { data = data.dup; }</div><div>}</div><div><br></div><div>If the copy target has weak or equal qualifier than the copy source, inout postblit is used, and it will always copy object by just bit blit (==memcpy).</div>
<div><br></div><div>If you need to avoid data sharing between mutable objects, you can add mutable postblit for that.</div><div><br></div><div>struct Array(T) {</div><div> T[] data;</div><div> this(this) { data = data.dup; }</div>
<div> this(this) inout { /* nothing to do.*/; }</div><div> this(this) unique { data = data.dup; }</div><div>}</div><div> </div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-color:rgb(204,204,204);border-left-style:solid;padding-left:1ex">
* The "unique expression" definition is quite strong and we should refine it and reuse in other contexts as well.<br></blockquote><div><br></div><div>OK, I'll separate the concept to other DIP.</div><div> </div>
<blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-color:rgb(204,204,204);border-left-style:solid;padding-left:1ex">
* The section "Overloading of qualified postblits" is great because it brings back the subtyping structure! It says: "If mutable postblit is defined, it is alwasy used for the copies: (a) mutable to mutable; (b) mutable to const". Indeed that is correct because mutable to const is a copy to a supertype. To respect subtyping, we should also add (c) immutable to const; (d) const to const; (e) immutable to immutable. That way we have a simple rule: "this(this) is invoked for all upcasts obtained by qualifiers". Wonderful!<br>
</blockquote><div><br></div><div>Yes. to support both simple case and complicated case, the rule is defined.</div><div> </div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-color:rgb(204,204,204);border-left-style:solid;padding-left:1ex">
* If we go forward with the idea above, we only need to treat the remaining cases of downcasts and cross-casts across the subtyping hierarchy:<br>
<br>
(a) downcasts: const(T) -> T and const(T) -> immutable(T)<br>
(b) cross-casts: immutable(T) -> T and T -> immutable(T)<br></blockquote><div><br></div><div><div>There is unique postblit to support them exactly what.</div></div><div><br></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-color:rgb(204,204,204);border-left-style:solid;padding-left:1ex">
* The use of subtyping above would replace the elaborate rules in section "Overloading of qualified postblits". In fact they seem to agree 95% of the time.</blockquote><div><br></div><div>The purpose of the section is to describe priority between postblits. It is simple:</div>
<div><div><br></div><div> (mutable == immutable) > inout > unique postblit</div><div><br></div></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-color:rgb(204,204,204);border-left-style:solid;padding-left:1ex">
By the way there are some confusing negations e.g. "if mutable postblit is not defined, it will be used for the copies". I assume the "not" should be removed?<br></blockquote><div><br></div><div>inout postblit is less specialized than (mutable|immutable) postblits. So "not" is necessary.</div>
<div> </div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-color:rgb(204,204,204);border-left-style:solid;padding-left:1ex">
* So we're left with the following postblitting rules as the maximum:<br>
<br>
struct T {<br>
this(this); // all upcasts including identity<br>
this(const this); // construct T from const(T)<br>
this(const this) immutable; // construct immutable(T) from const(T)<br>
this(immutable this); // construct T from immutable(T)<br>
this(this) immutable; // construct immutable(T) from T<br>
}<br>
<br>
Some could be missing, some could be deduced, but this is the total set.<br></blockquote><div><br></div><div>The DIP *does not* support to define different operations for "T from immutable(T)" and "immutable(T) from T".</div>
<div>They are always defined by "unique postblit".</div><div><br></div><div>I believe that supporting it will never make language more useful. For example, if an object supports immutable to mutable copy, but disables mutable to mutable copy, what benefit will be there?</div>
<div>In D, type qualifier is very important feature (especially 'immutable').<br></div><div>And that's why we should keep object copying rule (== postblit definition way) simple.</div><div><br></div><div>Note that, currently D has 9 qualifiers:</div>
<div>- (mutable)</div><div>- const</div><div>- inout</div><div>- inout const (added from 2.065. See issue 6930)</div><div>- shared</div><div>- shared const</div><div>- shared inout</div><div>- shared inout const (added from 2.065. See issue 6930)</div>
<div>- immutable</div><div><br></div><div>So, if we allow defining arbitrary postblit from A to B copy, you can define 9 * 9 = 81 postblits!</div><div>I don't want to see such horrible D code.</div><div> </div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-color:rgb(204,204,204);border-left-style:solid;padding-left:1ex">
* Consider a conversion like "this(this) immutable" which constructs an immutable(T) from a T. This is tricky to typecheck because fields of T have a mutable type when first read and immutable type after having been written to. That raises the question whether the entire notion of postblitting is too complicated for its own good. Should we leave it as is and go with classic C++-style copy construction in which source and destination are distinct objects? I think that would simplify both the language definition and its implementation.<br>
</blockquote><div><br></div><div>??? immutable postblit cannot be used for immutable(T) to T copy.</div><div>If you want to do it, you should define unique postblit. There's no way than others.</div><div> </div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-color:rgb(204,204,204);border-left-style:solid;padding-left:1ex">
* The section "Why 'const' postblit will called to copy arbitrary qualified object?" alludes to the subtyping relationship among qualifiers without stating it.</blockquote><div><br></div><div>If we select "this(this) unique" syntax for unique postblit, the section will be unnecessary.</div>
<div><br></div><div>Kenji Hara <br></div></div></div></div>