mirror of
https://git.code.sf.net/p/quake/quakeforge
synced 2025-04-06 01:11:59 +00:00
[qfcc] More work towards proper reference handling
Generic function parameters can now use references (currently via `@reference()`) for generic types. And seem to work. There's still a problem with chains: `atomicExchange(__imageTexel(...))` is dereferencing __imageTexel's return value.
This commit is contained in:
parent
e2f3a00a78
commit
9c3dc555a3
4 changed files with 87 additions and 44 deletions
|
@ -66,6 +66,8 @@ typedef struct genparam_s {
|
|||
typeeval_t *compute;
|
||||
int gentype; // index into function's list of types
|
||||
param_qual_t qual;
|
||||
bool is_reference;
|
||||
unsigned tag;
|
||||
} genparam_t;
|
||||
|
||||
typedef struct genfunc_s {
|
||||
|
|
|
@ -287,9 +287,15 @@ build_intrinsic_call (const expr_t *expr, symbol_t *fsym, const type_t *ftype,
|
|||
current_symtab = scope;
|
||||
list_gather (&call->intrinsic.operands, extra_args, extra_count);
|
||||
} else {
|
||||
for (int i = 0; i < arg_count; i++) {
|
||||
if (is_reference (get_type (arguments[i]))) {
|
||||
auto p = fsym->params;
|
||||
for (int i = 0; i < arg_count; i++, p = p->next) {
|
||||
auto arg_type = get_type (arguments[i]);
|
||||
if (is_reference (p->type) && !is_reference (arg_type)) {
|
||||
arguments[i] = address_expr (arguments[i], nullptr);
|
||||
} else if (!is_reference (p->type) && is_reference (arg_type)) {
|
||||
arguments[i] = pointer_deref (arguments[i]);
|
||||
} else {
|
||||
arguments[i] = arguments[i];
|
||||
}
|
||||
}
|
||||
list_gather (&call->intrinsic.operands, arguments, arg_count);
|
||||
|
|
|
@ -260,13 +260,41 @@ valid_type_list (const expr_t *expr, rua_ctx_t *ctx)
|
|||
return types;
|
||||
}
|
||||
|
||||
static symbol_t *
|
||||
get_gensym (const expr_t *type_expr, bool *is_reference, unsigned *tag,
|
||||
rua_ctx_t *ctx)
|
||||
{
|
||||
*is_reference = false;
|
||||
*tag = 0;
|
||||
if (!type_expr) {
|
||||
return nullptr;
|
||||
}
|
||||
if (type_expr->type == ex_type && type_expr->typ.op == QC_REFERENCE) {
|
||||
auto params = type_expr->typ.params;
|
||||
if (params && params->list.head) {
|
||||
*is_reference = true;
|
||||
if (params->list.head->next) {
|
||||
auto tag_expr = params->list.head->next->expr;
|
||||
tag_expr = expr_process (tag_expr, ctx);
|
||||
*tag = expr_integral (tag_expr);
|
||||
}
|
||||
type_expr = params->list.head->expr;
|
||||
}
|
||||
}
|
||||
if (type_expr->type != ex_symbol
|
||||
|| type_expr->symbol->sy_type != sy_type_param) {
|
||||
return nullptr;
|
||||
}
|
||||
return type_expr->symbol;
|
||||
}
|
||||
|
||||
static gentype_t
|
||||
make_gentype (const expr_t *expr, rua_ctx_t *ctx)
|
||||
{
|
||||
if (expr->type != ex_symbol || expr->symbol->sy_type != sy_type_param) {
|
||||
internal_error (expr, "expected generic type name");
|
||||
}
|
||||
auto sym = expr->symbol;
|
||||
bool is_reference = false;
|
||||
unsigned tag = 0;
|
||||
// strip off any reference type
|
||||
auto sym = get_gensym (expr, &is_reference, &tag, ctx);
|
||||
gentype_t gentype = {
|
||||
.name = save_string (sym->name),
|
||||
.valid_types = valid_type_list (sym->expr, ctx),
|
||||
|
@ -278,15 +306,16 @@ make_gentype (const expr_t *expr, rua_ctx_t *ctx)
|
|||
}
|
||||
|
||||
static int
|
||||
find_gentype (const expr_t *expr, genfunc_t *genfunc)
|
||||
find_gentype (const expr_t *expr, genfunc_t *genfunc, bool *is_reference,
|
||||
unsigned *tag, rua_ctx_t *ctx)
|
||||
{
|
||||
if (!expr || expr->type != ex_symbol) {
|
||||
auto sym = get_gensym (expr, is_reference, tag, ctx);
|
||||
if (!sym) {
|
||||
return -1;
|
||||
}
|
||||
const char *name = expr->symbol->name;
|
||||
for (int i = 0; i < genfunc->num_types; i++) {
|
||||
auto t = &genfunc->types[i];
|
||||
if (strcmp (name, t->name) == 0) {
|
||||
if (strcmp (sym->name, t->name) == 0) {
|
||||
return i;
|
||||
}
|
||||
}
|
||||
|
@ -296,7 +325,10 @@ find_gentype (const expr_t *expr, genfunc_t *genfunc)
|
|||
static genparam_t
|
||||
make_genparam (param_t *param, genfunc_t *genfunc, rua_ctx_t *ctx)
|
||||
{
|
||||
int gentype = find_gentype (param->type_expr, genfunc);
|
||||
bool is_reference = false;
|
||||
unsigned tag = 0;
|
||||
int gentype = find_gentype (param->type_expr, genfunc, &is_reference, &tag,
|
||||
ctx);
|
||||
typeeval_t *compute = nullptr;
|
||||
if (gentype < 0 && param->type_expr) {
|
||||
compute = build_type_function (param->type_expr,
|
||||
|
@ -308,6 +340,8 @@ make_genparam (param_t *param, genfunc_t *genfunc, rua_ctx_t *ctx)
|
|||
.compute = compute,
|
||||
.gentype = gentype,
|
||||
.qual = param->qual,
|
||||
.is_reference = is_reference,
|
||||
.tag = tag,
|
||||
};
|
||||
return genparam;
|
||||
}
|
||||
|
@ -335,11 +369,14 @@ parse_generic_function (const char *name, specifier_t spec, rua_ctx_t *ctx)
|
|||
bool found = false;
|
||||
for (auto q = &ret_param; q; q = q->next) {
|
||||
// skip complex expressions because they will be either fixed
|
||||
// or rely on earlier parameters
|
||||
if (!q->type_expr || q->type_expr->type != ex_symbol) {
|
||||
// or rely on earlier parameters, but need to check references
|
||||
bool is_reference;
|
||||
unsigned tag;
|
||||
auto gsym = get_gensym (q->type_expr, &is_reference, &tag, ctx);
|
||||
if (!gsym) {
|
||||
continue;
|
||||
}
|
||||
if (strcmp (q->type_expr->symbol->name, s->name) == 0) {
|
||||
if (strcmp (gsym->name, s->name) == 0) {
|
||||
num_gentype++;
|
||||
found = true;
|
||||
break;
|
||||
|
@ -372,12 +409,15 @@ parse_generic_function (const char *name, specifier_t spec, rua_ctx_t *ctx)
|
|||
for (auto s = generic_tab->symbols; s; s = s->next) {
|
||||
for (auto q = &ret_param; q; q = q->next) {
|
||||
// see complex expressions comment above
|
||||
if (!q->type_expr || q->type_expr->type != ex_symbol) {
|
||||
bool is_reference;
|
||||
unsigned tag;
|
||||
auto gsym = get_gensym (q->type_expr, &is_reference, &tag, ctx);
|
||||
if (!gsym) {
|
||||
continue;
|
||||
}
|
||||
if (strcmp (q->type_expr->symbol->name, s->name) == 0) {
|
||||
genfunc->types[num_gentype++] = make_gentype (q->type_expr,
|
||||
ctx);
|
||||
if (strcmp (gsym->name, s->name) == 0) {
|
||||
int ind = num_gentype++;
|
||||
genfunc->types[ind] = make_gentype (q->type_expr, ctx);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
@ -608,8 +648,13 @@ check_type (const type_t *type, callparm_t param, unsigned *cost, bool promote)
|
|||
return true;
|
||||
}
|
||||
if (is_reference (type)) {
|
||||
// pass by references is a free conversion, but no promotion
|
||||
return type_same (dereference_type (type), param.type);
|
||||
type = dereference_type (type);
|
||||
if (is_reference (param.type)) {
|
||||
return type_same (type, dereference_type (param.type));
|
||||
} else {
|
||||
// pass by references is a free conversion, but no promotion
|
||||
return type_same (type, param.type);
|
||||
}
|
||||
}
|
||||
if (is_reference (param.type)) {
|
||||
// dereferencing a reference is free so long as there's no
|
||||
|
@ -642,22 +687,8 @@ static const type_t * __attribute__((pure))
|
|||
select_type (gentype_t *gentype, callparm_t param)
|
||||
{
|
||||
for (auto t = gentype->valid_types; t && *t; t++) {
|
||||
if (*t == param.type) {
|
||||
return *t;
|
||||
}
|
||||
if (is_reference (*t) && dereference_type (*t) == param.type) {
|
||||
// pass value by reference: no promotion
|
||||
return *t;
|
||||
}
|
||||
auto pt = param.type;
|
||||
if (is_reference (pt)) {
|
||||
// pass reference by value: promotion ok
|
||||
pt = dereference_type (pt);
|
||||
}
|
||||
if (*t == pt) {
|
||||
return *t;
|
||||
}
|
||||
if (type_promotes (*t, pt)) {
|
||||
unsigned cost = 0;
|
||||
if (check_type (*t, param, &cost, true)) {
|
||||
return *t;
|
||||
}
|
||||
}
|
||||
|
@ -744,11 +775,15 @@ compute_param_type (const genparam_t *param, int param_ind,
|
|||
calltype->types, fexpr, ctx);
|
||||
}
|
||||
int ind = param->gentype;
|
||||
if (!call_types[ind] && param_ind >= 0) {
|
||||
call_types[ind] = select_type (&genfunc->types[ind],
|
||||
call_params[param_ind]);
|
||||
auto type = call_types[ind];
|
||||
if (!type && param_ind >= 0) {
|
||||
type = select_type (&genfunc->types[ind], call_params[param_ind]);
|
||||
call_types[ind] = type;
|
||||
}
|
||||
return call_types[ind];
|
||||
if (param->is_reference) {
|
||||
type = tagged_reference_type (param->tag, type);
|
||||
}
|
||||
return type;
|
||||
}
|
||||
|
||||
static symbol_t *
|
||||
|
|
|
@ -1206,8 +1206,8 @@ SRC_LINE
|
|||
"#define _image(d,m,a,s) __image(,d,m,a,s),__image(i,d,m,a,s),__image(u,d,m,a,s)\n"
|
||||
"#define gvec4 @vector(gimage.sample_type, 4)" "\n"
|
||||
"#define gvec4MS @vector(gimageMS.sample_type, 4)" "\n"
|
||||
"#define IMAGE_PARAMS gimage image, gimage.image_coord P" "\n"
|
||||
"#define IMAGE_PARAMS_MS gimageMS image, gimageMS.image_coord P, int sample" "\n"
|
||||
"#define IMAGE_PARAMS @reference(gimage) image, gimage.image_coord P" "\n"
|
||||
"#define IMAGE_PARAMS_MS @reference(gimageMS) image, gimageMS.image_coord P, int sample" "\n"
|
||||
"@generic(gimage=[_image(1D,,,)," "\n"
|
||||
" _image(1D,,Array,)," "\n"
|
||||
" _image(2D,,,)," "\n"
|
||||
|
@ -1226,8 +1226,8 @@ SRC_LINE
|
|||
"gvec4MS imageLoad(readonly IMAGE_PARAMS_MS) = " SPV(OpImageRead) ";" "\n"
|
||||
"void imageStore(writeonly IMAGE_PARAMS, gvec4 data) = " SPV(OpImageWrite) ";" "\n"
|
||||
"void imageStore(writeonly IMAGE_PARAMS_MS, gvec4MS data) = " SPV(OpImageWrite) ";" "\n"
|
||||
"@pointer(gimage.sample_type, StorageClass.Image) __imageTexel(IMAGE_PARAMS, int sample) = " SPV(OpImageTexelPointer) ";" "\n"
|
||||
"@pointer(gimageMS.sample_type, StorageClass.Image) __imageTexel(IMAGE_PARAMS_MS) = " SPV(OpImageTexelPointer) ";" "\n"
|
||||
"@reference(gimage.sample_type, StorageClass.Image) __imageTexel(IMAGE_PARAMS, int sample) = " SPV(OpImageTexelPointer) ";" "\n"
|
||||
"@reference(gimageMS.sample_type, StorageClass.Image) __imageTexel(IMAGE_PARAMS_MS) = " SPV(OpImageTexelPointer) ";" "\n"
|
||||
"#define __imageAtomic(op,type) \\" "\n"
|
||||
"type imageAtomic##op(IMAGE_PARAMS, type data) \\" "\n"
|
||||
"{ \\" "\n"
|
||||
|
|
Loading…
Reference in a new issue