Merge remote branch 'origin/master' into pipe-video
[mesa.git] / src / gallium / drivers / nouveau / nouveau_fence.c
1 /*
2 * Copyright 2010 Christoph Bumiller
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 shall be included in
12 * all copies or substantial portions of the Software.
13 *
14 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
17 * THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
18 * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF
19 * OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
20 * SOFTWARE.
21 */
22
23 #include "util/u_double_list.h"
24
25 #include "nouveau_screen.h"
26 #include "nouveau_fence.h"
27
28 #include "nouveau/nouveau_pushbuf.h"
29
30 #ifdef PIPE_OS_UNIX
31 #include <sched.h>
32 #endif
33
34 boolean
35 nouveau_fence_new(struct nouveau_screen *screen, struct nouveau_fence **fence,
36 boolean emit)
37 {
38 *fence = CALLOC_STRUCT(nouveau_fence);
39 if (!*fence)
40 return FALSE;
41
42 (*fence)->screen = screen;
43 (*fence)->ref = 1;
44 LIST_INITHEAD(&(*fence)->work);
45
46 if (emit)
47 nouveau_fence_emit(*fence);
48
49 return TRUE;
50 }
51
52 static void
53 nouveau_fence_trigger_work(struct nouveau_fence *fence)
54 {
55 struct nouveau_fence_work *work, *tmp;
56
57 LIST_FOR_EACH_ENTRY_SAFE(work, tmp, &fence->work, list) {
58 work->func(work->data);
59 LIST_DEL(&work->list);
60 FREE(work);
61 }
62 }
63
64 boolean
65 nouveau_fence_work(struct nouveau_fence *fence,
66 void (*func)(void *), void *data)
67 {
68 struct nouveau_fence_work *work;
69
70 if (!fence || fence->state == NOUVEAU_FENCE_STATE_SIGNALLED) {
71 func(data);
72 return TRUE;
73 }
74
75 work = CALLOC_STRUCT(nouveau_fence_work);
76 if (!work)
77 return FALSE;
78 work->func = func;
79 work->data = data;
80 LIST_ADD(&work->list, &fence->work);
81 return TRUE;
82 }
83
84 void
85 nouveau_fence_emit(struct nouveau_fence *fence)
86 {
87 struct nouveau_screen *screen = fence->screen;
88
89 fence->sequence = ++screen->fence.sequence;
90
91 assert(fence->state == NOUVEAU_FENCE_STATE_AVAILABLE);
92
93 screen->fence.emit(&screen->base, fence->sequence);
94
95 ++fence->ref;
96
97 if (screen->fence.tail)
98 screen->fence.tail->next = fence;
99 else
100 screen->fence.head = fence;
101
102 screen->fence.tail = fence;
103
104 fence->state = NOUVEAU_FENCE_STATE_EMITTED;
105 }
106
107 void
108 nouveau_fence_del(struct nouveau_fence *fence)
109 {
110 struct nouveau_fence *it;
111 struct nouveau_screen *screen = fence->screen;
112
113 if (fence->state == NOUVEAU_FENCE_STATE_EMITTED ||
114 fence->state == NOUVEAU_FENCE_STATE_FLUSHED) {
115 if (fence == screen->fence.head) {
116 screen->fence.head = fence->next;
117 if (!screen->fence.head)
118 screen->fence.tail = NULL;
119 } else {
120 for (it = screen->fence.head; it && it->next != fence; it = it->next);
121 it->next = fence->next;
122 if (screen->fence.tail == fence)
123 screen->fence.tail = it;
124 }
125 }
126
127 if (!LIST_IS_EMPTY(&fence->work)) {
128 debug_printf("WARNING: deleting fence with work still pending !\n");
129 nouveau_fence_trigger_work(fence);
130 }
131
132 FREE(fence);
133 }
134
135 void
136 nouveau_fence_update(struct nouveau_screen *screen, boolean flushed)
137 {
138 struct nouveau_fence *fence;
139 struct nouveau_fence *next = NULL;
140 u32 sequence = screen->fence.update(&screen->base);
141
142 if (screen->fence.sequence_ack == sequence)
143 return;
144 screen->fence.sequence_ack = sequence;
145
146 for (fence = screen->fence.head; fence; fence = next) {
147 next = fence->next;
148 sequence = fence->sequence;
149
150 fence->state = NOUVEAU_FENCE_STATE_SIGNALLED;
151
152 nouveau_fence_trigger_work(fence);
153 nouveau_fence_ref(NULL, &fence);
154
155 if (sequence == screen->fence.sequence_ack)
156 break;
157 }
158 screen->fence.head = next;
159 if (!next)
160 screen->fence.tail = NULL;
161
162 if (flushed) {
163 for (fence = next; fence; fence = fence->next)
164 fence->state = NOUVEAU_FENCE_STATE_FLUSHED;
165 }
166 }
167
168 #define NOUVEAU_FENCE_MAX_SPINS (1 << 31)
169
170 boolean
171 nouveau_fence_signalled(struct nouveau_fence *fence)
172 {
173 struct nouveau_screen *screen = fence->screen;
174
175 if (fence->state >= NOUVEAU_FENCE_STATE_EMITTED)
176 nouveau_fence_update(screen, FALSE);
177
178 return fence->state == NOUVEAU_FENCE_STATE_SIGNALLED;
179 }
180
181 boolean
182 nouveau_fence_wait(struct nouveau_fence *fence)
183 {
184 struct nouveau_screen *screen = fence->screen;
185 uint32_t spins = 0;
186
187 if (fence->state < NOUVEAU_FENCE_STATE_EMITTED) {
188 nouveau_fence_emit(fence);
189
190 if (fence == screen->fence.current)
191 nouveau_fence_new(screen, &screen->fence.current, FALSE);
192 }
193 if (fence->state < NOUVEAU_FENCE_STATE_FLUSHED)
194 FIRE_RING(screen->channel);
195
196 do {
197 nouveau_fence_update(screen, FALSE);
198
199 if (fence->state == NOUVEAU_FENCE_STATE_SIGNALLED)
200 return TRUE;
201 spins++;
202 #ifdef PIPE_OS_UNIX
203 if (!(spins % 8)) /* donate a few cycles */
204 sched_yield();
205 #endif
206 } while (spins < NOUVEAU_FENCE_MAX_SPINS);
207
208 debug_printf("Wait on fence %u (ack = %u, next = %u) timed out !\n",
209 fence->sequence,
210 screen->fence.sequence_ack, screen->fence.sequence);
211
212 return FALSE;
213 }
214
215 void
216 nouveau_fence_next(struct nouveau_screen *screen)
217 {
218 nouveau_fence_emit(screen->fence.current);
219 nouveau_fence_new(screen, &screen->fence.current, FALSE);
220 }