static void
add_var_xfb_outputs(nir_xfb_info *xfb,
nir_variable *var,
+ unsigned buffer,
unsigned *location,
unsigned *offset,
const struct glsl_type *type)
unsigned length = glsl_get_length(type);
const struct glsl_type *child_type = glsl_get_array_element(type);
for (unsigned i = 0; i < length; i++)
- add_var_xfb_outputs(xfb, var, location, offset, child_type);
+ add_var_xfb_outputs(xfb, var, buffer, location, offset, child_type);
} else if (glsl_type_is_struct(type)) {
unsigned length = glsl_get_length(type);
for (unsigned i = 0; i < length; i++) {
const struct glsl_type *child_type = glsl_get_struct_field(type, i);
- add_var_xfb_outputs(xfb, var, location, offset, child_type);
+ add_var_xfb_outputs(xfb, var, buffer, location, offset, child_type);
}
} else {
- assert(var->data.xfb_buffer < NIR_MAX_XFB_BUFFERS);
- if (xfb->buffers_written & (1 << var->data.xfb_buffer)) {
- assert(xfb->strides[var->data.xfb_buffer] == var->data.xfb_stride);
- assert(xfb->buffer_to_stream[var->data.xfb_buffer] == var->data.stream);
+ assert(buffer < NIR_MAX_XFB_BUFFERS);
+ if (xfb->buffers_written & (1 << buffer)) {
+ assert(xfb->strides[buffer] == var->data.xfb_stride);
+ assert(xfb->buffer_to_stream[buffer] == var->data.stream);
} else {
- xfb->buffers_written |= (1 << var->data.xfb_buffer);
- xfb->strides[var->data.xfb_buffer] = var->data.xfb_stride;
- xfb->buffer_to_stream[var->data.xfb_buffer] = var->data.stream;
+ xfb->buffers_written |= (1 << buffer);
+ xfb->strides[buffer] = var->data.xfb_stride;
+ xfb->buffer_to_stream[buffer] = var->data.stream;
}
assert(var->data.stream < NIR_MAX_XFB_STREAMS);
for (unsigned s = 0; s < attrib_slots; s++) {
nir_xfb_output_info *output = &xfb->outputs[xfb->output_count++];
- output->buffer = var->data.xfb_buffer;
+ output->buffer = buffer;
output->offset = *offset + s * 16;
output->location = *location;
output->component_mask = (comp_mask >> (s * 4)) & 0xf;
/* Compute the number of outputs we have. This is simply the number of
* cumulative locations consumed by all the variables. If a location is
* represented by multiple variables, then they each count separately in
- * number of outputs.
+ * number of outputs. This is only an estimate as some variables may have
+ * an xfb_buffer but not an output so it may end up larger than we need but
+ * it should be good enough for allocation.
*/
unsigned num_outputs = 0;
nir_foreach_variable(var, &shader->outputs) {
- if (var->data.explicit_xfb_buffer &&
- var->data.explicit_offset) {
-
+ if (var->data.explicit_xfb_buffer)
num_outputs += glsl_count_attribute_slots(var->type, false);
- }
}
if (num_outputs == 0)
return NULL;
/* Walk the list of outputs and add them to the array */
nir_foreach_variable(var, &shader->outputs) {
- if (var->data.explicit_xfb_buffer &&
- var->data.explicit_offset) {
+ if (!var->data.explicit_xfb_buffer)
+ continue;
+
+ unsigned location = var->data.location;
- unsigned location = var->data.location;
+ if (var->data.explicit_offset) {
unsigned offset = var->data.offset;
- add_var_xfb_outputs(xfb, var, &location, &offset, var->type);
+ add_var_xfb_outputs(xfb, var, var->data.xfb_buffer,
+ &location, &offset, var->type);
+ } else if (glsl_type_is_array(var->type) &&
+ glsl_type_is_struct(glsl_without_array(var->type))) {
+ unsigned aoa_size = glsl_get_aoa_size(var->type);
+ const struct glsl_type *stype = glsl_without_array(var->type);
+ unsigned nfields = glsl_get_length(stype);
+ for (unsigned b = 0; b < aoa_size; b++) {
+ for (unsigned f = 0; f < nfields; f++) {
+ int foffset = glsl_get_struct_field_offset(stype, f);
+ const struct glsl_type *ftype = glsl_get_struct_field(stype, f);
+ if (foffset < 0) {
+ location += glsl_count_attribute_slots(ftype, false);
+ continue;
+ }
+
+ unsigned offset = foffset;
+ add_var_xfb_outputs(xfb, var, var->data.xfb_buffer + b,
+ &location, &offset, ftype);
+ }
+ }
}
}
- assert(xfb->output_count == num_outputs);
/* Everything is easier in the state setup code if the list is sorted in
* order of output offset.