1use std::sync::Arc;
2
3use indexmap::IndexMap;
4use mir_types::{Location, Name, Type};
5use rustc_hash::FxHashMap;
6use serde::{Deserialize, Serialize};
7
8mod interned_types {
15 use super::*;
16 use std::sync::OnceLock;
17
18 fn intern_string() -> Arc<Type> {
19 Arc::new(Type::string())
20 }
21
22 fn intern_int() -> Arc<Type> {
23 Arc::new(Type::int())
24 }
25
26 fn intern_float() -> Arc<Type> {
27 Arc::new(Type::float())
28 }
29
30 fn intern_bool() -> Arc<Type> {
31 Arc::new(Type::bool())
32 }
33
34 fn intern_mixed() -> Arc<Type> {
35 Arc::new(Type::mixed())
36 }
37
38 fn intern_null() -> Arc<Type> {
39 Arc::new(Type::null())
40 }
41
42 fn intern_void() -> Arc<Type> {
43 Arc::new(Type::void())
44 }
45
46 static STRING: OnceLock<Arc<Type>> = OnceLock::new();
47 static INT: OnceLock<Arc<Type>> = OnceLock::new();
48 static FLOAT: OnceLock<Arc<Type>> = OnceLock::new();
49 static BOOL: OnceLock<Arc<Type>> = OnceLock::new();
50 static MIXED: OnceLock<Arc<Type>> = OnceLock::new();
51 static NULL: OnceLock<Arc<Type>> = OnceLock::new();
52 static VOID: OnceLock<Arc<Type>> = OnceLock::new();
53
54 pub fn string() -> Arc<Type> {
55 STRING.get_or_init(intern_string).clone()
56 }
57
58 pub fn int() -> Arc<Type> {
59 INT.get_or_init(intern_int).clone()
60 }
61
62 pub fn float() -> Arc<Type> {
63 FLOAT.get_or_init(intern_float).clone()
64 }
65
66 pub fn bool() -> Arc<Type> {
67 BOOL.get_or_init(intern_bool).clone()
68 }
69
70 pub fn mixed() -> Arc<Type> {
71 MIXED.get_or_init(intern_mixed).clone()
72 }
73
74 pub fn null() -> Arc<Type> {
75 NULL.get_or_init(intern_null).clone()
76 }
77
78 pub fn void() -> Arc<Type> {
79 VOID.get_or_init(intern_void).clone()
80 }
81
82 static GLOBAL_UNION_INTERN: std::sync::OnceLock<dashmap::DashMap<Type, Arc<Type>>> =
95 std::sync::OnceLock::new();
96
97 fn global_intern_table() -> &'static dashmap::DashMap<Type, Arc<Type>> {
98 GLOBAL_UNION_INTERN.get_or_init(dashmap::DashMap::default)
99 }
100
101 pub fn intern_or_wrap(union: Type) -> Arc<Type> {
103 if union.types.len() == 1 && !union.possibly_undefined && !union.from_docblock {
106 match &union.types[0] {
107 mir_types::Atomic::TString => return string(),
108 mir_types::Atomic::TInt => return int(),
109 mir_types::Atomic::TFloat => return float(),
110 mir_types::Atomic::TBool => return bool(),
111 mir_types::Atomic::TMixed => return mixed(),
112 mir_types::Atomic::TNull => return null(),
113 mir_types::Atomic::TVoid => return void(),
114 _ => {}
115 }
116 }
117 if union.types.is_empty() {
120 return Arc::new(union);
121 }
122 let table = global_intern_table();
124 if let Some(existing) = table.get(&union) {
125 return Arc::clone(existing.value());
126 }
127 let arc = Arc::new(union.clone());
128 match table.entry(union) {
132 dashmap::mapref::entry::Entry::Occupied(o) => Arc::clone(o.get()),
133 dashmap::mapref::entry::Entry::Vacant(v) => {
134 v.insert(Arc::clone(&arc));
135 arc
136 }
137 }
138 }
139}
140
141#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Serialize, Deserialize)]
146pub enum Visibility {
147 Public,
148 Protected,
149 Private,
150}
151
152impl Visibility {
153 pub fn is_at_least(&self, required: Visibility) -> bool {
154 *self <= required
155 }
156}
157
158impl std::fmt::Display for Visibility {
159 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
160 match self {
161 Visibility::Public => write!(f, "public"),
162 Visibility::Protected => write!(f, "protected"),
163 Visibility::Private => write!(f, "private"),
164 }
165 }
166}
167
168#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
169pub struct TemplateParam {
170 pub name: Name,
171 pub bound: Option<Type>,
172 pub defining_entity: Name,
174 pub variance: mir_types::Variance,
175}
176
177#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
178pub struct FnParam {
179 pub name: Name,
180 #[serde(
184 deserialize_with = "deserialize_param_type",
185 serialize_with = "serialize_param_type"
186 )]
187 pub ty: Option<Arc<Type>>,
188 pub has_default: bool,
191 pub is_variadic: bool,
192 pub is_byref: bool,
193 pub is_optional: bool,
194}
195
196impl std::hash::Hash for FnParam {
197 fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
198 self.name.hash(state);
199 self.has_default.hash(state);
200 self.is_variadic.hash(state);
201 self.is_byref.hash(state);
202 self.is_optional.hash(state);
203 self.ty.as_deref().hash(state);
207 }
208}
209
210fn deserialize_param_type<'de, D>(deserializer: D) -> Result<Option<Arc<Type>>, D::Error>
212where
213 D: serde::Deserializer<'de>,
214{
215 Option::<Type>::deserialize(deserializer).map(|opt| opt.map(interned_types::intern_or_wrap))
216}
217
218fn serialize_param_type<S>(value: &Option<Arc<Type>>, serializer: S) -> Result<S::Ok, S::Error>
219where
220 S: serde::Serializer,
221{
222 let opt = value.as_ref().map(|arc| (**arc).clone());
223 opt.serialize(serializer)
224}
225
226fn deserialize_return_type<'de, D>(deserializer: D) -> Result<Option<Arc<Type>>, D::Error>
227where
228 D: serde::Deserializer<'de>,
229{
230 Option::<Type>::deserialize(deserializer).map(|opt| opt.map(interned_types::intern_or_wrap))
231}
232
233fn serialize_return_type<S>(value: &Option<Arc<Type>>, serializer: S) -> Result<S::Ok, S::Error>
234where
235 S: serde::Serializer,
236{
237 let opt = value.as_ref().map(|arc| (**arc).clone());
238 opt.serialize(serializer)
239}
240
241fn deserialize_params<'de, D>(deserializer: D) -> Result<Arc<[FnParam]>, D::Error>
242where
243 D: serde::Deserializer<'de>,
244{
245 Vec::<FnParam>::deserialize(deserializer).map(|v| Arc::from(v.into_boxed_slice()))
246}
247
248fn default_imports() -> Arc<FxHashMap<Name, Name>> {
249 Arc::new(FxHashMap::default())
250}
251
252fn deserialize_imports<'de, D>(deserializer: D) -> Result<Arc<FxHashMap<Name, Name>>, D::Error>
257where
258 D: serde::Deserializer<'de>,
259{
260 let raw = FxHashMap::<String, String>::deserialize(deserializer)?;
261 let mut out: FxHashMap<Name, Name> =
262 FxHashMap::with_capacity_and_hasher(raw.len(), Default::default());
263 for (k, v) in raw {
264 out.insert(Name::new(&k), Name::new(&v));
265 }
266 Ok(Arc::new(out))
267}
268
269fn serialize_imports<S>(
273 value: &Arc<FxHashMap<Name, Name>>,
274 serializer: S,
275) -> Result<S::Ok, S::Error>
276where
277 S: serde::Serializer,
278{
279 use serde::ser::SerializeMap;
280 let mut map = serializer.serialize_map(Some(value.len()))?;
281 for (k, v) in value.iter() {
282 map.serialize_entry(k.as_str(), v.as_str())?;
283 }
284 map.end()
285}
286
287fn serialize_params<S>(value: &Arc<[FnParam]>, serializer: S) -> Result<S::Ok, S::Error>
288where
289 S: serde::Serializer,
290{
291 value.as_ref().serialize(serializer)
292}
293
294pub fn wrap_param_type(ty: Option<Type>) -> Option<Arc<Type>> {
296 ty.map(interned_types::intern_or_wrap)
297}
298
299pub fn wrap_return_type(ty: Option<Type>) -> Option<Arc<Type>> {
301 ty.map(interned_types::intern_or_wrap)
302}
303
304#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
309pub enum AssertionKind {
310 Assert,
311 AssertIfTrue,
312 AssertIfFalse,
313}
314
315#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
316pub struct Assertion {
317 pub kind: AssertionKind,
318 pub param: Arc<str>,
319 pub ty: Type,
320}
321
322#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
327pub struct MethodDef {
328 pub name: Arc<str>,
329 pub fqcn: Arc<str>,
330 #[serde(
331 deserialize_with = "deserialize_params",
332 serialize_with = "serialize_params"
333 )]
334 pub params: Arc<[FnParam]>,
335 #[serde(
339 deserialize_with = "deserialize_return_type",
340 serialize_with = "serialize_return_type"
341 )]
342 pub return_type: Option<Arc<Type>>,
343 #[serde(
348 deserialize_with = "deserialize_return_type",
349 serialize_with = "serialize_return_type"
350 )]
351 pub inferred_return_type: Option<Arc<Type>>,
352 pub visibility: Visibility,
353 pub is_static: bool,
354 pub is_abstract: bool,
355 pub is_final: bool,
356 pub is_constructor: bool,
357 pub template_params: Vec<TemplateParam>,
358 pub assertions: Vec<Assertion>,
359 pub throws: Vec<Arc<str>>,
360 pub deprecated: Option<Arc<str>>,
361 pub is_internal: bool,
362 pub is_pure: bool,
363 pub location: Option<Location>,
364 #[serde(default)]
367 pub docstring: Option<Arc<str>>,
368 #[serde(default)]
371 pub is_virtual: bool,
372}
373
374impl MethodDef {
375 pub fn effective_return_type(&self) -> Option<&Type> {
376 self.return_type
377 .as_deref()
378 .or(self.inferred_return_type.as_deref())
379 }
380}
381
382#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
387pub struct PropertyDef {
388 pub name: Arc<str>,
389 pub ty: Option<Type>,
390 pub inferred_ty: Option<Type>,
391 pub visibility: Visibility,
392 pub is_static: bool,
393 pub is_readonly: bool,
394 pub default: Option<Type>,
395 pub location: Option<Location>,
396}
397
398#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
403pub struct ConstantDef {
404 pub name: Arc<str>,
405 pub ty: Type,
406 pub visibility: Option<Visibility>,
407 #[serde(default)]
408 pub is_final: bool,
409 pub location: Option<Location>,
410}
411
412#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
417pub struct ClassDef {
418 pub fqcn: Arc<str>,
419 pub short_name: Arc<str>,
420 pub parent: Option<Arc<str>>,
421 pub interfaces: Vec<Arc<str>>,
422 pub traits: Vec<Arc<str>>,
423 pub own_methods: IndexMap<Arc<str>, Arc<MethodDef>>,
424 pub own_properties: IndexMap<Arc<str>, PropertyDef>,
425 pub own_constants: IndexMap<Arc<str>, ConstantDef>,
426 #[serde(default)]
427 pub mixins: Vec<Arc<str>>,
428 pub template_params: Vec<TemplateParam>,
429 pub extends_type_args: Vec<Type>,
431 #[serde(default)]
433 pub implements_type_args: Vec<(Arc<str>, Vec<Type>)>,
434 pub is_abstract: bool,
435 pub is_final: bool,
436 pub is_readonly: bool,
437 pub deprecated: Option<Arc<str>>,
438 pub is_internal: bool,
439 pub location: Option<Location>,
440 #[serde(default)]
444 pub trait_use_locations: Vec<(Arc<str>, Location)>,
445 #[serde(default)]
447 pub type_aliases: FxHashMap<Arc<str>, Type>,
448 #[serde(default)]
450 pub pending_import_types: Vec<(Arc<str>, Arc<str>, Arc<str>)>,
451}
452
453impl ClassDef {
454 pub fn get_method(&self, name: &str) -> Option<&MethodDef> {
455 self.own_methods.get(name).map(Arc::as_ref).or_else(|| {
459 self.own_methods
460 .iter()
461 .find(|(k, _)| k.as_ref().eq_ignore_ascii_case(name))
462 .map(|(_, v)| v.as_ref())
463 })
464 }
465
466 pub fn get_property(&self, name: &str) -> Option<&PropertyDef> {
467 self.own_properties.get(name)
468 }
469}
470
471#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
476pub struct InterfaceDef {
477 pub fqcn: Arc<str>,
478 pub short_name: Arc<str>,
479 pub extends: Vec<Arc<str>>,
480 pub own_methods: IndexMap<Arc<str>, Arc<MethodDef>>,
481 pub own_constants: IndexMap<Arc<str>, ConstantDef>,
482 pub template_params: Vec<TemplateParam>,
483 pub location: Option<Location>,
484}
485
486#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
491pub struct TraitDef {
492 pub fqcn: Arc<str>,
493 pub short_name: Arc<str>,
494 pub own_methods: IndexMap<Arc<str>, Arc<MethodDef>>,
495 pub own_properties: IndexMap<Arc<str>, PropertyDef>,
496 pub own_constants: IndexMap<Arc<str>, ConstantDef>,
497 pub template_params: Vec<TemplateParam>,
498 pub traits: Vec<Arc<str>>,
500 pub location: Option<Location>,
501 #[serde(default)]
503 pub require_extends: Vec<Arc<str>>,
504 #[serde(default)]
506 pub require_implements: Vec<Arc<str>>,
507}
508
509#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
514pub struct EnumCaseDef {
515 pub name: Arc<str>,
516 pub value: Option<Type>,
517 pub location: Option<Location>,
518}
519
520#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
521pub struct EnumDef {
522 pub fqcn: Arc<str>,
523 pub short_name: Arc<str>,
524 pub scalar_type: Option<Type>,
525 pub interfaces: Vec<Arc<str>>,
526 pub cases: IndexMap<Arc<str>, EnumCaseDef>,
527 pub own_methods: IndexMap<Arc<str>, Arc<MethodDef>>,
528 pub own_constants: IndexMap<Arc<str>, ConstantDef>,
529 pub location: Option<Location>,
530}
531
532#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
537pub struct FunctionDef {
538 pub fqn: Arc<str>,
539 pub short_name: Arc<str>,
540 #[serde(
541 deserialize_with = "deserialize_params",
542 serialize_with = "serialize_params"
543 )]
544 pub params: Arc<[FnParam]>,
545 #[serde(
548 deserialize_with = "deserialize_return_type",
549 serialize_with = "serialize_return_type"
550 )]
551 pub return_type: Option<Arc<Type>>,
552 #[serde(
555 deserialize_with = "deserialize_return_type",
556 serialize_with = "serialize_return_type"
557 )]
558 pub inferred_return_type: Option<Arc<Type>>,
559 pub template_params: Vec<TemplateParam>,
560 pub assertions: Vec<Assertion>,
561 pub throws: Vec<Arc<str>>,
562 pub deprecated: Option<Arc<str>>,
563 pub is_pure: bool,
564 pub location: Option<Location>,
565 #[serde(default)]
568 pub docstring: Option<Arc<str>>,
569}
570
571impl FunctionDef {
572 pub fn effective_return_type(&self) -> Option<&Type> {
573 self.return_type
574 .as_deref()
575 .or(self.inferred_return_type.as_deref())
576 }
577}
578
579#[derive(Debug, Clone, Default, PartialEq, serde::Serialize, serde::Deserialize)]
589pub struct StubSlice {
590 pub classes: Vec<Arc<ClassDef>>,
591 pub interfaces: Vec<Arc<InterfaceDef>>,
592 pub traits: Vec<Arc<TraitDef>>,
593 pub enums: Vec<Arc<EnumDef>>,
594 pub functions: Vec<Arc<FunctionDef>>,
595 #[serde(default)]
596 pub constants: Vec<(Arc<str>, Type)>,
597 #[serde(default)]
600 pub file: Option<Arc<str>>,
601 #[serde(default)]
605 pub global_vars: Vec<(Arc<str>, Type)>,
606 #[serde(default)]
610 pub namespace: Option<Arc<str>>,
611 #[serde(
621 deserialize_with = "deserialize_imports",
622 serialize_with = "serialize_imports"
623 )]
624 #[serde(default = "default_imports")]
625 pub imports: Arc<FxHashMap<Name, Name>>,
626 #[serde(skip)]
629 pub is_deduped: bool,
630}
631
632use std::sync::Mutex;
637
638type ParamCache = Mutex<FxHashMap<Vec<FnParam>, Arc<[FnParam]>>>;
639
640static PARAM_DEDUP_CACHE: std::sync::OnceLock<ParamCache> = std::sync::OnceLock::new();
644
645pub fn deduplicate_params_in_slice(slice: &mut StubSlice) {
653 let cache: &ParamCache = PARAM_DEDUP_CACHE.get_or_init(|| Mutex::new(FxHashMap::default()));
654 let mut canonical_params = cache.lock().unwrap();
655
656 let mut deduplicate = |params: &mut Arc<[FnParam]>| {
657 if let Some(existing) = canonical_params.get(params.as_ref()) {
658 *params = existing.clone();
659 } else {
660 canonical_params.insert(params.as_ref().to_vec(), params.clone());
661 }
662 };
663
664 for cls in &mut slice.classes {
666 for method in Arc::make_mut(cls).own_methods.values_mut() {
667 deduplicate(&mut Arc::make_mut(method).params);
668 }
669 }
670
671 for iface in &mut slice.interfaces {
673 for method in Arc::make_mut(iface).own_methods.values_mut() {
674 deduplicate(&mut Arc::make_mut(method).params);
675 }
676 }
677
678 for tr in &mut slice.traits {
680 for method in Arc::make_mut(tr).own_methods.values_mut() {
681 deduplicate(&mut Arc::make_mut(method).params);
682 }
683 }
684
685 for en in &mut slice.enums {
687 for method in Arc::make_mut(en).own_methods.values_mut() {
688 deduplicate(&mut Arc::make_mut(method).params);
689 }
690 }
691
692 for func in &mut slice.functions {
694 deduplicate(&mut Arc::make_mut(func).params);
695 }
696 slice.is_deduped = true;
697}