oxc_allocator 0.127.0

A collection of JavaScript tools written in Rust.
Documentation
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
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
//! `Alloc` trait.

// Copyright 2015 The Rust Project Developers.
// See the COPYRIGHT file at the top-level directory of this distribution and at http://rust-lang.org/COPYRIGHT.
//
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or http://www.apache.org/licenses/LICENSE-2.0>
// or the MIT license <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your option.
// This file may not be copied, modified, or distributed except according to those terms.

#![expect(clippy::undocumented_unsafe_blocks)]
#![allow(unstable_name_collisions)]
#![allow(dead_code)]

use std::{
    alloc::{Layout, LayoutError},
    cmp, fmt, mem,
    ptr::{self, NonNull},
};

#[cold]
fn new_layout_err() -> LayoutError {
    Layout::from_size_align(1, 3).unwrap_err()
}

#[cold]
#[inline(never)]
pub fn handle_alloc_error(layout: Layout) -> ! {
    panic!("encountered allocation error: {layout:?}")
}

// TODO: `Layout::repeat` was stabilized in Rust 1.95. Remove this trait and use the
// standard library method directly once MSRV reaches 1.95.
pub trait UnstableLayoutMethods {
    fn padding_needed_for(&self, align: usize) -> usize;
    fn repeat(&self, n: usize) -> Result<(Layout, usize), LayoutError>;
    fn array<T>(n: usize) -> Result<Layout, LayoutError>;
}

impl UnstableLayoutMethods for Layout {
    fn padding_needed_for(&self, align: usize) -> usize {
        let len = self.size();

        // Rounded up value is:
        //   len_rounded_up = (len + align - 1) & !(align - 1);
        // and then we return the padding difference: `len_rounded_up - len`.
        //
        // We use modular arithmetic throughout:
        //
        // 1. align is guaranteed to be > 0, so align - 1 is always valid
        //
        // 2. `len + align - 1` can overflow by at most `align - 1`, so the &-mask with `!(align - 1)` will
        //    ensure that in the case of overflow, `len_rounded_up` will itself be 0. Thus the returned
        //    padding, when added to `len`, yields 0, which trivially satisfies the alignment `align`.
        //
        // (Of course, attempts to allocate blocks of memory whose size and padding overflow in the above
        // manner should cause the allocator to yield an error anyway.)

        let len_rounded_up = len.wrapping_add(align).wrapping_sub(1) & !align.wrapping_sub(1);
        len_rounded_up.wrapping_sub(len)
    }

    fn repeat(&self, n: usize) -> Result<(Layout, usize), LayoutError> {
        let padded_size = self
            .size()
            .checked_add(self.padding_needed_for(self.align()))
            .ok_or_else(new_layout_err)?;
        let alloc_size = padded_size.checked_mul(n).ok_or_else(new_layout_err)?;

        unsafe {
            // self.align is already known to be valid and alloc_size has been padded already
            Ok((Layout::from_size_align_unchecked(alloc_size, self.align()), padded_size))
        }
    }

    fn array<T>(n: usize) -> Result<Layout, LayoutError> {
        UnstableLayoutMethods::repeat(&Layout::new::<T>(), n).map(|(k, offs)| {
            debug_assert!(offs == mem::size_of::<T>());
            k
        })
    }
}

/// Represents the combination of a starting address and a total capacity of the returned block.
// #[unstable(feature = "allocator_api", issue = "32838")]
#[derive(Debug)]
pub struct Excess(pub NonNull<u8>, pub usize);

fn size_align<T>() -> (usize, usize) {
    (mem::size_of::<T>(), mem::align_of::<T>())
}

/// The `AllocErr` error indicates an allocation failure that may be due to resource exhaustion or to
/// something wrong when combining the given input arguments with this allocator.
// #[unstable(feature = "allocator_api", issue = "32838")]
#[derive(Clone, PartialEq, Eq, Debug)]
pub struct AllocErr;

// (we need this for downstream impl of trait Error)
// #[unstable(feature = "allocator_api", issue = "32838")]
impl fmt::Display for AllocErr {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        f.write_str("memory allocation failed")
    }
}

