musli_allocator/lib.rs
1//! [<img alt="github" src="https://img.shields.io/badge/github-udoprog/musli-8da0cb?style=for-the-badge&logo=github" height="20">](https://github.com/udoprog/musli)
2//! [<img alt="crates.io" src="https://img.shields.io/crates/v/musli-allocator.svg?style=for-the-badge&color=fc8d62&logo=rust" height="20">](https://crates.io/crates/musli-allocator)
3//! [<img alt="docs.rs" src="https://img.shields.io/badge/docs.rs-musli--allocator-66c2a5?style=for-the-badge&logoColor=white&logo=" height="20">](https://docs.rs/musli-allocator)
4//!
5//! Allocation support for [Müsli].
6//!
7//! This crate contains two types of allocators:
8//! * The [`System`] allocator, which uses the system allocation facilities.
9//! Particularly [`std::alloc::System`].
10//! * The [`Stack`] allocator, which can allocate buffers from a fixed-size
11//! slice.
12//!
13//! <br>
14//!
15//! ## Examples
16//!
17//! ```
18//! use musli::{Allocator, Buf};
19//!
20//! musli_allocator::with(|alloc| {
21//! let mut a = alloc.alloc().expect("allocation a failed");
22//! let mut b = alloc.alloc().expect("allocation b failed");
23//!
24//! b.write(b"He11o");
25//! a.write(b.as_slice());
26//!
27//! assert_eq!(a.as_slice(), b"He11o");
28//! assert_eq!(a.len(), 5);
29//!
30//! a.write(b" W0rld");
31//!
32//! assert_eq!(a.as_slice(), b"He11o W0rld");
33//! assert_eq!(a.len(), 11);
34//!
35//! let mut c = alloc.alloc().expect("allocation c failed");
36//! c.write(b"!");
37//! a.write(c.as_slice());
38//!
39//! assert_eq!(a.as_slice(), b"He11o W0rld!");
40//! assert_eq!(a.len(), 12);
41//! });
42//! ```
43//!
44//! [Müsli]: <https://docs.rs/musli>
45//! [`std::alloc::System`]: https://doc.rust-lang.org/std/alloc/struct.System.html
46
47#![deny(missing_docs)]
48#![no_std]
49#![cfg_attr(doc_cfg, feature(doc_cfg))]
50
51#[cfg_attr(test, macro_use)]
52#[cfg(feature = "std")]
53extern crate std;
54
55#[cfg(feature = "alloc")]
56extern crate alloc;
57
58#[cfg(test)]
59mod tests;
60
61#[cfg(feature = "alloc")]
62mod system;
63
64#[cfg(feature = "alloc")]
65#[cfg_attr(doc_cfg, doc(cfg(feature = "alloc")))]
66pub use self::system::System;
67
68mod disabled;
69pub use self::disabled::Disabled;
70
71mod stack;
72#[doc(inline)]
73pub use self::stack::{Stack, StackBuffer};
74
75/// The default stack buffer size for the default allocator provided through
76/// [`with`].
77pub const DEFAULT_STACK_BUFFER: usize = 4096;
78
79/// The default allocator.
80#[cfg(feature = "alloc")]
81pub type Default<'a> = System;
82
83/// The default allocator.
84#[cfg(not(feature = "alloc"))]
85pub type Default<'a> = Stack<'a>;
86
87/// Call the given closure with the default allocator.
88///
89/// This is useful if you want to write application which are agnostic to
90/// whether the `alloc` feature is or isn't enabled.
91///
92/// * If the `alloc` feature is enabled, this is the [`System`] allocator.
93/// * If the `alloc` feature is disabled, this is the [`Stack`] allocator with
94/// [`DEFAULT_STACK_BUFFER`] bytes allocated on the stack.
95///
96/// # Examples
97///
98/// ```
99/// use musli::{Allocator, Buf};
100///
101/// musli_allocator::with(|alloc| {
102/// let mut a = alloc.alloc().expect("allocation a failed");
103/// let mut b = alloc.alloc().expect("allocation b failed");
104///
105/// b.write(b"He11o");
106/// a.write(b.as_slice());
107///
108/// assert_eq!(a.as_slice(), b"He11o");
109/// assert_eq!(a.len(), 5);
110///
111/// a.write(b" W0rld");
112///
113/// assert_eq!(a.as_slice(), b"He11o W0rld");
114/// assert_eq!(a.len(), 11);
115///
116/// let mut c = alloc.alloc().expect("allocation c failed");
117/// c.write(b"!");
118/// a.write(c.as_slice());
119///
120/// assert_eq!(a.as_slice(), b"He11o W0rld!");
121/// assert_eq!(a.len(), 12);
122/// });
123/// ```
124#[inline(always)]
125#[track_caller]
126pub fn with<F, O>(f: F) -> O
127where
128 F: FnOnce(&Default<'_>) -> O,
129{
130 with_impl(f)
131}
132
133#[cfg(feature = "alloc")]
134#[inline(always)]
135#[track_caller]
136fn with_impl<F, O>(f: F) -> O
137where
138 F: FnOnce(&System) -> O,
139{
140 let alloc = System::new();
141 f(&alloc)
142}
143
144#[cfg(not(feature = "alloc"))]
145#[inline(always)]
146#[track_caller]
147fn with_impl<F, O>(f: F) -> O
148where
149 F: FnOnce(&Stack<'_>) -> O,
150{
151 let mut buf = StackBuffer::<DEFAULT_STACK_BUFFER>::new();
152 let alloc = Stack::new(&mut buf);
153 f(&alloc)
154}