mirror of
https://gitlab.winehq.org/wine/vkd3d.git
synced 2024-11-21 16:46:41 -08:00
vkd3d-shader/hlsl: Support all complex initalizers.
Signed-off-by: Francisco Casas <fcasas@codeweavers.com> Signed-off-by: Giovanni Mascellani <gmascellani@codeweavers.com> Signed-off-by: Zebediah Figura <zfigura@codeweavers.com> Signed-off-by: Henri Verbeet <hverbeet@codeweavers.com> Signed-off-by: Alexandre Julliard <julliard@winehq.org>
This commit is contained in:
parent
e2b4f01a9f
commit
246ff14511
@ -232,6 +232,85 @@ static struct hlsl_type *hlsl_new_type(struct hlsl_ctx *ctx, const char *name, e
|
||||
return type;
|
||||
}
|
||||
|
||||
/* Returns the register offset of a given component within a type, given its index.
|
||||
* *comp_type will be set to the type of the component. */
|
||||
unsigned int hlsl_compute_component_offset(struct hlsl_ctx *ctx, struct hlsl_type *type,
|
||||
unsigned int idx, struct hlsl_type **comp_type)
|
||||
{
|
||||
switch (type->type)
|
||||
{
|
||||
case HLSL_CLASS_SCALAR:
|
||||
case HLSL_CLASS_VECTOR:
|
||||
{
|
||||
assert(idx < type->dimx * type->dimy);
|
||||
*comp_type = hlsl_get_scalar_type(ctx, type->base_type);
|
||||
return idx;
|
||||
}
|
||||
case HLSL_CLASS_MATRIX:
|
||||
{
|
||||
unsigned int minor, major, x = idx % type->dimx, y = idx / type->dimx;
|
||||
|
||||
assert(idx < type->dimx * type->dimy);
|
||||
|
||||
if (hlsl_type_is_row_major(type))
|
||||
{
|
||||
minor = x;
|
||||
major = y;
|
||||
}
|
||||
else
|
||||
{
|
||||
minor = y;
|
||||
major = x;
|
||||
}
|
||||
|
||||
*comp_type = hlsl_get_scalar_type(ctx, type->base_type);
|
||||
return 4 * major + minor;
|
||||
}
|
||||
|
||||
case HLSL_CLASS_ARRAY:
|
||||
{
|
||||
unsigned int elem_comp_count = hlsl_type_component_count(type->e.array.type);
|
||||
unsigned int array_idx = idx / elem_comp_count;
|
||||
unsigned int idx_in_elem = idx % elem_comp_count;
|
||||
|
||||
assert(array_idx < type->e.array.elements_count);
|
||||
|
||||
return array_idx * hlsl_type_get_array_element_reg_size(type->e.array.type) +
|
||||
hlsl_compute_component_offset(ctx, type->e.array.type, idx_in_elem, comp_type);
|
||||
}
|
||||
|
||||
case HLSL_CLASS_STRUCT:
|
||||
{
|
||||
struct hlsl_struct_field *field;
|
||||
|
||||
LIST_FOR_EACH_ENTRY(field, type->e.elements, struct hlsl_struct_field, entry)
|
||||
{
|
||||
unsigned int elem_comp_count = hlsl_type_component_count(field->type);
|
||||
|
||||
if (idx < elem_comp_count)
|
||||
{
|
||||
return field->reg_offset +
|
||||
hlsl_compute_component_offset(ctx, field->type, idx, comp_type);
|
||||
}
|
||||
idx -= elem_comp_count;
|
||||
}
|
||||
|
||||
assert(0);
|
||||
return 0;
|
||||
}
|
||||
|
||||
case HLSL_CLASS_OBJECT:
|
||||
{
|
||||
assert(idx == 0);
|
||||
*comp_type = type;
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
assert(0);
|
||||
return 0;
|
||||
}
|
||||
|
||||
struct hlsl_type *hlsl_new_array_type(struct hlsl_ctx *ctx, struct hlsl_type *basic_type, unsigned int array_size)
|
||||
{
|
||||
struct hlsl_type *type;
|
||||
|
@ -786,6 +786,8 @@ struct hlsl_type *hlsl_type_clone(struct hlsl_ctx *ctx, struct hlsl_type *old,
|
||||
unsigned int default_majority, unsigned int modifiers);
|
||||
unsigned int hlsl_type_component_count(struct hlsl_type *type);
|
||||
unsigned int hlsl_type_get_array_element_reg_size(const struct hlsl_type *type);
|
||||
unsigned int hlsl_compute_component_offset(struct hlsl_ctx *ctx, struct hlsl_type *type,
|
||||
unsigned int idx, struct hlsl_type **comp_type);
|
||||
unsigned int hlsl_type_get_sm4_offset(const struct hlsl_type *type, unsigned int offset);
|
||||
bool hlsl_types_are_equal(const struct hlsl_type *t1, const struct hlsl_type *t2);
|
||||
|
||||
|
@ -1450,77 +1450,44 @@ static bool add_increment(struct hlsl_ctx *ctx, struct list *instrs, bool decrem
|
||||
return true;
|
||||
}
|
||||
|
||||
static void initialize_numeric_var(struct hlsl_ctx *ctx, struct hlsl_ir_var *var,
|
||||
struct parse_initializer *initializer, unsigned int reg_offset, struct hlsl_type *type,
|
||||
unsigned int *initializer_offset)
|
||||
static void initialize_var_components(struct hlsl_ctx *ctx, struct list *instrs,
|
||||
struct hlsl_ir_var *dst, unsigned int *store_index, struct hlsl_ir_node *src,
|
||||
const struct vkd3d_shader_location *loc)
|
||||
{
|
||||
unsigned int i;
|
||||
unsigned int src_comp_count = hlsl_type_component_count(src->data_type);
|
||||
unsigned int k;
|
||||
|
||||
if (type->type == HLSL_CLASS_MATRIX)
|
||||
hlsl_fixme(ctx, &var->loc, "Matrix initializer.");
|
||||
|
||||
for (i = 0; i < type->dimx; i++)
|
||||
for (k = 0; k < src_comp_count; ++k)
|
||||
{
|
||||
struct hlsl_type *dst_comp_type, *src_comp_type;
|
||||
unsigned int dst_reg_offset, src_reg_offset;
|
||||
struct hlsl_ir_store *store;
|
||||
struct hlsl_ir_constant *c;
|
||||
struct hlsl_ir_node *node;
|
||||
struct hlsl_ir_load *load;
|
||||
struct hlsl_ir_node *conv;
|
||||
|
||||
node = initializer->args[*initializer_offset];
|
||||
*initializer_offset += 1;
|
||||
dst_reg_offset = hlsl_compute_component_offset(ctx, dst->data_type, *store_index, &dst_comp_type);
|
||||
src_reg_offset = hlsl_compute_component_offset(ctx, src->data_type, k, &src_comp_type);
|
||||
|
||||
if (!(node = add_implicit_conversion(ctx, initializer->instrs, node,
|
||||
hlsl_get_scalar_type(ctx, type->base_type), &node->loc)))
|
||||
if (!(c = hlsl_new_uint_constant(ctx, src_reg_offset, loc)))
|
||||
return;
|
||||
list_add_tail(instrs, &c->node.entry);
|
||||
|
||||
if (!(load = add_load(ctx, instrs, src, &c->node, src_comp_type, *loc)))
|
||||
return;
|
||||
|
||||
if (!(c = hlsl_new_uint_constant(ctx, reg_offset + i, &node->loc)))
|
||||
return;
|
||||
list_add_tail(initializer->instrs, &c->node.entry);
|
||||
|
||||
if (!(store = hlsl_new_store(ctx, var, &c->node, node, 0, node->loc)))
|
||||
if (!(conv = add_implicit_conversion(ctx, instrs, &load->node, dst_comp_type, loc)))
|
||||
return;
|
||||
|
||||
list_add_tail(initializer->instrs, &store->node.entry);
|
||||
}
|
||||
}
|
||||
if (!(c = hlsl_new_uint_constant(ctx, dst_reg_offset, loc)))
|
||||
return;
|
||||
list_add_tail(instrs, &c->node.entry);
|
||||
|
||||
static void struct_var_initializer(struct hlsl_ctx *ctx, struct hlsl_ir_var *var,
|
||||
struct parse_initializer *initializer)
|
||||
{
|
||||
struct hlsl_type *type = var->data_type;
|
||||
struct hlsl_struct_field *field;
|
||||
unsigned int i = 0;
|
||||
if (!(store = hlsl_new_store(ctx, dst, &c->node, conv, 0, *loc)))
|
||||
return;
|
||||
list_add_tail(instrs, &store->node.entry);
|
||||
|
||||
if (initializer_size(initializer) != hlsl_type_component_count(type))
|
||||
{
|
||||
hlsl_error(ctx, &var->loc, VKD3D_SHADER_ERROR_HLSL_WRONG_PARAMETER_COUNT,
|
||||
"Expected %u components in initializer, but got %u.",
|
||||
hlsl_type_component_count(type), initializer_size(initializer));
|
||||
return;
|
||||
}
|
||||
|
||||
LIST_FOR_EACH_ENTRY(field, type->e.elements, struct hlsl_struct_field, entry)
|
||||
{
|
||||
struct hlsl_ir_node *node = initializer->args[i];
|
||||
struct hlsl_ir_store *store;
|
||||
struct hlsl_ir_constant *c;
|
||||
|
||||
if (i++ >= initializer->args_count)
|
||||
break;
|
||||
|
||||
if (hlsl_type_component_count(field->type) == hlsl_type_component_count(node->data_type))
|
||||
{
|
||||
if (!(c = hlsl_new_uint_constant(ctx, field->reg_offset, &node->loc)))
|
||||
break;
|
||||
list_add_tail(initializer->instrs, &c->node.entry);
|
||||
|
||||
if (!(store = hlsl_new_store(ctx, var, &c->node, node, 0, node->loc)))
|
||||
break;
|
||||
list_add_tail(initializer->instrs, &store->node.entry);
|
||||
}
|
||||
else
|
||||
{
|
||||
hlsl_fixme(ctx, &node->loc, "Implicit cast in structure initializer.");
|
||||
}
|
||||
++*store_index;
|
||||
}
|
||||
}
|
||||
|
||||
@ -1648,51 +1615,23 @@ static struct list *declare_vars(struct hlsl_ctx *ctx, struct hlsl_type *basic_t
|
||||
if (v->initializer.braces)
|
||||
{
|
||||
unsigned int size = initializer_size(&v->initializer);
|
||||
unsigned int initializer_offset = 0;
|
||||
unsigned int store_index = 0;
|
||||
unsigned int k;
|
||||
|
||||
if (type->type <= HLSL_CLASS_LAST_NUMERIC && type->dimx * type->dimy != size)
|
||||
if (hlsl_type_component_count(type) != size)
|
||||
{
|
||||
hlsl_error(ctx, &v->loc, VKD3D_SHADER_ERROR_HLSL_WRONG_PARAMETER_COUNT,
|
||||
"Expected %u components in numeric initializer, but got %u.",
|
||||
type->dimx * type->dimy, v->initializer.args_count);
|
||||
"Expected %u components in initializer, but got %u.",
|
||||
hlsl_type_component_count(type), size);
|
||||
free_parse_initializer(&v->initializer);
|
||||
vkd3d_free(v);
|
||||
continue;
|
||||
}
|
||||
|
||||
if ((type->type == HLSL_CLASS_STRUCT || type->type == HLSL_CLASS_ARRAY)
|
||||
&& hlsl_type_component_count(type) != size)
|
||||
for (k = 0; k < v->initializer.args_count; ++k)
|
||||
{
|
||||
hlsl_error(ctx, &v->loc, VKD3D_SHADER_ERROR_HLSL_WRONG_PARAMETER_COUNT,
|
||||
"Expected %u components in initializer, but got %u.", hlsl_type_component_count(type), size);
|
||||
free_parse_initializer(&v->initializer);
|
||||
vkd3d_free(v);
|
||||
continue;
|
||||
}
|
||||
|
||||
if (type->type > HLSL_CLASS_LAST_NUMERIC && type->type != HLSL_CLASS_STRUCT)
|
||||
{
|
||||
FIXME("Initializers for non scalar/struct variables not supported yet.\n");
|
||||
free_parse_initializer(&v->initializer);
|
||||
vkd3d_free(v);
|
||||
continue;
|
||||
}
|
||||
|
||||
if (type->type == HLSL_CLASS_STRUCT)
|
||||
{
|
||||
struct_var_initializer(ctx, var, &v->initializer);
|
||||
}
|
||||
else
|
||||
{
|
||||
if (v->initializer.args_count != size)
|
||||
{
|
||||
hlsl_fixme(ctx, &v->loc, "Flatten initializer.");
|
||||
free_parse_initializer(&v->initializer);
|
||||
vkd3d_free(v);
|
||||
continue;
|
||||
}
|
||||
|
||||
initialize_numeric_var(ctx, var, &v->initializer, 0, type, &initializer_offset);
|
||||
initialize_var_components(ctx, v->initializer.instrs, var,
|
||||
&store_index, v->initializer.args[k], &v->initializer.args[k]->loc);
|
||||
}
|
||||
}
|
||||
else
|
||||
|
@ -1660,7 +1660,7 @@ bool hlsl_offset_from_deref(struct hlsl_ctx *ctx, const struct hlsl_deref *deref
|
||||
if (*offset >= deref->var->data_type->reg_size)
|
||||
{
|
||||
hlsl_error(ctx, &deref->offset.node->loc, VKD3D_SHADER_ERROR_HLSL_OFFSET_OUT_OF_BOUNDS,
|
||||
"Dereference is out of bounds.");
|
||||
"Dereference is out of bounds. %u/%u", *offset, deref->var->data_type->reg_size);
|
||||
return false;
|
||||
}
|
||||
|
||||
|
@ -6,7 +6,7 @@ float4 main() : sv_target
|
||||
}
|
||||
|
||||
[test]
|
||||
todo draw quad
|
||||
draw quad
|
||||
probe all rgba (1, 2, 3, 4)
|
||||
|
||||
|
||||
@ -24,7 +24,7 @@ float4 main() : sv_target
|
||||
}
|
||||
|
||||
[test]
|
||||
todo draw quad
|
||||
draw quad
|
||||
probe all rgba (4, 5, 6, 7)
|
||||
|
||||
|
||||
@ -38,7 +38,7 @@ float4 main() : sv_target
|
||||
|
||||
[test]
|
||||
draw quad
|
||||
todo probe all rgba (40, 10, 20, 30)
|
||||
probe all rgba (40, 10, 20, 30)
|
||||
|
||||
|
||||
[pixel shader]
|
||||
@ -56,7 +56,7 @@ float4 main() : sv_target
|
||||
}
|
||||
|
||||
[test]
|
||||
todo draw quad
|
||||
draw quad
|
||||
probe all rgba (1.0, 2.0, 3.0, 4.0)
|
||||
|
||||
|
||||
@ -69,5 +69,5 @@ float4 main() : sv_target
|
||||
}
|
||||
|
||||
[test]
|
||||
todo draw quad
|
||||
draw quad
|
||||
probe all rgba (1.0, 2.0, 3.0, 4.0)
|
||||
|
@ -10,7 +10,7 @@ float4 main() : sv_target
|
||||
|
||||
[test]
|
||||
draw quad
|
||||
todo probe all rgba (17, 18, 19, 20)
|
||||
probe all rgba (17, 18, 19, 20)
|
||||
|
||||
|
||||
[pixel shader fail]
|
||||
@ -57,7 +57,7 @@ float4 main() : sv_target
|
||||
}
|
||||
|
||||
[test]
|
||||
todo draw quad
|
||||
draw quad
|
||||
probe all rgba (22, 23, 24, 25)
|
||||
|
||||
|
||||
|
@ -11,7 +11,7 @@ float4 main() : SV_TARGET
|
||||
|
||||
[test]
|
||||
draw quad
|
||||
todo probe all rgba (21, 22, 23, 24)
|
||||
probe all rgba (21, 22, 23, 24)
|
||||
|
||||
|
||||
[pixel shader]
|
||||
@ -32,4 +32,4 @@ float4 main() : SV_TARGET
|
||||
|
||||
[test]
|
||||
draw quad
|
||||
todo probe all rgba (71, 72, 73, 74)
|
||||
probe all rgba (71, 72, 73, 74)
|
||||
|
@ -24,7 +24,7 @@ float4 main() : sv_target
|
||||
|
||||
[test]
|
||||
draw quad
|
||||
todo probe all rgba (21, 22, 23, 24)
|
||||
probe all rgba (21, 22, 23, 24)
|
||||
|
||||
|
||||
[pixel shader]
|
||||
@ -52,5 +52,5 @@ float4 main() : sv_target
|
||||
}
|
||||
|
||||
[test]
|
||||
todo draw quad
|
||||
draw quad
|
||||
probe all rgba (21, 22, 23, 24)
|
||||
|
@ -12,7 +12,7 @@ float4 main() : SV_TARGET
|
||||
|
||||
[test]
|
||||
draw quad
|
||||
todo probe all rgba (21, 22, 23, 24)
|
||||
probe all rgba (21, 22, 23, 24)
|
||||
|
||||
|
||||
[pixel shader]
|
||||
@ -34,4 +34,4 @@ float4 main() : SV_TARGET
|
||||
|
||||
[test]
|
||||
draw quad
|
||||
todo probe all rgba (61, 62, 63, 64)
|
||||
probe all rgba (61, 62, 63, 64)
|
||||
|
@ -22,7 +22,7 @@ float4 main() : sv_target
|
||||
}
|
||||
|
||||
[test]
|
||||
todo draw quad
|
||||
draw quad
|
||||
probe all rgba (41, 42, 43, 44)
|
||||
|
||||
|
||||
@ -52,7 +52,7 @@ float4 main() : sv_target
|
||||
}
|
||||
|
||||
[test]
|
||||
todo draw quad
|
||||
draw quad
|
||||
probe all rgba (4311, 4312, 4313, 4314)
|
||||
|
||||
|
||||
@ -80,5 +80,5 @@ float4 main() : sv_target
|
||||
}
|
||||
|
||||
[test]
|
||||
todo draw quad
|
||||
draw quad
|
||||
probe all rgba (21, 22, 23, 24)
|
||||
|
Loading…
Reference in New Issue
Block a user