But the original code only did 0, so maybe it's better to do it as a followup patch, so that functional change is separate from refactoring?
Gražvydas On Tue, Aug 25, 2015 at 12:30 AM, Marek Olšák <mar...@gmail.com> wrote: > The "workaround for a missing scissor enable on r600" must be applied > to all scissors, not just 0. That is, if scissor.enable is changed, > all scissors must be marked as dirty. > > Marek > > On Mon, Aug 24, 2015 at 2:24 AM, Grazvydas Ignotas <nota...@gmail.com> wrote: >> As suggested by Marek Olšák, we can use single atom to track all scissor >> states. This will allow to simplify dirty atom handling later. >> --- >> src/gallium/drivers/r600/evergreen_state.c | 33 ++++++++++++++------- >> src/gallium/drivers/r600/r600_blit.c | 2 +- >> src/gallium/drivers/r600/r600_hw_context.c | 4 ++- >> src/gallium/drivers/r600/r600_pipe.h | 8 +++--- >> src/gallium/drivers/r600/r600_state.c | 43 >> +++++++++++++++++----------- >> src/gallium/drivers/r600/r600_state_common.c | 6 ++-- >> 6 files changed, 60 insertions(+), 36 deletions(-) >> >> diff --git a/src/gallium/drivers/r600/evergreen_state.c >> b/src/gallium/drivers/r600/evergreen_state.c >> index 6a91d47..fbdcc9c 100644 >> --- a/src/gallium/drivers/r600/evergreen_state.c >> +++ b/src/gallium/drivers/r600/evergreen_state.c >> @@ -892,27 +892,39 @@ static void evergreen_set_scissor_states(struct >> pipe_context *ctx, >> const struct pipe_scissor_state >> *state) >> { >> struct r600_context *rctx = (struct r600_context *)ctx; >> + struct r600_scissor_state *rstate = &rctx->scissor; >> int i; >> >> for (i = start_slot; i < start_slot + num_scissors; i++) { >> - rctx->scissor[i].scissor = state[i - start_slot]; >> - r600_mark_atom_dirty(rctx, &rctx->scissor[i].atom); >> + rstate->scissor[i] = state[i - start_slot]; >> + rstate->dirty_mask |= 1 << i; >> } >> + rstate->atom.num_dw = util_bitcount(rstate->dirty_mask) * 4; >> + r600_mark_atom_dirty(rctx, &rstate->atom); >> } >> >> static void evergreen_emit_scissor_state(struct r600_context *rctx, struct >> r600_atom *atom) >> { >> struct radeon_winsys_cs *cs = rctx->b.rings.gfx.cs; >> - struct r600_scissor_state *rstate = (struct r600_scissor_state >> *)atom; >> - struct pipe_scissor_state *state = &rstate->scissor; >> - unsigned offset = rstate->idx * 4 * 2; >> + struct r600_scissor_state *rstate = &rctx->scissor; >> + struct pipe_scissor_state *state; >> + uint32_t dirty_mask; >> + unsigned i, offset; >> uint32_t tl, br; >> >> - evergreen_get_scissor_rect(rctx, state->minx, state->miny, >> state->maxx, state->maxy, &tl, &br); >> + dirty_mask = rstate->dirty_mask; >> + while (dirty_mask != 0) { >> + i = u_bit_scan(&dirty_mask); >> + state = &rstate->scissor[i]; >> + evergreen_get_scissor_rect(rctx, state->minx, state->miny, >> state->maxx, state->maxy, &tl, &br); >> >> - r600_write_context_reg_seq(cs, R_028250_PA_SC_VPORT_SCISSOR_0_TL + >> offset, 2); >> - radeon_emit(cs, tl); >> - radeon_emit(cs, br); >> + offset = i * 4 * 2; >> + r600_write_context_reg_seq(cs, >> R_028250_PA_SC_VPORT_SCISSOR_0_TL + offset, 2); >> + radeon_emit(cs, tl); >> + radeon_emit(cs, br); >> + } >> + rstate->dirty_mask = 0; >> + rstate->atom.num_dw = 0; >> } >> >> /** >> @@ -3484,11 +3496,10 @@ void evergreen_init_state_functions(struct >> r600_context *rctx) >> r600_init_atom(rctx, &rctx->dsa_state.atom, id++, >> r600_emit_cso_state, 0); >> r600_init_atom(rctx, &rctx->poly_offset_state.atom, id++, >> evergreen_emit_polygon_offset, 6); >> r600_init_atom(rctx, &rctx->rasterizer_state.atom, id++, >> r600_emit_cso_state, 0); >> + r600_init_atom(rctx, &rctx->scissor.atom, id++, >> evergreen_emit_scissor_state, 0); >> for (i = 0; i < R600_MAX_VIEWPORTS; i++) { >> r600_init_atom(rctx, &rctx->viewport[i].atom, id++, >> r600_emit_viewport_state, 8); >> - r600_init_atom(rctx, &rctx->scissor[i].atom, id++, >> evergreen_emit_scissor_state, 4); >> rctx->viewport[i].idx = i; >> - rctx->scissor[i].idx = i; >> } >> r600_init_atom(rctx, &rctx->stencil_ref.atom, id++, >> r600_emit_stencil_ref, 4); >> r600_init_atom(rctx, &rctx->vertex_fetch_shader.atom, id++, >> evergreen_emit_vertex_fetch_shader, 5); >> diff --git a/src/gallium/drivers/r600/r600_blit.c >> b/src/gallium/drivers/r600/r600_blit.c >> index 22a0950..6ecb8ee 100644 >> --- a/src/gallium/drivers/r600/r600_blit.c >> +++ b/src/gallium/drivers/r600/r600_blit.c >> @@ -66,7 +66,7 @@ static void r600_blitter_begin(struct pipe_context *ctx, >> enum r600_blitter_op op >> >> if (op & R600_SAVE_FRAGMENT_STATE) { >> util_blitter_save_viewport(rctx->blitter, >> &rctx->viewport[0].state); >> - util_blitter_save_scissor(rctx->blitter, >> &rctx->scissor[0].scissor); >> + util_blitter_save_scissor(rctx->blitter, >> &rctx->scissor.scissor[0]); >> util_blitter_save_fragment_shader(rctx->blitter, >> rctx->ps_shader); >> util_blitter_save_blend(rctx->blitter, >> rctx->blend_state.cso); >> util_blitter_save_depth_stencil_alpha(rctx->blitter, >> rctx->dsa_state.cso); >> diff --git a/src/gallium/drivers/r600/r600_hw_context.c >> b/src/gallium/drivers/r600/r600_hw_context.c >> index 6445151..6710cd3 100644 >> --- a/src/gallium/drivers/r600/r600_hw_context.c >> +++ b/src/gallium/drivers/r600/r600_hw_context.c >> @@ -308,8 +308,10 @@ void r600_begin_new_cs(struct r600_context *ctx) >> r600_mark_atom_dirty(ctx, &ctx->poly_offset_state.atom); >> r600_mark_atom_dirty(ctx, &ctx->vgt_state.atom); >> r600_mark_atom_dirty(ctx, &ctx->sample_mask.atom); >> + ctx->scissor.dirty_mask = (1 << R600_MAX_VIEWPORTS) - 1; >> + ctx->scissor.atom.num_dw = R600_MAX_VIEWPORTS * 4; >> + r600_mark_atom_dirty(ctx, &ctx->scissor.atom); >> for (i = 0; i < R600_MAX_VIEWPORTS; i++) { >> - r600_mark_atom_dirty(ctx, &ctx->scissor[i].atom); >> r600_mark_atom_dirty(ctx, &ctx->viewport[i].atom); >> } >> if (ctx->b.chip_class < EVERGREEN) { >> diff --git a/src/gallium/drivers/r600/r600_pipe.h >> b/src/gallium/drivers/r600/r600_pipe.h >> index 384ba80..549a7f6 100644 >> --- a/src/gallium/drivers/r600/r600_pipe.h >> +++ b/src/gallium/drivers/r600/r600_pipe.h >> @@ -36,7 +36,7 @@ >> #include "util/list.h" >> #include "util/u_transfer.h" >> >> -#define R600_NUM_ATOMS 75 >> +#define R600_NUM_ATOMS 60 >> >> #define R600_MAX_VIEWPORTS 16 >> >> @@ -385,9 +385,9 @@ struct r600_cso_state >> struct r600_scissor_state >> { >> struct r600_atom atom; >> - struct pipe_scissor_state scissor; >> + struct pipe_scissor_state scissor[R600_MAX_VIEWPORTS]; >> + uint32_t dirty_mask; >> bool enable; /* r6xx only */ >> - int idx; >> }; >> >> struct r600_fetch_shader { >> @@ -450,7 +450,7 @@ struct r600_context { >> struct r600_poly_offset_state poly_offset_state; >> struct r600_cso_state rasterizer_state; >> struct r600_sample_mask sample_mask; >> - struct r600_scissor_state scissor[R600_MAX_VIEWPORTS]; >> + struct r600_scissor_state scissor; >> struct r600_seamless_cube_map seamless_cube_map; >> struct r600_config_state config_state; >> struct r600_stencil_ref_state stencil_ref; >> diff --git a/src/gallium/drivers/r600/r600_state.c >> b/src/gallium/drivers/r600/r600_state.c >> index 5cc2283..88f8da5 100644 >> --- a/src/gallium/drivers/r600/r600_state.c >> +++ b/src/gallium/drivers/r600/r600_state.c >> @@ -769,21 +769,32 @@ static void r600_set_polygon_stipple(struct >> pipe_context *ctx, >> static void r600_emit_scissor_state(struct r600_context *rctx, struct >> r600_atom *atom) >> { >> struct radeon_winsys_cs *cs = rctx->b.rings.gfx.cs; >> - struct r600_scissor_state *rstate = (struct r600_scissor_state >> *)atom; >> - struct pipe_scissor_state *state = &rstate->scissor; >> - unsigned offset = rstate->idx * 4 * 2; >> + struct r600_scissor_state *rstate = &rctx->scissor; >> + struct pipe_scissor_state *state; >> + uint32_t dirty_mask; >> + unsigned i, offset; >> >> - if (rctx->b.chip_class != R600 || rctx->scissor[0].enable) { >> - r600_write_context_reg_seq(cs, >> R_028250_PA_SC_VPORT_SCISSOR_0_TL + offset, 2); >> - radeon_emit(cs, S_028240_TL_X(state->minx) | >> S_028240_TL_Y(state->miny) | >> - S_028240_WINDOW_OFFSET_DISABLE(1)); >> - radeon_emit(cs, S_028244_BR_X(state->maxx) | >> S_028244_BR_Y(state->maxy)); >> - } else { >> + if (rctx->b.chip_class == R600 && !rctx->scissor.enable) { >> r600_write_context_reg_seq(cs, >> R_028250_PA_SC_VPORT_SCISSOR_0_TL, 2); >> radeon_emit(cs, S_028240_TL_X(0) | S_028240_TL_Y(0) | >> S_028240_WINDOW_OFFSET_DISABLE(1)); >> radeon_emit(cs, S_028244_BR_X(8192) | S_028244_BR_Y(8192)); >> + return; >> + } >> + >> + dirty_mask = rstate->dirty_mask; >> + while (dirty_mask != 0) >> + { >> + i = u_bit_scan(&dirty_mask); >> + offset = i * 4 * 2; >> + state = &rstate->scissor[i]; >> + r600_write_context_reg_seq(cs, >> R_028250_PA_SC_VPORT_SCISSOR_0_TL + offset, 2); >> + radeon_emit(cs, S_028240_TL_X(state->minx) | >> S_028240_TL_Y(state->miny) | >> + S_028240_WINDOW_OFFSET_DISABLE(1)); >> + radeon_emit(cs, S_028244_BR_X(state->maxx) | >> S_028244_BR_Y(state->maxy)); >> } >> + rstate->dirty_mask = 0; >> + rstate->atom.num_dw = 0; >> } >> >> static void r600_set_scissor_states(struct pipe_context *ctx, >> @@ -792,18 +803,19 @@ static void r600_set_scissor_states(struct >> pipe_context *ctx, >> const struct pipe_scissor_state *state) >> { >> struct r600_context *rctx = (struct r600_context *)ctx; >> + struct r600_scissor_state *rstate = &rctx->scissor; >> int i; >> >> for (i = start_slot ; i < start_slot + num_scissors; i++) { >> - rctx->scissor[i].scissor = state[i - start_slot]; >> + rstate->scissor[i] = state[i - start_slot]; >> + rstate->dirty_mask |= 1 << i; >> } >> + rstate->atom.num_dw = util_bitcount(rstate->dirty_mask) * 4; >> >> - if (rctx->b.chip_class == R600 && !rctx->scissor[0].enable) >> + if (rctx->b.chip_class == R600 && !rstate->enable) >> return; >> >> - for (i = start_slot ; i < start_slot + num_scissors; i++) { >> - r600_mark_atom_dirty(rctx, &rctx->scissor[i].atom); >> - } >> + r600_mark_atom_dirty(rctx, &rstate->atom); >> } >> >> static struct r600_resource *r600_buffer_create_helper(struct r600_screen >> *rscreen, >> @@ -3065,10 +3077,9 @@ void r600_init_state_functions(struct r600_context >> *rctx) >> r600_init_atom(rctx, &rctx->dsa_state.atom, id++, >> r600_emit_cso_state, 0); >> r600_init_atom(rctx, &rctx->poly_offset_state.atom, id++, >> r600_emit_polygon_offset, 6); >> r600_init_atom(rctx, &rctx->rasterizer_state.atom, id++, >> r600_emit_cso_state, 0); >> + r600_init_atom(rctx, &rctx->scissor.atom, id++, >> r600_emit_scissor_state, 0); >> for (i = 0;i < R600_MAX_VIEWPORTS; i++) { >> - r600_init_atom(rctx, &rctx->scissor[i].atom, id++, >> r600_emit_scissor_state, 4); >> r600_init_atom(rctx, &rctx->viewport[i].atom, id++, >> r600_emit_viewport_state, 8); >> - rctx->scissor[i].idx = i; >> rctx->viewport[i].idx = i; >> } >> r600_init_atom(rctx, &rctx->config_state.atom, id++, >> r600_emit_config_state, 3); >> diff --git a/src/gallium/drivers/r600/r600_state_common.c >> b/src/gallium/drivers/r600/r600_state_common.c >> index a05dd83..66dc534 100644 >> --- a/src/gallium/drivers/r600/r600_state_common.c >> +++ b/src/gallium/drivers/r600/r600_state_common.c >> @@ -346,9 +346,9 @@ static void r600_bind_rs_state(struct pipe_context *ctx, >> void *state) >> >> /* Workaround for a missing scissor enable on r600. */ >> if (rctx->b.chip_class == R600 && >> - rs->scissor_enable != rctx->scissor[0].enable) { >> - rctx->scissor[0].enable = rs->scissor_enable; >> - r600_mark_atom_dirty(rctx, &rctx->scissor[0].atom); >> + rs->scissor_enable != rctx->scissor.enable) { >> + rctx->scissor.enable = rs->scissor_enable; >> + r600_mark_atom_dirty(rctx, &rctx->scissor.atom); >> } >> >> /* Re-emit PA_SC_LINE_STIPPLE. */ >> -- >> 1.9.1 >> >> _______________________________________________ >> mesa-dev mailing list >> mesa-dev@lists.freedesktop.org >> http://lists.freedesktop.org/mailman/listinfo/mesa-dev _______________________________________________ mesa-dev mailing list mesa-dev@lists.freedesktop.org http://lists.freedesktop.org/mailman/listinfo/mesa-dev