[Mesa-dev] [PATCH 11/13] gallium/upload_mgr: don't unmap buffers if persistent mappings are supported

Roland Scheidegger sroland at vmware.com
Thu Jan 30 08:33:46 PST 2014


Am 30.01.2014 02:20, schrieb Marek Olšák:
> From: Marek Olšák <marek.olsak at amd.com>
> 
> ---
>  src/gallium/auxiliary/util/u_upload_mgr.c | 65 ++++++++++++++++++++++++-------
>  1 file changed, 51 insertions(+), 14 deletions(-)
> 
> diff --git a/src/gallium/auxiliary/util/u_upload_mgr.c b/src/gallium/auxiliary/util/u_upload_mgr.c
> index 7349d00..ccaab27 100644
> --- a/src/gallium/auxiliary/util/u_upload_mgr.c
> +++ b/src/gallium/auxiliary/util/u_upload_mgr.c
> @@ -44,6 +44,8 @@ struct u_upload_mgr {
>     unsigned default_size;  /* Minimum size of the upload buffer, in bytes. */
>     unsigned alignment;     /* Alignment of each sub-allocation. */
>     unsigned bind;          /* Bitmask of PIPE_BIND_* flags. */
> +   unsigned map_flags;     /* Bitmask of PIPE_TRANSFER_* flags. */
> +   boolean map_persistent; /* If persistent mappings are supported. */
>  
>     struct pipe_resource *buffer;   /* Upload buffer. */
>     struct pipe_transfer *transfer; /* Transfer object for the upload buffer. */
> @@ -67,20 +69,39 @@ struct u_upload_mgr *u_upload_create( struct pipe_context *pipe,
>     upload->default_size = default_size;
>     upload->alignment = alignment;
>     upload->bind = bind;
> -   upload->buffer = NULL;
> +
> +   upload->map_persistent =
> +      pipe->screen->get_param(pipe->screen,
> +                              PIPE_CAP_BUFFER_TRANSFER_PERSISTENT_COHERENT);
> +
> +   if (upload->map_persistent) {
> +      upload->map_flags = PIPE_TRANSFER_WRITE |
> +                          PIPE_TRANSFER_PERSISTENT |
> +                          PIPE_TRANSFER_COHERENT;
> +   }
> +   else {
> +      upload->map_flags = PIPE_TRANSFER_WRITE |
> +                          PIPE_TRANSFER_UNSYNCHRONIZED |
> +                          PIPE_TRANSFER_FLUSH_EXPLICIT;
> +   }
>  
>     return upload;
>  }
>  
> -void u_upload_unmap( struct u_upload_mgr *upload )
> +
> +static void upload_unmap_internal(struct u_upload_mgr *upload, boolean destroying)
>  {
> +   if (!destroying && upload->map_persistent)
> +      return;
> +
>     if (upload->transfer) {
>        struct pipe_box *box = &upload->transfer->box;
> -      if ((int) upload->offset > box->x) {
>  
> +      if (!upload->map_persistent && (int) upload->offset > box->x) {
>           pipe_buffer_flush_mapped_range(upload->pipe, upload->transfer,
>                                          box->x, upload->offset - box->x);
>        }
> +
>        pipe_transfer_unmap(upload->pipe, upload->transfer);
>        upload->transfer = NULL;
>        upload->map = NULL;
> @@ -88,10 +109,16 @@ void u_upload_unmap( struct u_upload_mgr *upload )
>  }
>  
>  
> +void u_upload_unmap( struct u_upload_mgr *upload )
> +{
> +   upload_unmap_internal(upload, FALSE);
> +}
> +
> +
>  static void u_upload_release_buffer(struct u_upload_mgr *upload)
>  {
>     /* Unmap and unreference the upload buffer. */
> -   u_upload_unmap(upload);
> +   upload_unmap_internal(upload, TRUE);
>     pipe_resource_reference( &upload->buffer, NULL );
>     upload->size = 0;
>  }
> @@ -108,6 +135,8 @@ static enum pipe_error
>  u_upload_alloc_buffer( struct u_upload_mgr *upload,
>                         unsigned min_size )
>  {
> +   struct pipe_screen *screen = upload->pipe->screen;
> +   struct pipe_resource buffer;
>     unsigned size;
>  
>     /* Release the old buffer, if present:
> @@ -118,19 +147,29 @@ u_upload_alloc_buffer( struct u_upload_mgr *upload,
>      */
>     size = align(MAX2(upload->default_size, min_size), 4096);
>  
> -   upload->buffer = pipe_buffer_create( upload->pipe->screen,
> -                                        upload->bind,
> -                                        PIPE_USAGE_STREAM,
> -                                        size );
> +   memset(&buffer, 0, sizeof buffer);
> +   buffer.target = PIPE_BUFFER;
> +   buffer.format = PIPE_FORMAT_R8_UNORM; /* want TYPELESS or similar */
> +   buffer.bind = upload->bind;
> +   buffer.usage = PIPE_USAGE_STREAM;
> +   buffer.width0 = size;
> +   buffer.height0 = 1;
> +   buffer.depth0 = 1;
> +   buffer.array_size = 1;
> +
> +   if (upload->map_persistent) {
> +      buffer.flags = PIPE_RESOURCE_FLAG_TRANSFER_PERSISTENT |
> +                     PIPE_RESOURCE_FLAG_TRANSFER_COHERENT;
> +   }
> +
> +   upload->buffer = screen->resource_create(screen, &buffer);
>     if (upload->buffer == NULL) {
>        return PIPE_ERROR_OUT_OF_MEMORY;
>     }
>  
>     /* Map the new buffer. */
>     upload->map = pipe_buffer_map_range(upload->pipe, upload->buffer,
> -                                       0, size,
> -                                       PIPE_TRANSFER_WRITE |
> -                                       PIPE_TRANSFER_FLUSH_EXPLICIT,
> +                                       0, size, upload->map_flags,
>                                         &upload->transfer);
>     if (upload->map == NULL) {
>        upload->transfer = NULL;
> @@ -176,9 +215,7 @@ enum pipe_error u_upload_alloc( struct u_upload_mgr *upload,
>     if (!upload->map) {
>        upload->map = pipe_buffer_map_range(upload->pipe, upload->buffer,
>  					  offset, upload->size - offset,
> -					  PIPE_TRANSFER_WRITE |
> -					  PIPE_TRANSFER_FLUSH_EXPLICIT |
> -					  PIPE_TRANSFER_UNSYNCHRONIZED,
> +                                          upload->map_flags,
>  					  &upload->transfer);
>        if (!upload->map) {
>           upload->transfer = NULL;
> 

Is this actually a win over just remapping unsynchronized buffers here?
Just curious...

Roland


More information about the mesa-dev mailing list