/// The `CannotReallocInPlace` error is used when `grow_in_place` or `shrink_in_place` were unable to reuse
/// the given memory block for a requested layout.
// #[unstable(feature = "allocator_api", issue = "32838")]
#[derive(Clone, PartialEq, Eq, Debug)]
pub struct CannotReallocInPlace;

// #[unstable(feature = "allocator_api", issue = "32838")]
impl CannotReallocInPlace {
    #[expect(clippy::unused_self, reason = "part of public API")]
    pub fn description(&self) -> &'static str {
        "cannot reallocate allocator's memory in place"
    }
}

// (we need this for downstream impl of trait Error)
// #[unstable(feature = "allocator_api", issue = "32838")]
impl fmt::Display for CannotReallocInPlace {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        write!(f, "{}", self.description())
    }
}

/// An implementation of `Alloc` can allocate, reallocate, and deallocate arbitrary blocks of data described
/// via `Layout`.
///
/// Some of the methods require that a memory block be *currently allocated* via an allocator. This means that:
///
/// * the starting address for that memory block was previously returned by a previous call to an allocation
///   method (`alloc`, `alloc_zeroed`, `alloc_excess`, `alloc_one`, `alloc_array`) or reallocation method
///   (`realloc`, `realloc_excess`, or `realloc_array`), and
///
/// * the memory block has not been subsequently deallocated, where blocks are deallocated either by being
///   passed to a deallocation method (`dealloc`, `dealloc_one`, `dealloc_array`) or by being passed to a
///   reallocation method (see above) that returns `Ok`.
///
/// A note regarding zero-sized types and zero-sized layouts: many methods in the `Alloc` trait state that
/// allocation requests must be non-zero size, or else undefined behavior can result.
///
/// * However, some higher-level allocation methods (`alloc_one`, `alloc_array`) are well-defined on zero-sized
///   types and can optionally support them: it is left up to the implementor whether to return `Err`, or to
///   return `Ok` with some pointer.
///
/// * If an `Alloc` implementation chooses to return `Ok` in this case (i.e. the pointer denotes a zero-sized
///   inaccessible block) then that returned pointer must be considered "currently allocated". On such an
///   allocator, *all* methods that take currently-allocated pointers as inputs must accept these zero-sized
///   pointers, *without* causing undefined behavior.
///
/// * In other words, if a zero-sized pointer can flow out of an allocator, then that allocator must likewise
///   accept that pointer flowing back into its deallocation and reallocation methods.
///
/// Some of the methods require that a layout *fit* a memory block. What it means for a layout to "fit" a
/// memory block means (or equivalently, for a memory block to "fit" a layout) is that the following two
/// conditions must hold:
///
/// 1. The block's starting address must be aligned to `layout.align()`.
///
/// 2. The block's size must fall in the range `[use_min, use_max]`, where:
///
///    * `use_min` is `self.usable_size(layout).0`, and
///
///    * `use_max` is the capacity that was (or would have been) returned when (if) the block was allocated
///      via a call to `alloc_excess` or `realloc_excess`.
///
/// Note that:
///
///  * the size of the layout most recently used to allocate the block is guaranteed to be in the range
///    `[use_min, use_max]`, and
///
///  * a lower-bound on `use_max` can be safely approximated by a call to `usable_size`.
///
///  * if a layout `k` fits a memory block (denoted by `ptr`) currently allocated via an allocator `a`, then it
///    is legal to use that layout to deallocate it, i.e. `a.dealloc(ptr, k);`.
///
/// # SAFETY
///
/// The `Alloc` trait is an `unsafe` trait for a number of reasons, and implementors must ensure that they
/// adhere to these contracts:
///
/// * Pointers returned from allocation functions must point to valid memory and retain their validity until at
///   least the instance of `Alloc` is dropped itself.
///
/// * `Layout` queries and calculations in general must be correct. Callers of this trait are allowed to rely
///   on the contracts defined on each method, and implementors must ensure such contracts remain true.
///
/// Note that this list may get tweaked over time as clarifications are made in the future.
// #[unstable(feature = "allocator_api", issue = "32838")]
pub unsafe trait Alloc {
    // (Note: some existing allocators have unspecified but well-defined behavior in response to a zero size
    // allocation request ; e.g. in C, `malloc` of 0 will either return a null pointer or a unique pointer,
    // but will not have arbitrary undefined behavior.
    // However in jemalloc for example, `mallocx(0)` is documented as undefined behavior.)

