summaryrefslogtreecommitdiffstats
path: root/src/gallium/drivers/nvfx/nvfx_buffer.c
blob: 76b7f4507c45f15485ccb79ff04d9134366809c6 (plain)
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;
	}
}