<div dir="ltr"><br><div class="gmail_extra"><br><div class="gmail_quote">On Wed, Sep 6, 2017 at 12:41 PM, Joe Groff via swift-evolution <span dir="ltr">&lt;<a href="mailto:swift-evolution@swift.org" target="_blank">swift-evolution@swift.org</a>&gt;</span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">&gt; Currently, memory is deallocated by an instance method on UnsafeMutablePointer, deallocate(count:). Like much of the Swift pointer API, performing this operation on a buffer pointer requires extracting baseAddress! and count. It is very common for the allocation code above to be immediately followed by:<br>
&gt;<br>
&gt; defer<br>
&gt;<br>
&gt; {<br>
&gt;     buffer.<br>
&gt; baseAddress?.deallocate(<wbr>capacity: buffer.count<br>
&gt; )<br>
&gt; }<br>
&gt;<br>
&gt; This method is extremely problematic because nearly all users, on first seeing the signature of deallocate(capacity:), will naturally conclude from the capacity label that deallocate(capacity:) is equivalent to some kind of realloc()that can only shrink the buffer. However this is not the actual behavior — deallocate(capacity:) actually ignores the capacity argument and just calls free() on self. The current API is not only awkward and suboptimal, it is misleading. You can write perfectly legal Swift code that shouldn’t segfault, but still can, for example<br>
&gt;<br>
&gt; var ptr = UnsafeMutablePointer&lt;UInt8&gt;.<wbr>allocate(capacity: 1000000<br>
&gt; )<br>
&gt; ptr.<br>
&gt; initialize(to: 13, count: 1000000<br>
&gt; )<br>
&gt; ptr.<br>
&gt; deallocate(capacity: 500000) // deallocate the second half of the memory block<br>
&gt; ptr[0] // segmentation fault<br>
&gt; where the first 500000 addresses should still be valid if the documentation is to be read literally.<br>
<br>
The fact that the Swift runtime currently uses malloc/free is an implementation detail. Tracking deallocation size is a very useful optimization for better allocator backends, and C++ underwent an ABI break to make it so that sized delete can be supported. Maybe we can change the name to `deallocate(allocatedCapacity:<wbr>)` to make it clear that it isn&#39;t resizing the memory, and/or make the capacity argument optional so that you can pay for the overhead of the allocator deriving the size if it&#39;s inconvenient for the calling code to carry the size around, but we shouldn&#39;t remove the functionality altogether.<br>
<br>
-Joe<br>
<div class="HOEnZb"><div class="h5">______________________________<wbr>_________________<br>
swift-evolution mailing list<br>
<a href="mailto:swift-evolution@swift.org">swift-evolution@swift.org</a><br>
<a href="https://lists.swift.org/mailman/listinfo/swift-evolution" rel="noreferrer" target="_blank">https://lists.swift.org/<wbr>mailman/listinfo/swift-<wbr>evolution</a></div></div></blockquote><div><br></div><div>The idea is to get the house in order by removing all parameters from <span style="font-family:monospace,monospace">deallocate()</span>, since that’s what it really does right now. Then, in the future, if Swift gets a more sophisticated allocator backend, a new method like <span style="font-family:monospace,monospace">deallocate(capacity:)</span> or <span style="font-family:monospace,monospace">reallocate(toCapacity:)</span> could be added without conflicting with the currently named <span style="font-family:monospace,monospace">deallocate(capacity:)</span>. However, using the function signature to pretend that it does something it can’t actually do right now is extremely dangerous.<br></div></div><br></div></div>