How many elements are in the array?

static int2 Arr[] = {
  int2(0,1),
  int2(2,3),
  int4(4,5,6,7),
};

A. 3

B. 4

C. Clearly this is an error right?

Answer!

B!

In the days before DXC, HLSL’s type system took heavy inspiration from functional programming languages. This history has some big impacts that are still with HLSL today.

One of those impacts is what I call HLSL’s “floatly-typed” nature: every scalar value can be viewed as a float or collection of floats. The other is HLSL’s somewhat weak typing of structures.

Both of these impacted how aggregate initialization is defined in HLSL.

To try and simplify the wording, HLSL ignores braces {} in initializer lists and flattens all the argument structures down to an ordering if individual scalar elements. The same transformation is then performed to the initialization target. If the number of scalar elements to initialize on the left equals the number os scalar arguments on the right, and each argument can be converted to the corresponding element type on the left; you have a valid initializer.

This gets a bit complicated with incomplete array types because you don’t know the full size of the array from just looking at the assignment target, but the logic is basically the same. It flattens the array element type, flattens the initializer, and figures out how many elements can be initialized.

See it here in Compiler Explorer.

The discussion of this code snippet sparked some of my favorite conversations with users to date.

Bernie wants clang - Contribution from https://github.com/TheRealMJP

Pretty overwhelmingly users expressed a desire that this be an error (or at least a loud warning), because the behavior is pretty unintuitive.

It’s also behavior that is extremely simple to rely on in ways that you really don’t expect.

The HLSL team is tracking a proposal to change this in a future language version, however this behavior will be supported in Clang.

Anakin is making Clang weird

We’ve found that removing this language support is massively disruptive to codebases migrating to Clang, so we’re supporting the old behavior in Clang and planning to support tooling to assist in removing this behavior in the future.