Cthulhu  0.2.10
Cthulhu compiler collection
query.c
Go to the documentation of this file.
1 // SPDX-License-Identifier: LGPL-3.0-only
2 
4 #include "cthulhu/util/util.h"
5 
6 #include "cthulhu/tree/tree.h"
7 #include "cthulhu/tree/query.h"
8 
9 #include "std/vector.h"
10 
11 #include "base/panic.h"
12 #include <stdio.h>
13 
14 static tree_t *select_decl_imported(tree_t *sema, const size_t *tags, size_t len, const char *name, tree_t **mod)
15 {
16  CTASSERT(tags != NULL);
17  CTASSERT(len > 0);
18 
19  for (size_t i = 0; i < len; i++)
20  {
21  tree_t *decl = tree_module_find(sema, tags[i], name, mod);
22  if (decl != NULL)
23  {
24  return decl;
25  }
26  }
27 
28  return NULL;
29 }
30 
31 static tree_t *select_module(tree_t *sema, search_t search, const char *name, bool *imported)
32 {
33  CTASSERT(imported != NULL);
34 
35  tree_t *context = NULL;
36 
37  tree_t *inner = select_decl_imported(sema, search.tags, search.count, name, &context);
38  *imported = (inner != context);
39  return inner;
40 }
41 
42 static search_t get_module_search(const decl_search_t *search)
43 {
44  search_t result = {
45  .tags = search->module_tags,
46  .count = search->module_count
47  };
48  return result;
49 }
50 
51 static search_t get_decl_search(const decl_search_t *search)
52 {
53  search_t result = {
54  .tags = search->decl_tags,
55  .count = search->decl_count
56  };
57  return result;
58 }
59 
60 static bool is_public(const tree_t *decl)
61 {
62  const tree_attribs_t *attrib = tree_get_attrib(decl);
63  return attrib->visibility == eVisiblePublic;
64 }
65 
66 tree_t *util_search_namespace(tree_t *sema, search_t search, const node_t *node, const vector_t *path, bool *is_imported)
67 {
68  CTASSERT(sema != NULL);
69  CTASSERT(vector_len(path) > 0);
70  CTASSERT(is_imported != NULL);
71 
72  size_t len = vector_len(path);
73  tree_t *ns = sema;
74  for (size_t i = 0; i < len - 1; i++)
75  {
76  if (!tree_is(ns, eTreeDeclModule))
77  {
78  return tree_raise(node, sema->reports, &kEvent_MalformedTypeName, "expected a namespace but got `%s` instead", tree_to_string(ns));
79  }
80 
81  const char *segment = vector_get(path, i);
82  ns = select_module(ns, search, segment, is_imported);
83  if (ns == NULL)
84  {
85  return tree_raise(node, sema->reports, &kEvent_SymbolNotFound, "namespace `%s` not found", segment);
86  }
87  }
88 
89  return ns;
90 }
91 
92 tree_t *util_search_path(tree_t *sema, const decl_search_t *search, const node_t *node, vector_t *path)
93 {
94  CTASSERTF(sema != NULL && search != NULL, "(sema = %p, search = %p)", (void*)sema, (void*)search);
95  CTASSERT(vector_len(path) > 0);
96 
97  bool is_imported = false;
98  size_t len = vector_len(path);
99  tree_t *ns = sema;
100  for (size_t i = 0; i < len - 1; i++)
101  {
102  const char *segment = vector_get(path, i);
103  ns = select_module(ns, get_module_search(search), segment, &is_imported);
104  if (ns == NULL)
105  {
106  return tree_raise(node, sema->reports, &kEvent_SymbolNotFound, "namespace `%s` not found", segment);
107  }
108  }
109 
110  const char *name = vector_tail(path);
111  tree_t *decl = util_select_decl(ns, get_decl_search(search), name);
112  if (decl == NULL)
113  {
114  return tree_raise(node, sema->reports, &kEvent_SymbolNotFound, "decl `%s` not found", name);
115  }
116 
117  if (is_imported && !is_public(decl))
118  {
119  return tree_raise(node, sema->reports, &kEvent_SymbolNotVisible, "decl `%s` is not public", name);
120  }
121 
122  return decl;
123 }
124 
125 tree_t *util_search_qualified(tree_t *sema, const decl_search_t *search, const node_t *node, const char *mod, const char *name)
126 {
127  bool is_imported = false;
128  tree_t *ns = select_module(sema, get_module_search(search), mod, &is_imported);
129  if (ns == NULL)
130  {
131  return tree_raise(node, sema->reports, &kEvent_SymbolNotFound, "namespace `%s` not found", mod);
132  }
133 
134  tree_t *decl = util_select_decl(ns, get_decl_search(search), name);
135  if (decl == NULL)
136  {
137  return tree_raise(node, sema->reports, &kEvent_SymbolNotFound, "decl `%s` not found", name);
138  }
139 
140  if (is_imported && !is_public(decl))
141  {
142  return tree_raise(node, sema->reports, &kEvent_SymbolNotVisible, "decl `%s` is not public", name);
143  }
144 
145  return decl;
146 }
const tree_attribs_t * tree_get_attrib(const tree_t *self)
Definition: query.c:83
bool tree_is(const tree_t *self, tree_kind_t kind)
Definition: query.c:91
const char * tree_to_string(const tree_t *self)
Definition: query.c:40
#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 * util_search_qualified(tree_t *sema, const decl_search_t *search, const node_t *node, const char *mod, const char *name)
search for a decl inside a module
Definition: query.c:125
CT_UTIL_API void * util_select_decl(tree_t *sema, search_t search, const char *name)
search for a declaration by name in a set of tags
Definition: util.c:16
tree_t * util_search_namespace(tree_t *sema, search_t search, const node_t *node, const vector_t *path, bool *is_imported)
search for a namespace given a path, ignoring the last element in the path
Definition: query.c:66
tree_t * util_search_path(tree_t *sema, const decl_search_t *search, const node_t *node, vector_t *path)
search for a decl given a path
Definition: query.c:92
CT_TREE_API 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
CT_TREE_API tree_t * tree_raise(const node_t *node, logger_t *reports, const diagnostic_t *diagnostic, const char *message,...)
Definition: tree.c:76
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_PUREFN CT_STD_API void * vector_tail(const vector_t *vector)
get the last element of a vector
Definition: vector.c:143
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
const size_t * decl_tags
Definition: util.h:46
size_t module_count
Definition: util.h:44
const size_t * module_tags
Definition: util.h:43
size_t decl_count
Definition: util.h:47
a position in a source file
Definition: node.h:23
Definition: util.h:28
size_t count
Definition: util.h:30
const size_t * tags
Definition: util.h:29
tree_visibility_t visibility
the visibility of the declaration
Definition: tree.h:52
Definition: tree.h:67
logger_t * reports
Definition: tree.h:227
a generic vector of pointers
Definition: vector.c:16