mirror of
https://gitlab.winehq.org/wine/vkd3d.git
synced 2024-11-21 16:46:41 -08:00
vkd3d-shader/tpf: Use 'movc' to implement ternary operator.
Signed-off-by: Nikolay Sivov <nsivov@codeweavers.com>
This commit is contained in:
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_LERP] = "lerp",
|
||||
[HLSL_OP3_MOVC] = "movc",
|
||||
[HLSL_OP3_TERNARY] = "ternary",
|
||||
};
|
||||
|
||||
return op_names[op];
|
||||
|
@ -552,6 +552,9 @@ enum hlsl_ir_expr_op
|
||||
|
||||
HLSL_OP3_DP2ADD,
|
||||
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
|
||||
|
@ -6562,6 +6562,7 @@ conditional_expr:
|
||||
struct hlsl_ir_node *cond = node_from_block($1);
|
||||
struct hlsl_ir_node *first = node_from_block($3);
|
||||
struct hlsl_ir_node *second = node_from_block($5);
|
||||
struct hlsl_ir_node *args[HLSL_MAX_OPERANDS] = { 0 };
|
||||
struct hlsl_type *common_type;
|
||||
|
||||
hlsl_block_add_block($1, $3);
|
||||
@ -6578,7 +6579,10 @@ conditional_expr:
|
||||
if (!(second = add_implicit_conversion(ctx, $1, second, common_type, &@5)))
|
||||
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;
|
||||
$$ = $1;
|
||||
}
|
||||
|
@ -2391,6 +2391,54 @@ static bool lower_round(struct hlsl_ctx *ctx, struct hlsl_ir_node *instr, void *
|
||||
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)
|
||||
{
|
||||
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);
|
||||
sort_synthetic_separated_samplers_first(ctx);
|
||||
|
||||
if (profile->major_version >= 4)
|
||||
hlsl_transform_ir(ctx, lower_ternary, body, NULL);
|
||||
if (profile->major_version < 4)
|
||||
{
|
||||
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);
|
||||
}
|
||||
|
||||
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,
|
||||
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,
|
||||
@ -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 *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;
|
||||
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);
|
||||
break;
|
||||
|
||||
case HLSL_OP3_MOVC:
|
||||
write_sm4_ternary_op(tpf, VKD3D_SM4_OP_MOVC, &expr->node, arg1, arg2, arg3);
|
||||
break;
|
||||
|
||||
default:
|
||||
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
|
||||
draw quad
|
||||
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)
|
||||
|
Loading…
x
Reference in New Issue
Block a user