mirror of
https://gitlab.freedesktop.org/mesa/mesa.git
synced 2025-12-24 17:30:12 +01:00
anv/cmd_buffer: check for NULL framebuffer
This can happen when we record a VkCmdDraw in a secondary buffer that was created inheriting from the primary buffer, but with the framebuffer set to NULL in the VkCommandBufferInheritanceInfo. Vulkan 1.1.81 spec says that "the application must ensure (using scissor if neccesary) that all rendering is contained in the render area [...] [which] must be contained within the framebuffer dimesions". While this should be done by the application, commit465e5a86added the clamp to the framebuffer size, in case of application does not do it. But this requires to know the framebuffer dimensions. If we do not have a framebuffer at that moment, the best compromise we can do is to just apply the scissor as it is, and let the application to ensure the rendering is contained in the render area. v2: do not clamp to framebuffer if there isn't a framebuffer v3 (Jason): - clamp earlier in the conditional - clamp to render area if command buffer is primary v4: clamp also x and y to render area (Jason) v5: rename used variables (Jason) Fixes:465e5a86("anv: Clamp scissors to the framebuffer boundary") CC: Jason Ekstrand <jason@jlekstrand.net> Reviewed-by: Jason Ekstrand <jason@jlekstrand.net>
This commit is contained in:
parent
6c64413b6f
commit
1ad26f9417
1 changed files with 29 additions and 5 deletions
|
|
@ -70,12 +70,36 @@ gen7_cmd_buffer_emit_scissor(struct anv_cmd_buffer *cmd_buffer)
|
|||
};
|
||||
|
||||
const int max = 0xffff;
|
||||
|
||||
uint32_t y_min = s->offset.y;
|
||||
uint32_t x_min = s->offset.x;
|
||||
uint32_t y_max = s->offset.y + s->extent.height - 1;
|
||||
uint32_t x_max = s->offset.x + s->extent.width - 1;
|
||||
|
||||
/* Do this math using int64_t so overflow gets clamped correctly. */
|
||||
if (cmd_buffer->level == VK_COMMAND_BUFFER_LEVEL_PRIMARY) {
|
||||
y_min = clamp_int64((uint64_t) y_min,
|
||||
cmd_buffer->state.render_area.offset.y, max);
|
||||
x_min = clamp_int64((uint64_t) x_min,
|
||||
cmd_buffer->state.render_area.offset.x, max);
|
||||
y_max = clamp_int64((uint64_t) y_max, 0,
|
||||
cmd_buffer->state.render_area.offset.y +
|
||||
cmd_buffer->state.render_area.extent.height - 1);
|
||||
x_max = clamp_int64((uint64_t) x_max, 0,
|
||||
cmd_buffer->state.render_area.offset.x +
|
||||
cmd_buffer->state.render_area.extent.width - 1);
|
||||
} else if (fb) {
|
||||
y_min = clamp_int64((uint64_t) y_min, 0, max);
|
||||
x_min = clamp_int64((uint64_t) x_min, 0, max);
|
||||
y_max = clamp_int64((uint64_t) y_max, 0, fb->height - 1);
|
||||
x_max = clamp_int64((uint64_t) x_max, 0, fb->width - 1);
|
||||
}
|
||||
|
||||
struct GEN7_SCISSOR_RECT scissor = {
|
||||
/* Do this math using int64_t so overflow gets clamped correctly. */
|
||||
.ScissorRectangleYMin = clamp_int64(s->offset.y, 0, max),
|
||||
.ScissorRectangleXMin = clamp_int64(s->offset.x, 0, max),
|
||||
.ScissorRectangleYMax = clamp_int64((uint64_t) s->offset.y + s->extent.height - 1, 0, fb->height - 1),
|
||||
.ScissorRectangleXMax = clamp_int64((uint64_t) s->offset.x + s->extent.width - 1, 0, fb->width - 1)
|
||||
.ScissorRectangleYMin = y_min,
|
||||
.ScissorRectangleXMin = x_min,
|
||||
.ScissorRectangleYMax = y_max,
|
||||
.ScissorRectangleXMax = x_max
|
||||
};
|
||||
|
||||
if (s->extent.width <= 0 || s->extent.height <= 0) {
|
||||
|
|
|
|||
Loading…
Add table
Reference in a new issue