    /// Returns a pointer meeting the size and alignment guarantees of `layout`.
    ///
    /// If this method returns an `Ok(addr)`, then the `addr` returned will be non-null address pointing to a
    /// block of storage suitable for holding an instance of `layout`.
    ///
    /// The returned block of storage may or may not have its contents initialized. (Extension subtraits might
    /// restrict this behavior, e.g. to ensure initialization to particular sets of bit patterns.)
    ///
    /// # SAFETY
    ///
    /// This function is unsafe because undefined behavior can result if the caller does not ensure that
    /// `layout` has non-zero size.
    ///
    /// (Extension subtraits might provide more specific bounds on behavior, e.g. guarantee a sentinel address
    /// or a null pointer in response to a zero-size allocation request.)
    ///
    /// # Errors
    ///
    /// Returning `Err` indicates that either memory is exhausted or `layout` does not meet allocator's size
    /// or alignment constraints.
    ///
    /// Implementations are encouraged to return `Err` on memory exhaustion rather than panicking or aborting,
    /// but this is not a strict requirement. (Specifically: it is *legal* to implement this trait atop an
    /// underlying native allocation library that aborts on memory exhaustion.)
    ///
    /// Clients wishing to abort computation in response to an allocation error are encouraged to call the
    /// [`handle_alloc_error`] function, rather than directly invoking `panic!` or similar.
    ///
    /// [`handle_alloc_error`]: ../../alloc/alloc/fn.handle_alloc_error.html
    unsafe fn alloc(&mut self, layout: Layout) -> Result<NonNull<u8>, AllocErr>;

    /// Deallocate the memory referenced by `ptr`.
    ///
    /// # SAFETY
    ///
    /// This function is unsafe because undefined behavior can result if the caller does not ensure all of the
    /// following:
    ///
    /// * `ptr` must denote a block of memory currently allocated via this allocator,
    ///
    /// * `layout` must *fit* that block of memory,
    ///
    /// * In addition to fitting the block of memory `layout`, the alignment of the `layout` must match the
    ///   alignment used to allocate that block of memory.
    unsafe fn dealloc(&mut self, ptr: NonNull<u8>, layout: Layout);

    // == ALLOCATOR-SPECIFIC QUANTITIES AND LIMITS ==
    // usable_size

    /// Returns bounds on the guaranteed usable size of a successful allocation created with the specified
    /// `layout`.
    ///
    /// In particular, if one has a memory block allocated via a given allocator `a` and layout `k` where
    /// `a.usable_size(k)` returns `(l, u)`, then one can pass that block to `a.dealloc()` with a layout in
    /// the size range [l, u].
    ///
    /// (All implementors of `usable_size` must ensure that `l <= k.size() <= u`)
    ///
    /// Both the lower- and upper-bounds (`l` and `u` respectively) are provided, because an allocator based
    /// on size classes could misbehave if one attempts to deallocate a block without providing a correct
    /// value for its size (i.e., one within the range `[l, u]`).
    ///
    /// Clients who wish to make use of excess capacity are encouraged to use the `alloc_excess` and
    /// `realloc_excess` instead, as this method is constrained to report conservative values that serve as
    /// valid bounds for *all possible* allocation method calls.
    ///
    /// However, for clients that do not wish to track the capacity returned by `alloc_excess` locally, this
    /// method is likely to produce useful results.
    #[inline]
    fn usable_size(&self, layout: &Layout) -> (usize, usize) {
        (layout.size(), layout.size())
    }

    // == METHODS FOR MEMORY REUSE ==
    // realloc. alloc_excess, realloc_excess

