aboutsummaryrefslogtreecommitdiff
path: root/src/check_builtin.cpp
diff options
context:
space:
mode:
authorgingerBill <bill@gingerbill.org>2021-10-20 01:51:16 +0100
committergingerBill <bill@gingerbill.org>2021-10-20 01:51:16 +0100
commit7faca7066c30d6e663b268dc1e8ec66710ae3dd5 (patch)
tree0a1b66625ec88dfc4abbc2e7ae0a1e0336eafebf /src/check_builtin.cpp
parent3eaac057da11d28cbedd7321f9f6368588b0b4ee (diff)
Add builtin `transpose`
Diffstat (limited to 'src/check_builtin.cpp')
-rw-r--r--src/check_builtin.cpp34
1 files changed, 32 insertions, 2 deletions
diff --git a/src/check_builtin.cpp b/src/check_builtin.cpp
index a04302d01..659a74ad7 100644
--- a/src/check_builtin.cpp
+++ b/src/check_builtin.cpp
@@ -1966,13 +1966,13 @@ bool check_builtin_procedure(CheckerContext *c, Operand *operand, Ast *call, i32
return false;
}
if (!is_operand_value(x)) {
- error(call, "'soa_unzip' expects an #soa slice");
+ error(call, "'%.*s' expects an #soa slice", LIT(builtin_name));
return false;
}
Type *t = base_type(x.type);
if (!is_type_soa_struct(t) || t->Struct.soa_kind != StructSoa_Slice) {
gbString s = type_to_string(x.type);
- error(call, "'soa_unzip' expects an #soa slice, got %s", s);
+ error(call, "'%.*s' expects an #soa slice, got %s", LIT(builtin_name), s);
gb_string_free(s);
return false;
}
@@ -1987,6 +1987,36 @@ bool check_builtin_procedure(CheckerContext *c, Operand *operand, Ast *call, i32
operand->mode = Addressing_Value;
break;
}
+
+ case BuiltinProc_transpose: {
+ Operand x = {};
+ check_expr(c, &x, ce->args[0]);
+ if (x.mode == Addressing_Invalid) {
+ return false;
+ }
+ if (!is_operand_value(x)) {
+ error(call, "'%.*s' expects a matrix or array", LIT(builtin_name));
+ return false;
+ }
+ Type *t = base_type(x.type);
+ if (!is_type_matrix(t) && !is_type_array(t)) {
+ gbString s = type_to_string(x.type);
+ error(call, "'%.*s' expects a matrix or array, got %s", LIT(builtin_name), s);
+ gb_string_free(s);
+ return false;
+ }
+
+ operand->mode = Addressing_Value;
+ if (is_type_array(t)) {
+ // Do nothing
+ operand->type = x.type;
+ } else {
+ GB_ASSERT(t->kind == Type_Matrix);
+ operand->type = alloc_type_matrix(t->Matrix.elem, t->Matrix.column_count, t->Matrix.row_count);
+ }
+ operand->type = check_matrix_type_hint(operand->type, type_hint);
+ break;
+ }
case BuiltinProc_simd_vector: {
Operand x = {};