Cthulhu  0.2.10
Cthulhu compiler collection
sema.c
Go to the documentation of this file.
1 // SPDX-License-Identifier: LGPL-3.0-only
2 
3 #include "arena/arena.h"
4 #include "common.h"
5 
6 #include "cthulhu/tree/query.h"
7 
8 #include "std/vector.h"
9 #include "std/map.h"
10 
11 #include "memory/memory.h"
12 #include "base/panic.h"
13 
14 static tree_t *tree_module_new(const node_t *node, const char *name,
15  tree_t *parent, tree_cookie_t *cookie,
16  logger_t *reports,
17  size_t decls, const size_t *sizes,
18  arena_t *arena)
19 {
20  CTASSERTF(decls >= eSemaCount, "module cannot be constructed with less than %d tags (%zu given)", eSemaCount, decls);
21  CTASSERT(reports != NULL);
22 
23  tree_t *self = tree_decl(eTreeDeclModule, node, NULL, name, eQualNone);
24  self->arena = arena;
25  self->parent = parent;
26  self->cookie = cookie;
27  self->reports = reports;
28 
29  self->tags = vector_of(decls, arena);
30  ARENA_IDENTIFY(self->tags, "module_tags", self, arena);
31 
32  for (size_t i = 0; i < decls; i++)
33  {
34  map_t *map = map_optimal(sizes[i], kTypeInfoString, arena);
35  ARENA_IDENTIFY(map, "module_tag", self, arena);
36  vector_set(self->tags, i, map);
37  }
38 
39  return self;
40 }
41 
42 tree_t *tree_module_root(logger_t *reports, tree_cookie_t *cookie, const node_t *node, const char *name, size_t decls, const size_t *sizes, arena_t *arena)
43 {
44  return tree_module_new(
45  node, name,
46  /* parent = */ NULL,
47  /* cookie = */ cookie,
48  /* reports = */ reports,
49  decls, sizes, arena);
50 }
51 
52 tree_t *tree_module(tree_t *parent, const node_t *node, const char *name, size_t decls, const size_t *sizes)
53 {
54  TREE_EXPECT(parent, eTreeDeclModule);
55 
56  return tree_module_new(node, name, parent, parent->cookie, parent->reports, decls, sizes, parent->arena);
57 }
58 
59 void *tree_module_get(tree_t *self, size_t tag, const char *name)
60 {
61  CTASSERT(name != NULL);
62 
63  // its ok to do an early return here and skip checking the parent module
64  // because parent modules will always have <= the tags of the child module
65  map_t *map = tree_module_tag(self, tag);
66  if (map == NULL) return NULL;
67 
68  tree_t *old = map_get(map, name);
69  if (old != NULL)
70  {
71  return old;
72  }
73 
74  if (self->parent != NULL)
75  {
76  return tree_module_get(self->parent, tag, name);
77  }
78 
79  return NULL;
80 }
81 
82 void *tree_module_find(tree_t *sema, size_t tag, const char *name, tree_t **module)
83 {
84  CTASSERT(sema != NULL);
85  CTASSERT(name != NULL);
86  CTASSERT(module != NULL);
87 
88  map_t *map = tree_module_tag(sema, tag);
89  if (map == NULL)
90  {
91  *module = NULL;
92  return NULL;
93  }
94 
95  tree_t *decl = map_get(map, name);
96  if (decl != NULL)
97  {
98  *module = sema;
99  return decl;
100  }
101 
102  if (sema->parent != NULL)
103  {
104  return tree_module_find(sema->parent, tag, name, module);
105  }
106 
107  *module = NULL;
108  return NULL;
109 }
110 
111 void *tree_module_set(tree_t *self, size_t tag, const char *name, void *value)
112 {
113  void *old = tree_module_get(self, tag, name);
114  if (old != NULL)
115  {
116  return old;
117  }
118 
119  map_t *map = tree_module_tag(self, tag);
120  map_set(map, name, value);
121 
122  return NULL;
123 }
124 
125 map_t *tree_module_tag(const tree_t *self, size_t tag)
126 {
127  TREE_EXPECT(self, eTreeDeclModule);
128 
129  size_t len = vector_len(self->tags);
130  if (tag >= len)
131  return NULL;
132 
133  return vector_get(self->tags, tag);
134 }
135 
137 {
138  TREE_EXPECT(sema, eTreeDeclModule);
139 
140  return sema->cookie;
141 }
CT_LOCAL tree_t * tree_decl(tree_kind_t kind, const node_t *node, const tree_t *type, const char *name, tree_quals_t quals)
Definition: tree.c:35
#define TREE_EXPECT(SELF, KIND)
Definition: common.h:22
CT_NODISCARD CT_STD_API map_t * map_optimal(size_t size, hash_info_t info, arena_t *arena)
create a new map with an optimal size
Definition: optimal.c:85
CT_STD_API void map_set(map_t *map, const void *key, void *value)
set a key-value pair in a map
Definition: map.c:294
CT_NODISCARD CT_PUREFN CT_STD_API void * map_get(const map_t *map, const void *key)
get a value from a map
Definition: map.c:324
#define ARENA_IDENTIFY(ptr, name, parent, arena)
rename and reparent a pointer in a custom allocator
Definition: arena.h:409
#define CTASSERT(expr)
assert a condition, prints the condition as a message
Definition: panic.h:130
#define CTASSERTF(expr,...)
assert a condition with a message and optional format arguments
Definition: panic.h:116
tree_t * tree_module(tree_t *parent, const node_t *node, const char *name, size_t decls, const size_t *sizes)
create a new module
Definition: sema.c:52
map_t * tree_module_tag(const tree_t *self, size_t tag)
Definition: sema.c:125
void * tree_module_find(tree_t *sema, size_t tag, const char *name, tree_t **module)
search for a declaration in a module also returns the module that the declaration was found in
Definition: sema.c:82
tree_cookie_t * tree_get_cookie(tree_t *sema)
return a resolution cookie
Definition: sema.c:136
tree_t * tree_module_root(logger_t *reports, tree_cookie_t *cookie, const node_t *node, const char *name, size_t decls, const size_t *sizes, arena_t *arena)
Definition: sema.c:42
void * tree_module_set(tree_t *self, size_t tag, const char *name, void *value)
set a declaration in the current module
Definition: sema.c:111
void * tree_module_get(tree_t *self, size_t tag, const char *name)
recursively search for a declaration in a module
Definition: sema.c:59
@ eSemaCount
Definition: tree.h:47
CT_STD_API const hash_info_t kTypeInfoString
type information for a c style string
Definition: typeinfo.c:35
CT_NODISCARD CT_PUREFN CT_STD_API void * vector_get(const vector_t *vector, size_t index)
get a value from a vector
Definition: vector.c:134
CT_NODISCARD CT_STD_API vector_t * vector_of(size_t len, arena_t *arena)
create a new vector with a specified length
Definition: vector.c:71
CT_STD_API void vector_set(vector_t *vector, size_t index, void *value)
set a value in a vector
Definition: vector.c:125
CT_NODISCARD CT_PUREFN CT_STD_API size_t vector_len(const vector_t *vector)
get the length of a vector
Definition: vector.c:152
an allocator object
Definition: arena.h:86
a logging sink
Definition: notify.c:14
an unordered hash map
Definition: map.h:38
a position in a source file
Definition: node.h:23
Definition: tree.h:67
tree_cookie_t * cookie
Definition: tree.h:225
logger_t * reports
Definition: tree.h:227
arena_t * arena
Definition: tree.h:223
tree_t * parent
Definition: tree.h:224