    /// Returns a pointer suitable for holding data described by a new layout with `layout`’s alignment and a
    /// size given by `new_size`. To accomplish this, this may extend or shrink the allocation referenced by
    /// `ptr` to fit the new layout.
    ///
    /// If this returns `Ok`, then ownership of the memory block referenced by `ptr` has been transferred to
    /// this allocator. The memory may or may not have been freed, and should be considered unusable (unless of
    /// course it was transferred back to the caller again via the return value of this method).
    ///
    /// If this method returns `Err`, then ownership of the memory block has not been transferred to this
    /// allocator, and the contents of the memory block are unaltered.
    ///
    /// # SAFETY
    ///
    /// This function is unsafe because undefined behavior can result if the caller does not ensure all of the
    /// following:
    ///
    /// * `ptr` must be currently allocated via this allocator,
    ///
    /// * `layout` must *fit* the `ptr` (see above). (The `new_size` argument need not fit it.)
    ///
    /// * `new_size` must be greater than zero
    ///
    /// * `new_size`, when rounded up to the nearest multiple of `layout.align()`, must not overflow (i.e. the
    ///   rounded value must be less than `usize::MAX`)
    ///
    /// (Extension subtraits might provide more specific bounds on behavior, e.g. guarantee a sentinel address
    /// or a null pointer in response to a zero-size allocation request.)
    ///
    /// # Errors
    ///
    /// Returns `Err` only if the new layout does not meet the allocator's size and alignment constraints of
    /// the allocator, or if reallocation otherwise fails.
    ///
    /// Implementations are encouraged to return `Err` on memory exhaustion rather than panicking or aborting,
    /// but this is not a strict requirement. (Specifically: it is *legal* to implement this trait atop an
    /// underlying native allocation library that aborts on memory exhaustion.)
    ///
    /// Clients wishing to abort computation in response to a reallocation error are encouraged to call the
    /// [`handle_alloc_error`] function, rather than directly invoking `panic!` or similar.
    ///
    /// [`handle_alloc_error`]: ../../alloc/alloc/fn.handle_alloc_error.html
    unsafe fn realloc(
        &mut self,
        ptr: NonNull<u8>,
        layout: Layout,
        new_size: usize,
    ) -> Result<NonNull<u8>, AllocErr> {
        let old_size = layout.size();

        #[expect(clippy::collapsible_else_if)]
        if new_size >= old_size {
            if unsafe { self.grow_in_place(ptr, layout, new_size) }.is_ok() {
                return Ok(ptr);
            }
        } else {
            if unsafe { self.shrink_in_place(ptr, layout, new_size) }.is_ok() {
                return Ok(ptr);
            }
        }

        // otherwise, fall back on alloc + copy + dealloc
        let new_layout = unsafe { Layout::from_size_align_unchecked(new_size, layout.align()) };
        let result = unsafe { self.alloc(new_layout) };
        if let Ok(new_ptr) = result {
            unsafe {
                ptr::copy_nonoverlapping(
                    ptr.as_ptr(),
                    new_ptr.as_ptr(),
                    cmp::min(old_size, new_size),
                );
                self.dealloc(ptr, layout);
            }
        }
        result
    }

    /// Behaves like `alloc`, but also ensures that the contents are set to zero before being returned.
    ///
    /// # SAFETY
    ///
    /// This function is unsafe for the same reasons that `alloc` is.
    ///
    /// # Errors
    ///
    /// Returning `Err` indicates that either memory is exhausted or `layout` does not meet allocator's size
    /// or alignment constraints, just as in `alloc`.
    ///
    /// Clients wishing to abort computation in response to an allocation error are encouraged to call the
    /// [`handle_alloc_error`] function, rather than directly invoking `panic!` or similar.
    ///
    /// [`handle_alloc_error`]: ../../alloc/alloc/fn.handle_alloc_error.html
    unsafe fn alloc_zeroed(&mut self, layout: Layout) -> Result<NonNull<u8>, AllocErr> {
        let size = layout.size();
        let p = unsafe { self.alloc(layout) };
        if let Ok(p) = p {
            unsafe { ptr::write_bytes(p.as_ptr(), 0, size) };
        }
        p
    }

    /// Behaves like `alloc`, but also returns the whole size of the returned block. For some `layout` inputs,
    /// like arrays, this may include extra storage usable for additional data.
    ///
    /// # SAFETY
    ///
    /// This function is unsafe for the same reasons that `alloc` is.
    ///
    /// # Errors
    ///
    /// Returning `Err` indicates that either memory is exhausted or `layout` does not meet allocator's size
    /// or alignment constraints, just as in `alloc`.
    ///
    /// Clients wishing to abort computation in response to an allocation error are encouraged to call the
    /// [`handle_alloc_error`] function, rather than directly invoking `panic!` or similar.
    ///
    /// [`handle_alloc_error`]: ../../alloc/alloc/fn.handle_alloc_error.html
    unsafe fn alloc_excess(&mut self, layout: Layout) -> Result<Excess, AllocErr> {
        let usable_size = self.usable_size(&layout);
        unsafe { self.alloc(layout) }.map(|p| Excess(p, usable_size.1))
    }

