1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
|
#include "util/u_inlines.h"
#include "util/u_memory.h"
#include "util/u_math.h"
#include "nouveau/nouveau_screen.h"
#include "nouveau/nouveau_winsys.h"
#include "nvfx_resource.h"
#include "nvfx_screen.h"
void nvfx_buffer_destroy(struct pipe_screen *pscreen,
struct pipe_resource *presource)
{
struct nvfx_buffer *buffer = nvfx_buffer(presource);
if(!(buffer->base.base.flags & NVFX_RESOURCE_FLAG_USER))
align_free(buffer->data);
nouveau_screen_bo_release(pscreen, buffer->base.bo);
FREE(buffer);
}
struct pipe_resource *
nvfx_buffer_create(struct pipe_screen *pscreen,
const struct pipe_resource *template)
{
struct nvfx_screen* screen = nvfx_screen(pscreen);
struct nvfx_buffer* buffer;
buffer = CALLOC_STRUCT(nvfx_buffer);
if (!buffer)
return NULL;
buffer->base.base = *template;
buffer->base.base.flags |= NOUVEAU_RESOURCE_FLAG_LINEAR;
pipe_reference_init(&buffer->base.base.reference, 1);
buffer->base.base.screen = pscreen;
buffer->size = util_format_get_stride(template->format, template->width0);
buffer->bytes_to_draw_until_static = buffer->size * screen->static_reuse_threshold;
buffer->data = align_malloc(buffer->size, 16);
return &buffer->base.base;
}
struct pipe_resource *
nvfx_user_buffer_create(struct pipe_screen *pscreen,
void *ptr,
unsigned bytes,
unsigned usage)
{
struct nvfx_screen* screen = nvfx_screen(pscreen);
struct nvfx_buffer* buffer;
buffer = CALLOC_STRUCT(nvfx_buffer);
if (!buffer)
return NULL;
pipe_reference_init(&buffer->base.base.reference, 1);
buffer->base.base.flags =
NOUVEAU_RESOURCE_FLAG_LINEAR | NVFX_RESOURCE_FLAG_USER;
buffer->base.base.screen = pscreen;
buffer->base.base.format = PIPE_FORMAT_R8_UNORM;
buffer->base.base.usage = PIPE_USAGE_IMMUTABLE;
buffer->base.base.bind = usage;
buffer->base.base.width0 = bytes;
buffer->base.base.height0 = 1;
buffer->base.base.depth0 = 1;
buffer->base.base.array_size = 1;
buffer->data = ptr;
buffer->size = bytes;
buffer->bytes_to_draw_until_static = bytes * screen->static_reuse_threshold;
buffer->dirty_end = bytes;
return &buffer->base.base;
}
void nvfx_buffer_upload(struct nvfx_buffer* buffer)
{
unsigned dirty = buffer->dirty_end - buffer->dirty_begin;
if(!buffer->base.bo)
{
buffer->base.bo = nouveau_screen_bo_new(buffer->base.base.screen,
16,
buffer->base.base.usage,
buffer->base.base.bind,
buffer->base.base.width0);
}
if(dirty)
{
// TODO: may want to use a temporary in some cases
nouveau_bo_map(buffer->base.bo, NOUVEAU_BO_WR
| (buffer->dirty_unsynchronized ? NOUVEAU_BO_NOSYNC : 0));
memcpy((uint8_t*)buffer->base.bo->map + buffer->dirty_begin, buffer->data + buffer->dirty_begin, dirty);
nouveau_bo_unmap(buffer->base.bo);
buffer->dirty_begin = buffer->dirty_end = 0;
}
}
|