1use core::hash::{Hash, Hasher};
2
3use crate::{
4 binding_model::{self},
5 FastIndexMap,
6};
7
8#[derive(Debug, Copy, Clone, PartialEq, Eq)]
10pub enum Origin {
11 Pool,
13 Derived,
15}
16
17#[derive(Debug, Default, Clone, Eq)]
21pub struct EntryMap {
22 inner: FastIndexMap<u32, wgt::BindGroupLayoutEntry>,
25 sorted: bool,
31}
32
33impl PartialEq for EntryMap {
34 fn eq(&self, other: &Self) -> bool {
35 self.assert_sorted();
36 other.assert_sorted();
37
38 self.inner == other.inner
39 }
40}
41
42impl Hash for EntryMap {
43 fn hash<H: Hasher>(&self, state: &mut H) {
44 self.assert_sorted();
45
46 for entry in self.inner.values() {
51 entry.hash(state);
52 }
53 }
54}
55
56impl EntryMap {
57 fn assert_sorted(&self) {
58 assert!(self.sorted);
59 }
60
61 pub fn from_entries(
66 entries: &[wgt::BindGroupLayoutEntry],
67 ) -> Result<Self, binding_model::CreateBindGroupLayoutError> {
68 let mut inner = FastIndexMap::with_capacity_and_hasher(entries.len(), Default::default());
69 for entry in entries {
70 if inner.insert(entry.binding, *entry).is_some() {
71 return Err(binding_model::CreateBindGroupLayoutError::ConflictBinding(
72 entry.binding,
73 ));
74 }
75 }
76 inner.sort_unstable_keys();
77
78 Ok(Self {
79 inner,
80 sorted: true,
81 })
82 }
83
84 pub fn len(&self) -> usize {
86 self.inner.len()
87 }
88
89 pub fn get(&self, binding: u32) -> Option<&wgt::BindGroupLayoutEntry> {
91 self.inner.get(&binding)
92 }
93
94 pub fn indices(&self) -> impl ExactSizeIterator<Item = u32> + '_ {
96 self.inner.keys().copied()
97 }
98
99 pub fn values(&self) -> impl ExactSizeIterator<Item = &wgt::BindGroupLayoutEntry> + '_ {
101 self.inner.values()
102 }
103
104 pub fn iter(&self) -> impl ExactSizeIterator<Item = (&u32, &wgt::BindGroupLayoutEntry)> + '_ {
105 self.inner.iter()
106 }
107
108 pub fn is_empty(&self) -> bool {
109 self.inner.is_empty()
110 }
111
112 pub fn contains_key(&self, key: u32) -> bool {
113 self.inner.contains_key(&key)
114 }
115
116 pub fn entry(&mut self, key: u32) -> indexmap::map::Entry<'_, u32, wgt::BindGroupLayoutEntry> {
117 self.sorted = false;
118 self.inner.entry(key)
119 }
120
121 pub fn sort(&mut self) {
122 self.inner.sort_unstable_keys();
123 self.sorted = true;
124 }
125}