[qfcc] Check declarator type chains

Arrays of functions or functions that return arrays or functions aren't
valid. While working on how to get generics in properly, I finally
understood what's going on with function types in the specifier, and I
think I'll be able to sort out function pointers vs prototypes, too.
This commit is contained in:
Bill Currie 2024-08-21 08:23:12 +09:00
parent 856761c4a9
commit a00b17bf71
2 changed files with 31 additions and 2 deletions

View file

@ -16,7 +16,7 @@ string ty_meta_name[10] = {
//FIXME use pr_type_names.h, but need to fix unsigned, and add missing types
#define field .int
#define func void()(void)
#define func @function(void)
#define ptr void *
#define uint unsigned
#define ulong unsigned long

View file

@ -421,6 +421,17 @@ function_spec (specifier_t spec, param_t *params)
spec = default_type (spec, spec.sym);
}
spec.sym->params = params;
if (spec.sym->type) {
if (is_func (spec.sym->type)) {
error (0, "'%s' declared as a function returning a function",
spec.sym->name);
} else if (is_array (spec.sym->type)) {
error (0, "declaration of '%s' as array of functions",
spec.sym->name);
} else if (!is_ptr (spec.sym->type) && !is_field (spec.sym->type)) {
internal_error (0, "unexpected type");
}
}
spec.sym->type = append_type (spec.sym->type, parse_params (0, params));
spec.is_function = true; //FIXME do proper void(*)() -> ev_func
spec.is_generic = generic_scope;
@ -433,6 +444,16 @@ static specifier_t
array_spec (specifier_t spec, unsigned size)
{
spec = default_type (spec, spec.sym);
if (spec.sym->type) {
if (is_func (spec.sym->type)) {
error (0, "'%s' declared as function returning an array",
spec.sym->name);
} else if (!is_ptr (spec.sym->type)
&& !is_array (spec.sym->type)
&& !is_field (spec.sym->type)) {
internal_error (0, "unexpected type");
}
}
spec.sym->type = append_type (spec.sym->type, array_type (0, size));
return spec;
}
@ -440,6 +461,14 @@ array_spec (specifier_t spec, unsigned size)
static specifier_t
pointer_spec (specifier_t quals, specifier_t spec)
{
if (spec.sym->type) {
if (!is_func (spec.sym->type)
&& !is_ptr (spec.sym->type)
&& !is_array (spec.sym->type)
&& !is_field (spec.sym->type)) {
internal_error (0, "unexpected type");
}
}
spec.sym->type = append_type (spec.sym->type, pointer_type (0));
return spec;
}
@ -688,7 +717,7 @@ qc_param_list
;
// quakec function parameters cannot use a typedef as the return type
// in the first parameter (really, they should't use any as standard quakec
// doesn't support typedef, but that seems overly restrictive), howevery,
// doesn't support typedef, but that seems overly restrictive), however,
// they can stil use a typedef first parameter. This is due to grammar issues
qc_first_param
: typespec identifier