aboutsummaryrefslogtreecommitdiff
path: root/enums.c
diff options
context:
space:
mode:
authorBruce Hill <bruce@bruce-hill.com>2024-02-24 15:24:44 -0500
committerBruce Hill <bruce@bruce-hill.com>2024-02-24 15:24:44 -0500
commit106704b9564b50e95dfce50b7a7471e73b64a78e (patch)
tree509e94ff49f76c9c2edc75c20fe1da9bfd600912 /enums.c
parentc7470346457550467bafd9199185c5d0e8481f02 (diff)
Improve enums with metamethods
Diffstat (limited to 'enums.c')
-rw-r--r--enums.c136
1 files changed, 136 insertions, 0 deletions
diff --git a/enums.c b/enums.c
new file mode 100644
index 00000000..890bf551
--- /dev/null
+++ b/enums.c
@@ -0,0 +1,136 @@
+
+#include <ctype.h>
+#include <gc/cord.h>
+#include <gc.h>
+#include <stdio.h>
+
+#include "ast.h"
+#include "builtins/string.h"
+#include "compile.h"
+#include "structs.h"
+#include "environment.h"
+#include "typecheck.h"
+#include "util.h"
+
+static CORD compile_str_method(env_t *env, ast_t *ast)
+{
+ auto def = Match(ast, EnumDef);
+ CORD str_func = CORD_all("static CORD ", def->name, "__as_str(", def->name, "_t *obj, bool use_color) {\n"
+ "\tif (!obj) return \"", def->name, "\";\n"
+ "switch (obj->$tag) {\n");
+ for (tag_ast_t *tag = def->tags; tag; tag = tag->next) {
+ str_func = CORD_all(str_func, "\tcase $tag$", def->name, "$", tag->name, ": return CORD_all(use_color ? \"\\x1b[36;1m",
+ def->name, ".", tag->name, "\\x1b[m(\" : \"", def->name, ".", tag->name, "(\"");
+
+ if (tag->secret) {
+ str_func = CORD_cat(str_func, ", \"***)\");\n");
+ continue;
+ }
+
+ for (arg_ast_t *field = tag->fields; field; field = field->next) {
+ type_t *field_t = parse_type_ast(env, field->type);
+ CORD field_str = expr_as_string(env, CORD_all("obj->", tag->name, ".", field->name), field_t, "use_color");
+ str_func = CORD_all(str_func, ", \"", field->name, "=\", ", field_str);
+ if (field->next) str_func = CORD_cat(str_func, ", \", \"");
+ }
+ str_func = CORD_cat(str_func, ", \")\");\n");
+ }
+ str_func = CORD_cat(str_func, "\t}\n}\n");
+ return str_func;
+}
+
+static CORD compile_compare_method(env_t *env, ast_t *ast)
+{
+ auto def = Match(ast, EnumDef);
+ CORD cmp_func = CORD_all("static int ", def->name, "__compare(const ", def->name, "_t *x, const ", def->name,
+ "_t *y, const TypeInfo *info) {\n"
+ "int diff = x->$tag - y->$tag;\n"
+ "if (diff) return diff;\n"
+ "switch (x->$tag) {\n");
+ for (tag_ast_t *tag = def->tags; tag; tag = tag->next) {
+ type_t *tag_type = Table_str_get(env->types, heap_strf("%s$%s", def->name, tag->name));
+ cmp_func = CORD_all(cmp_func, "\tcase $tag$", def->name, "$", tag->name, ": "
+ "return generic_compare(&x->", tag->name, ", &y->", tag->name, ", ", compile_type_info(env, tag_type), ");\n");
+ }
+ cmp_func = CORD_all(cmp_func, "}\n}\n");
+ return cmp_func;
+}
+
+static CORD compile_equals_method(env_t *env, ast_t *ast)
+{
+ auto def = Match(ast, EnumDef);
+ CORD eq_func = CORD_all("static bool ", def->name, "__equal(const ", def->name, "_t *x, const ", def->name,
+ "_t *y, const TypeInfo *info) {\n"
+ "if (x->$tag != y->$tag) return no;\n"
+ "switch (x->$tag) {\n");
+ for (tag_ast_t *tag = def->tags; tag; tag = tag->next) {
+ type_t *tag_type = Table_str_get(env->types, heap_strf("%s$%s", def->name, tag->name));
+ eq_func = CORD_all(eq_func, "\tcase $tag$", def->name, "$", tag->name, ": "
+ "return generic_equal(&x->", tag->name, ", &y->", tag->name, ", ", compile_type_info(env, tag_type), ");\n");
+ }
+ eq_func = CORD_all(eq_func, "}\n}\n");
+ return eq_func;
+}
+
+static CORD compile_hash_method(env_t *env, ast_t *ast)
+{
+ auto def = Match(ast, EnumDef);
+ CORD hash_func = CORD_all("static uint32_t ", def->name, "__hash(const ", def->name, "_t *obj, const TypeInfo *info) {\n"
+ "uint32_t hashes[2] = {(uint32_t)obj->$tag};\n"
+ "switch (obj->$tag) {\n");
+ for (tag_ast_t *tag = def->tags; tag; tag = tag->next) {
+ type_t *tag_type = Table_str_get(env->types, heap_strf("%s$%s", def->name, tag->name));
+ hash_func = CORD_all(hash_func, "\tcase $tag$", def->name, "$", tag->name, ": "
+ "hashes[1] = generic_hash(&obj->", tag->name, ", ", compile_type_info(env, tag_type), ");\n"
+ "break;\n");
+ }
+ hash_func = CORD_all(hash_func, "}\n"
+ "uint32_t hash;\n"
+ "halfsiphash(&hashes, sizeof(hashes), SSS_HASH_VECTOR, (uint8_t*)&hash, sizeof(hash));\n"
+ "return hash;\n}\n");
+ return hash_func;
+}
+
+void compile_enum_def(env_t *env, ast_t *ast)
+{
+ auto def = Match(ast, EnumDef);
+ CORD_appendf(&env->code->typedefs, "typedef struct %s_s %s_t;\n", def->name, def->name);
+ CORD_appendf(&env->code->typedefs, "typedef struct { TypeInfo type; } %s_namespace_t;\n", def->name);
+ CORD_appendf(&env->code->typedefs, "extern %s_namespace_t %s;\n", def->name, def->name);
+ CORD enum_def = CORD_all("struct ", def->name, "_s {\n"
+ "\tenum {");
+ for (tag_ast_t *tag = def->tags; tag; tag = tag->next) {
+ CORD_appendf(&enum_def, "$tag$%s$%s = %ld", def->name, tag->name, tag->value);
+ if (tag->next) enum_def = CORD_cat(enum_def, ", ");
+ }
+ enum_def = CORD_cat(enum_def, "} $tag;\n"
+ "union {\n");
+ for (tag_ast_t *tag = def->tags; tag; tag = tag->next) {
+ compile_struct_def(env, WrapAST(ast, StructDef, .name=heap_strf("%s$%s", def->name, tag->name), .fields=tag->fields));
+ enum_def = CORD_all(enum_def, def->name, "$", tag->name, "_t ", tag->name, ";\n");
+ CORD_appendf(&env->code->typedefs, "#define %s__%s(...) ((%s_t){$tag$%s$%s, .%s={__VA_ARGS__}})\n",
+ def->name, tag->name, def->name, def->name, tag->name, tag->name);
+ }
+ enum_def = CORD_cat(enum_def, "};\n};\n");
+ env->code->typecode = CORD_cat(env->code->typecode, enum_def);
+
+ type_t *t = Table_str_get(env->types, def->name);
+ CORD typeinfo = CORD_asprintf("public %s_namespace_t %s = {{%zu, %zu, {.tag=CustomInfo, .CustomInfo={",
+ def->name, def->name, type_size(t), type_align(t));
+
+ env->code->funcs = CORD_all(
+ env->code->funcs,
+ compile_str_method(env, ast),
+ compile_equals_method(env, ast), compile_compare_method(env, ast),
+ compile_hash_method(env, ast));
+ typeinfo = CORD_all(
+ typeinfo,
+ ".as_str=(void*)", def->name, "__as_str, "
+ ".equal=(void*)", def->name, "__equal, "
+ ".hash=(void*)", def->name, "__hash, "
+ ".compare=(void*)", def->name, "__compare");
+ typeinfo = CORD_cat(typeinfo, "}}}};\n");
+ env->code->typeinfos = CORD_all(env->code->typeinfos, typeinfo);
+}
+
+// vim: ts=4 sw=0 et cino=L2,l1,(0,W4,m1,\:0