Skip to content
Snippets Groups Projects
Commit 3eb99323 authored by Mike Blumenkrantz's avatar Mike Blumenkrantz :lifter: Committed by Marge Bot
Browse files

aux/draw: add a util function for reading back indirect draw params


the return type of this is a bit clunky because instance values can change,
but it's simpler to just return the full draw info struct than to force the
caller to keep pulling from arrays or whatever

Reviewed-by: default avatarRob Clark <robdclark@chromium.org>
Part-of: <!10973>
parent 854d93f7
No related branches found
No related tags found
No related merge requests found
......@@ -125,6 +125,66 @@ util_draw_max_index(
return max_index + 1;
}
struct u_indirect_params *
util_draw_indirect_read(struct pipe_context *pipe,
const struct pipe_draw_info *info_in,
const struct pipe_draw_indirect_info *indirect,
unsigned *num_draws)
{
struct pipe_transfer *transfer;
uint32_t *params;
struct u_indirect_params *draws;
unsigned num_params = info_in->index_size ? 5 : 4;
assert(indirect);
assert(!indirect->count_from_stream_output);
uint32_t draw_count = indirect->draw_count;
if (indirect->indirect_draw_count) {
struct pipe_transfer *dc_transfer;
uint32_t *dc_param = pipe_buffer_map_range(pipe,
indirect->indirect_draw_count,
indirect->indirect_draw_count_offset,
4, PIPE_MAP_READ, &dc_transfer);
if (!dc_transfer) {
debug_printf("%s: failed to map indirect draw count buffer\n", __FUNCTION__);
return NULL;
}
if (dc_param[0] < draw_count)
draw_count = dc_param[0];
pipe_buffer_unmap(pipe, dc_transfer);
}
draws = malloc(sizeof(struct u_indirect_params) * draw_count);
if (!draws)
return NULL;
if (indirect->stride)
num_params = MIN2(indirect->stride / 4, num_params);
params = pipe_buffer_map_range(pipe,
indirect->buffer,
indirect->offset,
(num_params * indirect->draw_count) * sizeof(uint32_t),
PIPE_MAP_READ,
&transfer);
if (!transfer) {
debug_printf("%s: failed to map indirect buffer\n", __FUNCTION__);
free(draws);
return NULL;
}
for (unsigned i = 0; i < draw_count; i++) {
memcpy(&draws[i].info, info_in, sizeof(struct pipe_draw_info));
draws[i].draw.count = params[0];
draws[i].info.instance_count = params[1];
draws[i].draw.start = params[2];
draws[i].draw.index_bias = info_in->index_size ? params[3] : 0;
draws[i].info.start_instance = info_in->index_size ? params[4] : params[3];
params += indirect->stride / 4;
}
pipe_buffer_unmap(pipe, transfer);
*num_draws = draw_count;
return draws;
}
/* This extracts the draw arguments from the indirect resource,
* puts them into a new instance of pipe_draw_info, and calls draw_vbo on it.
......
......@@ -148,6 +148,17 @@ util_draw_elements_instanced(struct pipe_context *pipe,
pipe->draw_vbo(pipe, &info, 0, NULL, &draw, 1);
}
struct u_indirect_params {
struct pipe_draw_info info;
struct pipe_draw_start_count_bias draw;
};
/* caller must free the return value */
struct u_indirect_params *
util_draw_indirect_read(struct pipe_context *pipe,
const struct pipe_draw_info *info_in,
const struct pipe_draw_indirect_info *indirect,
unsigned *num_draws);
/* This converts an indirect draw into a direct draw by mapping the indirect
* buffer, extracting its arguments, and calling pipe->draw_vbo.
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment