Issue 2072: Unclear wording about capacity of temporary buffers (original) (raw)
This page is a snapshot from the LWG issues list, see the Library Active Issues List for more information and the meaning of C++17 status.
2072. Unclear wording about capacity of temporary buffers
Section: 99 [depr.temporary.buffer] Status: C++17 Submitter: Kazutoshi Satoda Opened: 2011-08-10 Last modified: 2025-03-13
Priority: 3
View all other issues in [depr.temporary.buffer].
View all issues with C++17 status.
Discussion:
According to [temporary.buffer] p1+2:
template pair<T*, ptrdiff_t> get_temporary_buffer(ptrdiff_t n) noexcept;
-1- Effects: Obtains a pointer to storage sufficient to store up to
nadjacentTobjects. It is implementation-defined whether over-aligned types are supported (3.11).-2- Returns: A pair containing the buffer's address and capacity (in the units of
sizeof(T)), or a pair of 0 values if no storage can be obtained or ifn <= 0.
I read this as prohibiting to return a buffer of which capacity is less than n, because such a buffer is not sufficient to store n objects.
The corresponding description in SGI STLis clear on this point, but I think it is a bit too verbose:
(for the return value, a pair
P) [...] the buffer pointed to byP.firstis large enough to holdP.secondobjects of typeT.P.secondis greater than or equal to 0, and less than or equal tolen.
There seems to be two different targets of the "up to n" modification: The capacity of obtained buffer, and the actual number that the caller will store into the buffer.
First I read as the latter, and got surprised seeing that libstdc++ implementation can return a smaller buffer. I started searching aboutget_temporary_buffer(). After reading a quote from TC++PL atstackoverflow, I realized that the former is intended.
Such misinterpretation seems common:
- The above question is likely started from same misinterpretation.
- JIS standard (Japanese translation of ISO/IEC standard) says nothing like "up to". I think the editor misinterpreted the original wording, and omitted words for "up to" as it is redundant. (If a buffer is sufficient to store
nobjects, it is also sufficient to store up tonobjects.) - Rogue Wave implementation doesn't return smaller buffer, instead, it can return larger buffer on some circumstances. Apache STDCXX is a derived version of that implementation, and publicly accessible:
Specializations of the
get_temporary_buffer()function template attempt to allocate a region of storage sufficiently large to store at leastnadjacent objects of typeT.
I know one commercial compiler package based on Rogue Wave implementation, and its implementation is essentially same as the above.
[2014-05-18, Daniel comments and suggests concrete wording]
The provided wording attempts to clarify the discussed capacity freedom, but it also makes it clearer that the returned memory is just "raw memory", which is currently not really clear. In addition the wording clarifies that the deallocatingreturn_temporary_buffer function does not throw exceptions, which I believe is the intention when the preconditions of the functions are satisfied. Then, my understanding is that we can provide to return_temporary_buffer a null pointer value if that was the value, get_temporary_buffer() had returned. Furthermore, as STL noticed, the current wording seemingly allows multiple invocations of return_temporary_buffer with the same value returned by get_temporary_buffer; this should be constrained similar to the wording we have for operator delete (unfortunately we miss such wording for allocators).
[2015-05, Lenexa]
MC: move to ready? in favor: 14, opposed: 0, abstain: 0
Proposed resolution:
This wording is relative to N3936.
- Change [temporary.buffer] as indicated:
template
pair<T*, ptrdiff_t> get_temporary_buffer(ptrdiff_t n) noexcept;-1- Effects: Obtains a pointer to uninitialized, contiguous storage for
_N_adjacent objects of typeT, for some non-negative number_N_.Obtains a pointer to storage sufficient to store up toIt is implementation-defined whether over-aligned types are supported (3.11).nadjacentTobjects.-?- Remarks: Calling
get_temporary_bufferwith a positive numbernis a non-binding request to return storage fornobjects of typeT. In this case, an implementation is permitted to return instead storage for a non-negative number_N_of such objects, where_N_ != n(including_N_ == 0). [Note: The request is non-binding to allow latitude for implementation-specific optimizations of its memory management. — _end note_].-2- Returns: If
n <= 0or if no storage could be obtained, returns a pairPsuch thatP.firstis a null pointer value andP.second == 0; otherwise returns a pairPsuch thatP.firstrefers to the address of the uninitialized storage andP.secondrefers to its capacity_N_(in the units ofsizeof(T)).Apaircontaining the buffer's address and capacity (in the units ofsizeof(T)), or a pair of 0 values if no storage can be obtained or ifn <= 0.template void return_temporary_buffer(T* p);
-3- Effects: Deallocates the
buffer to whichstorage referenced byppointsp.-4- Requires:
The buffer shall have been previously allocated bypshall be a pointer value returned by an earlier call toget_temporary_bufferwhich has not been invalidated by an intervening call toreturn_temporary_buffer(T*).-?- Throws: Nothing.