1. Introduction
[P0593] introduced the ability for objects to be implicitly created under
certain circumstances. In doing so, it added the requirement to allocators
that, when 
This example is incorrect.
After the lifetime of the 
First, if 
represents the address A of a byte in memory. An object X that is within its lifetime and whose type is similar top is located at the address A. All bytes of storage that would be reachable through the result are reachable throughT (see below).p 
The result of 
If 
2. Problem
The fact that an example is incorrect is not a problem, since it can be fixed.
The problem is that there is no proper way to do what the example hopes to
accomplish. To the author’s knowledge, there is no reasonable standard way to
allocate a 
This problem applies to any allocator which returns memory from an allocated 
Allocators based on 
T * allocate ( size_type n ) { return * reinterpret_cast < T ( * )[] > ( :: operator new ( sizeof ( T ) * n )); } 
In this implementation, 
There is no equivalent for user-defined functions that create a 
In addition to suitable created objects being insufficient, there is no other
(reasonable) way to obtain a pointer to a 
In particular, the effects are undefined in the following cases:
- [...]
- If an incomplete type is used as a template argument when instantiating a template component or evaluating a concept, unless specifically allowed for that component.
No specific type 
However, a non-constant template argument to 
template < typename T , std :: size_t Min , std :: size_t Max > auto do_launder_array ( void * p , std :: size_t n ) -> T ( * )[] { if constexpr ( Min == Max ) { std :: abort (); } else { if ( n == Min ) { return std :: launder ( reinterpret_cast < T ( * )[ Min ] > ( p )); } else { return :: do_launder_array < T , Min + 1 , Max > ( p , n ); } } } inline constexpr std :: size_t max_launder_array_size = 899 ; template < typename T > auto launder_array ( void * p , std :: size_t n ) -> T ( * )[] { return :: do_launder_array < T , 1 , max_launder_array_size > ( p , n ); } // In some allocator: T * allocate ( size_type n ) { return *:: launder_array < T > ( somehow_allocate_byte_array ( n ), n ); } 
This works by determining, at runtime, the type 
3. Proposed Solution
The solution proposed here is to allow laundering pointers to arrays of unknown bound. This allows acquiring a pointer to an implicitly created array, without requiring that any elements have been created.
Since 
With this change, the following would be a conforming implementation of 
T * allocate ( size_type n ) { return * std :: launder ( reinterpret_cast < T ( * )[] > ( somehow_allocate_byte_array ( n ))); } 
4. Proposed Wording
4.1. 17.6.5 Pointer optimization barrier [ptr.launder]
template < class T > [[ nodiscard ]] constexpr T * launder ( T * p ) noexcept ; 
Mandates:is true.! is_function_v < T > && ! is_void_v < T > Preconditions:
represents the address A of a byte in memory. An object X that is within its lifetime and whose type is similar top is located at the address A. All bytes of storage that would be reachable through the result are reachable throughT (see below).p Returns: A value of type
that points to X.T * Remarks:
may be a type that is an array of unknown bound. An invocation of this function may be used in a core constant expression if and only if the (converted) value of its argument may be used in place of the function invocation. A byte of storage b is reachable through a pointer value that points to an object Y if there is an object Z, pointer-interconvertible with Y, such that b is within the storage occupied by Z, or the immediately-enclosing array object if Z is an array element.T