1//! Trait for defining the flavor of a syntree.
23use core::ops::DerefMut;
45use crate::index::{Index, Length, TreeIndex};
6use crate::pointer::{Pointer, Width};
78/// Storage being used in a tree.
9pub trait Storage<T>
10where
11Self: Sized + DerefMut<Target = [T]>,
12{
13/// A storage error.
14type Error: 'static;
1516/// Empty storage.
17const EMPTY: Self;
1819/// Constructstorage with the given capacity.
20fn with_capacity(capacity: usize) -> Result<Self, Self::Error>;
2122/// Get the capacity of the underlying storage.
23fn capacity(&self) -> usize;
2425/// Push an item into storage.
26fn push(&mut self, item: T) -> Result<(), Self::Error>;
27}
2829#[cfg(feature = "alloc")]
30impl<T> Storage<T> for alloc::vec::Vec<T> {
31type Error = core::convert::Infallible;
3233const EMPTY: Self = alloc::vec::Vec::new();
3435#[inline]
36fn with_capacity(capacity: usize) -> Result<Self, Self::Error> {
37Ok(alloc::vec::Vec::with_capacity(capacity))
38 }
3940#[inline]
41fn capacity(&self) -> usize {
42 alloc::vec::Vec::capacity(self)
43 }
4445#[inline]
46fn push(&mut self, item: T) -> Result<(), Self::Error> {
47 alloc::vec::Vec::push(self, item);
48Ok(())
49 }
50}
5152/// Declare a new flavor.
53///
54/// The available type parameters are:
55/// * `type Index` which declares the index to use.
56/// * `type Width` which declares the width to use, defaults to `usize`.
57///
58/// # Examples
59///
60/// ```
61/// use syntree::{Empty, EmptyVec, TreeIndex};
62///
63/// syntree::flavor! {
64/// struct FlavorEmpty {
65/// type Index = Empty;
66/// type Indexes = EmptyVec<TreeIndex<Self>>;
67/// }
68/// }
69///
70/// syntree::flavor! {
71/// struct FlavorU32 {
72/// type Index = u32;
73/// type Width = u32;
74/// }
75/// }
76/// ```
77#[macro_export]
78macro_rules! flavor {
79 (
80 $(#[doc = $doc:literal])*
81$vis:vis struct $ty:ident {
82type Index = $index:ty;
83 $(type Width = $width:ty;)?
84$(type Storage = $storage:ty;)?
85$(type Indexes = $indexes:ty;)?
86}
87 ) => {
88 $(#[doc = $doc])*
89#[non_exhaustive]
90$vis struct $ty;
9192impl $crate::Flavor for $ty {
93type Error = core::convert::Infallible;
94type Index = $index;
95type Length = <$index as $crate::Index>::Length;
96type Width = $crate::flavor!(@width $($width)*);
97type Pointer = $crate::flavor!(@pointer $($width)*);
98type Storage<T> = $crate::macro_support::Vec<T>;
99type Indexes = $crate::flavor!(@indexes $($indexes)*);
100 }
101 };
102103 (@width $ty:ty) => { $ty };
104 (@width) => { usize };
105 (@pointer $ty:ty) => { <$ty as $crate::pointer::Width>::Pointer };
106 (@pointer) => { <usize as $crate::pointer::Width>::Pointer };
107 (@indexes $ty:ty) => { $ty };
108 (@indexes) => { $crate::macro_support::DefaultIndexes<Self> };
109}
110111flavor! {
112/// The default flavor of a tree.
113 ///
114 /// This corresponds to a `u32` index with a `usize` width.
115pub struct FlavorDefault {
116type Index = u32;
117type Width = usize;
118 }
119}
120121/// The flavor of a tree.
122///
123/// This should not be implemented directly, instead see the [flavor!] macro.
124///
125/// The Index associated type is constrained by the [Index] trait, and
126/// determines the numerical bounds of [spans] the tree.
127///
128/// The `Width` associated type determines the bounds of pointers in the tree
129/// through the [Width] trait, this decides how many elements that can be stored
130/// in the tree.
131///
132/// [spans]: crate::Span
133pub trait Flavor {
134/// The error raised by the type of the tree.
135type Error;
136/// The type of an index used by a tree.
137type Index: Index<Length = Self::Length>;
138/// The length used in the flavor.
139type Length: Length;
140/// The width used in the flavor.
141type Width: Width<Pointer = Self::Pointer>;
142/// The pointer in use.
143type Pointer: Pointer;
144/// The storage type used in the tree.
145type Storage<T>: Storage<T, Error = Self::Error>;
146/// How indexes are stored in the tree.
147type Indexes: Storage<TreeIndex<Self>, Error = Self::Error>;
148}