2 * Mesa 3-D graphics library
5 * Copyright (C) 2010 LunarG Inc.
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:
14 * The above copyright notice and this permission notice shall be included
15 * in all copies or substantial portions of the Software.
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.
26 * Chia-I Wu <olv@lunarg.com>
29 #include "state_tracker/st_api.h"
31 #include "pipe/p_context.h"
32 #include "pipe/p_screen.h"
33 #include "util/u_memory.h"
34 #include "util/u_inlines.h"
36 #include "vg_manager.h"
37 #include "vg_context.h"
38 #include "vg_tracker.h" /* for st_resize_framebuffer */
42 * Flush the front buffer if the current context renders to the front buffer.
45 vg_manager_flush_frontbuffer(struct vg_context *ctx)
47 struct st_framebuffer *stfb = ctx->draw_buffer;
52 /* st_public.h is used */
54 struct pipe_screen *screen = ctx->pipe->screen;
55 if (screen->flush_frontbuffer) {
56 screen->flush_frontbuffer(screen,
57 stfb->strb->surface, ctx->pipe->priv);
62 switch (stfb->strb_att) {
63 case ST_ATTACHMENT_FRONT_LEFT:
64 case ST_ATTACHMENT_FRONT_RIGHT:
65 stfb->iface->flush_front(stfb->iface, stfb->strb_att);
73 * Re-validate the framebuffer.
76 vg_manager_validate_framebuffer(struct vg_context *ctx)
78 struct pipe_screen *screen = ctx->pipe->screen;
79 struct st_framebuffer *stfb = ctx->draw_buffer;
80 struct st_renderbuffer *rb;
81 struct pipe_resource *pt;
83 /* no binding surface */
87 /* st_public.h is used */
89 struct pipe_screen *screen = ctx->pipe->screen;
90 if (screen->update_buffer)
91 screen->update_buffer(screen, ctx->pipe->priv);
95 if (!p_atomic_read(&ctx->draw_buffer_invalid))
99 if (!stfb->iface->validate(stfb->iface, &stfb->strb_att, 1, &pt) || !pt)
103 if (rb->texture == pt) {
104 pipe_resource_reference(&pt, NULL);
108 /* unreference existing ones */
109 pipe_surface_reference(&rb->surface, NULL);
110 pipe_resource_reference(&rb->texture, NULL);
113 rb->surface = screen->get_tex_surface(screen, rb->texture, 0, 0, 0,
114 PIPE_BIND_RENDER_TARGET |
115 PIPE_BIND_BLIT_SOURCE |
116 PIPE_BIND_BLIT_DESTINATION);
118 rb->width = rb->surface->width;
119 rb->height = rb->surface->height;
121 st_resize_framebuffer(stfb, rb->width, rb->height);
123 p_atomic_set(&ctx->draw_buffer_invalid, FALSE);
128 vg_context_notify_invalid_framebuffer(struct st_context_iface *stctxi,
129 struct st_framebuffer_iface *stfbi)
131 struct vg_context *ctx = (struct vg_context *) stctxi;
132 p_atomic_set(&ctx->draw_buffer_invalid, TRUE);
136 vg_context_flush(struct st_context_iface *stctxi, unsigned flags,
137 struct pipe_fence_handle **fence)
139 struct vg_context *ctx = (struct vg_context *) stctxi;
140 ctx->pipe->flush(ctx->pipe, flags, fence);
141 if (flags & PIPE_FLUSH_RENDER_CACHE)
142 vg_manager_flush_frontbuffer(ctx);
146 vg_context_destroy(struct st_context_iface *stctxi)
148 struct vg_context *ctx = (struct vg_context *) stctxi;
149 vg_destroy_context(ctx);
152 static struct st_context_iface *
153 vg_api_create_context(struct st_api *stapi, struct st_manager *smapi,
154 const struct st_visual *visual,
155 struct st_context_iface *shared_stctxi)
157 struct vg_context *shared_ctx = (struct vg_context *) shared_stctxi;
158 struct vg_context *ctx;
159 struct pipe_context *pipe;
161 pipe = smapi->screen->context_create(smapi->screen, NULL);
164 ctx = vg_create_context(pipe, NULL, shared_ctx);
170 ctx->iface.destroy = vg_context_destroy;
172 ctx->iface.notify_invalid_framebuffer =
173 vg_context_notify_invalid_framebuffer;
174 ctx->iface.flush = vg_context_flush;
176 ctx->iface.teximage = NULL;
177 ctx->iface.copy = NULL;
179 ctx->iface.st_context_private = (void *) smapi;
184 static struct st_renderbuffer *
185 create_renderbuffer(enum pipe_format format)
187 struct st_renderbuffer *strb;
189 strb = CALLOC_STRUCT(st_renderbuffer);
191 strb->format = format;
197 destroy_renderbuffer(struct st_renderbuffer *strb)
199 pipe_surface_reference(&strb->surface, NULL);
200 pipe_resource_reference(&strb->texture, NULL);
205 * Decide the buffer to render to.
207 static enum st_attachment_type
208 choose_attachment(struct st_framebuffer_iface *stfbi)
210 enum st_attachment_type statt;
212 statt = stfbi->visual->render_buffer;
213 if (statt != ST_ATTACHMENT_INVALID) {
214 /* use the buffer given by the visual, unless it is unavailable */
215 if (!st_visual_have_buffers(stfbi->visual, 1 << statt)) {
217 case ST_ATTACHMENT_BACK_LEFT:
218 statt = ST_ATTACHMENT_FRONT_LEFT;
220 case ST_ATTACHMENT_BACK_RIGHT:
221 statt = ST_ATTACHMENT_FRONT_RIGHT;
227 if (!st_visual_have_buffers(stfbi->visual, 1 << statt))
228 statt = ST_ATTACHMENT_INVALID;
236 * Bind the context to the given framebuffers.
239 vg_context_bind_framebuffers(struct st_context_iface *stctxi,
240 struct st_framebuffer_iface *stdrawi,
241 struct st_framebuffer_iface *streadi)
243 struct vg_context *ctx = (struct vg_context *) stctxi;
244 struct st_framebuffer *stfb;
245 enum st_attachment_type strb_att;
247 /* the draw and read framebuffers must be the same */
248 if (stdrawi != streadi)
251 p_atomic_set(&ctx->draw_buffer_invalid, TRUE);
253 strb_att = (stdrawi) ? choose_attachment(stdrawi) : ST_ATTACHMENT_INVALID;
255 if (ctx->draw_buffer) {
256 stfb = ctx->draw_buffer;
258 /* free the existing fb */
260 stfb->strb_att != strb_att ||
261 stfb->strb->format != stdrawi->visual->color_format ||
262 stfb->dsrb->format != stdrawi->visual->depth_stencil_format) {
263 destroy_renderbuffer(stfb->strb);
264 destroy_renderbuffer(stfb->dsrb);
267 ctx->draw_buffer = NULL;
274 if (strb_att == ST_ATTACHMENT_INVALID)
277 /* create a new fb */
278 if (!ctx->draw_buffer) {
279 stfb = CALLOC_STRUCT(st_framebuffer);
283 stfb->strb = create_renderbuffer(stdrawi->visual->color_format);
289 stfb->dsrb = create_renderbuffer(stdrawi->visual->depth_stencil_format);
298 stfb->strb_att = strb_att;
300 ctx->draw_buffer = stfb;
303 ctx->draw_buffer->iface = stdrawi;
309 vg_api_make_current(struct st_api *stapi, struct st_context_iface *stctxi,
310 struct st_framebuffer_iface *stdrawi,
311 struct st_framebuffer_iface *streadi)
313 struct vg_context *ctx = (struct vg_context *) stctxi;
316 vg_context_bind_framebuffers(stctxi, stdrawi, streadi);
317 vg_set_current_context(ctx);
322 static struct st_context_iface *
323 vg_api_get_current(struct st_api *stapi)
325 struct vg_context *ctx = vg_current_context();
327 return (ctx) ? &ctx->iface : NULL;
331 vg_api_is_visual_supported(struct st_api *stapi,
332 const struct st_visual *visual)
334 /* the impl requires a depth/stencil buffer */
335 if (visual->depth_stencil_format == PIPE_FORMAT_NONE)
342 vg_api_get_proc_address(struct st_api *stapi, const char *procname)
345 return (st_proc_t) NULL;
349 vg_api_destroy(struct st_api *stapi)
354 static struct st_api *
355 vg_module_create_api(void)
357 struct st_api *stapi;
359 stapi = CALLOC_STRUCT(st_api);
361 stapi->destroy = vg_api_destroy;
362 stapi->get_proc_address = vg_api_get_proc_address;
363 stapi->is_visual_supported = vg_api_is_visual_supported;
365 stapi->create_context = vg_api_create_context;
366 stapi->make_current = vg_api_make_current;
367 stapi->get_current = vg_api_get_current;
373 PUBLIC const struct st_module st_module_OpenVG = {
374 .api = ST_API_OPENVG,
375 .create_api = vg_module_create_api,