    /// Behaves like `realloc`, but also returns the whole size of the returned block. For some `layout`
    /// inputs, like arrays, this may include extra storage usable for additional data.
    ///
    /// # SAFETY
    ///
    /// This function is unsafe for the same reasons that `realloc` is.
    ///
    /// # Errors
    ///
    /// Returning `Err` indicates that either memory is exhausted or `layout` does not meet allocator's size
    /// or alignment constraints, just as in `realloc`.
    ///
    /// Clients wishing to abort computation in response to a reallocation error are encouraged to call the
    /// [`handle_alloc_error`] function, rather than directly invoking `panic!` or similar.
    ///
    /// [`handle_alloc_error`]: ../../alloc/alloc/fn.handle_alloc_error.html
    unsafe fn realloc_excess(
        &mut self,
        ptr: NonNull<u8>,
        layout: Layout,
        new_size: usize,
    ) -> Result<Excess, AllocErr> {
        let new_layout = unsafe { Layout::from_size_align_unchecked(new_size, layout.align()) };
        let usable_size = self.usable_size(&new_layout);
        unsafe { self.realloc(ptr, layout, new_size) }.map(|p| Excess(p, usable_size.1))
    }

    /// Attempts to extend the allocation referenced by `ptr` to fit `new_size`.
    ///
    /// If this returns `Ok`, then the allocator has asserted that the memory block referenced by `ptr` now
    /// fits `new_size`, and thus can be used to carry data of a layout of that size and same alignment as
    /// `layout`. (The allocator is allowed to expend effort to accomplish this, such as extending the memory
    /// block to include successor blocks, or virtual memory tricks.)
    ///
    /// Regardless of what this method returns, ownership of the memory block referenced by `ptr` has not been
    /// transferred, and the contents of the memory block are unaltered.
    ///
    /// # SAFETY
    ///
    /// This function is unsafe because undefined behavior can result if the caller does not ensure all of the
    /// following:
    ///
    /// * `ptr` must be currently allocated via this allocator,
    ///
    /// * `layout` must *fit* the `ptr` (see above); note the `new_size` argument need not fit it,
    ///
    /// * `new_size` must not be less than `layout.size()`,
    ///
    /// # Errors
    ///
    /// Returns `Err(CannotReallocInPlace)` when the allocator is unable to assert that the memory block
    /// referenced by `ptr` could fit `layout`.
    ///
    /// Note that one cannot pass `CannotReallocInPlace` to the `handle_alloc_error` function; clients are
    /// expected either to be able to recover from `grow_in_place` failures without aborting, or to fall back
    /// on another reallocation method before resorting to an abort.
    unsafe fn grow_in_place(
        &mut self,
        ptr: NonNull<u8>,
        layout: Layout,
        new_size: usize,
    ) -> Result<(), CannotReallocInPlace> {
        let _ = ptr; // this default implementation doesn't care about the actual address
        debug_assert!(new_size >= layout.size());
        let (_l, u) = self.usable_size(&layout);
        // _l <= layout.size()                       [guaranteed by usable_size()]
        //       layout.size() <= new_layout.size()  [required by this method]
        if new_size <= u { Ok(()) } else { Err(CannotReallocInPlace) }
    }

