Allow using an untyped empty array/set/table literal for places where

the expected type is known
This commit is contained in:
Bruce Hill 2025-04-03 15:40:57 -04:00
parent 8d173710fe
commit 8ab991fba5
3 changed files with 31 additions and 12 deletions

View File

@ -866,6 +866,7 @@ static CORD _compile_statement(env_t *env, ast_t *ast)
return compile_statement(env, WrapAST(ast, DocTest, .expr=decl->value, .expected=test->expected, .skip_source=test->skip_source)); return compile_statement(env, WrapAST(ast, DocTest, .expr=decl->value, .expected=test->expected, .skip_source=test->skip_source));
CORD var = CORD_all("_$", Match(decl->var, Var)->name); CORD var = CORD_all("_$", Match(decl->var, Var)->name);
type_t *t = get_type(env, decl->value); type_t *t = get_type(env, decl->value);
if (!t) code_err(decl->value, "I couldn't figure out the type of this value!");
CORD val_code = compile_maybe_incref(env, decl->value, t); CORD val_code = compile_maybe_incref(env, decl->value, t);
if (t->tag == FunctionType) { if (t->tag == FunctionType) {
assert(promote(env, decl->value, &val_code, t, Type(ClosureType, t))); assert(promote(env, decl->value, &val_code, t, Type(ClosureType, t)));
@ -1840,6 +1841,16 @@ CORD compile_to_type(env_t *env, ast_t *ast, type_t *t)
} }
} else if (ast->tag == None && Match(ast, None)->type == NULL) { } else if (ast->tag == None && Match(ast, None)->type == NULL) {
return compile_none(t); return compile_none(t);
} else if (t->tag == ArrayType && ast->tag == Array && !Match(ast, Array)->item_type && !Match(ast, Array)->items) {
return compile(env, ast);
} else if (t->tag == TableType && ast->tag == Table) {
auto table = Match(ast, Table);
if (!table->key_type && !table->value_type && !table->default_value && !table->fallback && !table->entries)
return compile(env, ast);
} else if (t->tag == SetType && ast->tag == Set) {
auto set = Match(ast, Set);
if (!set->item_type && !set->items)
return compile(env, ast);
} }
type_t *actual = get_type(env, ast); type_t *actual = get_type(env, ast);
@ -2695,13 +2706,13 @@ CORD compile(env_t *env, ast_t *ast)
"})"); "})");
} }
case Array: { case Array: {
type_t *array_type = get_type(env, ast);
type_t *item_type = Match(array_type, ArrayType)->item_type;
auto array = Match(ast, Array); auto array = Match(ast, Array);
if (!array->items) if (!array->items)
return "(Array_t){.length=0}"; return "(Array_t){.length=0}";
type_t *array_type = get_type(env, ast);
type_t *item_type = Match(array_type, ArrayType)->item_type;
int64_t n = 0; int64_t n = 0;
for (ast_list_t *item = array->items; item; item = item->next) { for (ast_list_t *item = array->items; item; item = item->next) {
++n; ++n;
@ -3758,6 +3769,7 @@ CORD compile(env_t *env, ast_t *ast)
CORD compile_type_info(type_t *t) CORD compile_type_info(type_t *t)
{ {
if (t == NULL) compiler_err(NULL, NULL, NULL, "Attempt to compile a NULL type");
if (t == PATH_TYPE) return "&Path$info"; if (t == PATH_TYPE) return "&Path$info";
else if (t == PATH_TYPE_TYPE) return "&PathType$info"; else if (t == PATH_TYPE_TYPE) return "&PathType$info";

View File

@ -710,15 +710,13 @@ PARSER(parse_array) {
whitespace(&pos); whitespace(&pos);
expect_closing(ctx, &pos, "]", "I wasn't able to parse the rest of this array"); expect_closing(ctx, &pos, "]", "I wasn't able to parse the rest of this array");
if (!item_type && !items)
parser_err(ctx, start, pos, "Empty arrays must specify what type they would contain (e.g. [:Int])");
REVERSE_LIST(items); REVERSE_LIST(items);
return NewAST(ctx->file, start, pos, Array, .item_type=item_type, .items=items); return NewAST(ctx->file, start, pos, Array, .item_type=item_type, .items=items);
} }
PARSER(parse_table) { PARSER(parse_table) {
const char *start = pos; const char *start = pos;
if (match(&pos, "{/}")) return NULL;
if (!match(&pos, "{")) return NULL; if (!match(&pos, "{")) return NULL;
whitespace(&pos); whitespace(&pos);
@ -759,9 +757,6 @@ PARSER(parse_table) {
REVERSE_LIST(entries); REVERSE_LIST(entries);
if (!key_type && !value_type && !entries)
return NULL;
whitespace(&pos); whitespace(&pos);
ast_t *fallback = NULL, *default_value = NULL; ast_t *fallback = NULL, *default_value = NULL;
@ -798,6 +793,9 @@ PARSER(parse_table) {
PARSER(parse_set) { PARSER(parse_set) {
const char *start = pos; const char *start = pos;
if (match(&pos, "{/}"))
return NewAST(ctx->file, start, pos, Set);
if (!match(&pos, "{")) return NULL; if (!match(&pos, "{")) return NULL;
whitespace(&pos); whitespace(&pos);
@ -833,9 +831,6 @@ PARSER(parse_set) {
REVERSE_LIST(items); REVERSE_LIST(items);
if (!item_type && !items)
return NULL;
whitespace(&pos); whitespace(&pos);
expect_closing(ctx, &pos, "}", "I wasn't able to parse the rest of this set"); expect_closing(ctx, &pos, "}", "I wasn't able to parse the rest of this set");

View File

@ -739,6 +739,10 @@ type_t *get_type(env_t *env, ast_t *ast)
} }
if (has_stack_memory(item_type)) if (has_stack_memory(item_type))
code_err(ast, "Arrays cannot hold stack references, because the array may outlive the stack frame the reference was created in."); code_err(ast, "Arrays cannot hold stack references, because the array may outlive the stack frame the reference was created in.");
if (!item_type)
code_err(ast, "I couldn't figure out the item type for this array!");
return Type(ArrayType, .item_type=item_type); return Type(ArrayType, .item_type=item_type);
} }
case Set: { case Set: {
@ -766,6 +770,10 @@ type_t *get_type(env_t *env, ast_t *ast)
item_type = item_merged; item_type = item_merged;
} }
} }
if (!item_type)
code_err(ast, "I couldn't figure out the item type for this set!");
if (has_stack_memory(item_type)) if (has_stack_memory(item_type))
code_err(ast, "Sets cannot hold stack references because the set may outlive the reference's stack frame."); code_err(ast, "Sets cannot hold stack references because the set may outlive the reference's stack frame.");
return Type(SetType, .item_type=item_type); return Type(SetType, .item_type=item_type);
@ -809,6 +817,10 @@ type_t *get_type(env_t *env, ast_t *ast)
value_type = val_merged; value_type = val_merged;
} }
} }
if (!key_type || !value_type)
code_err(ast, "I couldn't figure out the key and value types for this table!");
if (has_stack_memory(key_type) || has_stack_memory(value_type)) if (has_stack_memory(key_type) || has_stack_memory(value_type))
code_err(ast, "Tables cannot hold stack references because the table may outlive the reference's stack frame."); code_err(ast, "Tables cannot hold stack references because the table may outlive the reference's stack frame.");
return Type(TableType, .key_type=key_type, .value_type=value_type, .default_value=table->default_value, .env=env); return Type(TableType, .key_type=key_type, .value_type=value_type, .default_value=table->default_value, .env=env);