mirror of
https://git.code.sf.net/p/quake/quakeforge
synced 2025-04-04 16:31:30 +00:00
[qfcc] Improve handling of branched types
This makes using structs in interface blocks possible by auto-casting whenever possible. Now my compute shaders compile, but there's another duplicate type sneaking in somehow.
This commit is contained in:
parent
5690fd028d
commit
3337d45aa0
7 changed files with 41 additions and 3 deletions
|
@ -56,8 +56,11 @@ typedef struct {
|
|||
const expr_t *(*vector_compare)(int op, const expr_t *e1, const expr_t *e2);
|
||||
const expr_t *(*shift_op)(int op, const expr_t *e1, const expr_t *e2);
|
||||
const expr_t *(*test_expr) (const expr_t *expr);
|
||||
const expr_t *(*cast_expr) (const type_t *dstType, const expr_t *expr);
|
||||
const expr_t *(*check_types_compatible) (const expr_t *dst,
|
||||
const expr_t *src);
|
||||
bool (*type_assignable) (const type_t *dst, const type_t *src);
|
||||
bool (*init_type_ok) (const type_t *dst, const type_t *src);
|
||||
|
||||
bool (*setup_intrinsic_symtab) (symtab_t *symtab);
|
||||
|
||||
|
|
|
@ -42,6 +42,7 @@
|
|||
#include "tools/qfcc/include/diagnostic.h"
|
||||
#include "tools/qfcc/include/expr.h"
|
||||
#include "tools/qfcc/include/struct.h"
|
||||
#include "tools/qfcc/include/target.h"
|
||||
#include "tools/qfcc/include/type.h"
|
||||
#include "tools/qfcc/include/value.h"
|
||||
|
||||
|
@ -148,6 +149,12 @@ cast_expr (const type_t *dstType, const expr_t *e)
|
|||
&& (is_short (srcType) || is_ushort (srcType))
|
||||
// [u]short is always width 0
|
||||
&& type_width (dstType) == 1)) {
|
||||
if (current_target.cast_expr) {
|
||||
auto c = current_target.cast_expr (dstType, e);
|
||||
if (c) {
|
||||
return c;
|
||||
}
|
||||
}
|
||||
return cast_error (e, srcType, dstType);
|
||||
}
|
||||
if (is_array (srcType)) {
|
||||
|
|
|
@ -685,8 +685,12 @@ check_type (const type_t *type, callparm_t param, unsigned *cost, bool promote)
|
|||
bool demotes = param.implicit && type_demotes (type, param.type);
|
||||
if (demotes) {
|
||||
*cost += 1;
|
||||
return true;
|
||||
}
|
||||
// allow any final checks
|
||||
if (!type_assignable (type, param.type)) {
|
||||
return false;
|
||||
}
|
||||
return demotes;
|
||||
}
|
||||
*cost += 2;
|
||||
return true;
|
||||
|
|
|
@ -21,9 +21,11 @@
|
|||
#include "tools/qfcc/include/strpool.h"
|
||||
#include "tools/qfcc/include/struct.h"
|
||||
#include "tools/qfcc/include/symtab.h"
|
||||
#include "tools/qfcc/include/target.h"
|
||||
#include "tools/qfcc/include/type.h"
|
||||
#include "tools/qfcc/include/value.h"
|
||||
|
||||
target_t current_target;
|
||||
struct dstring_s;
|
||||
options_t options;
|
||||
int num_linenos;
|
||||
|
|
|
@ -295,6 +295,10 @@ init_type_ok (const type_t *dstType, const expr_t *init)
|
|||
&& (type_size (dstType) == type_size (init_type))) {
|
||||
return true;
|
||||
}
|
||||
if (current_target.init_type_ok
|
||||
&& current_target.init_type_ok (dstType, init_type)) {
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
|
|
|
@ -2682,6 +2682,18 @@ spirv_types_logically_match (const type_t *dst, const type_t *src)
|
|||
return false;
|
||||
}
|
||||
|
||||
static const expr_t *
|
||||
spirv_cast_expr (const type_t *dstType, const expr_t *expr)
|
||||
{
|
||||
auto src_type = get_type (expr);
|
||||
if (!spirv_types_logically_match (dstType, src_type)) {
|
||||
return nullptr;
|
||||
}
|
||||
auto cast = new_unary_expr ('C', expr);
|
||||
cast->expr.type = dstType;
|
||||
return cast;
|
||||
}
|
||||
|
||||
static const expr_t *
|
||||
spirv_check_types_compatible (const expr_t *dst, const expr_t *src)
|
||||
{
|
||||
|
@ -2691,8 +2703,7 @@ spirv_check_types_compatible (const expr_t *dst, const expr_t *src)
|
|||
if (!spirv_types_logically_match (dst_type, src_type)) {
|
||||
return nullptr;
|
||||
}
|
||||
auto cast = new_unary_expr ('C', src);
|
||||
cast->expr.type = dst_type;
|
||||
auto cast = spirv_cast_expr (dst_type, src);
|
||||
return assign_expr (dst, cast);
|
||||
}
|
||||
|
||||
|
@ -2718,5 +2729,8 @@ target_t spirv_target = {
|
|||
// ruamoko and spirv are mostly compatible for bools other than lbool
|
||||
// but that's handled by spirv_mirror_bool
|
||||
.test_expr = ruamoko_test_expr,
|
||||
.cast_expr = spirv_cast_expr,
|
||||
.check_types_compatible = spirv_check_types_compatible,
|
||||
.type_assignable = spirv_types_logically_match,
|
||||
.init_type_ok = spirv_types_logically_match,
|
||||
};
|
||||
|
|
|
@ -60,6 +60,7 @@
|
|||
#include "tools/qfcc/include/strpool.h"
|
||||
#include "tools/qfcc/include/struct.h"
|
||||
#include "tools/qfcc/include/symtab.h"
|
||||
#include "tools/qfcc/include/target.h"
|
||||
#include "tools/qfcc/include/type.h"
|
||||
|
||||
#define EV_TYPE(t) \
|
||||
|
@ -1752,6 +1753,9 @@ type_assignable (const type_t *dst, const type_t *src)
|
|||
if (is_lbool (dst) && is_bool (src)) {
|
||||
return true;
|
||||
}
|
||||
if (current_target.type_assignable) {
|
||||
return current_target.type_assignable (dst, src);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in a new issue