    /// Attempts to shrink the allocation referenced by `ptr` to fit `new_size`.
    ///
    /// If this returns `Ok`, then the allocator has asserted that the memory block referenced by `ptr` now
    /// fits `new_size`, and thus can only be used to carry data of that smaller layout. (The allocator is
    /// allowed to take advantage of this, carving off portions of the block for reuse elsewhere.) The
    /// truncated contents of the block within the smaller layout are unaltered, and ownership of block has
    /// not been transferred.
    ///
    /// If this returns `Err`, then the memory block is considered to still represent the original (larger)
    /// `layout`. None of the block has been carved off for reuse elsewhere, ownership of the memory block has
    /// not been transferred, and the contents of the memory block are unaltered.
    ///
    /// # SAFETY
    ///
    /// This function is unsafe because undefined behavior can result if the caller does not ensure all of the
    /// following:
    ///
    /// * `ptr` must be currently allocated via this allocator,
    ///
    /// * `layout` must *fit* the `ptr` (see above); note the `new_size` argument need not fit it,
    ///
    /// * `new_size` must not be greater than `layout.size()` (and must be greater than zero),
    ///
    /// # Errors
    ///
    /// Returns `Err(CannotReallocInPlace)` when the allocator is unable to assert that the memory block
    /// referenced by `ptr` could fit `layout`.
    ///
    /// Note that one cannot pass `CannotReallocInPlace` to the `handle_alloc_error` function; clients are
    /// expected either to be able to recover from `shrink_in_place` failures without aborting, or to fall
    /// back on another reallocation method before resorting to an abort.
    unsafe fn shrink_in_place(
        &mut self,
        ptr: NonNull<u8>,
        layout: Layout,
        new_size: usize,
    ) -> Result<(), CannotReallocInPlace> {
        let _ = ptr; // this default implementation doesn't care about the actual address
        debug_assert!(new_size <= layout.size());
        let (l, _u) = self.usable_size(&layout);
        //                      layout.size() <= _u  [guaranteed by usable_size()]
        // new_layout.size() <= layout.size()        [required by this method]
        if l <= new_size { Ok(()) } else { Err(CannotReallocInPlace) }
    }

    // == COMMON USAGE PATTERNS ==
    // alloc_one, dealloc_one, alloc_array, realloc_array. dealloc_array

    /// Allocates a block suitable for holding an instance of `T`.
    ///
    /// Captures a common usage pattern for allocators.
    ///
    /// The returned block is suitable for passing to the `alloc`/`realloc` methods of this allocator.
    ///
    /// Note to implementors: If this returns `Ok(ptr)`, then `ptr` must be considered "currently allocated"
    /// and must be acceptable input to methods such as `realloc` or `dealloc`, *even if* `T` is a zero-sized
    /// type. In other words, if your `Alloc` implementation overrides this method in a manner that can return
    /// a zero-sized `ptr`, then all reallocation and deallocation methods need to be similarly overridden to
    /// accept such values as input.
    ///
    /// # Errors
    ///
    /// Returning `Err` indicates that either memory is exhausted or `T` does not meet allocator's size or
    /// alignment constraints.
    ///
    /// For zero-sized `T`, may return either of `Ok` or `Err`, but will *not* yield undefined behavior.
    ///
    /// Clients wishing to abort computation in response to an allocation error are encouraged to call the
    /// [`handle_alloc_error`] function, rather than directly invoking `panic!` or similar.
    ///
    /// [`handle_alloc_error`]: ../../alloc/alloc/fn.handle_alloc_error.html
    fn alloc_one<T>(&mut self) -> Result<NonNull<T>, AllocErr>
    where
        Self: Sized,
    {
        let k = Layout::new::<T>();
        if k.size() > 0 { unsafe { self.alloc(k).map(NonNull::cast) } } else { Err(AllocErr) }
    }

    /// Deallocates a block suitable for holding an instance of `T`.
    ///
    /// The given block must have been produced by this allocator, and must be suitable for storing a `T` (in
    /// terms of alignment as well as minimum and maximum size); otherwise yields undefined behavior.
    ///
    /// Captures a common usage pattern for allocators.
    ///
    /// # SAFETY
    ///
    /// This function is unsafe because undefined behavior can result if the caller does not ensure both:
    ///
    /// * `ptr` must denote a block of memory currently allocated via this allocator
    ///
    /// * the layout of `T` must *fit* that block of memory
    unsafe fn dealloc_one<T>(&mut self, ptr: NonNull<T>)
    where
        Self: Sized,
    {
        let k = Layout::new::<T>();
        if k.size() > 0 {
            unsafe { self.dealloc(ptr.cast(), k) };
        }
    }

