Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
PojavLauncherTeam
GitHub Repository: PojavLauncherTeam/mesa
Path: blob/21.2-virgl/src/gallium/drivers/freedreno/a4xx/fd4_gmem.c
4574 views
1
/*
2
* Copyright (C) 2014 Rob Clark <[email protected]>
3
*
4
* Permission is hereby granted, free of charge, to any person obtaining a
5
* copy of this software and associated documentation files (the "Software"),
6
* to deal in the Software without restriction, including without limitation
7
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
8
* and/or sell copies of the Software, and to permit persons to whom the
9
* Software is furnished to do so, subject to the following conditions:
10
*
11
* The above copyright notice and this permission notice (including the next
12
* paragraph) shall be included in all copies or substantial portions of the
13
* Software.
14
*
15
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
18
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
* SOFTWARE.
22
*
23
* Authors:
24
* Rob Clark <[email protected]>
25
*/
26
27
#include "pipe/p_state.h"
28
#include "util/format/u_format.h"
29
#include "util/u_inlines.h"
30
#include "util/u_memory.h"
31
#include "util/u_string.h"
32
33
#include "freedreno_draw.h"
34
#include "freedreno_resource.h"
35
#include "freedreno_state.h"
36
37
#include "fd4_context.h"
38
#include "fd4_draw.h"
39
#include "fd4_emit.h"
40
#include "fd4_format.h"
41
#include "fd4_gmem.h"
42
#include "fd4_program.h"
43
#include "fd4_zsa.h"
44
45
static void
46
fd4_gmem_emit_set_prog(struct fd_context *ctx, struct fd4_emit *emit,
47
struct fd_program_stateobj *prog)
48
{
49
emit->skip_consts = true;
50
emit->key.vs = prog->vs;
51
emit->key.fs = prog->fs;
52
emit->prog = fd4_program_state(
53
ir3_cache_lookup(ctx->shader_cache, &emit->key, &ctx->debug));
54
/* reset the fd4_emit_get_*p cache */
55
emit->vs = NULL;
56
emit->fs = NULL;
57
}
58
59
static void
60
emit_mrt(struct fd_ringbuffer *ring, unsigned nr_bufs,
61
struct pipe_surface **bufs, const uint32_t *bases, uint32_t bin_w,
62
bool decode_srgb)
63
{
64
enum a4xx_tile_mode tile_mode;
65
unsigned i;
66
67
if (bin_w) {
68
tile_mode = 2;
69
} else {
70
tile_mode = TILE4_LINEAR;
71
}
72
73
for (i = 0; i < A4XX_MAX_RENDER_TARGETS; i++) {
74
enum a4xx_color_fmt format = 0;
75
enum a3xx_color_swap swap = WZYX;
76
bool srgb = false;
77
struct fd_resource *rsc = NULL;
78
uint32_t stride = 0;
79
uint32_t base = 0;
80
uint32_t offset = 0;
81
82
if ((i < nr_bufs) && bufs[i]) {
83
struct pipe_surface *psurf = bufs[i];
84
enum pipe_format pformat = psurf->format;
85
86
rsc = fd_resource(psurf->texture);
87
88
/* In case we're drawing to Z32F_S8, the "color" actually goes to
89
* the stencil
90
*/
91
if (rsc->stencil) {
92
rsc = rsc->stencil;
93
pformat = rsc->b.b.format;
94
if (bases)
95
bases++;
96
}
97
98
format = fd4_pipe2color(pformat);
99
swap = fd4_pipe2swap(pformat);
100
101
if (decode_srgb)
102
srgb = util_format_is_srgb(pformat);
103
else
104
pformat = util_format_linear(pformat);
105
106
debug_assert(psurf->u.tex.first_layer == psurf->u.tex.last_layer);
107
108
offset = fd_resource_offset(rsc, psurf->u.tex.level,
109
psurf->u.tex.first_layer);
110
111
if (bin_w) {
112
stride = bin_w << fdl_cpp_shift(&rsc->layout);
113
114
if (bases) {
115
base = bases[i];
116
}
117
} else {
118
stride = fd_resource_pitch(rsc, psurf->u.tex.level);
119
}
120
} else if ((i < nr_bufs) && bases) {
121
base = bases[i];
122
}
123
124
OUT_PKT0(ring, REG_A4XX_RB_MRT_BUF_INFO(i), 3);
125
OUT_RING(ring, A4XX_RB_MRT_BUF_INFO_COLOR_FORMAT(format) |
126
A4XX_RB_MRT_BUF_INFO_COLOR_TILE_MODE(tile_mode) |
127
A4XX_RB_MRT_BUF_INFO_COLOR_BUF_PITCH(stride) |
128
A4XX_RB_MRT_BUF_INFO_COLOR_SWAP(swap) |
129
COND(srgb, A4XX_RB_MRT_BUF_INFO_COLOR_SRGB));
130
if (bin_w || (i >= nr_bufs) || !bufs[i]) {
131
OUT_RING(ring, base);
132
OUT_RING(ring, A4XX_RB_MRT_CONTROL3_STRIDE(stride));
133
} else {
134
OUT_RELOC(ring, rsc->bo, offset, 0, 0);
135
/* RB_MRT[i].CONTROL3.STRIDE not emitted by c2d..
136
* not sure if we need to skip it for bypass or
137
* not.
138
*/
139
OUT_RING(ring, A4XX_RB_MRT_CONTROL3_STRIDE(0));
140
}
141
}
142
}
143
144
static bool
145
use_hw_binning(struct fd_batch *batch)
146
{
147
const struct fd_gmem_stateobj *gmem = batch->gmem_state;
148
149
if ((gmem->maxpw * gmem->maxph) > 32)
150
return false;
151
152
if ((gmem->maxpw > 15) || (gmem->maxph > 15))
153
return false;
154
155
return fd_binning_enabled && ((gmem->nbins_x * gmem->nbins_y) > 2);
156
}
157
158
/* transfer from gmem to system memory (ie. normal RAM) */
159
160
static void
161
emit_gmem2mem_surf(struct fd_batch *batch, bool stencil, uint32_t base,
162
struct pipe_surface *psurf)
163
{
164
struct fd_ringbuffer *ring = batch->gmem;
165
struct fd_resource *rsc = fd_resource(psurf->texture);
166
enum pipe_format pformat = psurf->format;
167
uint32_t offset, pitch;
168
169
if (!rsc->valid)
170
return;
171
172
if (stencil) {
173
debug_assert(rsc->stencil);
174
rsc = rsc->stencil;
175
pformat = rsc->b.b.format;
176
}
177
178
offset =
179
fd_resource_offset(rsc, psurf->u.tex.level, psurf->u.tex.first_layer);
180
pitch = fd_resource_pitch(rsc, psurf->u.tex.level);
181
182
debug_assert(psurf->u.tex.first_layer == psurf->u.tex.last_layer);
183
184
OUT_PKT0(ring, REG_A4XX_RB_COPY_CONTROL, 4);
185
OUT_RING(ring, A4XX_RB_COPY_CONTROL_MSAA_RESOLVE(MSAA_ONE) |
186
A4XX_RB_COPY_CONTROL_MODE(RB_COPY_RESOLVE) |
187
A4XX_RB_COPY_CONTROL_GMEM_BASE(base));
188
OUT_RELOC(ring, rsc->bo, offset, 0, 0); /* RB_COPY_DEST_BASE */
189
OUT_RING(ring, A4XX_RB_COPY_DEST_PITCH_PITCH(pitch));
190
OUT_RING(ring, A4XX_RB_COPY_DEST_INFO_TILE(TILE4_LINEAR) |
191
A4XX_RB_COPY_DEST_INFO_FORMAT(fd4_pipe2color(pformat)) |
192
A4XX_RB_COPY_DEST_INFO_COMPONENT_ENABLE(0xf) |
193
A4XX_RB_COPY_DEST_INFO_ENDIAN(ENDIAN_NONE) |
194
A4XX_RB_COPY_DEST_INFO_SWAP(fd4_pipe2swap(pformat)));
195
196
fd4_draw(batch, ring, DI_PT_RECTLIST, IGNORE_VISIBILITY,
197
DI_SRC_SEL_AUTO_INDEX, 2, 1, INDEX4_SIZE_8_BIT, 0, 0, NULL);
198
}
199
200
static void
201
fd4_emit_tile_gmem2mem(struct fd_batch *batch,
202
const struct fd_tile *tile) assert_dt
203
{
204
struct fd_context *ctx = batch->ctx;
205
const struct fd_gmem_stateobj *gmem = batch->gmem_state;
206
struct fd_ringbuffer *ring = batch->gmem;
207
struct pipe_framebuffer_state *pfb = &batch->framebuffer;
208
struct fd4_emit emit = {
209
.debug = &ctx->debug,
210
.vtx = &ctx->solid_vbuf_state,
211
};
212
fd4_gmem_emit_set_prog(ctx, &emit, &ctx->solid_prog);
213
214
OUT_PKT0(ring, REG_A4XX_RB_DEPTH_CONTROL, 1);
215
OUT_RING(ring, A4XX_RB_DEPTH_CONTROL_ZFUNC(FUNC_NEVER));
216
217
OUT_PKT0(ring, REG_A4XX_RB_STENCIL_CONTROL, 2);
218
OUT_RING(ring, A4XX_RB_STENCIL_CONTROL_FUNC(FUNC_NEVER) |
219
A4XX_RB_STENCIL_CONTROL_FAIL(STENCIL_KEEP) |
220
A4XX_RB_STENCIL_CONTROL_ZPASS(STENCIL_KEEP) |
221
A4XX_RB_STENCIL_CONTROL_ZFAIL(STENCIL_KEEP) |
222
A4XX_RB_STENCIL_CONTROL_FUNC_BF(FUNC_NEVER) |
223
A4XX_RB_STENCIL_CONTROL_FAIL_BF(STENCIL_KEEP) |
224
A4XX_RB_STENCIL_CONTROL_ZPASS_BF(STENCIL_KEEP) |
225
A4XX_RB_STENCIL_CONTROL_ZFAIL_BF(STENCIL_KEEP));
226
OUT_RING(ring, 0x00000000); /* RB_STENCIL_CONTROL2 */
227
228
OUT_PKT0(ring, REG_A4XX_RB_STENCILREFMASK, 2);
229
OUT_RING(ring, 0xff000000 | A4XX_RB_STENCILREFMASK_STENCILREF(0) |
230
A4XX_RB_STENCILREFMASK_STENCILMASK(0) |
231
A4XX_RB_STENCILREFMASK_STENCILWRITEMASK(0xff));
232
OUT_RING(ring, 0xff000000 | A4XX_RB_STENCILREFMASK_BF_STENCILREF(0) |
233
A4XX_RB_STENCILREFMASK_BF_STENCILMASK(0) |
234
A4XX_RB_STENCILREFMASK_BF_STENCILWRITEMASK(0xff));
235
236
OUT_PKT0(ring, REG_A4XX_GRAS_SU_MODE_CONTROL, 1);
237
OUT_RING(ring, A4XX_GRAS_SU_MODE_CONTROL_LINEHALFWIDTH(0));
238
239
fd_wfi(batch, ring);
240
241
OUT_PKT0(ring, REG_A4XX_GRAS_CL_CLIP_CNTL, 1);
242
OUT_RING(ring, 0x80000); /* GRAS_CL_CLIP_CNTL */
243
244
OUT_PKT0(ring, REG_A4XX_GRAS_CL_VPORT_XOFFSET_0, 6);
245
OUT_RING(ring, A4XX_GRAS_CL_VPORT_XOFFSET_0((float)pfb->width / 2.0));
246
OUT_RING(ring, A4XX_GRAS_CL_VPORT_XSCALE_0((float)pfb->width / 2.0));
247
OUT_RING(ring, A4XX_GRAS_CL_VPORT_YOFFSET_0((float)pfb->height / 2.0));
248
OUT_RING(ring, A4XX_GRAS_CL_VPORT_YSCALE_0(-(float)pfb->height / 2.0));
249
OUT_RING(ring, A4XX_GRAS_CL_VPORT_ZOFFSET_0(0.0));
250
OUT_RING(ring, A4XX_GRAS_CL_VPORT_ZSCALE_0(1.0));
251
252
OUT_PKT0(ring, REG_A4XX_RB_RENDER_CONTROL, 1);
253
OUT_RING(ring, A4XX_RB_RENDER_CONTROL_DISABLE_COLOR_PIPE | 0xa); /* XXX */
254
255
OUT_PKT0(ring, REG_A4XX_GRAS_SC_CONTROL, 1);
256
OUT_RING(ring, A4XX_GRAS_SC_CONTROL_RENDER_MODE(RB_RESOLVE_PASS) |
257
A4XX_GRAS_SC_CONTROL_MSAA_DISABLE |
258
A4XX_GRAS_SC_CONTROL_MSAA_SAMPLES(MSAA_ONE) |
259
A4XX_GRAS_SC_CONTROL_RASTER_MODE(1));
260
261
OUT_PKT0(ring, REG_A4XX_PC_PRIM_VTX_CNTL, 1);
262
OUT_RING(ring, A4XX_PC_PRIM_VTX_CNTL_PROVOKING_VTX_LAST);
263
264
OUT_PKT0(ring, REG_A4XX_GRAS_ALPHA_CONTROL, 1);
265
OUT_RING(ring, 0x00000002);
266
267
OUT_PKT0(ring, REG_A4XX_GRAS_SC_WINDOW_SCISSOR_BR, 2);
268
OUT_RING(ring, A4XX_GRAS_SC_WINDOW_SCISSOR_BR_X(pfb->width - 1) |
269
A4XX_GRAS_SC_WINDOW_SCISSOR_BR_Y(pfb->height - 1));
270
OUT_RING(ring, A4XX_GRAS_SC_WINDOW_SCISSOR_TL_X(0) |
271
A4XX_GRAS_SC_WINDOW_SCISSOR_TL_Y(0));
272
273
OUT_PKT0(ring, REG_A4XX_VFD_INDEX_OFFSET, 2);
274
OUT_RING(ring, 0); /* VFD_INDEX_OFFSET */
275
OUT_RING(ring, 0); /* ??? UNKNOWN_2209 */
276
277
fd4_program_emit(ring, &emit, 0, NULL);
278
fd4_emit_vertex_bufs(ring, &emit);
279
280
if (batch->resolve & (FD_BUFFER_DEPTH | FD_BUFFER_STENCIL)) {
281
struct fd_resource *rsc = fd_resource(pfb->zsbuf->texture);
282
if (!rsc->stencil || (batch->resolve & FD_BUFFER_DEPTH))
283
emit_gmem2mem_surf(batch, false, gmem->zsbuf_base[0], pfb->zsbuf);
284
if (rsc->stencil && (batch->resolve & FD_BUFFER_STENCIL))
285
emit_gmem2mem_surf(batch, true, gmem->zsbuf_base[1], pfb->zsbuf);
286
}
287
288
if (batch->resolve & FD_BUFFER_COLOR) {
289
unsigned i;
290
for (i = 0; i < pfb->nr_cbufs; i++) {
291
if (!pfb->cbufs[i])
292
continue;
293
if (!(batch->resolve & (PIPE_CLEAR_COLOR0 << i)))
294
continue;
295
emit_gmem2mem_surf(batch, false, gmem->cbuf_base[i], pfb->cbufs[i]);
296
}
297
}
298
299
OUT_PKT0(ring, REG_A4XX_GRAS_SC_CONTROL, 1);
300
OUT_RING(ring, A4XX_GRAS_SC_CONTROL_RENDER_MODE(RB_RENDERING_PASS) |
301
A4XX_GRAS_SC_CONTROL_MSAA_DISABLE |
302
A4XX_GRAS_SC_CONTROL_MSAA_SAMPLES(MSAA_ONE) |
303
A4XX_GRAS_SC_CONTROL_RASTER_MODE(0));
304
}
305
306
/* transfer from system memory to gmem */
307
308
static void
309
emit_mem2gmem_surf(struct fd_batch *batch, const uint32_t *bases,
310
struct pipe_surface **bufs, uint32_t nr_bufs, uint32_t bin_w)
311
{
312
struct fd_ringbuffer *ring = batch->gmem;
313
struct pipe_surface *zsbufs[2];
314
315
emit_mrt(ring, nr_bufs, bufs, bases, bin_w, false);
316
317
if (bufs[0] && (bufs[0]->format == PIPE_FORMAT_Z32_FLOAT_S8X24_UINT)) {
318
/* The gmem_restore_tex logic will put the first buffer's stencil
319
* as color. Supply it with the proper information to make that
320
* happen.
321
*/
322
zsbufs[0] = zsbufs[1] = bufs[0];
323
bufs = zsbufs;
324
nr_bufs = 2;
325
}
326
327
fd4_emit_gmem_restore_tex(ring, nr_bufs, bufs);
328
329
fd4_draw(batch, ring, DI_PT_RECTLIST, IGNORE_VISIBILITY,
330
DI_SRC_SEL_AUTO_INDEX, 2, 1, INDEX4_SIZE_8_BIT, 0, 0, NULL);
331
}
332
333
static void
334
fd4_emit_tile_mem2gmem(struct fd_batch *batch,
335
const struct fd_tile *tile) assert_dt
336
{
337
struct fd_context *ctx = batch->ctx;
338
const struct fd_gmem_stateobj *gmem = batch->gmem_state;
339
struct fd_ringbuffer *ring = batch->gmem;
340
struct pipe_framebuffer_state *pfb = &batch->framebuffer;
341
struct fd4_emit emit = {
342
.debug = &ctx->debug,
343
.vtx = &ctx->blit_vbuf_state,
344
.sprite_coord_enable = 1,
345
.no_decode_srgb = true,
346
};
347
/* NOTE: They all use the same VP, this is for vtx bufs. */
348
fd4_gmem_emit_set_prog(ctx, &emit, &ctx->blit_prog[0]);
349
350
unsigned char mrt_comp[A4XX_MAX_RENDER_TARGETS] = {0};
351
float x0, y0, x1, y1;
352
unsigned bin_w = tile->bin_w;
353
unsigned bin_h = tile->bin_h;
354
unsigned i;
355
356
/* write texture coordinates to vertexbuf: */
357
x0 = ((float)tile->xoff) / ((float)pfb->width);
358
x1 = ((float)tile->xoff + bin_w) / ((float)pfb->width);
359
y0 = ((float)tile->yoff) / ((float)pfb->height);
360
y1 = ((float)tile->yoff + bin_h) / ((float)pfb->height);
361
362
OUT_PKT3(ring, CP_MEM_WRITE, 5);
363
OUT_RELOC(ring, fd_resource(ctx->blit_texcoord_vbuf)->bo, 0, 0, 0);
364
OUT_RING(ring, fui(x0));
365
OUT_RING(ring, fui(y0));
366
OUT_RING(ring, fui(x1));
367
OUT_RING(ring, fui(y1));
368
369
for (i = 0; i < A4XX_MAX_RENDER_TARGETS; i++) {
370
mrt_comp[i] = ((i < pfb->nr_cbufs) && pfb->cbufs[i]) ? 0xf : 0;
371
372
OUT_PKT0(ring, REG_A4XX_RB_MRT_CONTROL(i), 1);
373
OUT_RING(ring, A4XX_RB_MRT_CONTROL_ROP_CODE(ROP_COPY) |
374
A4XX_RB_MRT_CONTROL_COMPONENT_ENABLE(0xf));
375
376
OUT_PKT0(ring, REG_A4XX_RB_MRT_BLEND_CONTROL(i), 1);
377
OUT_RING(
378
ring,
379
A4XX_RB_MRT_BLEND_CONTROL_RGB_SRC_FACTOR(FACTOR_ONE) |
380
A4XX_RB_MRT_BLEND_CONTROL_RGB_BLEND_OPCODE(BLEND_DST_PLUS_SRC) |
381
A4XX_RB_MRT_BLEND_CONTROL_RGB_DEST_FACTOR(FACTOR_ZERO) |
382
A4XX_RB_MRT_BLEND_CONTROL_ALPHA_SRC_FACTOR(FACTOR_ONE) |
383
A4XX_RB_MRT_BLEND_CONTROL_ALPHA_BLEND_OPCODE(BLEND_DST_PLUS_SRC) |
384
A4XX_RB_MRT_BLEND_CONTROL_ALPHA_DEST_FACTOR(FACTOR_ZERO));
385
}
386
387
OUT_PKT0(ring, REG_A4XX_RB_RENDER_COMPONENTS, 1);
388
OUT_RING(ring, A4XX_RB_RENDER_COMPONENTS_RT0(mrt_comp[0]) |
389
A4XX_RB_RENDER_COMPONENTS_RT1(mrt_comp[1]) |
390
A4XX_RB_RENDER_COMPONENTS_RT2(mrt_comp[2]) |
391
A4XX_RB_RENDER_COMPONENTS_RT3(mrt_comp[3]) |
392
A4XX_RB_RENDER_COMPONENTS_RT4(mrt_comp[4]) |
393
A4XX_RB_RENDER_COMPONENTS_RT5(mrt_comp[5]) |
394
A4XX_RB_RENDER_COMPONENTS_RT6(mrt_comp[6]) |
395
A4XX_RB_RENDER_COMPONENTS_RT7(mrt_comp[7]));
396
397
OUT_PKT0(ring, REG_A4XX_RB_RENDER_CONTROL, 1);
398
OUT_RING(ring, 0x8); /* XXX RB_RENDER_CONTROL */
399
400
OUT_PKT0(ring, REG_A4XX_RB_DEPTH_CONTROL, 1);
401
OUT_RING(ring, A4XX_RB_DEPTH_CONTROL_ZFUNC(FUNC_LESS));
402
403
OUT_PKT0(ring, REG_A4XX_GRAS_CL_CLIP_CNTL, 1);
404
OUT_RING(ring, 0x280000); /* XXX GRAS_CL_CLIP_CNTL */
405
406
OUT_PKT0(ring, REG_A4XX_GRAS_SU_MODE_CONTROL, 1);
407
OUT_RING(ring, A4XX_GRAS_SU_MODE_CONTROL_LINEHALFWIDTH(0) |
408
A4XX_GRAS_SU_MODE_CONTROL_RENDERING_PASS);
409
410
OUT_PKT0(ring, REG_A4XX_GRAS_CL_VPORT_XOFFSET_0, 6);
411
OUT_RING(ring, A4XX_GRAS_CL_VPORT_XOFFSET_0((float)bin_w / 2.0));
412
OUT_RING(ring, A4XX_GRAS_CL_VPORT_XSCALE_0((float)bin_w / 2.0));
413
OUT_RING(ring, A4XX_GRAS_CL_VPORT_YOFFSET_0((float)bin_h / 2.0));
414
OUT_RING(ring, A4XX_GRAS_CL_VPORT_YSCALE_0(-(float)bin_h / 2.0));
415
OUT_RING(ring, A4XX_GRAS_CL_VPORT_ZOFFSET_0(0.0));
416
OUT_RING(ring, A4XX_GRAS_CL_VPORT_ZSCALE_0(1.0));
417
418
OUT_PKT0(ring, REG_A4XX_GRAS_SC_WINDOW_SCISSOR_BR, 2);
419
OUT_RING(ring, A4XX_GRAS_SC_WINDOW_SCISSOR_BR_X(bin_w - 1) |
420
A4XX_GRAS_SC_WINDOW_SCISSOR_BR_Y(bin_h - 1));
421
OUT_RING(ring, A4XX_GRAS_SC_WINDOW_SCISSOR_TL_X(0) |
422
A4XX_GRAS_SC_WINDOW_SCISSOR_TL_Y(0));
423
424
OUT_PKT0(ring, REG_A4XX_GRAS_SC_SCREEN_SCISSOR_TL, 2);
425
OUT_RING(ring, A4XX_GRAS_SC_SCREEN_SCISSOR_TL_X(0) |
426
A4XX_GRAS_SC_SCREEN_SCISSOR_TL_Y(0));
427
OUT_RING(ring, A4XX_GRAS_SC_SCREEN_SCISSOR_BR_X(bin_w - 1) |
428
A4XX_GRAS_SC_SCREEN_SCISSOR_BR_Y(bin_h - 1));
429
430
OUT_PKT0(ring, REG_A4XX_RB_MODE_CONTROL, 1);
431
OUT_RING(ring, A4XX_RB_MODE_CONTROL_WIDTH(gmem->bin_w) |
432
A4XX_RB_MODE_CONTROL_HEIGHT(gmem->bin_h));
433
434
OUT_PKT0(ring, REG_A4XX_RB_STENCIL_CONTROL, 2);
435
OUT_RING(ring, A4XX_RB_STENCIL_CONTROL_FUNC(FUNC_ALWAYS) |
436
A4XX_RB_STENCIL_CONTROL_FAIL(STENCIL_KEEP) |
437
A4XX_RB_STENCIL_CONTROL_ZPASS(STENCIL_KEEP) |
438
A4XX_RB_STENCIL_CONTROL_ZFAIL(STENCIL_KEEP) |
439
A4XX_RB_STENCIL_CONTROL_FUNC_BF(FUNC_ALWAYS) |
440
A4XX_RB_STENCIL_CONTROL_FAIL_BF(STENCIL_KEEP) |
441
A4XX_RB_STENCIL_CONTROL_ZPASS_BF(STENCIL_KEEP) |
442
A4XX_RB_STENCIL_CONTROL_ZFAIL_BF(STENCIL_KEEP));
443
OUT_RING(ring, 0x00000000); /* RB_STENCIL_CONTROL2 */
444
445
OUT_PKT0(ring, REG_A4XX_GRAS_SC_CONTROL, 1);
446
OUT_RING(ring, A4XX_GRAS_SC_CONTROL_RENDER_MODE(RB_RENDERING_PASS) |
447
A4XX_GRAS_SC_CONTROL_MSAA_DISABLE |
448
A4XX_GRAS_SC_CONTROL_MSAA_SAMPLES(MSAA_ONE) |
449
A4XX_GRAS_SC_CONTROL_RASTER_MODE(1));
450
451
OUT_PKT0(ring, REG_A4XX_PC_PRIM_VTX_CNTL, 1);
452
OUT_RING(ring, A4XX_PC_PRIM_VTX_CNTL_PROVOKING_VTX_LAST |
453
A4XX_PC_PRIM_VTX_CNTL_VAROUT(1));
454
455
OUT_PKT0(ring, REG_A4XX_VFD_INDEX_OFFSET, 2);
456
OUT_RING(ring, 0); /* VFD_INDEX_OFFSET */
457
OUT_RING(ring, 0); /* ??? UNKNOWN_2209 */
458
459
fd4_emit_vertex_bufs(ring, &emit);
460
461
/* for gmem pitch/base calculations, we need to use the non-
462
* truncated tile sizes:
463
*/
464
bin_w = gmem->bin_w;
465
bin_h = gmem->bin_h;
466
467
if (fd_gmem_needs_restore(batch, tile, FD_BUFFER_COLOR)) {
468
fd4_gmem_emit_set_prog(ctx, &emit, &ctx->blit_prog[pfb->nr_cbufs - 1]);
469
fd4_program_emit(ring, &emit, pfb->nr_cbufs, pfb->cbufs);
470
emit_mem2gmem_surf(batch, gmem->cbuf_base, pfb->cbufs, pfb->nr_cbufs,
471
bin_w);
472
}
473
474
if (fd_gmem_needs_restore(batch, tile,
475
FD_BUFFER_DEPTH | FD_BUFFER_STENCIL)) {
476
switch (pfb->zsbuf->format) {
477
case PIPE_FORMAT_Z32_FLOAT_S8X24_UINT:
478
case PIPE_FORMAT_Z32_FLOAT:
479
if (pfb->zsbuf->format == PIPE_FORMAT_Z32_FLOAT)
480
fd4_gmem_emit_set_prog(ctx, &emit, &ctx->blit_z);
481
else
482
fd4_gmem_emit_set_prog(ctx, &emit, &ctx->blit_zs);
483
484
OUT_PKT0(ring, REG_A4XX_RB_DEPTH_CONTROL, 1);
485
OUT_RING(ring, A4XX_RB_DEPTH_CONTROL_Z_ENABLE |
486
A4XX_RB_DEPTH_CONTROL_Z_WRITE_ENABLE |
487
A4XX_RB_DEPTH_CONTROL_ZFUNC(FUNC_ALWAYS) |
488
A4XX_RB_DEPTH_CONTROL_EARLY_Z_DISABLE);
489
490
OUT_PKT0(ring, REG_A4XX_GRAS_ALPHA_CONTROL, 1);
491
OUT_RING(ring, A4XX_GRAS_ALPHA_CONTROL_ALPHA_TEST_ENABLE);
492
493
OUT_PKT0(ring, REG_A4XX_GRAS_CL_CLIP_CNTL, 1);
494
OUT_RING(ring, 0x80000); /* GRAS_CL_CLIP_CNTL */
495
496
break;
497
default:
498
/* Non-float can use a regular color write. It's split over 8-bit
499
* components, so half precision is always sufficient.
500
*/
501
fd4_gmem_emit_set_prog(ctx, &emit, &ctx->blit_prog[0]);
502
break;
503
}
504
fd4_program_emit(ring, &emit, 1, &pfb->zsbuf);
505
emit_mem2gmem_surf(batch, gmem->zsbuf_base, &pfb->zsbuf, 1, bin_w);
506
}
507
508
OUT_PKT0(ring, REG_A4XX_GRAS_SC_CONTROL, 1);
509
OUT_RING(ring, A4XX_GRAS_SC_CONTROL_RENDER_MODE(RB_RENDERING_PASS) |
510
A4XX_GRAS_SC_CONTROL_MSAA_SAMPLES(MSAA_ONE) |
511
A4XX_GRAS_SC_CONTROL_RASTER_MODE(0));
512
513
OUT_PKT0(ring, REG_A4XX_RB_MODE_CONTROL, 1);
514
OUT_RING(ring, A4XX_RB_MODE_CONTROL_WIDTH(gmem->bin_w) |
515
A4XX_RB_MODE_CONTROL_HEIGHT(gmem->bin_h) |
516
0x00010000); /* XXX */
517
}
518
519
static void
520
patch_draws(struct fd_batch *batch, enum pc_di_vis_cull_mode vismode)
521
{
522
unsigned i;
523
for (i = 0; i < fd_patch_num_elements(&batch->draw_patches); i++) {
524
struct fd_cs_patch *patch = fd_patch_element(&batch->draw_patches, i);
525
*patch->cs = patch->val | DRAW4(0, 0, 0, vismode);
526
}
527
util_dynarray_clear(&batch->draw_patches);
528
}
529
530
/* for rendering directly to system memory: */
531
static void
532
fd4_emit_sysmem_prep(struct fd_batch *batch) assert_dt
533
{
534
struct pipe_framebuffer_state *pfb = &batch->framebuffer;
535
struct fd_ringbuffer *ring = batch->gmem;
536
537
fd4_emit_restore(batch, ring);
538
539
OUT_PKT0(ring, REG_A4XX_RB_FRAME_BUFFER_DIMENSION, 1);
540
OUT_RING(ring, A4XX_RB_FRAME_BUFFER_DIMENSION_WIDTH(pfb->width) |
541
A4XX_RB_FRAME_BUFFER_DIMENSION_HEIGHT(pfb->height));
542
543
emit_mrt(ring, pfb->nr_cbufs, pfb->cbufs, NULL, 0, true);
544
545
/* setup scissor/offset for current tile: */
546
OUT_PKT0(ring, REG_A4XX_RB_BIN_OFFSET, 1);
547
OUT_RING(ring, A4XX_RB_BIN_OFFSET_X(0) | A4XX_RB_BIN_OFFSET_Y(0));
548
549
OUT_PKT0(ring, REG_A4XX_GRAS_SC_SCREEN_SCISSOR_TL, 2);
550
OUT_RING(ring, A4XX_GRAS_SC_SCREEN_SCISSOR_TL_X(0) |
551
A4XX_GRAS_SC_SCREEN_SCISSOR_TL_Y(0));
552
OUT_RING(ring, A4XX_GRAS_SC_SCREEN_SCISSOR_BR_X(pfb->width - 1) |
553
A4XX_GRAS_SC_SCREEN_SCISSOR_BR_Y(pfb->height - 1));
554
555
OUT_PKT0(ring, REG_A4XX_RB_MODE_CONTROL, 1);
556
OUT_RING(ring, A4XX_RB_MODE_CONTROL_WIDTH(0) |
557
A4XX_RB_MODE_CONTROL_HEIGHT(0) | 0x00c00000); /* XXX */
558
559
OUT_PKT0(ring, REG_A4XX_RB_RENDER_CONTROL, 1);
560
OUT_RING(ring, 0x8);
561
562
patch_draws(batch, IGNORE_VISIBILITY);
563
}
564
565
static void
566
update_vsc_pipe(struct fd_batch *batch) assert_dt
567
{
568
struct fd_context *ctx = batch->ctx;
569
const struct fd_gmem_stateobj *gmem = batch->gmem_state;
570
struct fd4_context *fd4_ctx = fd4_context(ctx);
571
struct fd_ringbuffer *ring = batch->gmem;
572
int i;
573
574
OUT_PKT0(ring, REG_A4XX_VSC_SIZE_ADDRESS, 1);
575
OUT_RELOC(ring, fd4_ctx->vsc_size_mem, 0, 0, 0); /* VSC_SIZE_ADDRESS */
576
577
OUT_PKT0(ring, REG_A4XX_VSC_PIPE_CONFIG_REG(0), 8);
578
for (i = 0; i < 8; i++) {
579
const struct fd_vsc_pipe *pipe = &gmem->vsc_pipe[i];
580
OUT_RING(ring, A4XX_VSC_PIPE_CONFIG_REG_X(pipe->x) |
581
A4XX_VSC_PIPE_CONFIG_REG_Y(pipe->y) |
582
A4XX_VSC_PIPE_CONFIG_REG_W(pipe->w) |
583
A4XX_VSC_PIPE_CONFIG_REG_H(pipe->h));
584
}
585
586
OUT_PKT0(ring, REG_A4XX_VSC_PIPE_DATA_ADDRESS_REG(0), 8);
587
for (i = 0; i < 8; i++) {
588
if (!ctx->vsc_pipe_bo[i]) {
589
ctx->vsc_pipe_bo[i] = fd_bo_new(
590
ctx->dev, 0x40000, 0, "vsc_pipe[%u]", i);
591
}
592
OUT_RELOC(ring, ctx->vsc_pipe_bo[i], 0, 0,
593
0); /* VSC_PIPE_DATA_ADDRESS[i] */
594
}
595
596
OUT_PKT0(ring, REG_A4XX_VSC_PIPE_DATA_LENGTH_REG(0), 8);
597
for (i = 0; i < 8; i++) {
598
OUT_RING(ring, fd_bo_size(ctx->vsc_pipe_bo[i]) -
599
32); /* VSC_PIPE_DATA_LENGTH[i] */
600
}
601
}
602
603
static void
604
emit_binning_pass(struct fd_batch *batch) assert_dt
605
{
606
const struct fd_gmem_stateobj *gmem = batch->gmem_state;
607
struct pipe_framebuffer_state *pfb = &batch->framebuffer;
608
struct fd_ringbuffer *ring = batch->gmem;
609
int i;
610
611
uint32_t x1 = gmem->minx;
612
uint32_t y1 = gmem->miny;
613
uint32_t x2 = gmem->minx + gmem->width - 1;
614
uint32_t y2 = gmem->miny + gmem->height - 1;
615
616
OUT_PKT0(ring, REG_A4XX_PC_BINNING_COMMAND, 1);
617
OUT_RING(ring, A4XX_PC_BINNING_COMMAND_BINNING_ENABLE);
618
619
OUT_PKT0(ring, REG_A4XX_GRAS_SC_CONTROL, 1);
620
OUT_RING(ring, A4XX_GRAS_SC_CONTROL_RENDER_MODE(RB_TILING_PASS) |
621
A4XX_GRAS_SC_CONTROL_MSAA_DISABLE |
622
A4XX_GRAS_SC_CONTROL_MSAA_SAMPLES(MSAA_ONE) |
623
A4XX_GRAS_SC_CONTROL_RASTER_MODE(0));
624
625
OUT_PKT0(ring, REG_A4XX_RB_FRAME_BUFFER_DIMENSION, 1);
626
OUT_RING(ring, A4XX_RB_FRAME_BUFFER_DIMENSION_WIDTH(pfb->width) |
627
A4XX_RB_FRAME_BUFFER_DIMENSION_HEIGHT(pfb->height));
628
629
/* setup scissor/offset for whole screen: */
630
OUT_PKT0(ring, REG_A4XX_RB_BIN_OFFSET, 1);
631
OUT_RING(ring, A4XX_RB_BIN_OFFSET_X(x1) | A4XX_RB_BIN_OFFSET_Y(y1));
632
633
OUT_PKT0(ring, REG_A4XX_GRAS_SC_SCREEN_SCISSOR_TL, 2);
634
OUT_RING(ring, A4XX_GRAS_SC_SCREEN_SCISSOR_TL_X(x1) |
635
A4XX_GRAS_SC_SCREEN_SCISSOR_TL_Y(y1));
636
OUT_RING(ring, A4XX_GRAS_SC_SCREEN_SCISSOR_BR_X(x2) |
637
A4XX_GRAS_SC_SCREEN_SCISSOR_BR_Y(y2));
638
639
for (i = 0; i < A4XX_MAX_RENDER_TARGETS; i++) {
640
OUT_PKT0(ring, REG_A4XX_RB_MRT_CONTROL(i), 1);
641
OUT_RING(ring, A4XX_RB_MRT_CONTROL_ROP_CODE(ROP_CLEAR) |
642
A4XX_RB_MRT_CONTROL_COMPONENT_ENABLE(0xf));
643
}
644
645
/* emit IB to binning drawcmds: */
646
fd4_emit_ib(ring, batch->binning);
647
648
fd_reset_wfi(batch);
649
fd_wfi(batch, ring);
650
651
/* and then put stuff back the way it was: */
652
653
OUT_PKT0(ring, REG_A4XX_PC_BINNING_COMMAND, 1);
654
OUT_RING(ring, 0x00000000);
655
656
OUT_PKT0(ring, REG_A4XX_GRAS_SC_CONTROL, 1);
657
OUT_RING(ring, A4XX_GRAS_SC_CONTROL_RENDER_MODE(RB_RENDERING_PASS) |
658
A4XX_GRAS_SC_CONTROL_MSAA_DISABLE |
659
A4XX_GRAS_SC_CONTROL_MSAA_SAMPLES(MSAA_ONE) |
660
A4XX_GRAS_SC_CONTROL_RASTER_MODE(0));
661
662
fd_event_write(batch, ring, CACHE_FLUSH);
663
fd_wfi(batch, ring);
664
}
665
666
/* before first tile */
667
static void
668
fd4_emit_tile_init(struct fd_batch *batch) assert_dt
669
{
670
struct fd_ringbuffer *ring = batch->gmem;
671
struct pipe_framebuffer_state *pfb = &batch->framebuffer;
672
const struct fd_gmem_stateobj *gmem = batch->gmem_state;
673
674
fd4_emit_restore(batch, ring);
675
676
OUT_PKT0(ring, REG_A4XX_VSC_BIN_SIZE, 1);
677
OUT_RING(ring, A4XX_VSC_BIN_SIZE_WIDTH(gmem->bin_w) |
678
A4XX_VSC_BIN_SIZE_HEIGHT(gmem->bin_h));
679
680
update_vsc_pipe(batch);
681
682
fd_wfi(batch, ring);
683
OUT_PKT0(ring, REG_A4XX_RB_FRAME_BUFFER_DIMENSION, 1);
684
OUT_RING(ring, A4XX_RB_FRAME_BUFFER_DIMENSION_WIDTH(pfb->width) |
685
A4XX_RB_FRAME_BUFFER_DIMENSION_HEIGHT(pfb->height));
686
687
if (use_hw_binning(batch)) {
688
OUT_PKT0(ring, REG_A4XX_RB_MODE_CONTROL, 1);
689
OUT_RING(ring, A4XX_RB_MODE_CONTROL_WIDTH(gmem->bin_w) |
690
A4XX_RB_MODE_CONTROL_HEIGHT(gmem->bin_h));
691
692
OUT_PKT0(ring, REG_A4XX_RB_RENDER_CONTROL, 1);
693
OUT_RING(ring, A4XX_RB_RENDER_CONTROL_BINNING_PASS |
694
A4XX_RB_RENDER_CONTROL_DISABLE_COLOR_PIPE | 0x8);
695
696
/* emit hw binning pass: */
697
emit_binning_pass(batch);
698
699
patch_draws(batch, USE_VISIBILITY);
700
} else {
701
patch_draws(batch, IGNORE_VISIBILITY);
702
}
703
704
OUT_PKT0(ring, REG_A4XX_RB_MODE_CONTROL, 1);
705
OUT_RING(ring, A4XX_RB_MODE_CONTROL_WIDTH(gmem->bin_w) |
706
A4XX_RB_MODE_CONTROL_HEIGHT(gmem->bin_h) |
707
A4XX_RB_MODE_CONTROL_ENABLE_GMEM);
708
}
709
710
/* before mem2gmem */
711
static void
712
fd4_emit_tile_prep(struct fd_batch *batch, const struct fd_tile *tile)
713
{
714
struct fd_ringbuffer *ring = batch->gmem;
715
struct pipe_framebuffer_state *pfb = &batch->framebuffer;
716
const struct fd_gmem_stateobj *gmem = batch->gmem_state;
717
718
if (pfb->zsbuf) {
719
struct fd_resource *rsc = fd_resource(pfb->zsbuf->texture);
720
uint32_t cpp = rsc->layout.cpp;
721
722
OUT_PKT0(ring, REG_A4XX_RB_DEPTH_INFO, 3);
723
OUT_RING(ring, A4XX_RB_DEPTH_INFO_DEPTH_BASE(gmem->zsbuf_base[0]) |
724
A4XX_RB_DEPTH_INFO_DEPTH_FORMAT(
725
fd4_pipe2depth(pfb->zsbuf->format)));
726
OUT_RING(ring, A4XX_RB_DEPTH_PITCH(cpp * gmem->bin_w));
727
OUT_RING(ring, A4XX_RB_DEPTH_PITCH2(cpp * gmem->bin_w));
728
729
OUT_PKT0(ring, REG_A4XX_RB_STENCIL_INFO, 2);
730
if (rsc->stencil) {
731
OUT_RING(ring,
732
A4XX_RB_STENCIL_INFO_SEPARATE_STENCIL |
733
A4XX_RB_STENCIL_INFO_STENCIL_BASE(gmem->zsbuf_base[1]));
734
OUT_RING(ring, A4XX_RB_STENCIL_PITCH(rsc->stencil->layout.cpp *
735
gmem->bin_w));
736
} else {
737
OUT_RING(ring, 0x00000000);
738
OUT_RING(ring, 0x00000000);
739
}
740
} else {
741
OUT_PKT0(ring, REG_A4XX_RB_DEPTH_INFO, 3);
742
OUT_RING(ring, 0x00000000);
743
OUT_RING(ring, 0x00000000);
744
OUT_RING(ring, 0x00000000);
745
746
OUT_PKT0(ring, REG_A4XX_RB_STENCIL_INFO, 2);
747
OUT_RING(ring, 0); /* RB_STENCIL_INFO */
748
OUT_RING(ring, 0); /* RB_STENCIL_PITCH */
749
}
750
751
OUT_PKT0(ring, REG_A4XX_GRAS_DEPTH_CONTROL, 1);
752
if (pfb->zsbuf) {
753
OUT_RING(ring, A4XX_GRAS_DEPTH_CONTROL_FORMAT(
754
fd4_pipe2depth(pfb->zsbuf->format)));
755
} else {
756
OUT_RING(ring, A4XX_GRAS_DEPTH_CONTROL_FORMAT(DEPTH4_NONE));
757
}
758
}
759
760
/* before IB to rendering cmds: */
761
static void
762
fd4_emit_tile_renderprep(struct fd_batch *batch,
763
const struct fd_tile *tile) assert_dt
764
{
765
struct fd_context *ctx = batch->ctx;
766
struct fd4_context *fd4_ctx = fd4_context(ctx);
767
struct fd_ringbuffer *ring = batch->gmem;
768
const struct fd_gmem_stateobj *gmem = batch->gmem_state;
769
struct pipe_framebuffer_state *pfb = &batch->framebuffer;
770
771
uint32_t x1 = tile->xoff;
772
uint32_t y1 = tile->yoff;
773
uint32_t x2 = tile->xoff + tile->bin_w - 1;
774
uint32_t y2 = tile->yoff + tile->bin_h - 1;
775
776
if (use_hw_binning(batch)) {
777
const struct fd_vsc_pipe *pipe = &gmem->vsc_pipe[tile->p];
778
struct fd_bo *pipe_bo = ctx->vsc_pipe_bo[tile->p];
779
780
assert(pipe->w && pipe->h);
781
782
fd_event_write(batch, ring, HLSQ_FLUSH);
783
fd_wfi(batch, ring);
784
785
OUT_PKT0(ring, REG_A4XX_PC_VSTREAM_CONTROL, 1);
786
OUT_RING(ring, A4XX_PC_VSTREAM_CONTROL_SIZE(pipe->w * pipe->h) |
787
A4XX_PC_VSTREAM_CONTROL_N(tile->n));
788
789
OUT_PKT3(ring, CP_SET_BIN_DATA, 2);
790
OUT_RELOC(ring, pipe_bo, 0, 0,
791
0); /* BIN_DATA_ADDR <- VSC_PIPE[p].DATA_ADDRESS */
792
OUT_RELOC(ring, fd4_ctx->vsc_size_mem, /* BIN_SIZE_ADDR <-
793
VSC_SIZE_ADDRESS + (p * 4) */
794
(tile->p * 4), 0, 0);
795
} else {
796
OUT_PKT0(ring, REG_A4XX_PC_VSTREAM_CONTROL, 1);
797
OUT_RING(ring, 0x00000000);
798
}
799
800
OUT_PKT3(ring, CP_SET_BIN, 3);
801
OUT_RING(ring, 0x00000000);
802
OUT_RING(ring, CP_SET_BIN_1_X1(x1) | CP_SET_BIN_1_Y1(y1));
803
OUT_RING(ring, CP_SET_BIN_2_X2(x2) | CP_SET_BIN_2_Y2(y2));
804
805
emit_mrt(ring, pfb->nr_cbufs, pfb->cbufs, gmem->cbuf_base, gmem->bin_w,
806
true);
807
808
/* setup scissor/offset for current tile: */
809
OUT_PKT0(ring, REG_A4XX_RB_BIN_OFFSET, 1);
810
OUT_RING(ring, A4XX_RB_BIN_OFFSET_X(tile->xoff) |
811
A4XX_RB_BIN_OFFSET_Y(tile->yoff));
812
813
OUT_PKT0(ring, REG_A4XX_GRAS_SC_SCREEN_SCISSOR_TL, 2);
814
OUT_RING(ring, A4XX_GRAS_SC_SCREEN_SCISSOR_TL_X(x1) |
815
A4XX_GRAS_SC_SCREEN_SCISSOR_TL_Y(y1));
816
OUT_RING(ring, A4XX_GRAS_SC_SCREEN_SCISSOR_BR_X(x2) |
817
A4XX_GRAS_SC_SCREEN_SCISSOR_BR_Y(y2));
818
819
OUT_PKT0(ring, REG_A4XX_RB_RENDER_CONTROL, 1);
820
OUT_RING(ring, 0x8);
821
}
822
823
void
824
fd4_gmem_init(struct pipe_context *pctx) disable_thread_safety_analysis
825
{
826
struct fd_context *ctx = fd_context(pctx);
827
828
ctx->emit_sysmem_prep = fd4_emit_sysmem_prep;
829
ctx->emit_tile_init = fd4_emit_tile_init;
830
ctx->emit_tile_prep = fd4_emit_tile_prep;
831
ctx->emit_tile_mem2gmem = fd4_emit_tile_mem2gmem;
832
ctx->emit_tile_renderprep = fd4_emit_tile_renderprep;
833
ctx->emit_tile_gmem2mem = fd4_emit_tile_gmem2mem;
834
}
835
836