mirror of
https://gitlab.winehq.org/wine/vkd3d.git
synced 2025-09-12 18:50:22 -07:00
vkd3d-shader/tpf: Use 'movc' to implement ternary operator.
Signed-off-by: Nikolay Sivov <nsivov@codeweavers.com>
This commit is contained in:
committed by
Alexandre Julliard
parent
2fb0c2d187
commit
1002a6b357
Notes:
Alexandre Julliard
2023-09-07 23:01:33 +02:00
Approved-by: Giovanni Mascellani (@giomasce) Approved-by: Zebediah Figura (@zfigura) Approved-by: Henri Verbeet (@hverbeet) Approved-by: Alexandre Julliard (@julliard) Merge-Request: https://gitlab.winehq.org/wine/vkd3d/-/merge_requests/268
@@ -2487,6 +2487,8 @@ const char *debug_hlsl_expr_op(enum hlsl_ir_expr_op op)
|
|||||||
|
|
||||||
[HLSL_OP3_DP2ADD] = "dp2add",
|
[HLSL_OP3_DP2ADD] = "dp2add",
|
||||||
[HLSL_OP3_LERP] = "lerp",
|
[HLSL_OP3_LERP] = "lerp",
|
||||||
|
[HLSL_OP3_MOVC] = "movc",
|
||||||
|
[HLSL_OP3_TERNARY] = "ternary",
|
||||||
};
|
};
|
||||||
|
|
||||||
return op_names[op];
|
return op_names[op];
|
||||||
|
@@ -552,6 +552,9 @@ enum hlsl_ir_expr_op
|
|||||||
|
|
||||||
HLSL_OP3_DP2ADD,
|
HLSL_OP3_DP2ADD,
|
||||||
HLSL_OP3_LERP,
|
HLSL_OP3_LERP,
|
||||||
|
/* TERNARY is used specifically for ternary operator, and later lowered according to profile to e.g. MOVC. */
|
||||||
|
HLSL_OP3_MOVC,
|
||||||
|
HLSL_OP3_TERNARY,
|
||||||
};
|
};
|
||||||
|
|
||||||
#define HLSL_MAX_OPERANDS 3
|
#define HLSL_MAX_OPERANDS 3
|
||||||
|
@@ -6562,6 +6562,7 @@ conditional_expr:
|
|||||||
struct hlsl_ir_node *cond = node_from_block($1);
|
struct hlsl_ir_node *cond = node_from_block($1);
|
||||||
struct hlsl_ir_node *first = node_from_block($3);
|
struct hlsl_ir_node *first = node_from_block($3);
|
||||||
struct hlsl_ir_node *second = node_from_block($5);
|
struct hlsl_ir_node *second = node_from_block($5);
|
||||||
|
struct hlsl_ir_node *args[HLSL_MAX_OPERANDS] = { 0 };
|
||||||
struct hlsl_type *common_type;
|
struct hlsl_type *common_type;
|
||||||
|
|
||||||
hlsl_block_add_block($1, $3);
|
hlsl_block_add_block($1, $3);
|
||||||
@@ -6578,7 +6579,10 @@ conditional_expr:
|
|||||||
if (!(second = add_implicit_conversion(ctx, $1, second, common_type, &@5)))
|
if (!(second = add_implicit_conversion(ctx, $1, second, common_type, &@5)))
|
||||||
YYABORT;
|
YYABORT;
|
||||||
|
|
||||||
if (!hlsl_add_conditional(ctx, $1, cond, first, second))
|
args[0] = cond;
|
||||||
|
args[1] = first;
|
||||||
|
args[2] = second;
|
||||||
|
if (!add_expr(ctx, $1, HLSL_OP3_TERNARY, args, common_type, &@1))
|
||||||
YYABORT;
|
YYABORT;
|
||||||
$$ = $1;
|
$$ = $1;
|
||||||
}
|
}
|
||||||
|
@@ -2391,6 +2391,54 @@ static bool lower_round(struct hlsl_ctx *ctx, struct hlsl_ir_node *instr, void *
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Use 'movc' for the ternary operator. */
|
||||||
|
static bool lower_ternary(struct hlsl_ctx *ctx, struct hlsl_ir_node *instr, void *context)
|
||||||
|
{
|
||||||
|
struct hlsl_ir_node *operands[HLSL_MAX_OPERANDS], *replacement;
|
||||||
|
struct hlsl_ir_node *zero, *cond, *first, *second;
|
||||||
|
struct hlsl_constant_value zero_value = { 0 };
|
||||||
|
struct hlsl_ir_expr *expr;
|
||||||
|
struct hlsl_type *type;
|
||||||
|
|
||||||
|
if (instr->type != HLSL_IR_EXPR)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
expr = hlsl_ir_expr(instr);
|
||||||
|
if (expr->op != HLSL_OP3_TERNARY)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
cond = expr->operands[0].node;
|
||||||
|
first = expr->operands[1].node;
|
||||||
|
second = expr->operands[2].node;
|
||||||
|
|
||||||
|
if (cond->data_type->base_type == HLSL_TYPE_FLOAT)
|
||||||
|
{
|
||||||
|
if (!(zero = hlsl_new_constant(ctx, cond->data_type, &zero_value, &instr->loc)))
|
||||||
|
return false;
|
||||||
|
list_add_tail(&instr->entry, &zero->entry);
|
||||||
|
|
||||||
|
memset(operands, 0, sizeof(operands));
|
||||||
|
operands[0] = zero;
|
||||||
|
operands[1] = cond;
|
||||||
|
type = cond->data_type;
|
||||||
|
type = hlsl_get_numeric_type(ctx, type->class, HLSL_TYPE_BOOL, type->dimx, type->dimy);
|
||||||
|
if (!(cond = hlsl_new_expr(ctx, HLSL_OP2_NEQUAL, operands, type, &instr->loc)))
|
||||||
|
return false;
|
||||||
|
list_add_before(&instr->entry, &cond->entry);
|
||||||
|
}
|
||||||
|
|
||||||
|
memset(operands, 0, sizeof(operands));
|
||||||
|
operands[0] = cond;
|
||||||
|
operands[1] = first;
|
||||||
|
operands[2] = second;
|
||||||
|
if (!(replacement = hlsl_new_expr(ctx, HLSL_OP3_MOVC, operands, first->data_type, &instr->loc)))
|
||||||
|
return false;
|
||||||
|
list_add_before(&instr->entry, &replacement->entry);
|
||||||
|
|
||||||
|
hlsl_replace_node(instr, replacement);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
static bool lower_casts_to_bool(struct hlsl_ctx *ctx, struct hlsl_ir_node *instr, void *context)
|
static bool lower_casts_to_bool(struct hlsl_ctx *ctx, struct hlsl_ir_node *instr, void *context)
|
||||||
{
|
{
|
||||||
struct hlsl_type *type = instr->data_type, *arg_type;
|
struct hlsl_type *type = instr->data_type, *arg_type;
|
||||||
@@ -4349,6 +4397,8 @@ int hlsl_emit_bytecode(struct hlsl_ctx *ctx, struct hlsl_ir_function_decl *entry
|
|||||||
hlsl_transform_ir(ctx, track_object_components_usage, body, NULL);
|
hlsl_transform_ir(ctx, track_object_components_usage, body, NULL);
|
||||||
sort_synthetic_separated_samplers_first(ctx);
|
sort_synthetic_separated_samplers_first(ctx);
|
||||||
|
|
||||||
|
if (profile->major_version >= 4)
|
||||||
|
hlsl_transform_ir(ctx, lower_ternary, body, NULL);
|
||||||
if (profile->major_version < 4)
|
if (profile->major_version < 4)
|
||||||
{
|
{
|
||||||
hlsl_transform_ir(ctx, lower_division, body, NULL);
|
hlsl_transform_ir(ctx, lower_division, body, NULL);
|
||||||
|
@@ -4341,6 +4341,26 @@ static void write_sm4_binary_op_with_two_destinations(const struct tpf_writer *t
|
|||||||
write_sm4_instruction(tpf, &instr);
|
write_sm4_instruction(tpf, &instr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void write_sm4_ternary_op(const struct tpf_writer *tpf, enum vkd3d_sm4_opcode opcode,
|
||||||
|
const struct hlsl_ir_node *dst, const struct hlsl_ir_node *src1, const struct hlsl_ir_node *src2,
|
||||||
|
const struct hlsl_ir_node *src3)
|
||||||
|
{
|
||||||
|
struct sm4_instruction instr;
|
||||||
|
|
||||||
|
memset(&instr, 0, sizeof(instr));
|
||||||
|
instr.opcode = opcode;
|
||||||
|
|
||||||
|
sm4_dst_from_node(&instr.dsts[0], dst);
|
||||||
|
instr.dst_count = 1;
|
||||||
|
|
||||||
|
sm4_src_from_node(&instr.srcs[0], src1, instr.dsts[0].writemask);
|
||||||
|
sm4_src_from_node(&instr.srcs[1], src2, instr.dsts[0].writemask);
|
||||||
|
sm4_src_from_node(&instr.srcs[2], src3, instr.dsts[0].writemask);
|
||||||
|
instr.src_count = 3;
|
||||||
|
|
||||||
|
write_sm4_instruction(tpf, &instr);
|
||||||
|
}
|
||||||
|
|
||||||
static void write_sm4_ld(const struct tpf_writer *tpf, const struct hlsl_ir_node *dst,
|
static void write_sm4_ld(const struct tpf_writer *tpf, const struct hlsl_ir_node *dst,
|
||||||
const struct hlsl_deref *resource, const struct hlsl_ir_node *coords,
|
const struct hlsl_deref *resource, const struct hlsl_ir_node *coords,
|
||||||
const struct hlsl_ir_node *sample_index, const struct hlsl_ir_node *texel_offset,
|
const struct hlsl_ir_node *sample_index, const struct hlsl_ir_node *texel_offset,
|
||||||
@@ -4702,6 +4722,7 @@ static void write_sm4_expr(const struct tpf_writer *tpf, const struct hlsl_ir_ex
|
|||||||
{
|
{
|
||||||
const struct hlsl_ir_node *arg1 = expr->operands[0].node;
|
const struct hlsl_ir_node *arg1 = expr->operands[0].node;
|
||||||
const struct hlsl_ir_node *arg2 = expr->operands[1].node;
|
const struct hlsl_ir_node *arg2 = expr->operands[1].node;
|
||||||
|
const struct hlsl_ir_node *arg3 = expr->operands[2].node;
|
||||||
const struct hlsl_type *dst_type = expr->node.data_type;
|
const struct hlsl_type *dst_type = expr->node.data_type;
|
||||||
struct vkd3d_string_buffer *dst_type_string;
|
struct vkd3d_string_buffer *dst_type_string;
|
||||||
|
|
||||||
@@ -5127,6 +5148,10 @@ static void write_sm4_expr(const struct tpf_writer *tpf, const struct hlsl_ir_ex
|
|||||||
&expr->node, arg1, arg2);
|
&expr->node, arg1, arg2);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case HLSL_OP3_MOVC:
|
||||||
|
write_sm4_ternary_op(tpf, VKD3D_SM4_OP_MOVC, &expr->node, arg1, arg2, arg3);
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
hlsl_fixme(tpf->ctx, &expr->node.loc, "SM4 %s expression.", debug_hlsl_expr_op(expr->op));
|
hlsl_fixme(tpf->ctx, &expr->node.loc, "SM4 %s expression.", debug_hlsl_expr_op(expr->op));
|
||||||
}
|
}
|
||||||
|
@@ -46,3 +46,18 @@ float4 main() : sv_target
|
|||||||
uniform 0 float4 1.0 0.0 0.0 0.0
|
uniform 0 float4 1.0 0.0 0.0 0.0
|
||||||
draw quad
|
draw quad
|
||||||
probe all rgba (0.5, 0.6, 0.7, 0.0)
|
probe all rgba (0.5, 0.6, 0.7, 0.0)
|
||||||
|
|
||||||
|
[pixel shader]
|
||||||
|
float4 x, y, z;
|
||||||
|
|
||||||
|
float4 main() : sv_target
|
||||||
|
{
|
||||||
|
return x ? y : z;
|
||||||
|
}
|
||||||
|
|
||||||
|
[test]
|
||||||
|
uniform 0 float4 0.0 1.0 0.0 -3.0
|
||||||
|
uniform 4 float4 1.0 2.0 3.0 4.0
|
||||||
|
uniform 8 float4 5.0 6.0 7.0 8.0
|
||||||
|
draw quad
|
||||||
|
probe all rgba (5.0, 2.0, 7.0, 4.0)
|
||||||
|
Reference in New Issue
Block a user