rune/runtime/value/
rtti.rs
1use core::borrow::Borrow;
2use core::cmp::Ordering;
3use core::hash;
4
5use rust_alloc::sync::Arc;
6
7use serde::{Deserialize, Serialize};
8
9use crate::alloc::prelude::*;
10use crate::alloc::HashMap;
11use crate::item::Item;
12use crate::runtime::{FieldMap, TypeInfo, Value};
13use crate::{Hash, ItemBuf};
14
15#[doc(hidden)]
17pub struct Accessor<'a> {
18 pub(crate) fields: &'a HashMap<Box<str>, usize>,
19 pub(crate) data: &'a [Value],
20}
21
22impl Accessor<'_> {
23 #[doc(hidden)]
25 pub fn get<Q>(&self, key: &Q) -> Option<&Value>
26 where
27 Box<str>: Borrow<Q>,
28 Q: hash::Hash + Eq + ?Sized,
29 {
30 self.data.get(*self.fields.get(key)?)
31 }
32}
33
34#[derive(Debug, Clone, Copy, Serialize, Deserialize)]
36#[serde(rename_all = "kebab-case")]
37pub(crate) enum RttiKind {
38 Empty,
40 Tuple,
42 Struct,
44}
45
46#[derive(Debug, Serialize, Deserialize)]
48#[non_exhaustive]
49pub struct Rtti {
50 pub(crate) kind: RttiKind,
52 pub(crate) hash: Hash,
54 pub(crate) variant_hash: Hash,
56 pub(crate) item: ItemBuf,
58 pub(crate) fields: FieldMap<Box<str>, usize>,
60}
61
62impl Rtti {
63 #[inline]
65 pub(crate) fn is(&self, hash: Hash, variant_hash: Hash) -> bool {
66 self.hash == hash && self.variant_hash == variant_hash
67 }
68
69 #[inline]
71 pub fn item(&self) -> &Item {
72 &self.item
73 }
74
75 #[inline]
77 pub fn type_hash(&self) -> Hash {
78 self.hash
79 }
80
81 #[inline]
83 pub fn type_info(self: Arc<Self>) -> TypeInfo {
84 TypeInfo::rtti(self)
85 }
86}
87
88impl PartialEq for Rtti {
89 fn eq(&self, other: &Self) -> bool {
90 self.hash == other.hash && self.variant_hash == other.variant_hash
91 }
92}
93
94impl Eq for Rtti {}
95
96impl hash::Hash for Rtti {
97 fn hash<H: hash::Hasher>(&self, state: &mut H) {
98 self.hash.hash(state);
99 self.variant_hash.hash(state);
100 }
101}
102
103impl PartialOrd for Rtti {
104 fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
105 Some(self.cmp(other))
106 }
107}
108
109impl Ord for Rtti {
110 #[inline]
111 fn cmp(&self, other: &Self) -> Ordering {
112 self.hash
113 .cmp(&other.hash)
114 .then_with(|| self.variant_hash.cmp(&other.variant_hash))
115 }
116}