File: CXXTypeAwareAllocators.rst

package info (click to toggle)
llvm-toolchain-21 1%3A21.1.3-1
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 2,236,320 kB
  • sloc: cpp: 7,619,338; ansic: 1,433,952; asm: 1,058,735; python: 252,125; f90: 94,671; objc: 70,753; lisp: 42,813; pascal: 18,401; sh: 10,094; ml: 5,111; perl: 4,720; awk: 3,523; makefile: 3,397; javascript: 2,272; xml: 892; fortran: 770
file content (155 lines) | stat: -rw-r--r-- 5,248 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
=========================
C++ Type Aware Allocators
=========================

.. contents::
   :local:

Introduction
============

Clang includes an implementation of P2719 "Type-aware allocation and deallocation
functions".

This is a feature that extends the semantics of `new`, `new[]`, `delete` and
`delete[]` operators to expose the type being allocated to the operator. This
can be used to customize allocation of types without needing to modify the
type declaration, or via template definitions fully generic type aware
allocators.

P2719 introduces a type-identity tag as valid parameter type for all allocation
operators. This tag is a default initialized value of type `std::type_identity<T>`
where T is the type being allocated or deallocated.  Unlike the other placement
arguments this tag is passed as the first parameter to the operator.

The most basic use case is as follows

.. code-block:: c++

  #include <new>
  #include <type_traits>

  struct S {
   // ...
  };

  void *operator new(std::type_identity<S>, size_t, std::align_val_t);
  void operator delete(std::type_identity<S>, void *, size_t, std::align_val_t);

  void f() {
    S *s = new S; // calls ::operator new(std::type_identity<S>(), sizeof(S), alignof(S))
    delete s; // calls ::operator delete(std::type_identity<S>(), s, sizeof(S), alignof(S))
  }

While this functionality alone is powerful and useful, the true power comes
by using templates. In addition to adding the type-identity tag, P2719 allows
the tag parameter to be a dependent specialization of `std::type_identity`,
updates the overload resolution rules to support full template deduction and
constraint semantics, and updates the definition of usual deallocation functions
to include `operator delete` definitions that are templatized on the
type-identity tag.

This allows arbitrarily constrained definitions of the operators that resolve
as would be expected for any other template function resolution, e.g (only
showing `operator new` for brevity)

.. code-block:: c++

   template <typename T, unsigned Size> struct Array {
     T buffer[Size];
   };

   // Starting with a concrete type
   void *operator new(std::type_identity<Array<int, 5>>, size_t, std::align_val_t);

   // Only care about five element arrays
   template <typename T>
   void *operator new(std::type_identity<Array<T, 5>>, size_t, std::align_val_t);

   // An array of N floats
   template <unsigned N>
   void *operator new(std::type_identity<Array<float, N>>, size_t, std::align_val_t);

   // Any array
   template <typename T, unsigned N>
   void *operator new(std::type_identity<Array<T, N>>, size_t, std::align_val_t);

   // A handy concept
   template <typename T> concept Polymorphic = std::is_polymorphic_v<T>;

   // Only applies is T is Polymorphic
   template <Polymorphic T, unsigned N>
   void *operator new(std::type_identity<Array<T, N>>, size_t, std::align_val_t);

   // Any even length array
   template <typename T, unsigned N>
   void *operator new(std::type_identity<Array<T, N>>, size_t, std::align_val_t)
       requires(N%2 == 0);

Operator selection then proceeds according to the usual rules for choosing
the best/most constrained match.

Any declaration of a type aware operator new or operator delete must include a
matching complimentary operator defined in the same scope.

Notes
=====

Unconstrained Global Operators
------------------------------

Declaring an unconstrained type aware global operator `new` or `delete` (or
`[]` variants) creates numerous hazards, similar to, but different from, those
created by attempting to replace the non-type aware global operators. For that
reason unconstrained operators are strongly discouraged.

Mismatching Constraints
-----------------------

When declaring global type aware operators you should ensure the constraints
applied to new and delete match exactly, and declare them together. This
limits the risk of having mismatching operators selected due to differing
constraints resulting in changes to prioritization when determining the most
viable candidate.

Declarations Across Libraries
-----------------------------

Declaring a typed allocator for a type in a separate TU or library creates
similar hazards as different libraries and TUs may see (or select) different
definitions.

Under this model something like this would be risky

.. code-block:: c++

  template<typename T>
  void *operator new(std::type_identity<std::vector<T>>, size_t, std::align_val_t);

However this hazard is not present simply due to the use of the a type from
another library:

.. code-block:: c++

  template<typename T>
  struct MyType {
    T thing;
  };
  template<typename T>
  void *operator new(std::type_identity<MyType<std::vector<T>>>, size_t, std::align_val_t);

Here we see `std::vector` being used, but that is not the actual type being
allocated.

Implicit and Placement Parameters
---------------------------------

Type aware allocators are always passed both the implicit alignment and size
parameters in all cases. Explicit placement parameters are supported after the
mandatory implicit parameters.

Publication
===========

`Type-aware allocation and deallocation functions <https://wg21.link/P2719>`_.
Louis Dionne, Oliver Hunt.