nir: Add support for generic pointers

The way they're handled is that deref->modes is treated as a bitfield of
possible modes.  Variables are required to have a specific mode and
derefs with deref_type_var are as well.

Reviewed-by: Jesse Natalie <jenatali@microsoft.com>
Reviewed-by: Caio Marcelo de Oliveira Filho <caio.oliveira@intel.com>
Part-of: <https://gitlab.freedesktop.org/mesa/mesa/-/merge_requests/6332>
This commit is contained in:
Jason Ekstrand
2020-08-15 00:28:55 -05:00
committed by Marge Bot
parent 9d377c01d0
commit d6415b5d2b
3 changed files with 28 additions and 7 deletions

View File

@@ -128,6 +128,10 @@ typedef enum {
nir_var_mem_ssbo = (1 << 7), nir_var_mem_ssbo = (1 << 7),
nir_var_mem_shared = (1 << 8), nir_var_mem_shared = (1 << 8),
nir_var_mem_global = (1 << 9), nir_var_mem_global = (1 << 9),
nir_var_mem_generic = (nir_var_shader_temp |
nir_var_function_temp |
nir_var_mem_shared |
nir_var_mem_global),
nir_var_mem_push_const = (1 << 10), /* not actually used for variables */ nir_var_mem_push_const = (1 << 10), /* not actually used for variables */
nir_var_mem_constant = (1 << 11), nir_var_mem_constant = (1 << 11),
nir_var_read_only_modes = nir_var_shader_in | nir_var_uniform | nir_var_read_only_modes = nir_var_shader_in | nir_var_uniform |

View File

@@ -723,9 +723,14 @@ print_deref_instr(nir_deref_instr *instr, print_state *state)
print_deref_link(instr, false, state); print_deref_link(instr, false, state);
fprintf(fp, " (%s %s) ", fprintf(fp, " (");
get_variable_mode_str(instr->modes, true), unsigned modes = instr->modes;
glsl_get_type_name(instr->type)); while (modes) {
int m = u_bit_scan(&modes);
fprintf(fp, "%s%s", get_variable_mode_str(1 << m, true),
modes ? "|" : "");
}
fprintf(fp, " %s) ", glsl_get_type_name(instr->type));
if (instr->deref_type != nir_deref_type_var && if (instr->deref_type != nir_deref_type_var &&
instr->deref_type != nir_deref_type_cast) { instr->deref_type != nir_deref_type_cast) {

View File

@@ -407,8 +407,6 @@ validate_var_use(nir_variable *var, validate_state *state)
static void static void
validate_deref_instr(nir_deref_instr *instr, validate_state *state) validate_deref_instr(nir_deref_instr *instr, validate_state *state)
{ {
validate_assert(state, util_bitcount(instr->modes) == 1);
if (instr->deref_type == nir_deref_type_var) { if (instr->deref_type == nir_deref_type_var) {
/* Variable dereferences are stupid simple. */ /* Variable dereferences are stupid simple. */
validate_assert(state, instr->modes == instr->var->data.mode); validate_assert(state, instr->modes == instr->var->data.mode);
@@ -420,8 +418,22 @@ validate_deref_instr(nir_deref_instr *instr, validate_state *state)
*/ */
validate_src(&instr->parent, state, 0, 0); validate_src(&instr->parent, state, 0, 0);
/* We just validate that the type and mode are there */ /* Most variable modes in NIR can only exist by themselves. */
validate_assert(state, instr->modes); if (instr->modes & ~nir_var_mem_generic)
validate_assert(state, util_bitcount(instr->modes) == 1);
nir_deref_instr *parent = nir_src_as_deref(instr->parent);
if (parent) {
/* Casts can change the mode but it can't change completely. The new
* mode must have some bits in common with the old.
*/
validate_assert(state, instr->modes & parent->modes);
} else {
/* If our parent isn't a deref, just assert the mode is there */
validate_assert(state, instr->modes != 0);
}
/* We just validate that the type is there */
validate_assert(state, instr->type); validate_assert(state, instr->type);
if (instr->cast.align_mul > 0) { if (instr->cast.align_mul > 0) {
validate_assert(state, util_is_power_of_two_nonzero(instr->cast.align_mul)); validate_assert(state, util_is_power_of_two_nonzero(instr->cast.align_mul));