    /// Allocates a block suitable for holding `n` instances of `T`.
    ///
    /// Captures a common usage pattern for allocators.
    ///
    /// The returned block is suitable for passing to the `alloc`/`realloc` methods of this allocator.
    ///
    /// Note to implementors: If this returns `Ok(ptr)`, then `ptr` must be considered "currently allocated"
    /// and must be acceptable input to methods such as `realloc` or `dealloc`, *even if* `T` is a zero-sized
    /// type. In other words, if your `Alloc` implementation overrides this method in a manner that can return
    /// a zero-sized `ptr`, then all reallocation and deallocation methods need to be similarly overridden to
    /// accept such values as input.
    ///
    /// # Errors
    ///
    /// Returning `Err` indicates that either memory is exhausted or `[T; n]` does not meet allocator's size
    /// or alignment constraints.
    ///
    /// For zero-sized `T` or `n == 0`, may return either of `Ok` or `Err`, but will *not* yield undefined
    /// behavior.
    ///
    /// Always returns `Err` on arithmetic overflow.
    ///
    /// Clients wishing to abort computation in response to an allocation error are encouraged to call the
    /// [`handle_alloc_error`] function, rather than directly invoking `panic!` or similar.
    ///
    /// [`handle_alloc_error`]: ../../alloc/alloc/fn.handle_alloc_error.html
    fn alloc_array<T>(&mut self, n: usize) -> Result<NonNull<T>, AllocErr>
    where
        Self: Sized,
    {
        match Layout::array::<T>(n) {
            Ok(layout) if layout.size() > 0 => unsafe { self.alloc(layout).map(NonNull::cast) },
            _ => Err(AllocErr),
        }
    }

    /// Reallocates a block previously suitable for holding `n_old` instances of `T`, returning a block
    /// suitable for holding `n_new` instances of `T`.
    ///
    /// Captures a common usage pattern for allocators.
    ///
    /// The returned block is suitable for passing to the `alloc`/`realloc` methods of this allocator.
    ///
    /// # SAFETY
    ///
    /// This function is unsafe because undefined behavior can result if the caller does not ensure all of the
    /// following:
    ///
    /// * `ptr` must be currently allocated via this allocator,
    ///
    /// * the layout of `[T; n_old]` must *fit* that block of memory
    ///
    /// # Errors
    ///
    /// Returning `Err` indicates that either memory is exhausted or `[T; n_new]` does not meet allocator's
    /// size or alignment constraints.
    ///
    /// For zero-sized `T` or `n_new == 0`, may return either of `Ok` or `Err`, but will *not* yield undefined
    /// behavior.
    ///
    /// Always returns `Err` on arithmetic overflow.
    ///
    /// Clients wishing to abort computation in response to a reallocation error are encouraged to call the
    /// [`handle_alloc_error`] function, rather than directly invoking `panic!` or similar.
    ///
    /// [`handle_alloc_error`]: ../../alloc/alloc/fn.handle_alloc_error.html
    unsafe fn realloc_array<T>(
        &mut self,
        ptr: NonNull<T>,
        n_old: usize,
        n_new: usize,
    ) -> Result<NonNull<T>, AllocErr>
    where
        Self: Sized,
    {
        match (Layout::array::<T>(n_old), Layout::array::<T>(n_new)) {
            (Ok(ref k_old), Ok(ref k_new)) if k_old.size() > 0 && k_new.size() > 0 => {
                debug_assert!(k_old.align() == k_new.align());
                unsafe { self.realloc(ptr.cast(), *k_old, k_new.size()) }.map(NonNull::cast)
            }
            _ => Err(AllocErr),
        }
    }

    /// Deallocates a block suitable for holding `n` instances of `T`.
    ///
    /// Captures a common usage pattern for allocators.
    ///
    /// # SAFETY
    ///
    /// This function is unsafe because undefined behavior can result if the caller does not ensure both:
    ///
    /// * `ptr` must denote a block of memory currently allocated via this allocator
    ///
    /// * the layout of `[T; n]` must *fit* that block of memory
    ///
    /// # Errors
    ///
    /// Returning `Err` indicates that either `[T; n]` or the given memory block does not meet allocator's
    /// size or alignment constraints.
    ///
    /// Always returns `Err` on arithmetic overflow.
    unsafe fn dealloc_array<T>(&mut self, ptr: NonNull<T>, n: usize) -> Result<(), AllocErr>
    where
        Self: Sized,
    {
        match Layout::array::<T>(n) {
            Ok(k) if k.size() > 0 => {
                unsafe { self.dealloc(ptr.cast(), k) };
                Ok(())
            }
            _ => Err(AllocErr),
        }
    }
}