diff options
Diffstat (limited to 'src/gallium/drivers/virgl')
-rw-r--r-- | src/gallium/drivers/virgl/virgl_buffer.c | 3 | ||||
-rw-r--r-- | src/gallium/drivers/virgl/virgl_context.c | 3 | ||||
-rw-r--r-- | src/gallium/drivers/virgl/virgl_encode.c | 11 | ||||
-rw-r--r-- | src/gallium/drivers/virgl/virgl_query.c | 5 | ||||
-rw-r--r-- | src/gallium/drivers/virgl/virgl_resource.c | 50 | ||||
-rw-r--r-- | src/gallium/drivers/virgl/virgl_resource.h | 3 | ||||
-rw-r--r-- | src/gallium/drivers/virgl/virgl_streamout.c | 4 |
7 files changed, 59 insertions, 20 deletions
diff --git a/src/gallium/drivers/virgl/virgl_buffer.c b/src/gallium/drivers/virgl/virgl_buffer.c index fce1ac496e7..284eeca4d8f 100644 --- a/src/gallium/drivers/virgl/virgl_buffer.c +++ b/src/gallium/drivers/virgl/virgl_buffer.c @@ -59,6 +59,9 @@ static void *virgl_buffer_transfer_map(struct pipe_context *ctx, return NULL; } + if (usage & PIPE_TRANSFER_WRITE) + util_range_add(&vbuf->valid_buffer_range, box->x, box->x + box->width); + *transfer = &trans->base; return trans->hw_res_map + trans->offset; } diff --git a/src/gallium/drivers/virgl/virgl_context.c b/src/gallium/drivers/virgl/virgl_context.c index 25f841a2b51..51055370346 100644 --- a/src/gallium/drivers/virgl/virgl_context.c +++ b/src/gallium/drivers/virgl/virgl_context.c @@ -954,7 +954,10 @@ static void virgl_resource_copy_region(struct pipe_context *ctx, struct virgl_resource *dres = virgl_resource(dst); struct virgl_resource *sres = virgl_resource(src); + if (dres->u.b.target == PIPE_BUFFER) + util_range_add(&dres->valid_buffer_range, dstx, dstx + src_box->width); virgl_resource_dirty(dres, dst_level); + virgl_encode_resource_copy_region(vctx, dres, dst_level, dstx, dsty, dstz, sres, src_level, diff --git a/src/gallium/drivers/virgl/virgl_encode.c b/src/gallium/drivers/virgl/virgl_encode.c index ee524c883d9..9c60b99bfb6 100644 --- a/src/gallium/drivers/virgl/virgl_encode.c +++ b/src/gallium/drivers/virgl/virgl_encode.c @@ -965,6 +965,9 @@ int virgl_encode_set_shader_buffers(struct virgl_context *ctx, virgl_encoder_write_dword(ctx->cbuf, buffers[i].buffer_offset); virgl_encoder_write_dword(ctx->cbuf, buffers[i].buffer_size); virgl_encoder_write_res(ctx, res); + + util_range_add(&res->valid_buffer_range, buffers[i].buffer_offset, + buffers[i].buffer_offset + buffers[i].buffer_size); virgl_resource_dirty(res, 0); } else { virgl_encoder_write_dword(ctx->cbuf, 0); @@ -989,6 +992,9 @@ int virgl_encode_set_hw_atomic_buffers(struct virgl_context *ctx, virgl_encoder_write_dword(ctx->cbuf, buffers[i].buffer_offset); virgl_encoder_write_dword(ctx->cbuf, buffers[i].buffer_size); virgl_encoder_write_res(ctx, res); + + util_range_add(&res->valid_buffer_range, buffers[i].buffer_offset, + buffers[i].buffer_offset + buffers[i].buffer_size); virgl_resource_dirty(res, 0); } else { virgl_encoder_write_dword(ctx->cbuf, 0); @@ -1017,6 +1023,11 @@ int virgl_encode_set_shader_images(struct virgl_context *ctx, virgl_encoder_write_dword(ctx->cbuf, images[i].u.buf.offset); virgl_encoder_write_dword(ctx->cbuf, images[i].u.buf.size); virgl_encoder_write_res(ctx, res); + + if (res->u.b.target == PIPE_BUFFER) { + util_range_add(&res->valid_buffer_range, images[i].u.buf.offset, + images[i].u.buf.offset + images[i].u.buf.size); + } virgl_resource_dirty(res, images[i].u.tex.level); } else { virgl_encoder_write_dword(ctx->cbuf, 0); diff --git a/src/gallium/drivers/virgl/virgl_query.c b/src/gallium/drivers/virgl/virgl_query.c index d69842b4427..df571ec4905 100644 --- a/src/gallium/drivers/virgl/virgl_query.c +++ b/src/gallium/drivers/virgl/virgl_query.c @@ -114,6 +114,10 @@ static struct pipe_query *virgl_create_query(struct pipe_context *ctx, query->result_size = (query_type == PIPE_QUERY_TIMESTAMP || query_type == PIPE_QUERY_TIME_ELAPSED) ? 8 : 4; + util_range_add(&query->buf->valid_buffer_range, 0, + sizeof(struct virgl_host_query_state)); + virgl_resource_dirty(query->buf, 0); + virgl_encoder_create_query(vctx, query->handle, pipe_to_virgl_query(query_type), index, query->buf, 0); @@ -156,7 +160,6 @@ static bool virgl_end_query(struct pipe_context *ctx, return false; host_state->query_state = VIRGL_QUERY_STATE_WAIT_HOST; - virgl_resource_dirty(query->buf, 0); query->ready = false; virgl_encoder_end_query(vctx, query->handle); diff --git a/src/gallium/drivers/virgl/virgl_resource.c b/src/gallium/drivers/virgl/virgl_resource.c index eaa50445d4f..5d639ce64fd 100644 --- a/src/gallium/drivers/virgl/virgl_resource.c +++ b/src/gallium/drivers/virgl/virgl_resource.c @@ -34,9 +34,6 @@ * - the resource is not referenced by the current cmdbuf * - the current cmdbuf has no draw/compute command that accesses the * resource (XXX there are also clear or blit commands) - * - the transfer is to an undefined region and we can assume the current - * cmdbuf has no command that accesses the region (XXX we cannot just check - * for overlapping transfers) */ static bool virgl_res_needs_flush(struct virgl_context *vctx, struct virgl_transfer *trans) @@ -61,19 +58,6 @@ static bool virgl_res_needs_flush(struct virgl_context *vctx, */ if (vctx->num_draws == 0 && vctx->num_compute == 0) return false; - - /* XXX Consider - * - * glBufferSubData(GL_ARRAY_BUFFER, 0, sizeof(float) * 3, data1); - * glFlush(); - * glDrawArrays(GL_TRIANGLES, 0, 3); - * glBufferSubData(GL_ARRAY_BUFFER, 0, sizeof(float) * 3, data2); - * glDrawArrays(GL_TRIANGLES, 0, 3); - * - * Both draws will see data2. - */ - if (!virgl_transfer_queue_is_queued(&vctx->queue, trans)) - return false; } return true; @@ -122,6 +106,26 @@ virgl_resource_transfer_prepare(struct virgl_context *vctx, readback = virgl_res_needs_readback(vctx, res, xfer->base.usage, xfer->base.level); + /* We need to wait for all cmdbufs, current or previous, that access the + * resource to finish, unless synchronization is disabled. Readback, which + * is yet another command and is transparent to the state trackers, should + * also be waited for. + */ + wait = !(xfer->base.usage & PIPE_TRANSFER_UNSYNCHRONIZED) || readback; + + /* When the transfer range consists of only uninitialized data, we can + * assume the GPU is not accessing the range and readback is unnecessary. + * We can proceed as if PIPE_TRANSFER_UNSYNCHRONIZED and + * PIPE_TRANSFER_DISCARD_RANGE are set. + */ + if (res->u.b.target == PIPE_BUFFER && + !util_ranges_intersect(&res->valid_buffer_range, xfer->base.box.x, + xfer->base.box.x + xfer->base.box.width)) { + flush = false; + readback = false; + wait = false; + } + /* XXX This is incorrect. Consider * * glTexImage2D(..., data1); @@ -185,10 +189,12 @@ static struct pipe_resource *virgl_resource_create(struct pipe_screen *screen, res->clean_mask = (1 << VR_MAX_TEXTURE_2D_LEVELS) - 1; - if (templ->target == PIPE_BUFFER) + if (templ->target == PIPE_BUFFER) { + util_range_init(&res->valid_buffer_range); virgl_buffer_init(res); - else + } else { virgl_texture_init(res); + } return &res->u.b; @@ -252,7 +258,8 @@ static bool virgl_buffer_transfer_extend(struct pipe_context *ctx, dummy_trans.offset = box->x; flush = virgl_res_needs_flush(vctx, &dummy_trans); - if (flush) + if (flush && util_ranges_intersect(&vbuf->valid_buffer_range, + box->x, box->x + box->width)) return false; queued = virgl_transfer_queue_extend(&vctx->queue, &dummy_trans); @@ -260,6 +267,7 @@ static bool virgl_buffer_transfer_extend(struct pipe_context *ctx, return false; memcpy(queued->hw_res_map + dummy_trans.offset, data, box->width); + util_range_add(&vbuf->valid_buffer_range, box->x, box->x + box->width); return true; } @@ -410,6 +418,10 @@ void virgl_resource_destroy(struct pipe_screen *screen, { struct virgl_screen *vs = virgl_screen(screen); struct virgl_resource *res = virgl_resource(resource); + + if (res->u.b.target == PIPE_BUFFER) + util_range_destroy(&res->valid_buffer_range); + vs->vws->resource_unref(vs->vws, res->hw_res); FREE(res); } diff --git a/src/gallium/drivers/virgl/virgl_resource.h b/src/gallium/drivers/virgl/virgl_resource.h index c17c753a596..f9a652367c2 100644 --- a/src/gallium/drivers/virgl/virgl_resource.h +++ b/src/gallium/drivers/virgl/virgl_resource.h @@ -50,6 +50,9 @@ struct virgl_resource { uint16_t clean_mask; struct virgl_hw_res *hw_res; struct virgl_resource_metadata metadata; + + /* For PIPE_BUFFER only. Data outside of this range are uninitialized. */ + struct util_range valid_buffer_range; }; enum virgl_transfer_map_type { diff --git a/src/gallium/drivers/virgl/virgl_streamout.c b/src/gallium/drivers/virgl/virgl_streamout.c index a467a6a3d02..125973d29fb 100644 --- a/src/gallium/drivers/virgl/virgl_streamout.c +++ b/src/gallium/drivers/virgl/virgl_streamout.c @@ -48,7 +48,11 @@ static struct pipe_stream_output_target *virgl_create_so_target( t->base.buffer_offset = buffer_offset; t->base.buffer_size = buffer_size; t->handle = handle; + + util_range_add(&res->valid_buffer_range, buffer_offset, + buffer_offset + buffer_size); virgl_resource_dirty(res, 0); + virgl_encoder_create_so_target(vctx, handle, res, buffer_offset, buffer_size); return &t->base; } |