iris: Fix iris_copy_region calls involving FCV_CCS_E

iris_copy_region improperly handles destinations that have
ISL_AUX_USAGE_FCV_CCS_E in use. To avoid corruption when copying to a
resource which has this aux usage, this function needs to:

   1. Remove existing fast-clear blocks that would be incompatible with
      the surface format that will be used by blorp_copy.

      This is actually a general rendering requirement that affects more
      aux usages than just FCV_CCS_E.

   2. Either avoid generating new fast-clear blocks that would be
      incompatible with the original surface format, or remove the newly
      generated incompatible fast-clear blocks.

      This is particular to FCV_CCS_E, which sometimes generates
      fast-clear blocks during a rendering operation. The generation is
      dependent on the surface format, the clear color, and the pixels
      being written.

iris_copy_region does step 1, only allowing fast-clear blocks that
represent the value of zero. These are compatible with every surface
format. This function does not do step 2 however, leading to rendering
corruption in certain cases.

Fix this by generally relying more on the standard resource preparation
functions, which account for this issue. Specifically, by using
iris_resource_prepare_render, steps 1 and 2 are both handled for us.

Closes: https://gitlab.freedesktop.org/mesa/mesa/-/issues/3732
Reviewed-by: Lionel Landwerlin <lionel.g.landwerlin@intel.com>
Part-of: <https://gitlab.freedesktop.org/mesa/mesa/-/merge_requests/24446>
This commit is contained in:
Nanley Chery 2023-05-05 14:57:03 -07:00 committed by Marge Bot
parent 3acdb29332
commit 60aebe8fa8

View file

@ -562,6 +562,76 @@ iris_blit(struct pipe_context *ctx, const struct pipe_blit_info *info)
iris_dirty_for_history(ice, (struct iris_resource *)info->dst.resource);
}
static enum isl_aux_usage
copy_region_aux_usage(struct iris_context *ice,
const struct iris_batch *batch,
struct iris_resource *res,
enum isl_format view_format,
unsigned level,
bool is_dest)
{
struct iris_screen *screen = (void *) ice->ctx.screen;
const struct intel_device_info *devinfo = screen->devinfo;
if (batch->name == IRIS_BATCH_RENDER) {
if (is_dest) {
return iris_resource_render_aux_usage(ice, res, view_format, level,
false);
} else {
return iris_resource_texture_aux_usage(ice, res, view_format, level,
1);
}
} else {
assert(batch->name == IRIS_BATCH_BLITTER);
if (is_dest) {
/* We only blit to images created with PIPE_BIND_PRIME_BLIT_DST.
* These are only created with dri3_alloc_render_buffer. That
* function also makes them linear, so they lack compression.
*
* Note: this code block could be substituted with the one below, but
* this setup clarifies that no additional handling of FCV is
* necessary.
*/
assert(res->base.b.bind & PIPE_BIND_PRIME_BLIT_DST);
assert(res->surf.tiling == ISL_TILING_LINEAR);
assert(res->aux.usage == ISL_AUX_USAGE_NONE);
return ISL_AUX_USAGE_NONE;
} else {
return devinfo->verx10 >= 125 ? res->aux.usage : ISL_AUX_USAGE_NONE;
}
}
}
static void
prepare_copy_region(struct iris_context *ice,
const struct iris_batch *batch,
struct iris_resource *res,
enum isl_format view_format,
uint32_t level,
uint32_t start_layer, uint32_t layer_count,
enum isl_aux_usage aux_usage,
bool is_dest)
{
if (batch->name == IRIS_BATCH_RENDER) {
if (is_dest) {
iris_resource_prepare_render(ice, res, view_format, level,
start_layer, layer_count, aux_usage);
} else {
iris_resource_prepare_texture(ice, res, view_format, level, 1,
start_layer, layer_count);
}
} else {
assert(batch->name == IRIS_BATCH_BLITTER);
iris_resource_prepare_access(ice, res, level, 1,
start_layer, layer_count, aux_usage,
isl_aux_usage_has_fast_clears(aux_usage) &&
clear_color_is_fully_zero(res));
}
}
static void
get_copy_region_aux_settings(struct iris_context *ice,
const struct iris_batch *batch,
@ -686,12 +756,10 @@ iris_copy_region(struct blorp_context *blorp,
blorp_copy_get_formats(&screen->isl_dev, &src_res->surf, &dst_res->surf,
&src_fmt, &dst_fmt);
enum isl_aux_usage src_aux_usage, dst_aux_usage;
bool src_clear_supported, dst_clear_supported;
get_copy_region_aux_settings(ice, batch, src_res, src_level,
&src_aux_usage, &src_clear_supported, false);
get_copy_region_aux_settings(ice, batch, dst_res, dst_level,
&dst_aux_usage, &dst_clear_supported, true);
enum isl_aux_usage src_aux_usage =
copy_region_aux_usage(ice, batch, src_res, src_fmt, src_level, false);
enum isl_aux_usage dst_aux_usage =
copy_region_aux_usage(ice, batch, dst_res, dst_fmt, dst_level, true);
if (iris_batch_references(batch, src_res->bo))
tex_cache_flush_hack(batch, src_fmt, src_res->surf.format);
@ -730,12 +798,10 @@ iris_copy_region(struct blorp_context *blorp,
} else {
// XXX: what about one surface being a buffer and not the other?
iris_resource_prepare_access(ice, src_res, src_level, 1,
src_box->z, src_box->depth,
src_aux_usage, src_clear_supported);
iris_resource_prepare_access(ice, dst_res, dst_level, 1,
dstz, src_box->depth,
dst_aux_usage, dst_clear_supported);
prepare_copy_region(ice, batch, src_res, src_fmt, src_level,
src_box->z, src_box->depth, src_aux_usage, false);
prepare_copy_region(ice, batch, dst_res, dst_fmt, dst_level,
dstz, src_box->depth, dst_aux_usage, true);
iris_emit_buffer_barrier_for(batch, src_res->bo,
IRIS_DOMAIN_SAMPLER_READ);