st/egl: Remove.
[mesa.git] / src / gallium / state_trackers / vega / vg_manager.c
1 /*
2 * Mesa 3-D graphics library
3 *
4 * Copyright 2009 VMware, Inc. All Rights Reserved.
5 * Copyright (C) 2010 LunarG Inc.
6 *
7 * Permission is hereby granted, free of charge, to any person obtaining a
8 * copy of this software and associated documentation files (the "Software"),
9 * to deal in the Software without restriction, including without limitation
10 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
11 * and/or sell copies of the Software, and to permit persons to whom the
12 * Software is furnished to do so, subject to the following conditions:
13 *
14 * The above copyright notice and this permission notice shall be included
15 * in all copies or substantial portions of the Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
20 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
21 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
22 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
23 * DEALINGS IN THE SOFTWARE.
24 *
25 * Authors:
26 * Chia-I Wu <olv@lunarg.com>
27 */
28
29 #include "state_tracker/st_api.h"
30
31 #include "pipe/p_context.h"
32 #include "pipe/p_screen.h"
33 #include "util/u_memory.h"
34 #include "util/u_inlines.h"
35 #include "util/u_box.h"
36 #include "util/u_surface.h"
37
38 #include "vg_api.h"
39 #include "vg_manager.h"
40 #include "vg_context.h"
41 #include "api.h"
42 #include "handle.h"
43
44 static boolean
45 vg_context_update_color_rb(struct vg_context *ctx, struct pipe_resource *pt)
46 {
47 struct st_renderbuffer *strb = ctx->draw_buffer->strb;
48 struct pipe_context *pipe = ctx->pipe;
49 struct pipe_surface surf_tmpl;
50
51 if (strb->texture == pt) {
52 pipe_resource_reference(&pt, NULL);
53 return FALSE;
54 }
55
56 /* unreference existing ones */
57 pipe_surface_reference(&strb->surface, NULL);
58 pipe_resource_reference(&strb->texture, NULL);
59 strb->width = strb->height = 0;
60
61 strb->texture = pt;
62
63 u_surface_default_template(&surf_tmpl, strb->texture);
64 strb->surface = pipe->create_surface(pipe, strb->texture, &surf_tmpl);
65
66 if (!strb->surface) {
67 pipe_resource_reference(&strb->texture, NULL);
68 return TRUE;
69 }
70
71 strb->width = pt->width0;
72 strb->height = pt->height0;
73
74 return TRUE;
75 }
76
77 /**
78 * Flush the front buffer if the current context renders to the front buffer.
79 */
80 void
81 vg_manager_flush_frontbuffer(struct vg_context *ctx)
82 {
83 struct st_framebuffer *stfb = ctx->draw_buffer;
84
85 if (!stfb)
86 return;
87
88 switch (stfb->strb_att) {
89 case ST_ATTACHMENT_FRONT_LEFT:
90 case ST_ATTACHMENT_FRONT_RIGHT:
91 stfb->iface->flush_front(&ctx->iface, stfb->iface, stfb->strb_att);
92 break;
93 default:
94 break;
95 }
96 }
97
98 /**
99 * Re-validate the framebuffer.
100 */
101 void
102 vg_manager_validate_framebuffer(struct vg_context *ctx)
103 {
104 struct st_framebuffer *stfb = ctx->draw_buffer;
105 struct pipe_resource *pt;
106 int32_t new_stamp;
107
108 /* no binding surface */
109 if (!stfb)
110 return;
111
112 new_stamp = p_atomic_read(&stfb->iface->stamp);
113 if (stfb->iface_stamp != new_stamp) {
114 do {
115 /* validate the fb */
116 if (!stfb->iface->validate((struct st_context_iface *)ctx,
117 stfb->iface, &stfb->strb_att,
118 1, &pt) || !pt)
119 return;
120
121 stfb->iface_stamp = new_stamp;
122 new_stamp = p_atomic_read(&stfb->iface->stamp);
123
124 } while (stfb->iface_stamp != new_stamp);
125
126 if (vg_context_update_color_rb(ctx, pt) ||
127 stfb->width != pt->width0 ||
128 stfb->height != pt->height0)
129 ++stfb->stamp;
130
131 stfb->width = pt->width0;
132 stfb->height = pt->height0;
133 }
134
135 if (ctx->draw_stamp != stfb->stamp) {
136 ctx->state.dirty |= FRAMEBUFFER_DIRTY;
137 ctx->draw_stamp = stfb->stamp;
138 }
139 }
140
141 static void
142 vg_context_flush(struct st_context_iface *stctxi, unsigned flags,
143 struct pipe_fence_handle **fence)
144 {
145 struct vg_context *ctx = (struct vg_context *) stctxi;
146 unsigned pipe_flags = 0;
147
148 if (flags & ST_FLUSH_END_OF_FRAME) {
149 pipe_flags |= PIPE_FLUSH_END_OF_FRAME;
150 }
151
152 ctx->pipe->flush(ctx->pipe, fence, pipe_flags);
153 if (flags & ST_FLUSH_FRONT)
154 vg_manager_flush_frontbuffer(ctx);
155 }
156
157 static void
158 vg_context_destroy(struct st_context_iface *stctxi)
159 {
160 struct vg_context *ctx = (struct vg_context *) stctxi;
161 struct pipe_context *pipe = ctx->pipe;
162
163 vg_destroy_context(ctx);
164 pipe->destroy(pipe);
165 }
166
167 static struct st_context_iface *
168 vg_api_create_context(struct st_api *stapi, struct st_manager *smapi,
169 const struct st_context_attribs *attribs,
170 enum st_context_error *error,
171 struct st_context_iface *shared_stctxi)
172 {
173 struct vg_context *shared_ctx = (struct vg_context *) shared_stctxi;
174 struct vg_context *ctx;
175 struct pipe_context *pipe;
176
177 if (!(stapi->profile_mask & (1 << attribs->profile))) {
178 *error = ST_CONTEXT_ERROR_BAD_API;
179 return NULL;
180 }
181
182 /* only 1.0 is supported */
183 if (attribs->major > 1 || (attribs->major == 1 && attribs->minor > 0)) {
184 *error = ST_CONTEXT_ERROR_BAD_VERSION;
185 return NULL;
186 }
187
188 /* for VGHandle / pointer lookups */
189 init_handles();
190
191 pipe = smapi->screen->context_create(smapi->screen, NULL);
192 if (!pipe) {
193 *error = ST_CONTEXT_ERROR_NO_MEMORY;
194 return NULL;
195 }
196 ctx = vg_create_context(pipe, NULL, shared_ctx);
197 if (!ctx) {
198 pipe->destroy(pipe);
199 *error = ST_CONTEXT_ERROR_NO_MEMORY;
200 return NULL;
201 }
202
203 ctx->iface.destroy = vg_context_destroy;
204
205 ctx->iface.flush = vg_context_flush;
206
207 ctx->iface.teximage = NULL;
208 ctx->iface.copy = NULL;
209
210 ctx->iface.st_context_private = (void *) smapi;
211
212 return &ctx->iface;
213 }
214
215 static struct st_renderbuffer *
216 create_renderbuffer(enum pipe_format format)
217 {
218 struct st_renderbuffer *strb;
219
220 strb = CALLOC_STRUCT(st_renderbuffer);
221 if (strb)
222 strb->format = format;
223
224 return strb;
225 }
226
227 static void
228 destroy_renderbuffer(struct st_renderbuffer *strb)
229 {
230 pipe_surface_reference(&strb->surface, NULL);
231 pipe_resource_reference(&strb->texture, NULL);
232 FREE(strb);
233 }
234
235 /**
236 * Decide the buffer to render to.
237 */
238 static enum st_attachment_type
239 choose_attachment(struct st_framebuffer_iface *stfbi)
240 {
241 enum st_attachment_type statt;
242
243 statt = stfbi->visual->render_buffer;
244 if (statt != ST_ATTACHMENT_INVALID) {
245 /* use the buffer given by the visual, unless it is unavailable */
246 if (!st_visual_have_buffers(stfbi->visual, 1 << statt)) {
247 switch (statt) {
248 case ST_ATTACHMENT_BACK_LEFT:
249 statt = ST_ATTACHMENT_FRONT_LEFT;
250 break;
251 case ST_ATTACHMENT_BACK_RIGHT:
252 statt = ST_ATTACHMENT_FRONT_RIGHT;
253 break;
254 default:
255 break;
256 }
257
258 if (!st_visual_have_buffers(stfbi->visual, 1 << statt))
259 statt = ST_ATTACHMENT_INVALID;
260 }
261 }
262
263 return statt;
264 }
265
266 /**
267 * Bind the context to the given framebuffers.
268 */
269 static boolean
270 vg_context_bind_framebuffers(struct st_context_iface *stctxi,
271 struct st_framebuffer_iface *stdrawi,
272 struct st_framebuffer_iface *streadi)
273 {
274 struct vg_context *ctx = (struct vg_context *) stctxi;
275 struct st_framebuffer *stfb;
276 enum st_attachment_type strb_att;
277
278 /* the draw and read framebuffers must be the same */
279 if (stdrawi != streadi)
280 return FALSE;
281
282 strb_att = (stdrawi) ? choose_attachment(stdrawi) : ST_ATTACHMENT_INVALID;
283
284 if (ctx->draw_buffer) {
285 stfb = ctx->draw_buffer;
286
287 /* free the existing fb */
288 if (!stdrawi ||
289 stfb->strb_att != strb_att ||
290 stfb->strb->format != stdrawi->visual->color_format) {
291 destroy_renderbuffer(stfb->strb);
292 destroy_renderbuffer(stfb->dsrb);
293 FREE(stfb);
294
295 ctx->draw_buffer = NULL;
296 }
297 }
298
299 if (!stdrawi)
300 return TRUE;
301
302 if (strb_att == ST_ATTACHMENT_INVALID)
303 return FALSE;
304
305 /* create a new fb */
306 if (!ctx->draw_buffer) {
307 stfb = CALLOC_STRUCT(st_framebuffer);
308 if (!stfb)
309 return FALSE;
310
311 stfb->strb = create_renderbuffer(stdrawi->visual->color_format);
312 if (!stfb->strb) {
313 FREE(stfb);
314 return FALSE;
315 }
316
317 stfb->dsrb = create_renderbuffer(ctx->ds_format);
318 if (!stfb->dsrb) {
319 FREE(stfb->strb);
320 FREE(stfb);
321 return FALSE;
322 }
323
324 stfb->width = 0;
325 stfb->height = 0;
326 stfb->strb_att = strb_att;
327 stfb->stamp = 1;
328 stfb->iface_stamp = p_atomic_read(&stdrawi->stamp) - 1;
329
330 ctx->draw_buffer = stfb;
331 }
332
333 ctx->draw_buffer->iface = stdrawi;
334 ctx->draw_stamp = ctx->draw_buffer->stamp - 1;
335
336 return TRUE;
337 }
338
339 static boolean
340 vg_api_make_current(struct st_api *stapi, struct st_context_iface *stctxi,
341 struct st_framebuffer_iface *stdrawi,
342 struct st_framebuffer_iface *streadi)
343 {
344 struct vg_context *ctx = (struct vg_context *) stctxi;
345
346 if (stctxi)
347 vg_context_bind_framebuffers(stctxi, stdrawi, streadi);
348 vg_set_current_context(ctx);
349
350 return TRUE;
351 }
352
353 static struct st_context_iface *
354 vg_api_get_current(struct st_api *stapi)
355 {
356 struct vg_context *ctx = vg_current_context();
357
358 return (ctx) ? &ctx->iface : NULL;
359 }
360
361 static st_proc_t
362 vg_api_get_proc_address(struct st_api *stapi, const char *procname)
363 {
364 return api_get_proc_address(procname);
365 }
366
367 static void
368 vg_api_destroy(struct st_api *stapi)
369 {
370 }
371
372 static void
373 vg_api_query_versions(struct st_api *stapi, struct st_manager *sm,
374 struct st_config_options *options,
375 int *gl_core_version,
376 int *gl_compat_version,
377 int *gl_es1_version,
378 int *gl_es2_version)
379 {
380 }
381
382 static const struct st_api vg_api = {
383 "Vega " PACKAGE_VERSION,
384 ST_API_OPENVG,
385 ST_PROFILE_DEFAULT_MASK,
386 0,
387 vg_api_destroy,
388 vg_api_query_versions,
389 vg_api_get_proc_address,
390 vg_api_create_context,
391 vg_api_make_current,
392 vg_api_get_current,
393 };
394
395 const struct st_api *
396 vg_api_get(void)
397 {
398 return &vg_api;
399 }