1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
use rustc_hash::FxHashMap;

use debris_common::{CompilationId, CompileContext, Ident, Span};

use crate::{
    mir_context::MirContextId,
    mir_nodes::{MirNode, PropertyAccess},
    mir_object::{MirObject, MirObjectId},
};

#[derive(Debug)]
pub struct MirNamespace {
    /// The objects are indexed by `MirObjectId.id`
    /// For this reason, no object may ever be removed from this vec
    pub(super) objects: Vec<MirObject>,
    pub(super) compilation_id: CompilationId,
    pub(super) local_namespaces: Vec<MirLocalNamespace>,
}

impl MirNamespace {
    pub fn new(ctx: &CompileContext) -> Self {
        MirNamespace {
            objects: Vec::new(),
            local_namespaces: Vec::new(),
            compilation_id: ctx.compilation_id,
        }
    }

    /// Creates a new object. This object stores the current context id as the place where it was defined
    pub fn insert_object(&mut self, current_context_id: MirContextId) -> &mut MirObject {
        let id = MirObjectId::new(self.compilation_id, self.objects.len().try_into().unwrap());
        let object = MirObject::new_in(self, current_context_id, id);
        self.objects.push(object);
        self.objects.last_mut().unwrap()
    }

    pub fn get_obj_mut(&mut self, obj: MirObjectId) -> &mut MirObject {
        &mut self.objects[obj.id as usize]
    }

    pub fn get_obj(&self, obj: MirObjectId) -> &MirObject {
        &self.objects[obj.id as usize]
    }

    pub fn insert_local_namespace(&mut self) -> MirLocalNamespaceId {
        let id = self.local_namespaces.len();
        let namespace = MirLocalNamespace::default();
        self.local_namespaces.push(namespace);
        MirLocalNamespaceId(id)
    }

    pub fn get_local_namespace(&self, id: MirLocalNamespaceId) -> &MirLocalNamespace {
        &self.local_namespaces[id.0]
    }

    pub fn get_local_namespace_mut(&mut self, id: MirLocalNamespaceId) -> &mut MirLocalNamespace {
        &mut self.local_namespaces[id.0]
    }

    pub fn get_obj_namespace(&self, obj_id: MirObjectId) -> &MirLocalNamespace {
        let namespace_id = self.get_obj(obj_id).local_namespace_id;
        self.get_local_namespace(namespace_id)
    }

    pub fn get_obj_namespace_mut(&mut self, obj_id: MirObjectId) -> &mut MirLocalNamespace {
        let namespace_id = self.get_obj(obj_id).local_namespace_id;
        self.get_local_namespace_mut(namespace_id)
    }
}

#[derive(Debug, PartialEq, Eq, Clone, Copy)]
pub struct MirLocalNamespaceId(usize);

/// A tuple of the actual object and the span where it was first declared
type LocalNamespaceValue = (MirObjectId, Span);

/// Stores all the identifiers that are local to a given object or context.
#[derive(Debug, Default, Clone)]
pub struct MirLocalNamespace {
    /// All properties of this namespace.
    properties: FxHashMap<Ident, LocalNamespaceValue>,
}

impl MirLocalNamespace {
    pub fn iter(&self) -> impl Iterator<Item = (&Ident, &LocalNamespaceValue)> {
        self.properties.iter()
    }

    pub fn get_property(&self, ident: &Ident) -> Option<MirObjectId> {
        self.properties.get(ident).map(|(obj_id, _)| *obj_id)
    }

    pub fn property_get_or_insert(
        &mut self,
        nodes: &mut Vec<MirNode>,
        namespace: &mut MirNamespace,
        value_id: MirObjectId,
        ident: Ident,
        span: Span,
        current_context_id: MirContextId,
    ) -> MirObjectId {
        self.properties
            .entry(ident.clone())
            .or_insert_with(|| {
                let obj_id = namespace.insert_object(current_context_id).id;
                // namespace.add_maybe_erroneous_obj_info(obj_id, (ident, span));
                nodes.push(MirNode::PropertyAccess(PropertyAccess {
                    property_ident: ident,
                    span,
                    value_id,
                    target_id: obj_id,
                }));
                (obj_id, span)
            })
            .0
    }

    pub fn insert(&mut self, id: MirObjectId, ident: Ident, span: Span) {
        self.properties.insert(ident, (id, span));
    }

    pub fn is_empty(&self) -> bool {
        self.properties.is_empty()
    }
}