forked from eriksvedang/cakelisp
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathMetadata.cpp
483 lines (457 loc) · 17.2 KB
/
Metadata.cpp
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
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
#include <string>
// #include <unordered_map>
#include "Evaluator.hpp"
#include "Generators.hpp"
#include "Utilities.hpp"
enum LanguageRequirement
{
// Cakelisp's Evaluator will handle the invocation without generating any C/C++ code
LanguageRequirement_Evaluated,
// All C features are considered valid C++
LanguageRequirement_C,
// Only features which only work in C++ and do not work in C
LanguageRequirement_Cpp,
};
enum EvaluationTime
{
EvaluationTime_EvaluatedImmediately = 1 << 0,
EvaluationTime_CompileTime = 1 << 1,
EvaluationTime_Runtime = 1 << 2,
EvaluationTime_EvaluatedOnImport = 1 << 3,
};
enum GeneratorCategory
{
GeneratorCategory_Uncategorized,
GeneratorCategory_Math,
GeneratorCategory_Logic,
GeneratorCategory_Relational,
GeneratorCategory_Build,
GeneratorCategory_ControlFlow,
GeneratorCategory_Memory,
GeneratorCategory_Definitions,
GeneratorCategory_CodeGeneration,
};
struct GeneratorMetadata
{
const char* generatorName;
GeneratorCategory category;
LanguageRequirement language;
int evaluationTime; // EvaluationTime flags
int minimumArguments;
int maximumArguments;
const char* description;
};
const int MaxArgumentsUnlimited = 99;
GeneratorMetadata g_generatorMetadata[] = {
//
// Build
//
{"skip-build", GeneratorCategory_Build, LanguageRequirement_Evaluated,
EvaluationTime_EvaluatedImmediately, 0, 0,
"[DEPRECATED] Mark the current module to be excluded from the runtime build process. This is "
"necessary when the module does not provide any runtime code. For example, a module with only "
"compile-time function definitions would need to skip building."},
{
"add-c-search-directory-global",
},
{
"add-cakelisp-search-directory",
},
{
"add-library-search-directory",
},
{
"add-c-search-directory-module",
},
{
"add-library-runtime-search-directory",
},
{
"set-module-option",
},
{
"set-cakelisp-option",
},
{
"add-build-options-global",
},
{
"add-cpp-build-dependency",
},
{
"add-c-build-dependency",
},
{
"add-build-options",
},
{
"add-compile-time-hook",
},
{
"add-compile-time-hook-module",
},
{
"add-linker-options",
},
{"add-static-link-objects", GeneratorCategory_Build, LanguageRequirement_Evaluated,
EvaluationTime_EvaluatedImmediately, 0, MaxArgumentsUnlimited,
"Link additional objects, static libraries, or (on Windows) compiled resources. Modification "
"times of the files in this list will be checked and cause a re-link if they are newer than "
"the cached executable."},
{
"add-compiler-link-options",
},
{
"add-build-config-label",
},
{
"add-library-dependency",
},
{
"import",
},
{
"c-import",
},
{
"comptime-define-symbol",
},
{
"comptime-cond",
},
{
"ignore",
},
{"export", GeneratorCategory_Uncategorized, LanguageRequirement_Evaluated,
EvaluationTime_EvaluatedOnImport, 1, MaxArgumentsUnlimited,
"When any other module imports the current module, evaluate the statements within this export "
"scope in the context of the other module. This allows modules to e.g. 'infect' other modules "
"with settings necessary for the importer to build. Compare to (export-and-evaluate), which "
"evaluates both in the export-defining module and in importer contexts."},
{"export-and-evaluate", GeneratorCategory_Uncategorized, LanguageRequirement_Evaluated,
(EvaluationTime_EvaluatedImmediately | EvaluationTime_EvaluatedOnImport), 1,
MaxArgumentsUnlimited,
"Evaluate the contained statements. Additionally, when any other module imports the current "
"module, evaluate the statements within this export scope in the context of the other module. "
"This allows modules to e.g. 'infect' other modules with settings necessary for the importer "
"to build. Compare to (export), which only evaluates in importer contexts."},
//
// Math
//
{"decr", GeneratorCategory_Math, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 1, 1,
"Subtract one from the first argument and set it to the new value. The subtraction is a "
"pre-decrement. Equivalent to (set arg (- arg 1))"},
{"--", GeneratorCategory_Math, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 1, 1,
"Subtract one from the first argument and set it to the new value. The subtraction is a "
"pre-decrement. Equivalent to (set arg (- arg 1))"},
{"incr", GeneratorCategory_Math, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 1, 1,
"Add one to the first argument and set it to the new value. The addition is a pre-increment. "
"Equivalent to (set arg (+ arg 1))"},
{"++", GeneratorCategory_Math, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 1, 1,
"Add one to the first argument and set it to the new value. The addition is a pre-increment. "
"Equivalent to (set arg (+ arg 1))"},
{"mod", GeneratorCategory_Math, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, 2,
"Perform the modulo (%) operation on the first value with the second as the divisor. E.g. "
"(mod 5 2) = 1. The modulo operator returns the remainder after the division."},
{"%", GeneratorCategory_Math, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, 2,
"Perform the modulo (%) operation on the first value with the second as the divisor. E.g. (% "
"5 2) = 1. The modulo operator returns the remainder after the division."},
{"/", GeneratorCategory_Math, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, MaxArgumentsUnlimited,
"Divide the arguments. E.g. (/ 4 2) = 2."},
{"*", GeneratorCategory_Math, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, MaxArgumentsUnlimited,
"Multiply the arguments. E.g. (* 4 2) = 8."},
{"+", GeneratorCategory_Math, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, MaxArgumentsUnlimited,
"Add the arguments. E.g. (+ 4 2) = 6."},
{"-", GeneratorCategory_Math, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, MaxArgumentsUnlimited,
"Subtract the arguments. E.g. (- 4 2) = 2."},
{"bit-<<", GeneratorCategory_Math, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, 2,
"Shift the bits in the first argument to the left by the number indicated by the second "
"argument. E.g. (bit-<< 1 2) = 4."},
{"bit->>", GeneratorCategory_Math, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, 2,
"Shift the bits in the first argument to the right by the number indicated by the second "
"argument. E.g. (bit->> 4 2) = 1."},
{
"bit-or",
},
{
"bit-ones-complement",
},
{
"bit-xor",
},
{
"bit-and",
},
//
// Relational
//
{"=", GeneratorCategory_Relational, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, 2,
"Evaluate if the two arguments are equal. Evaluates as true if equal or false otherwise."},
{"!=", GeneratorCategory_Relational, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, 2,
"Evaluate if the two arguments are not equal. Evaluates as true if not equal or false if "
"equal."},
{">=", GeneratorCategory_Relational, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, 2,
"Evaluate if the first argument is greater than or equal to the second argument."},
{"<=", GeneratorCategory_Relational, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, 2,
"Evaluate if the first argument is less than or equal to the second argument."},
{">", GeneratorCategory_Relational, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, 2,
"Evaluate if the first argument is greater than the second argument."},
{"<", GeneratorCategory_Relational, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, 2,
"Evaluate if the first argument is less than the second argument."},
//
// Logical
//
{"and", GeneratorCategory_Logic, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, MaxArgumentsUnlimited,
"Evaluate to true if all the expression arguments result in true. Lazily evaluates, "
"i.e. evaluation will stop as soon as any expression evaluates to false."},
{"or", GeneratorCategory_Logic, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, MaxArgumentsUnlimited,
"Evaluate to true if any of the expression arguments result in true. Lazily evaluates, "
"i.e. evaluation will stop as soon as any expression evaluates to true."},
{"not", GeneratorCategory_Logic, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 1, 1,
"Invert the boolean result of the argument. E.g. (not true) = false, (not false) = true"},
//
// Control flow
//
{"return", GeneratorCategory_ControlFlow, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 0, 1,
"Return from the current function. If a return value is specified, return that value to the "
"caller."},
{"if", GeneratorCategory_ControlFlow, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, 3,
"If the first argument (condition) evaluates to true, execute the second argument (true "
"block). If the first argument evaluates to false, execute the third argument (false block), "
"if specified. Use (scope) in order to specify more than one statement in the true or false "
"blocks."},
{"when", GeneratorCategory_ControlFlow, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, 2,
"If the first argument (condition) evaluates to true, execute the second argument (true "
"block). This is a way to write e.g. (if condition (do-thing))"},
{"unless", GeneratorCategory_ControlFlow, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, 2,
"If the first argument (condition) evaluates to false, execute the second argument (false "
"block). This is a cleaner way to write e.g. (if (not condition) (do-thing))"},
{
"cond",
},
{
"break",
},
{
"while",
},
{
"?",
},
{
"for-in",
},
{
"continue",
},
{
"type-cast",
},
{
"call-on",
},
{
"scope",
},
{
"rename-builtin",
},
{
"block",
},
{
"type",
},
{
"call",
},
{
"call-on-ptr",
},
{
"comptime-error",
},
{"defer", GeneratorCategory_ControlFlow, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 1, MaxArgumentsUnlimited,
"Defer the statements in the body until the current scope is exited. This is useful to e.g. "
"(defer (free buffer)). defer will ensure that the code is called, both in \"natural\" scope "
"exits and \"explicit\" exits (those caused by return, continue, or break)."},
//
// Definitions
//
{"var", GeneratorCategory_Definitions, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, 3,
"Define a variable. The first argument is the name, the second is the type, and the optional "
"third argument is the initial value. When used in module scope (i.e., not within any "
"function), the variable will be local to the module and initialized on application startup."},
{"var-global", GeneratorCategory_Definitions, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, 3,
"Define a global variable. The first argument is the name, the second is the type, and the "
"optional third argument is the initial value. This is only valid in module scope (i.e., not "
"within any function). The variable will be exposed to other modules which import the current "
"module. It is initialized on application startup."},
{"var-static", GeneratorCategory_Definitions, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, 3,
"Define a static variable. The first argument is the name, the second is the type, and the "
"optional third argument is the initial value. This is only valid in body scope (i.e., within "
"a function). The variable will retain its value since the previous execution of the "
"function. It is initialized the first time the declaration is hit at runtime."},
{
"defgenerator",
},
{
"defmacro",
},
{
"defstruct",
},
{
"defstruct-local",
},
{
"def-function-signature",
},
{
"def-function-signature-global",
},
{
"defun",
},
{
"defun-local",
},
{
"defun-nodecl",
},
{
"defun-comptime",
},
{
"def-type-alias",
},
{
"def-type-alias-global",
},
{
"array",
},
//
// Memory
//
{
"set",
},
{"field", GeneratorCategory_Memory, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, MaxArgumentsUnlimited,
"field accesses the field of a structure. (field my-struct field) would generate "
"myStruct.field. If you need to access fields through a pointer, use (path) instead."},
{"path", GeneratorCategory_Memory, LanguageRequirement_C,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 3, MaxArgumentsUnlimited,
"path allows you to access nested fields, including through pointers. For example, (path "
"my-ptr > ptr-to-struct > inline-struct . field) would generate "
"myPtr->ptrToStruct->inlineStruct.field. Use (field) if you are not accessing memory "
"through pointers."},
{
"addr",
},
{
"deref",
},
{
"at",
},
//
// Code generation
//
{
"tokenize-push",
GeneratorCategory_CodeGeneration,
},
{"splice-point", GeneratorCategory_CodeGeneration, LanguageRequirement_Evaluated,
EvaluationTime_EvaluatedImmediately, 1, 1,
"Define a new named splice point. This point can be used at compile-time by functions like "
"ClearAndEvaluateAtSplicePoint() to evaluate code at specific locations in the module. A "
"similar function is ReplaceAndEvaluateDefinition(), which is required when replacing a "
"definition in order to notify the Environment to remove the old definition."},
//
// Misc. C helpers
//
{
"c-preprocessor-define",
},
{
"c-preprocessor-define-global",
},
//
// C++ helpers. TODO: Move to CppHelpers.cake instead?
//
{
"in",
},
{"new", GeneratorCategory_Memory, LanguageRequirement_Cpp,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 1, 1,
"Run new on the first argument, a type. Returns a pointer to the memory allocated after the "
"constructor has been called."},
{"delete", GeneratorCategory_Memory, LanguageRequirement_Cpp,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 1, 1,
"Run delete on the argument, calling its destructor then freeing the memory."},
{"new-array", GeneratorCategory_Memory, LanguageRequirement_Cpp,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 2, 2,
"Run new[] on the first argument, a type. The second argument is the number of instances to "
"create. Returns a pointer to the memory allocated after the constructor has been called for "
"each instance."},
{"delete-array", GeneratorCategory_Memory, LanguageRequirement_Cpp,
(EvaluationTime_CompileTime | EvaluationTime_Runtime), 1, 1,
"Run delete[] on the argument, which should be a pointer to an array created via new[]. Each "
"instance will have its destructor called, then the memory will be freed."},
};
void printBuiltInGeneratorMetadata(EvaluatorEnvironment* environment)
{
for (GeneratorIterator it = environment->generators.begin();
it != environment->generators.end(); ++it)
{
GeneratorMetadata* foundMetadata = nullptr;
for (unsigned int i = 0; i < ArraySize(g_generatorMetadata); ++i)
{
if (0 == it->first.compare(g_generatorMetadata[i].generatorName))
{
foundMetadata = &g_generatorMetadata[i];
break;
}
}
if (!foundMetadata)
Logf("warning: built-in generator %s is missing metadata\n", it->first.c_str());
}
for (unsigned int i = 0; i < ArraySize(g_generatorMetadata); ++i)
{
GeneratorMetadata* currentMetadata = &g_generatorMetadata[i];
Logf("------------------------------\n%s\n\t%s\n", currentMetadata->generatorName,
currentMetadata->description);
}
// TODO: Also check that nothing in the metadata is missing from the generators list.
}