Cthulhu  0.2.10
Cthulhu compiler collection
broker.c
Go to the documentation of this file.
1 // SPDX-License-Identifier: LGPL-3.0-only
2 
4 
5 #include "base/log.h"
6 #include "cthulhu/broker/scan.h"
8 
9 #include "arena/arena.h"
10 #include "base/panic.h"
11 #include "base/util.h"
12 #include "cthulhu/tree/tree.h"
13 #include "interop/compile.h"
14 #include "notify/notify.h"
15 #include "scan/node.h"
16 #include "std/map.h"
17 #include "std/vector.h"
18 
19 typedef struct broker_t
20 {
27 
28  // all languages
29  // vector_t<language_runtime_t*>
31 
32  // all targets
33  // vector_t<target_runtime_t*>
35 
36  // all plugins
37  // stored in an array rather than a map because
38  // they need to be executed in the order they were added
39  // vector_t<plugin_runtime_t*>
41 
42  // all translation units
43  // map_t<unit_id_t*, compile_unit_t*>
45 
46  // all builtin modules
48 } broker_t;
49 
50 static const size_t kDeclSizes[eSemaCount] = {
51  [eSemaValues] = 1,
52  [eSemaTypes] = 1,
53  [eSemaProcs] = 1,
54  [eSemaModules] = 64,
55 };
56 
57 #define OPT_EXEC(fn, ...) do { if (fn != NULL) fn(__VA_ARGS__); } while (0)
58 
60 static bool parse_ok(parse_result_t result, scan_t *scan, logger_t *logger)
61 {
62  switch (result.result)
63  {
64  case eParseOk: return true;
65  case eParseReject: return false;
66 
67  case eParseInitError:
68  msg_notify(logger, &kEvent_ParseInitFailed, node_new(scan, kNowhere), "failed to initialize parser");
69  return false;
70 
71  case eParseScanError:
72  msg_notify(logger, &kEvent_ScanFailed, node_new(scan, kNowhere), "failed to scan input");
73  return false;
74 
75  default: CT_NEVER("unknown parse result %d", result.result);
76  }
77 }
78 
79 static void resolve_module_tag(tree_t *mod, size_t tag)
80 {
81  map_iter_t iter = map_iter(tree_module_tag(mod, tag));
82  while (map_has_next(&iter))
83  {
84  map_entry_t entry = map_next(&iter);
85  tree_t *tree = entry.value;
86  tree_resolve(tree_get_cookie(mod), tree);
87  }
88 }
89 
90 static void resolve_module(tree_t *mod)
91 {
92  CTASSERT(mod != NULL);
93 
94  resolve_module_tag(mod, eSemaValues);
95  resolve_module_tag(mod, eSemaTypes);
96  resolve_module_tag(mod, eSemaProcs);
97 
99  while (map_has_next(&mods))
100  {
101  map_entry_t entry = map_next(&mods);
102  tree_t *tree = entry.value;
103  resolve_module(tree);
104  }
105 }
106 
107 static compile_unit_t *compile_unit_new(language_runtime_t *lang, arena_t *arena, void *ast, tree_t *tree)
108 {
109  CTASSERT(lang != NULL);
110  CTASSERT(arena != NULL);
111  CTASSERT(tree != NULL);
112  broker_t *broker = lang->broker;
113 
114  compile_unit_t *unit = ARENA_MALLOC(sizeof(compile_unit_t), "compilation unit", lang, arena);
115  unit->lang = lang;
116  unit->ast = ast;
117  unit->tree = tree;
118  tree_module_set(broker->root, eSemaModules, tree_get_name(tree), tree);
119 
120  ARENA_REPARENT(tree, unit, arena);
121 
122  return unit;
123 }
124 
128 
129 STA_DECL
130 broker_t *broker_new(const frontend_t *frontend, arena_t *arena)
131 {
132  CTASSERT(frontend != NULL);
133  CTASSERT(arena != NULL);
134 
135  module_info_t info = frontend->info;
136 
137  broker_t *broker = ARENA_MALLOC(sizeof(broker_t), "broker", NULL, arena);
138  broker->frontend = frontend;
139  broker->arena = arena;
140  broker->builtin = node_builtin(info.name, arena);
141  broker->logger = logger_new(arena);
142 
143  ARENA_REPARENT(broker->builtin, broker, arena);
144  ARENA_REPARENT(broker->logger, broker, arena);
145 
146  tree_cookie_t cookie = {
147  .reports = broker->logger,
148  .stack = vector_new(16, arena),
149  .types = vector_new(16, arena)
150  };
151 
152  ARENA_REPARENT(cookie.stack, broker, arena);
153 
154  broker->cookie = cookie;
155  broker->root = tree_module_root(broker->logger, &broker->cookie, broker->builtin, info.name, eSemaCount, kDeclSizes, arena);
156 
157  broker->langs = vector_new(8, arena);
158  broker->targets = vector_new(8, arena);
159  broker->plugins = vector_new(8, arena);
160 
161  broker->units = map_new(64, kTypeInfoText, arena);
162  broker->builtins = map_new(64, kTypeInfoText, arena);
163 
164  ARENA_REPARENT(broker->root, broker, arena);
165  ARENA_REPARENT(broker->langs, broker, arena);
166  ARENA_REPARENT(broker->targets, broker, arena);
167  ARENA_REPARENT(broker->plugins, broker, arena);
168  ARENA_REPARENT(broker->units, broker, arena);
169  ARENA_REPARENT(broker->builtins, broker, arena);
170 
171  return broker;
172 }
173 
174 STA_DECL
176 {
177  CTASSERT(broker != NULL);
178  CTASSERT(lang != NULL);
179 
180  arena_t *arena = broker->arena;
181  module_info_t info = lang->info;
182  CTASSERT(info.id != NULL);
183  CTASSERT(info.name != NULL);
184  CTASSERT(info.version.license != NULL);
185 
186  language_info_t builtin = lang->builtin;
187  CTASSERTF(builtin.name.text != NULL, "language '%s' did not specify a builtin name", info.name);
188  CTASSERTF(builtin.length >= eSemaCount, "language '%s' did not specify builtin decls", info.name);
189  CTASSERTF(builtin.decls != NULL, "language '%s' did not specify builtin decls", info.name);
190 
191  language_runtime_t *runtime = ARENA_MALLOC(sizeof(language_runtime_t), info.name, broker, arena);
192  runtime->info = lang;
193  runtime->broker = broker;
194 
195  // TODO: seperate arena per language
196  runtime->arena = arena;
197  runtime->logger = broker->logger;
198  runtime->ast_arena = arena;
199 
200  node_t *node = node_builtin(info.id, arena);
201  ARENA_REPARENT(node, runtime, arena);
202 
203  // all builtins for the language go into this module
204  tree_t *tree = tree_module(broker->root, node, info.id, builtin.length, builtin.decls);
205  ARENA_REPARENT(tree, runtime, arena);
206 
207  runtime->root = tree;
208  vector_push(&broker->langs, runtime);
209 
210  compile_unit_t *unit = compile_unit_new(runtime, arena, NULL, tree);
211 
212  map_set(broker->builtins, &builtin.name, unit);
213 
214  tree_module_set(broker->root, eSemaModules, info.id, tree);
215 
216  return runtime;
217 }
218 
219 STA_DECL
221 {
222  CTASSERT(broker != NULL);
223  CTASSERT(plugin != NULL);
224 
225  arena_t *arena = broker->arena;
226  module_info_t info = plugin->info;
227 
228  CTASSERT(info.id != NULL);
229  CTASSERT(info.name != NULL);
230  CTASSERT(info.version.license != NULL);
231 
232  plugin_runtime_t *runtime = ARENA_MALLOC(sizeof(plugin_runtime_t), info.name, broker, arena);
233  runtime->info = plugin;
234 
235  vector_push(&broker->plugins, runtime);
236 
237  return runtime;
238 }
239 
240 STA_DECL
242 {
243  CTASSERT(broker != NULL);
244  CTASSERT(target != NULL);
245 
246  arena_t *arena = broker->arena;
247  module_info_t info = target->info;
248 
249  CTASSERT(info.id != NULL);
250  CTASSERT(info.name != NULL);
251  CTASSERT(info.version.license != NULL);
252 
253  target_runtime_t *runtime = ARENA_MALLOC(sizeof(target_runtime_t), info.name, broker, arena);
254  runtime->info = target;
255  runtime->broker = broker;
256  runtime->arena = arena;
257  runtime->logger = broker->logger;
258 
259  vector_push(&broker->targets, runtime);
260 
261  return runtime;
262 }
263 
264 STA_DECL
265 void broker_init(broker_t *broker)
266 {
267  CTASSERT(broker != NULL);
268 
269  // init plugins
270  size_t len = vector_len(broker->plugins);
271  for (size_t i = 0; i < len; i++)
272  {
273  plugin_runtime_t *plugin = vector_get(broker->plugins, i);
274  OPT_EXEC(plugin->info->fn_create, plugin);
275  }
276 
277  // init targets
278  len = vector_len(broker->targets);
279  for (size_t i = 0; i < len; i++)
280  {
281  target_runtime_t *target = vector_get(broker->targets, i);
282  OPT_EXEC(target->info->fn_create, target);
283  }
284 
285  // init languages
286  len = vector_len(broker->langs);
287  for (size_t i = 0; i < len; i++)
288  {
289  language_runtime_t *lang = vector_get(broker->langs, i);
290  OPT_EXEC(lang->info->fn_create, lang, lang->root);
291  }
292 }
293 
294 STA_DECL
295 void broker_deinit(broker_t *broker)
296 {
297  CTASSERT(broker != NULL);
298 
299  // deinit languages
300  size_t len = vector_len(broker->langs);
301  for (size_t i = 0; i < len; i++)
302  {
303  language_runtime_t *lang = vector_get(broker->langs, i);
304  OPT_EXEC(lang->info->fn_destroy, lang);
305  }
306 
307  // deinit targets
308  len = vector_len(broker->targets);
309  for (size_t i = 0; i < len; i++)
310  {
311  target_runtime_t *target = vector_get(broker->targets, i);
312  OPT_EXEC(target->info->fn_destroy, target);
313  }
314 
315  // deinit plugins
316  len = vector_len(broker->plugins);
317  for (size_t i = 0; i < len; i++)
318  {
319  plugin_runtime_t *plugin = vector_get(broker->plugins, i);
320  OPT_EXEC(plugin->info->fn_destroy, plugin);
321  }
322 }
323 
324 STA_DECL
326 {
327  CTASSERT(broker != NULL);
328 
329  return broker->logger;
330 }
331 
332 STA_DECL
334 {
335  CTASSERT(broker != NULL);
336 
337  return broker->builtin;
338 }
339 
340 STA_DECL
342 {
343  CTASSERT(broker != NULL);
344 
345  return broker->arena;
346 }
347 
348 static void collect_units(vector_t **vec, map_t *map)
349 {
350  map_iter_t iter = map_iter(map);
351  while (map_has_next(&iter))
352  {
353  map_entry_t entry = map_next(&iter);
354  compile_unit_t *unit = entry.value;
355  CTASSERT(unit != NULL);
356 
357  vector_push(vec, unit->tree);
358  }
359 }
360 
361 STA_DECL
363 {
364  CTASSERT(broker != NULL);
365 
366  vector_t *modules = vector_new(64, broker->arena);
367  collect_units(&modules, broker->builtins);
368  collect_units(&modules, broker->units);
369 
370  ARENA_IDENTIFY(modules, "modules", broker, broker->arena);
371 
372  return modules;
373 }
374 
375 STA_DECL
377 {
378  CTASSERT(runtime != NULL);
379  CTASSERT(io != NULL);
380 
381  broker_t *broker = runtime->broker;
382 
383  const language_t *lang = runtime->info;
384  const module_info_t *info = &lang->info;
385 
386  scan_context_t *ctx = ARENA_MALLOC(sizeof(scan_context_t) + lang->context_size, "scan context", runtime, broker->arena);
387  ctx->logger = broker->logger;
388  ctx->arena = broker->arena;
389  ctx->string_arena = broker->arena;
390  ctx->ast_arena = broker->arena;
391 
392  // TODO: allow languages that dont use scanner callbacks
393  CTASSERTF(lang->scanner != NULL, "language '%s' did not specify a scanner", info->name);
394 
395  scan_t *scan = scan_io(info->name, io, broker->arena);
396  ARENA_REPARENT(scan, runtime, broker->arena);
397 
398  if (lang->fn_preparse != NULL)
399  {
400  lang->fn_preparse(runtime, ctx->user);
401  scan_set_context(scan, ctx->user); // for backwards compatibility
402  }
403  else
404  {
405  scan_set_context(scan, ctx);
406  }
407 
408  parse_result_t result = scan_buffer(scan, lang->scanner);
409  if (!parse_ok(result, scan, broker->logger))
410  {
411  return;
412  }
413 
414  CTASSERTF(lang->fn_postparse != NULL, "language '%s' did not specify a postparse function", info->name);
415  lang->fn_postparse(runtime, scan, result.tree);
416 }
417 
418 STA_DECL
420 {
421  CTASSERT(broker != NULL);
422 
423  map_iter_t iter = map_iter(broker->units);
424  while (map_has_next(&iter))
425  {
426  map_entry_t entry = map_next(&iter);
427  compile_unit_t *unit = entry.value;
428  CTASSERT(unit != NULL);
429 
430  if (unit->ast == NULL)
431  continue;
432 
433  language_runtime_t *lang = unit->lang;
434  CTASSERTF(lang != NULL, "unit '%s' has no associated language", tree_get_name(unit->tree));
435 
436  language_pass_t fn = lang->info->fn_passes[pass];
437  if (fn == NULL)
438  {
439  const language_t *it = lang->info;
440  const module_info_t *info = &it->info;
441  ctu_log("language '%s' does not implement pass '%s'", info->name, broker_pass_name(pass));
442  continue;
443  }
444 
445  OPT_EXEC(fn, lang, unit);
446  }
447 }
448 
449 STA_DECL
451 {
452  CTASSERT(broker != NULL);
453 
454  resolve_module(broker->root);
455 }
456 
460 
461 STA_DECL
462 void lang_add_unit(language_runtime_t *runtime, unit_id_t id, const node_t *node, void *ast, const size_t *decls, size_t length)
463 {
464  CTASSERT(runtime != NULL);
465  CTASSERT(id.text != NULL);
466  CTASSERT(id.length > 0);
467  CTASSERT(ast != NULL);
468  CTASSERT(decls != NULL);
469  CTASSERT(length >= eSemaCount);
470 
471  compile_unit_t *old = map_get(runtime->broker->units, &id);
472  if (old != NULL)
473  {
474  msg_notify(runtime->logger, &kEvent_ModuleConflict, tree_get_node(old->tree), "module '%s' already exists", id.text);
475  return;
476  }
477  arena_t *arena = runtime->arena;
478 
479  char *copy = arena_memdup(id.text, id.length, arena);
480  for (size_t i = 0; i < id.length; i++)
481  if (copy[i] == '\0')
482  copy[i] = '/';
483  copy[id.length] = '\0';
484 
485  tree_t *tree = tree_module(runtime->root, node, copy, length, decls);
486  ARENA_REPARENT(copy, tree, arena);
487 
488  compile_unit_t *unit = compile_unit_new(runtime, arena, ast, tree);
489 
490  text_view_t *key = arena_memdup(&id, sizeof(unit_id_t), arena);
491  map_set(runtime->broker->units, key, unit);
492 }
493 
495 {
496  CTASSERT(runtime != NULL);
497  CTASSERT(id.text != NULL);
498  CTASSERT(id.length > 0);
499 
500  broker_t *broker = runtime->broker;
501  compile_unit_t *builtin = map_get(broker->builtins, &id);
502  if (builtin != NULL)
503  {
504  return builtin;
505  }
506 
507  return map_get(broker->units, &id);
508 }
509 
510 STA_DECL
512 {
513  CTASSERT(unit != NULL);
514 
515  return unit->ast;
516 }
517 
518 STA_DECL
519 void unit_update(compile_unit_t *unit, void *ast, tree_t *tree)
520 {
521  CTASSERT(unit != NULL);
522  CTASSERT(ast != NULL);
523  CTASSERT(tree != NULL);
524 
525  CTASSERTF(unit->ast != NULL, "attempting to modify builtin module '%s'", tree_get_name(unit->tree));
526 
527  unit->ast = ast;
528  unit->tree = tree;
529 }
530 
531 STA_DECL
533 {
534  CTASSERT(parts != NULL);
535  CTASSERT(arena != NULL);
536 
537  size_t len = vector_len(parts);
538  size_t chars = 0;
539  for (size_t i = 0; i < len; i++)
540  {
541  const char *part = vector_get(parts, i);
542  chars += ctu_strlen(part);
543  }
544 
545  size_t size = chars + len - 1;
546  char *buf = ARENA_MALLOC(size + 1, "unit_id", NULL, arena);
547 
548  size_t offset = 0;
549  for (size_t i = 0; i < len; i++)
550  {
551  const char *part = vector_get(parts, i);
552  size_t part_len = ctu_strlen(part);
553  ctu_memcpy(buf + offset, part, part_len);
554  offset += part_len;
555  if (i + 1 < len)
556  {
557  buf[offset++] = '\0';
558  }
559  }
560 
561  buf[size] = '\0';
562  return text_view_make(buf, size);
563 }
564 
565 STA_DECL
566 void target_emit_tree(target_runtime_t *runtime, const tree_t *tree, target_emit_t *emit)
567 {
568  CTASSERT(runtime != NULL);
569  CTASSERT(tree != NULL);
570  CTASSERT(emit != NULL);
571 
572  const target_t *target = runtime->info;
573  CTASSERTF(target->fn_tree != NULL, "target '%s' does not implement tree emission", target->info.name);
574 
575  target->fn_tree(runtime, tree, emit);
576 }
577 
578 STA_DECL
580 {
581  CTASSERT(runtime != NULL);
582  CTASSERT(ssa != NULL);
583  CTASSERT(emit != NULL);
584 
585  const target_t *target = runtime->info;
586  CTASSERTF(target->fn_ssa != NULL, "target '%s' does not implement ssa emission", target->info.name);
587 
588  return target->fn_ssa(runtime, ssa, emit);
589 }
590 
591 static const char *const kPassNames[ePassCount] = {
592 #define BROKER_PASS(ID, STR) [ID] = (STR),
593 #include "cthulhu/broker/broker.inc"
594 };
595 
596 STA_DECL
598 {
599  CT_ASSERT_RANGE(pass, 0, ePassCount - 1);
600 
601  return kPassNames[pass];
602 }
603 
604 static const char *const kFileLayoutNames[eFileLayoutCount] = {
605 #define FILE_LAYOUT(ID, STR) [ID] = (STR),
606 #include "cthulhu/broker/broker.inc"
607 };
608 
609 STA_DECL
610 const char *file_layout_name(file_layout_t layout)
611 {
612  CT_ASSERT_RANGE(layout, 0, eFileLayoutCount - 1);
613 
614  return kFileLayoutNames[layout];
615 }
#define OPT_EXEC(fn,...)
Definition: broker.c:57
CT_NODISCARD CT_SCAN_API scan_t * scan_io(const char *language, io_t *io, arena_t *arena)
create a scanner from an io source
Definition: scan.c:121
CT_SCAN_API void scan_set_context(scan_t *scan, void *value)
get the context of a scanner
Definition: scan.c:80
CT_NODISCARD size_t size
Definition: scan.h:128
CT_PUREFN CT_TREE_API const node_t * tree_get_node(const tree_t *tree)
Definition: context.c:94
CT_PUREFN CT_TREE_API const char * tree_get_name(const tree_t *tree)
Definition: context.c:102
#define STA_DECL
sal2 annotation on function implementations to copy annotations from the declaration
CT_NOALIAS CT_BASE_API void * ctu_memcpy(STA_WRITES(size) void *CT_RESTRICT dst, STA_READS(size) const void *CT_RESTRICT src, size_t size)
copy memory from one location to another equivalent to memcpy but with safety checks
CT_NODISCARD CT_PUREFN CT_BASE_API size_t ctu_strlen(const char *str)
get the length of a string not including the null terminator equivalent to strlen but with safety che...
Definition: util.c:87
CT_CONSTFN CT_BASE_API text_view_t text_view_make(STA_READS(length) const char *text, size_t length)
create a new non-owning text array text must be at least length bytes long
STA_DECL const node_t * broker_get_node(broker_t *broker)
Definition: broker.c:333
STA_DECL emit_result_t target_emit_ssa(target_runtime_t *runtime, const ssa_result_t *ssa, target_emit_t *emit)
Definition: broker.c:579
STA_DECL void * unit_get_ast(compile_unit_t *unit)
Definition: broker.c:511
STA_DECL void broker_parse(language_runtime_t *runtime, io_t *io)
Definition: broker.c:376
STA_DECL void broker_run_pass(broker_t *broker, broker_pass_t pass)
Definition: broker.c:419
broker_pass_t
Definition: broker.h:56
STA_DECL void broker_resolve(broker_t *broker)
Definition: broker.c:450
void(* language_pass_t)(language_runtime_t *runtime, compile_unit_t *unit)
a language compilation pass
Definition: broker.h:114
STA_DECL void target_emit_tree(target_runtime_t *runtime, const tree_t *tree, target_emit_t *emit)
all plugin apis
Definition: broker.c:566
STA_DECL void lang_add_unit(language_runtime_t *runtime, unit_id_t id, const node_t *node, void *ast, const size_t *decls, size_t length)
all runtime apis
Definition: broker.c:462
STA_DECL void unit_update(compile_unit_t *unit, void *ast, tree_t *tree)
Definition: broker.c:519
file_layout_t
output folder structure
Definition: broker.h:91
compile_unit_t * lang_get_unit(language_runtime_t *runtime, unit_id_t id)
Definition: broker.c:494
STA_DECL language_runtime_t * broker_add_language(broker_t *broker, const language_t *lang)
Definition: broker.c:175
STA_DECL void broker_deinit(broker_t *broker)
Definition: broker.c:295
STA_DECL const char * file_layout_name(file_layout_t layout)
Definition: broker.c:610
STA_DECL broker_t * broker_new(const frontend_t *frontend, arena_t *arena)
Definition: broker.c:130
STA_DECL target_runtime_t * broker_add_target(broker_t *broker, const target_t *target)
Definition: broker.c:241
STA_DECL const char * broker_pass_name(broker_pass_t pass)
extra stuff
Definition: broker.c:597
STA_DECL void broker_init(broker_t *broker)
Definition: broker.c:265
STA_DECL plugin_runtime_t * broker_add_plugin(broker_t *broker, const plugin_t *plugin)
Definition: broker.c:220
STA_DECL vector_t * broker_get_modules(broker_t *broker)
get all the modules in the broker this does not include the root module
Definition: broker.c:362
STA_DECL arena_t * broker_get_arena(broker_t *broker)
Definition: broker.c:341
STA_DECL logger_t * broker_get_logger(broker_t *broker)
Definition: broker.c:325
STA_DECL text_view_t build_unit_id(const vector_t *parts, arena_t *arena)
Definition: broker.c:532
@ ePassCount
Definition: broker.h:60
CT_NODISCARD CT_PUREFN CT_STD_API bool map_has_next(const map_iter_t *iter)
check if a map iterator has more elements
Definition: map.c:509
CT_NODISCARD CT_PUREFN CT_STD_API map_iter_t map_iter(const map_t *map)
create a new map iterator
Definition: map.c:456
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_STD_API map_t * map_new(size_t size, hash_info_t info, arena_t *arena)
create a new map on the heap
Definition: map.c:113
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
CT_NODISCARD CT_NOALIAS CT_STD_API map_entry_t map_next(map_iter_t *iter)
get the next key-value pair from a map iterator
Definition: map.c:476
CT_INTEROP_API parse_result_t scan_buffer(scan_t *scan, const scan_callbacks_t *callbacks)
parse the contents of a scanner into a language specific ast
Definition: compile.c:26
@ eParseInitError
error initializing the scanner internals (our fault)
Definition: compile.h:79
@ eParseOk
parse was successful
Definition: compile.h:76
@ eParseScanError
entered invalid state during scanning (our fault)
Definition: compile.h:82
@ eParseReject
failed due to invalid input (users fault)
Definition: compile.h:85
CT_NODISCARD CT_SCAN_API node_t * node_builtin(const char *name, arena_t *arena)
get the builtin node node used for drivers that declare builtin symbols
Definition: node.c:11
CT_SCAN_API const where_t kNowhere
nowhere in a source file
Definition: node.c:8
CT_NODISCARD CT_SCAN_API node_t * node_new(const scan_t *scan, where_t where)
create a new node on the heap
Definition: node.c:40
CT_BASE_API void ctu_log(STA_FORMAT_STRING const char *fmt,...)
log a message
CT_NODISCARD CT_ARENA_API void * arena_memdup(STA_READS(size) const void *ptr, size_t size, arena_t *arena)
duplicate a memory region from a custom allocator duplicate a region of memory and return a pointer t...
#define ARENA_IDENTIFY(ptr, name, parent, arena)
rename and reparent a pointer in a custom allocator
Definition: arena.h:409
#define ARENA_REPARENT(arena, ptr, parent)
reparent a pointer in a custom allocator
Definition: arena.h:391
#define ARENA_MALLOC(size, name, parent, arena)
allocate memory from a custom allocator
Definition: arena.h:392
CT_NOTIFY_API event_builder_t msg_notify(INOUT_NOTNULL logger_t *logs, const diagnostic_t *diagnostic, const node_t *node, STA_FORMAT_STRING const char *fmt,...)
notify the logger of a new message
CT_NODISCARD CT_NOTIFY_API logger_t * logger_new(arena_t *arena)
create a new logger
Definition: notify.c:21
#define CT_ASSERT_RANGE(value, min, max)
assert that a value is in a range inclusive bounds check
Definition: panic.h:148
#define CT_NEVER(...)
assert that a code path is never reached
Definition: panic.h:136
#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
@ eFileLayoutCount
Definition: broker.h:94
CT_TREE_API 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
CT_TREE_API map_t * tree_module_tag(const tree_t *self, size_t tag)
Definition: sema.c:125
CT_TREE_API tree_cookie_t * tree_get_cookie(tree_t *sema)
return a resolution cookie
Definition: sema.c:136
CT_TREE_API 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
CT_TREE_API tree_t * tree_resolve(tree_cookie_t *cookie, const tree_t *decl)
Definition: decl.c:47
CT_TREE_API 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
@ eSemaCount
Definition: tree.h:47
@ eSemaValues
Definition: tree.h:36
@ eSemaProcs
Definition: tree.h:42
@ eSemaModules
Definition: tree.h:45
@ eSemaTypes
Definition: tree.h:39
CT_STD_API const hash_info_t kTypeInfoText
type information for a text_view_t
Definition: typeinfo.c:47
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_new(size_t size, arena_t *arena)
create a new vector with an initial capacity
Definition: vector.c:63
CT_STD_API void vector_push(vector_t **vector, void *value)
push a value onto the end of a vector
Definition: vector.c:108
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
map_t * builtins
Definition: broker.c:47
vector_t * plugins
Definition: broker.c:40
tree_cookie_t cookie
Definition: broker.c:25
node_t * builtin
Definition: broker.c:23
const frontend_t * frontend
Definition: broker.c:21
logger_t * logger
Definition: broker.c:24
tree_t * root
Definition: broker.c:26
vector_t * langs
Definition: broker.c:30
vector_t * targets
Definition: broker.c:34
arena_t * arena
Definition: broker.c:22
map_t * units
Definition: broker.c:44
tree_t * tree
the tree for this unit is NULL if the unit has not been forward declared yet
Definition: broker.h:289
language_runtime_t * lang
the language that this originated from
Definition: broker.h:281
void * ast
the ast for this unit is NULL if this is a builtin/precompiled unit
Definition: broker.h:285
the frontend running the mediator
Definition: broker.h:251
module_info_t info
information about the frontend
Definition: broker.h:253
io object implementation
Definition: impl.h:122
unit_id_t name
the name of the builtin module
Definition: broker.h:126
arena_t * arena
default memory arena
Definition: broker.h:266
const language_t * info
Definition: broker.h:262
arena_t * ast_arena
arena for the language ast
Definition: broker.h:269
broker_t * broker
Definition: broker.h:263
tree_t * root
the builtins module for this language
Definition: broker.h:275
logger_t * logger
logger
Definition: broker.h:272
a language driver support capabilities
Definition: broker.h:143
language_preparse_t fn_preparse
called before a file is parsed should return a pointer to a context that will be put into the scanner
Definition: broker.h:171
size_t context_size
the size of the scan context for this language
Definition: broker.h:158
const scan_callbacks_t * scanner
callbacks for the parser
Definition: broker.h:178
language_info_t builtin
builtin module configuration
Definition: broker.h:148
language_destroy_t fn_destroy
called at shutdown
Definition: broker.h:167
language_postparse_t fn_postparse
called after a file is parsed should produce translation units from a scanned ast
Definition: broker.h:175
module_info_t info
common information about the language
Definition: broker.h:145
language_pass_t fn_passes[ePassCount]
an array of passes to run on each translation unit
Definition: broker.h:181
language_create_t fn_create
called once at startup
Definition: broker.h:164
a logging sink
Definition: notify.c:14
a key-value pair in a map
Definition: map.h:175
void * value
the value of this entry
Definition: map.h:177
a map iterator handle
Definition: map.h:183
an unordered hash map
Definition: map.h:38
common information about anything the broker supports
Definition: broker.h:99
version_info_t version
the version of the module
Definition: broker.h:107
const char * id
unique id for the module
Definition: broker.h:101
const char * name
the human readable name for the module
Definition: broker.h:104
a position in a source file
Definition: node.h:23
parse_error_t result
Definition: compile.h:92
void * tree
Definition: compile.h:95
const plugin_t * info
Definition: broker.h:294
plugin support capabilities
Definition: broker.h:202
module_info_t info
information about the plugin
Definition: broker.h:204
plugin_destroy_t fn_destroy
called at shutdown
Definition: broker.h:210
plugin_create_t fn_create
called once at startup
Definition: broker.h:207
arena_t * ast_arena
Definition: scan.h:24
logger_t * logger
Definition: scan.h:20
char user[]
Definition: scan.h:26
arena_t * arena
Definition: scan.h:22
arena_t * string_arena
Definition: scan.h:23
a source file scanner
Definition: scan.h:24
broker_t * broker
Definition: broker.h:300
arena_t * arena
Definition: broker.h:302
logger_t * logger
Definition: broker.h:304
const target_t * info
Definition: broker.h:299
a codegen target backend
Definition: broker.h:232
target_tree_t fn_tree
generate from the tree form
Definition: broker.h:243
target_ssa_t fn_ssa
generate from the ssa form
Definition: broker.h:246
target_destroy_t fn_destroy
called at shutdown
Definition: broker.h:240
target_create_t fn_create
called once at startup
Definition: broker.h:237
module_info_t info
information about the target
Definition: broker.h:234
a non-owning view of text
Definition: text.h:24
Definition: tree.h:67
a generic vector of pointers
Definition: vector.c:16
const char * license
the license of this component
Definition: version_def.h:49