1mod node;
2mod node_children;
3mod node_text;
4mod text_info;
56pub(crate) use self::node::Node;
7pub(crate) use self::node_children::NodeChildren;
8pub(crate) use self::node_text::NodeText;
9pub(crate) use self::text_info::TextInfo;
1011// Type used for storing tree metadata, such as byte and char length.
12pub(crate) type Count = u64;
1314// Real constants used in release builds.
15#[cfg(not(any(test, feature = "small_chunks")))]
16mod constants {
17use super::{Node, TextInfo};
18use smallvec::SmallVec;
19use std::{
20 mem::{align_of, size_of},
21 sync::Arc,
22 };
2324// Because stdlib's max is not const for some reason.
25 // TODO: replace with stdlib max once it's const.
26const fn cmax(a: usize, b: usize) -> usize {
27if a > b {
28 a
29 } else {
30 b
31 }
32 }
3334// Aim for Node + Arc counters to be 1024 bytes. Keeping the nodes
35 // multiples of large powers of two makes it easier for the memory
36 // allocator to avoid fragmentation.
37const TARGET_TOTAL_SIZE: usize = 1024;
3839// Space that the strong and weak Arc counters take up in `ArcInner`.
40const ARC_COUNTERS_SIZE: usize = size_of::<std::sync::atomic::AtomicUsize>() * 2;
4142// Misc useful info that we need below.
43const NODE_CHILDREN_ALIGN: usize = cmax(align_of::<Arc<u8>>(), align_of::<TextInfo>());
44const NODE_TEXT_ALIGN: usize = align_of::<SmallVec<[u8; 16]>>();
45const START_OFFSET: usize = {
46const NODE_INNER_ALIGN: usize = cmax(NODE_CHILDREN_ALIGN, NODE_TEXT_ALIGN);
47// The +NODE_INNER_ALIGN is because of Node's enum discriminant.
48ARC_COUNTERS_SIZE + NODE_INNER_ALIGN
49 };
5051// Node maximums.
52#[doc(hidden)] // NOT PART OF THE PUBLIC API!
53pub const MAX_CHILDREN: usize = {
54let node_list_align = align_of::<Arc<u8>>();
55let info_list_align = align_of::<TextInfo>();
56let field_gap = if node_list_align >= info_list_align {
570
58} else {
59// This is over-conservative, because in reality it depends
60 // on the number of elements. But handling that is probably
61 // more complexity than it's worth.
62info_list_align - node_list_align
63 };
6465// The -NODE_CHILDREN_ALIGN is for the `len` field in `NodeChildrenInternal`.
66let target_size = TARGET_TOTAL_SIZE - START_OFFSET - NODE_CHILDREN_ALIGN - field_gap;
6768 target_size / (size_of::<Arc<u8>>() + size_of::<TextInfo>())
69 };
70#[doc(hidden)] // NOT PART OF THE PUBLIC API!
71pub const MAX_BYTES: usize = {
72let smallvec_overhead = size_of::<SmallVec<[u8; 16]>>() - 16;
73 TARGET_TOTAL_SIZE - START_OFFSET - smallvec_overhead
74 };
7576// Node minimums.
77 // Note: MIN_BYTES is intentionally a little smaller than half
78 // MAX_BYTES, to give a little wiggle room when on the edge of
79 // merging/splitting.
80#[doc(hidden)] // NOT PART OF THE PUBLIC API!
81pub const MIN_CHILDREN: usize = MAX_CHILDREN / 2;
82#[doc(hidden)] // NOT PART OF THE PUBLIC API!
83pub const MIN_BYTES: usize = (MAX_BYTES / 2) - (MAX_BYTES / 32);
8485// Compile-time assertion.
86const _: () = {
87assert!(
88 (ARC_COUNTERS_SIZE + size_of::<Node>()) == TARGET_TOTAL_SIZE,
89"`Node` is not the target size in memory.",
90 );
91 };
92}
9394// Smaller constants used in debug builds. These are different from release
95// in order to trigger deeper trees without having to use huge text data in
96// the tests.
97#[cfg(any(test, feature = "small_chunks"))]
98mod test_constants {
99#[doc(hidden)] // NOT PART OF THE PUBLIC API!
100pub const MAX_CHILDREN: usize = 5;
101#[doc(hidden)] // NOT PART OF THE PUBLIC API!
102pub const MIN_CHILDREN: usize = MAX_CHILDREN / 2;
103104// MAX_BYTES must be >= 4 to allow for 4-byte utf8 characters.
105#[doc(hidden)] // NOT PART OF THE PUBLIC API!
106pub const MAX_BYTES: usize = 9; // Note: can't be 8, because 3-byte characters.
107#[doc(hidden)] // NOT PART OF THE PUBLIC API!
108pub const MIN_BYTES: usize = (MAX_BYTES / 2) - (MAX_BYTES / 32);
109}
110111#[cfg(not(any(test, feature = "small_chunks")))]
112pub use self::constants::{MAX_BYTES, MAX_CHILDREN, MIN_BYTES, MIN_CHILDREN};
113114#[cfg(any(test, feature = "small_chunks"))]
115pub use self::test_constants::{MAX_BYTES, MAX_CHILDREN, MIN_BYTES, MIN_CHILDREN};