diff options
author | Jason Ekstrand <[email protected]> | 2018-03-16 12:08:15 -0700 |
---|---|---|
committer | Jason Ekstrand <[email protected]> | 2018-06-22 20:54:00 -0700 |
commit | d7b0be48ef1a5a769ab159113d22bd35df520ab2 (patch) | |
tree | d76571dd7bbadee93bd8a13bb6b402044338dc87 /src/compiler/nir | |
parent | 85f4149f8a1f0fac3ec4dd4d97fd3d07f1654a8a (diff) |
nir: Use deref instructions in lower_constant_initializers
Acked-by: Rob Clark <[email protected]>
Acked-by: Bas Nieuwenhuizen <[email protected]>
Acked-by: Dave Airlie <[email protected]>
Reviewed-by: Kenneth Graunke <[email protected]>
Diffstat (limited to 'src/compiler/nir')
-rw-r--r-- | src/compiler/nir/nir.c | 63 | ||||
-rw-r--r-- | src/compiler/nir/nir.h | 3 | ||||
-rw-r--r-- | src/compiler/nir/nir_lower_constant_initializers.c | 57 |
3 files changed, 39 insertions, 84 deletions
diff --git a/src/compiler/nir/nir.c b/src/compiler/nir/nir.c index 18bf7635e21..2b7d8982003 100644 --- a/src/compiler/nir/nir.c +++ b/src/compiler/nir/nir.c @@ -832,69 +832,6 @@ nir_deref_foreach_leaf(nir_deref_var *deref, return deref_foreach_leaf_copy_recur(©, ©.deref, cb, state); } -/* Returns a load_const instruction that represents the constant - * initializer for the given deref chain. The caller is responsible for - * ensuring that there actually is a constant initializer. - */ -nir_load_const_instr * -nir_deref_get_const_initializer_load(nir_shader *shader, nir_deref_var *deref) -{ - nir_constant *constant = deref->var->constant_initializer; - assert(constant); - - const nir_deref *tail = &deref->deref; - unsigned matrix_col = 0; - while (tail->child) { - switch (tail->child->deref_type) { - case nir_deref_type_array: { - nir_deref_array *arr = nir_deref_as_array(tail->child); - assert(arr->deref_array_type == nir_deref_array_type_direct); - if (glsl_type_is_matrix(tail->type)) { - assert(arr->deref.child == NULL); - matrix_col = arr->base_offset; - } else { - constant = constant->elements[arr->base_offset]; - } - break; - } - - case nir_deref_type_struct: { - constant = constant->elements[nir_deref_as_struct(tail->child)->index]; - break; - } - - default: - unreachable("Invalid deref child type"); - } - - tail = tail->child; - } - - unsigned bit_size = glsl_get_bit_size(tail->type); - nir_load_const_instr *load = - nir_load_const_instr_create(shader, glsl_get_vector_elements(tail->type), - bit_size); - - switch (glsl_get_base_type(tail->type)) { - case GLSL_TYPE_FLOAT: - case GLSL_TYPE_INT: - case GLSL_TYPE_UINT: - case GLSL_TYPE_FLOAT16: - case GLSL_TYPE_DOUBLE: - case GLSL_TYPE_INT16: - case GLSL_TYPE_UINT16: - case GLSL_TYPE_UINT64: - case GLSL_TYPE_INT64: - case GLSL_TYPE_BOOL: - load->value = constant->values[matrix_col]; - break; - default: - unreachable("Invalid immediate type"); - } - - return load; -} - static nir_const_value const_value_float(double d, unsigned bit_size) { diff --git a/src/compiler/nir/nir.h b/src/compiler/nir/nir.h index 6642977fa3e..376ed5f1d17 100644 --- a/src/compiler/nir/nir.h +++ b/src/compiler/nir/nir.h @@ -2248,9 +2248,6 @@ typedef bool (*nir_deref_foreach_leaf_cb)(nir_deref_var *deref, void *state); bool nir_deref_foreach_leaf(nir_deref_var *deref, nir_deref_foreach_leaf_cb cb, void *state); -nir_load_const_instr * -nir_deref_get_const_initializer_load(nir_shader *shader, nir_deref_var *deref); - nir_const_value nir_alu_binop_identity(nir_op binop, unsigned bit_size); /** diff --git a/src/compiler/nir/nir_lower_constant_initializers.c b/src/compiler/nir/nir_lower_constant_initializers.c index f4d4d70dac0..4e9cea46157 100644 --- a/src/compiler/nir/nir_lower_constant_initializers.c +++ b/src/compiler/nir/nir_lower_constant_initializers.c @@ -24,18 +24,44 @@ #include "nir.h" #include "nir_builder.h" -static bool -deref_apply_constant_initializer(nir_deref_var *deref, void *state) +static void +build_constant_load(nir_builder *b, nir_deref_instr *deref, nir_constant *c) { - struct nir_builder *b = state; - - nir_load_const_instr *initializer = - nir_deref_get_const_initializer_load(b->shader, deref); - nir_builder_instr_insert(b, &initializer->instr); - - nir_store_deref_var(b, deref, &initializer->def, 0xf); - - return true; + if (glsl_type_is_vector_or_scalar(deref->type)) { + nir_load_const_instr *load = + nir_load_const_instr_create(b->shader, + glsl_get_vector_elements(deref->type), + glsl_get_bit_size(deref->type)); + load->value = c->values[0]; + nir_builder_instr_insert(b, &load->instr); + nir_store_deref(b, deref, &load->def, ~0); + } else if (glsl_type_is_matrix(deref->type)) { + unsigned cols = glsl_get_matrix_columns(deref->type); + unsigned rows = glsl_get_vector_elements(deref->type); + unsigned bit_size = glsl_get_bit_size(deref->type); + for (unsigned i = 0; i < cols; i++) { + nir_load_const_instr *load = + nir_load_const_instr_create(b->shader, rows, bit_size); + load->value = c->values[i]; + nir_builder_instr_insert(b, &load->instr); + nir_store_deref(b, nir_build_deref_array(b, deref, nir_imm_int(b, i)), + &load->def, ~0); + } + } else if (glsl_type_is_struct(deref->type)) { + unsigned len = glsl_get_length(deref->type); + for (unsigned i = 0; i < len; i++) { + build_constant_load(b, nir_build_deref_struct(b, deref, i), + c->elements[i]); + } + } else { + assert(glsl_type_is_array(deref->type)); + unsigned len = glsl_get_length(deref->type); + for (unsigned i = 0; i < len; i++) { + build_constant_load(b, + nir_build_deref_array(b, deref, nir_imm_int(b, i)), + c->elements[i]); + } + } } static bool @@ -51,13 +77,8 @@ lower_const_initializer(struct nir_builder *b, struct exec_list *var_list) progress = true; - nir_deref_var deref; - deref.deref.deref_type = nir_deref_type_var, - deref.deref.child = NULL; - deref.deref.type = var->type, - deref.var = var; - - nir_deref_foreach_leaf(&deref, deref_apply_constant_initializer, b); + build_constant_load(b, nir_build_deref_var(b, var), + var->constant_initializer); var->constant_initializer = NULL; } |