diff options
author | Leo Tenenbaum <pommicket@gmail.com> | 2020-03-08 10:19:24 -0400 |
---|---|---|
committer | Leo Tenenbaum <pommicket@gmail.com> | 2020-03-08 10:19:24 -0400 |
commit | 13fe154980092cdc79c0df6922b164eb2137f3e3 (patch) | |
tree | 8fe714ee9afae2bb59c3ae4ff64070677b471904 | |
parent | 095b24cce985bb56baf128ebcfcf2d9476d3104f (diff) |
more varargs
-rw-r--r-- | main.c | 2 | ||||
-rw-r--r-- | test.toc | 3 | ||||
-rw-r--r-- | types.c | 28 |
3 files changed, 20 insertions, 13 deletions
@@ -8,7 +8,9 @@ /* TODO: +why aren't we computing table_index at the end? variadic fns +make sure you can't have a variadic function pointer #foreign variadic fns where #returns_code (function/struct body is a block, to be evaluated at compile time, which returns the actual statements -- you can use this for implementation of printf) @@ -1,6 +1,7 @@ #include "std/io.toc", io; -foo ::= fn(x: ..) { +foo ::= fn(x: ..) int { + 3 }; main ::= fn() { @@ -2187,21 +2187,25 @@ static Status types_expr(Typer *tr, Expression *e) { ret_type = f->type.fn.types; param_types = ret_type + 1; } + + bool has_varargs = fn_decl && (fn_decl->flags & FN_EXPR_HAS_VARARGS); /* check types of arguments */ for (size_t p = 0; p < nparams; ++p) { - Expression *arg = &arg_exprs[p]; - Type *expected = ¶m_types[p]; - Type *got = &arg->type; - if (!type_eq(expected, got)) { - char *estr = type_to_str(expected); - char *gstr = type_to_str(got); - err_print(arg->where, "Expected type %s as argument to function, but got %s.", estr, gstr); - return false; - } - if (got->flags & TYPE_IS_FLEXIBLE) { - /* "cast" */ - *got = *expected; + if (p != nparams-1 || !has_varargs) { + Expression *arg = &arg_exprs[p]; + Type *expected = ¶m_types[p]; + Type *got = &arg->type; + if (!type_eq(expected, got)) { + char *estr = type_to_str(expected); + char *gstr = type_to_str(got); + err_print(arg->where, "Expected type %s as argument to function, but got %s.", estr, gstr); + return false; + } + if (got->flags & TYPE_IS_FLEXIBLE) { + /* "cast" */ + *got = *expected; + } } } if (fn_type->constness) { |