1. Revision History
1.1. R3
Added EWG Aug 2020 telecon polling results.In the design, an ODR-use without a definition no longer requires ill-formedness. The intent is still the same, in that it typically results in a linker error.
Updated grammar and ill-formedness of wording.
Found an additional note in expr.new/10 that needed adjustment.
Deferred feature test macro decision to a future revision of P2198.
1.2. R2
Making the status of "placement" new more obvious ("placement" new is still required).Added words in the design to indicate that the declarations in 
Mentioning that pointer safety remains unchanged.
1.3. R1
Added polling results from Prague EWGI and EWG.Added discussion on feature test macro.
Added discussion on not fixing virtual destructors.
Declaring that 
Added wording.
Wording includes changes to library clauses, so added LEWG to the audience.
1.4. R0
R0 of this paper was extracted from [P1105R1].The proposed solution is different than the one proposed for P1105R1, but the motivation is the same. The solution from P1105R1 is still listed as a design alternative.
2. What is changing
On freestanding systems without default heap storage, the presence of the replaceable allocation functions (i.e. allocating:: operator  new nothrow_t align_val_t As a consequence of the above, coroutines that are relying on the global allocation functions will be ill-formed so long as those global allocation functions are not present.
No other core language features require 
3. What is staying the same
The replaceable deallocating:: operator  delete virtual operator  delete :: operator  delete virtual Calling 
The replaceable allocation functions will still be implicitly declared at global scope in each translation unit basic.stc.dynamic.
  Non-ODR-uses of the replaceable allocation functions are still 
permitted (e.g. inside of uninstantiated templates).  The declarations 
of 
"
Non-allocating placement 
Core language concepts of pointer safety remain unchanged. Note that the pointer safety library facilities util.dynamic.safety are not required to be present in freestanding implementations, and the author is not aware of any papers to make the pointer safety library facilities required in freestanding implementations.
Hosted implementations are unchanged.  Users of freestanding 
implementations can still provide implementations of the replaceable 
allocation and deallocation functions.  The behavior of 
4. Why?
4.1. No allocations allowed
In space constrained and/or real-time environments, there is often no free store. These environments often cannot tolerate the space overhead for the free store, or the non-determinism from using the free store. In these environments, it is a desirable property for accidental globalnew new FreeRTOS allows for both static and dynamic allocation of OS constructs [FreeRTOS_StaticVDynamic].  Static allocation in conjunction with a missing 
THREADX [THREADX] does not consider dynamic allocation a core service, and can be built without support for dynamic allocation in order to reduce application size. THREADX also distinguishes between byte allocation (general purpose) vs. block allocation (no-fragmentation elements of fixed size in a pool).
Also, by allowing a no-op 
4.2. No right way to allocate memory
In some target environments, there is no "right" way to allocate memory. In kernel and embedded domains, the implementer of the C++ toolchain doesn’t always know the "right" way to allocate memory on the target environment. This makes it difficult to provide an implementation for:: operator  new malloc As an example, in the Microsoft Windows kernel environment, there are two leading choices about where to get dynamic memory [MSPools]. Users can get memory from the non-paged pool, which is a safe, but scarce resource; or users can get memory from the paged pool, which is plentiful, but not accessible in many common kernel operations. Non-paged pool must be used any time the allocated memory needs to be accessible from an interrupt or from a "high IRQL" context. The author has had experience with both paged pool and non-paged pool as defaults, with the predictable outcome of crashes with paged pool defaults and OOM with non-paged pool defaults. The implementer of the C++ toolchain is not in a good position to make this choice for the user.
In the Linux kernel environment, 
Freestanding implementations are intended to run without the benefit of an operating system (basic.def.odr).
  However, the name of the function that supplies dynamic memory is 
usually an OS-specific detail.  The C++ implementation should not (and 
may not) know the name of the function to request memory.  The Windows 
kernel uses 
Today’s C++ implementations don’t provide 
4.3. What about allocators?
The C++20 freestanding library does not include allocators. [P1642R1] proposes adding allocator machinery to freestanding, but doesn’t addstd :: allocator Allocators are still useful in a less-than-minimal freestanding 
implementation.  In environments with dynamic memory, custom allocators 
can be written and used with standard containers, assuming that the 
containers are present in the implementation.  This could be done even 
if a global 
Some facilities in the standard library (e.g. 
4.4. virtual 
    A no-op :: operator  delete virtual virtual operator  delete virtual new delete virtual Ideally, if neither 
4.5. Why not fix virtual operator  delete 
    This paper attempts to standardize existing practice.  There is not any existing practice for "fixed" virtual  destructors operator  delete virtual 4.6. Feature test macro
A future revision of P2198 will discuss this in more depth, and provide a recommendation.In order to provide this macro, library implementations are going to require knowledge of the target environment. That knowledge may be via a list of known target platforms that are detected at build time, or by having the builder of the implementation supply that information in a configuration flag.
The most likely usage of a feature test macro for this feature is to conditionally define a custom 
Another likely usage is to fall back to an implementation that does not use the heap at all.
#if defined(__cpp_lib_no_default_operator_new) && __cpp_lib_no_default_operator_new >= 20200913 using my_container = fixed_capacity_vector ; #else using my_container = my_vector ; #endif 
This is an imprecise check. Even though there is no default operator new, there may be a user provided operator new that works fine.
If this feature test macro were provided in the positive (
#if defined(__cpp_lib_has_default_operator_new) && __cpp_lib_has_default_operator_new >= 20200913 using my_container = my_vector ; #else // spuriously triggers for C++20 and earlier code using my_container = fixed_capacity_vector ; #endif 
If this feature test macro were provided in the negative (
All these considerations are deserving of dedicated SG10 discussion with P2198, and should not hold up the progress of this paper.
4.7. Likely misuses and abuses
Users are likely to provide overloads of:: operator  new bad_alloc new 5. Experience
The proposed design has field experience in a micro-controller environment. GCC was used, and the language support library was intentionally omitted. A no-op:: operator  delete :: operator  delete virtual 6. Polling history
6.1. Aug 19, 2020, EWG, Telecon
After updating wording for “either/of”, mandates, and talking to SG10 about the feature test macro, P2013 is tentatively ready to be forwarded to CWG. Walter volunteers to check that this is done.SF/F/N/A/SA
2/16/1/0/0
6.2. Feb 14, 2020, EWG, Prague
We are interested in freestanding having an optional operator new, please come back with wordingSF/F/N/A/SA
8/10/7/0/0
6.3. Feb 12, 2020, EWGI, Prague
Given the time constraints of the committee, should we spend additional committee effort on P2013?SF/F/N/A/SA
7/6/1/0/0
Is a Feature test macro a valuable addition to this paper?
SF/F/N/A/SA
0/5/5/2/0
Do we believe that P2013 is sufficiently developed to be seen by EWG?
SF/F/N/A/SA
5/8/0/0/0
6.4. Jan 8, 2020 SG14 Telecon
Forward P2013 as is with the minor editing quotesSF/F/N/A/SA
9/10/0/0/0
approves to go to EWG
7. Design Alternatives
7.1. Alternative 0: All-or-nothing allocating :: operator  new 
    This option preserves much functionality, without using any novel techniques.  See above for further explanation. 
   7.2. Alternative 1: Optional throwing :: operator  new 
    Rather than making all the replaceable allocation functions optional, we could make just the throwing :: operator  new align_val_t nothrow_t The 
7.3. Alternative 2: No deallocation functions
The presence of the replaceable deallocation functions is implementation defined.virtual :: operator  delete operator  delete This alternative has the benefit of being zero overhead and very 
explicit, but it has troublesome consequences for implementations.  
There are several language support classes that have 
If we were to adopt this alternative, many users would provide a no-op 
7.3.1. Experience
This alternative has field experience. MSVC’s /kernel [kernel_switch] flag omits definitions for:: operator  new :: operator  delete :: operator  new :: operator  delete 7.4. Alternative 3: No deallocation functions and new ODR-used rules for virtual destructors
The presence of the replaceable deallocation functions is implementation defined. Changevirtual :: operator  delete new :: operator  delete 7.4.1. How could this virtual destructor ODR-use change be implemented?
First, this is only a problem that needs to be solved on systems without a default heap. This means that typical user-mode desktop and server implementations would be unaffected.Existing linkers already have the ability to take multiple 
identical virtual table implementations and pick one for use in the 
final binary.  A potential implementation strategy is for compilers and 
linkers to support a new "weaker" linkage.  When the default heap is 
disabled, the compiler would emit a vtable with a 
7.4.2. ABI impact
Mixing multiple object files into the same program should be fine, even if some of them have a default heap and some don’t. All the regular / "strong" linkage vtables should be identical, and all the "weaker" linkage vtables should be identical. If anyone in the program calls any form ofnew new delete Shared libraries are trickier.  Vtables aren’t always emitted into
 every translation unit.  Take shared library "leaf" that has a default 
heap.  It depends upon shared library "root" that does not have a 
default heap.  If a class with a virtual destructor is defined in 
"root", along with its "key function", then a call to 
Lack of a default heap should generally be considered a trait of the platform. Mixing this configuration shouldn’t be a common occurrence.
7.4.3. Experience
This alternative is novel, and does not have implementation or usage experience.8. Wording
This is based on the Pre-Prague 2020-01 mailing, N4849.8.1. basic.def.odr
Modify basic.def.odr/3.13.1 - A function is named by an expression or conversion if it is the selected member of an overload set (6.5, 12.4, 12.5) in an overload resolution performed as part of forming that expression or conversion, unless it is either:
3.1.1 -- a pure virtual function and either the expression is not an id-expression naming the function with an explicitly qualified name or the expression forms a pointer to member (7.6.2.1)
., or3.1.2 -- a call to a replaceable global allocation function during the evaluation of a constant expression.
[Note: This covers taking the address of functions (7.3.3, 7.6.2.1), calls to named functions (7.6.1.2), operator overloading (Clause 12), user-defined conversions (11.4.7.2), allocation functions for new-expressions (7.6.2.7), as well as non-default initialization (9.4). A constructor selected to copy or move an object of class type is considered to be named by an expression or conversion even if the call is actually elided by the implementation (11.10.5).— end note]
8.2. basic.stc.dynamic
Modify basic.stc.dynamic/2TheOn hosted implementations, the library provides default definitions for the global allocation and deallocation functions. Some global allocation and deallocation functions are replaceable (17.6.2). On freestanding implementations, the library provides default definitions for the non-replaceable allocation functions and all the global deallocation functions. It is implementation-defined whether a freestanding implementation’s library provides default definitions for the replaceable global allocation functions. Either all or none of the replaceable global allocation functions shall be defined by a freestanding implementation’s library. A C++ program shall provide at most one definition of a replaceable allocation or deallocation function. Any such function definition replaces the default version provided in the library (16.5.4.6). The following allocation and deallocation functions (17.6) are implicitly declared in global scope in each translation unit of a program.
8.3. expr.new
Modify expr.new/10A new-expression may obtain storage for the object by calling an allocation function ([basic.stc.dynamic.allocation]). If the new-expression terminates by throwing an exception, it may release storage by calling a deallocation function. If the allocated type is a non-array type, the allocation function’s name is operator new and the deallocation function’s name is operator delete. If the allocated type is an array type, the allocation function’s name is operator new[] and the deallocation function’s name is operator delete[]. [Note:AnA hosted implementation is required to provide default definitions for the global allocation functions ([basic.stc.dynamic], [new.delete.single], [new.delete.array]). A C++ program can provide alternative definitions of these functions ([replacement.functions]) and/or class-specific versions ([class.free]). The set of allocation and deallocation functions that may be called by a new-expression may include functions that do not perform allocation or deallocation; for example, see [new.delete.placement]. — end note]
8.4. new.delete
Modify new.delete17.6.2 Storage allocation and deallocation [new.delete]
1 Except where otherwise specified, the provisions of 6.7.5.4 apply to the library versions of operator new and operator delete. If the value of an alignment argument passed to any of these functions is not a valid alignment value, the behavior is undefined.
17.6.2.1 Allocating forms [allocating.forms]
1 Odr-use of the allocation functions in this subclause results in undefined behavior unless a definition of the odr-used allocation function has been provided. [Note: On hosted implementations, the library provides default definitions of these allocation functions ([basic.stc.dynamic]).- end note]
17.6.2.117.6.2.1.1 Single-object forms [new.delete.single]No changes
17.6.2.217.6.2.1.2 Array forms [new.delete.array]No changes
17.6.2.317.6.2.2 Non-allocating forms [new.delete.placement]No changes
17.6.2.417.6.2.3 Data races [new.delete.dataraces]No changes
9. Alternative Wording
9.1. new.delete
[allocating.forms]/1 uses undefined behavior. Making the program ill-formed would be preferable, but we cannot use ill-formed to diagnose linker errors. The following is the wording approach to use if we let the ODR violation provide the normative backing for this design. This results in IF;NDR when no definitions are present.17.6.2.1 Allocating forms [allocating.forms]
1 [Note: Odr-use of the allocation functions in this subclause results in a one-definition rule violation unless a definition of the odr-used allocation function has been provided. ([basic.def.odr]). On hosted implementations, the library provides default definitions of these allocation functions ([basic.stc.dynamic]).- end note]
10. Acknowledgments
Thank you to the many reviewers of this paper: Brandon Streiff, Irwan Djajadi, Joshua Cannon, Brad Keryan, Alfred Bratterud, Phil Hindman, Arthur O’Dwyer, Laurin-Luis Lehning, JF Bastien, Matthew Bentley, and Alisdair Meredith.Thank you to Daveed Vandevoorde and Walter Brown for providing feedback on the wording.