1 /**********************************************************
2 * Copyright 2009-2011 VMware, Inc. All rights reserved.
4 * Permission is hereby granted, free of charge, to any person
5 * obtaining a copy of this software and associated documentation
6 * files (the "Software"), to deal in the Software without
7 * restriction, including without limitation the rights to use, copy,
8 * modify, merge, publish, distribute, sublicense, and/or sell copies
9 * of the Software, and to permit persons to whom the Software is
10 * furnished to do so, subject to the following conditions:
12 * The above copyright notice and this permission notice shall be
13 * included in all copies or substantial portions of the Software.
15 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
16 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
17 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
18 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
19 * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
20 * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
21 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
24 *********************************************************
26 * Zack Rusin <zackr-at-vmware-dot-com>
27 * Thomas Hellstrom <thellstrom-at-vmware-dot-com>
29 #include "xa_context.h"
31 #include "cso_cache/cso_context.h"
32 #include "util/u_inlines.h"
33 #include "util/u_rect.h"
34 #include "util/u_surface.h"
35 #include "pipe/p_context.h"
38 xa_context_flush(struct xa_context
*ctx
)
40 if (ctx
->last_fence
) {
41 struct pipe_screen
*screen
= ctx
->xa
->screen
;
42 screen
->fence_reference(screen
, &ctx
->last_fence
, NULL
);
44 ctx
->pipe
->flush(ctx
->pipe
, &ctx
->last_fence
, 0);
47 XA_EXPORT
struct xa_context
*
48 xa_context_default(struct xa_tracker
*xa
)
50 return xa
->default_ctx
;
53 XA_EXPORT
struct xa_context
*
54 xa_context_create(struct xa_tracker
*xa
)
56 struct xa_context
*ctx
= calloc(1, sizeof(*ctx
));
59 ctx
->pipe
= xa
->screen
->context_create(xa
->screen
, NULL
, 0);
60 ctx
->cso
= cso_create_context(ctx
->pipe
, 0);
61 ctx
->shaders
= xa_shaders_create(ctx
);
62 renderer_init_state(ctx
);
68 xa_context_destroy(struct xa_context
*r
)
70 struct pipe_resource
**vsbuf
= &r
->vs_const_buffer
;
71 struct pipe_resource
**fsbuf
= &r
->fs_const_buffer
;
74 pipe_resource_reference(vsbuf
, NULL
);
77 pipe_resource_reference(fsbuf
, NULL
);
80 xa_shaders_destroy(r
->shaders
);
84 xa_ctx_sampler_views_destroy(r
);
86 pipe_surface_reference(&r
->srf
, NULL
);
89 cso_destroy_context(r
->cso
);
93 r
->pipe
->destroy(r
->pipe
);
98 xa_surface_dma(struct xa_context
*ctx
,
99 struct xa_surface
*srf
,
102 int to_surface
, struct xa_box
*boxes
, unsigned int num_boxes
)
104 struct pipe_transfer
*transfer
;
107 enum pipe_transfer_usage transfer_direction
;
108 struct pipe_context
*pipe
= ctx
->pipe
;
110 transfer_direction
= (to_surface
? PIPE_TRANSFER_WRITE
:
113 for (i
= 0; i
< num_boxes
; ++i
, ++boxes
) {
114 w
= boxes
->x2
- boxes
->x1
;
115 h
= boxes
->y2
- boxes
->y1
;
117 map
= pipe_transfer_map(pipe
, srf
->tex
, 0, 0,
118 transfer_direction
, boxes
->x1
, boxes
->y1
,
121 return -XA_ERR_NORES
;
124 util_copy_rect(map
, srf
->tex
->format
, transfer
->stride
,
125 0, 0, w
, h
, data
, pitch
, boxes
->x1
, boxes
->y1
);
127 util_copy_rect(data
, srf
->tex
->format
, pitch
,
128 boxes
->x1
, boxes
->y1
, w
, h
, map
, transfer
->stride
, 0,
131 pipe
->transfer_unmap(pipe
, transfer
);
137 xa_surface_map(struct xa_context
*ctx
,
138 struct xa_surface
*srf
, unsigned int usage
)
141 unsigned int gallium_usage
= 0;
142 struct pipe_context
*pipe
= ctx
->pipe
;
145 * A surface may only have a single map.
150 if (usage
& XA_MAP_READ
)
151 gallium_usage
|= PIPE_TRANSFER_READ
;
152 if (usage
& XA_MAP_WRITE
)
153 gallium_usage
|= PIPE_TRANSFER_WRITE
;
154 if (usage
& XA_MAP_MAP_DIRECTLY
)
155 gallium_usage
|= PIPE_TRANSFER_MAP_DIRECTLY
;
156 if (usage
& XA_MAP_UNSYNCHRONIZED
)
157 gallium_usage
|= PIPE_TRANSFER_UNSYNCHRONIZED
;
158 if (usage
& XA_MAP_DONTBLOCK
)
159 gallium_usage
|= PIPE_TRANSFER_DONTBLOCK
;
160 if (usage
& XA_MAP_DISCARD_WHOLE_RESOURCE
)
161 gallium_usage
|= PIPE_TRANSFER_DISCARD_WHOLE_RESOURCE
;
163 if (!(gallium_usage
& (PIPE_TRANSFER_READ_WRITE
)))
166 map
= pipe_transfer_map(pipe
, srf
->tex
, 0, 0,
168 srf
->tex
->width0
, srf
->tex
->height0
,
173 srf
->mapping_pipe
= pipe
;
178 xa_surface_unmap(struct xa_surface
*srf
)
181 struct pipe_context
*pipe
= srf
->mapping_pipe
;
183 pipe
->transfer_unmap(pipe
, srf
->transfer
);
184 srf
->transfer
= NULL
;
189 xa_ctx_srf_create(struct xa_context
*ctx
, struct xa_surface
*dst
)
191 struct pipe_screen
*screen
= ctx
->pipe
->screen
;
192 struct pipe_surface srf_templ
;
195 * Cache surfaces unless we change render target
198 if (ctx
->srf
->texture
== dst
->tex
)
201 pipe_surface_reference(&ctx
->srf
, NULL
);
204 if (!screen
->is_format_supported(screen
, dst
->tex
->format
,
205 PIPE_TEXTURE_2D
, 0, 0,
206 PIPE_BIND_RENDER_TARGET
))
207 return -XA_ERR_INVAL
;
209 u_surface_default_template(&srf_templ
, dst
->tex
);
210 ctx
->srf
= ctx
->pipe
->create_surface(ctx
->pipe
, dst
->tex
, &srf_templ
);
212 return -XA_ERR_NORES
;
218 xa_ctx_srf_destroy(struct xa_context
*ctx
)
221 * Cache surfaces unless we change render target.
222 * Final destruction on context destroy.
227 xa_copy_prepare(struct xa_context
*ctx
,
228 struct xa_surface
*dst
, struct xa_surface
*src
)
231 return -XA_ERR_INVAL
;
233 if (src
->tex
->format
!= dst
->tex
->format
) {
234 int ret
= xa_ctx_srf_create(ctx
, dst
);
235 if (ret
!= XA_ERR_NONE
)
237 renderer_copy_prepare(ctx
, ctx
->srf
, src
->tex
,
238 src
->fdesc
.xa_format
,
239 dst
->fdesc
.xa_format
);
240 ctx
->simple_copy
= 0;
242 ctx
->simple_copy
= 1;
246 xa_ctx_srf_destroy(ctx
);
252 xa_copy(struct xa_context
*ctx
,
253 int dx
, int dy
, int sx
, int sy
, int width
, int height
)
255 struct pipe_box src_box
;
257 xa_scissor_update(ctx
, dx
, dy
, dx
+ width
, dy
+ height
);
259 if (ctx
->simple_copy
) {
260 u_box_2d(sx
, sy
, width
, height
, &src_box
);
261 ctx
->pipe
->resource_copy_region(ctx
->pipe
,
262 ctx
->dst
->tex
, 0, dx
, dy
, 0,
266 renderer_copy(ctx
, dx
, dy
, sx
, sy
, width
, height
,
267 (float) ctx
->src
->tex
->width0
,
268 (float) ctx
->src
->tex
->height0
);
272 xa_copy_done(struct xa_context
*ctx
)
274 if (!ctx
->simple_copy
) {
275 renderer_draw_flush(ctx
);
280 bind_solid_blend_state(struct xa_context
*ctx
)
282 struct pipe_blend_state blend
;
284 memset(&blend
, 0, sizeof(struct pipe_blend_state
));
285 blend
.rt
[0].blend_enable
= 0;
286 blend
.rt
[0].colormask
= PIPE_MASK_RGBA
;
288 blend
.rt
[0].rgb_src_factor
= PIPE_BLENDFACTOR_ONE
;
289 blend
.rt
[0].alpha_src_factor
= PIPE_BLENDFACTOR_ONE
;
290 blend
.rt
[0].rgb_dst_factor
= PIPE_BLENDFACTOR_ZERO
;
291 blend
.rt
[0].alpha_dst_factor
= PIPE_BLENDFACTOR_ZERO
;
293 cso_set_blend(ctx
->cso
, &blend
);
297 xa_solid_prepare(struct xa_context
*ctx
, struct xa_surface
*dst
,
300 unsigned vs_traits
, fs_traits
;
301 struct xa_shader shader
;
304 ret
= xa_ctx_srf_create(ctx
, dst
);
305 if (ret
!= XA_ERR_NONE
)
308 if (ctx
->srf
->format
== PIPE_FORMAT_L8_UNORM
)
309 xa_pixel_to_float4_a8(fg
, ctx
->solid_color
);
311 xa_pixel_to_float4(fg
, ctx
->solid_color
);
312 ctx
->has_solid_src
= 1;
317 debug_printf("Color Pixel=(%d, %d, %d, %d), RGBA=(%f, %f, %f, %f)\n",
318 (fg
>> 24) & 0xff, (fg
>> 16) & 0xff,
319 (fg
>> 8) & 0xff, (fg
>> 0) & 0xff,
320 exa
->solid_color
[0], exa
->solid_color
[1],
321 exa
->solid_color
[2], exa
->solid_color
[3]);
324 vs_traits
= VS_SRC_SRC
| VS_COMPOSITE
;
325 fs_traits
= FS_SRC_SRC
| VS_COMPOSITE
;
327 renderer_bind_destination(ctx
, ctx
->srf
);
328 bind_solid_blend_state(ctx
);
329 cso_set_samplers(ctx
->cso
, PIPE_SHADER_FRAGMENT
, 0, NULL
);
330 cso_set_sampler_views(ctx
->cso
, PIPE_SHADER_FRAGMENT
, 0, NULL
);
332 shader
= xa_shaders_get(ctx
->shaders
, vs_traits
, fs_traits
);
333 cso_set_vertex_shader_handle(ctx
->cso
, shader
.vs
);
334 cso_set_fragment_shader_handle(ctx
->cso
, shader
.fs
);
336 renderer_begin_solid(ctx
);
338 xa_ctx_srf_destroy(ctx
);
343 xa_solid(struct xa_context
*ctx
, int x
, int y
, int width
, int height
)
345 xa_scissor_update(ctx
, x
, y
, x
+ width
, y
+ height
);
346 renderer_solid(ctx
, x
, y
, x
+ width
, y
+ height
);
350 xa_solid_done(struct xa_context
*ctx
)
352 renderer_draw_flush(ctx
);
354 ctx
->has_solid_src
= FALSE
;
355 ctx
->num_bound_samplers
= 0;
358 XA_EXPORT
struct xa_fence
*
359 xa_fence_get(struct xa_context
*ctx
)
361 struct xa_fence
*fence
= calloc(1, sizeof(*fence
));
362 struct pipe_screen
*screen
= ctx
->xa
->screen
;
369 if (ctx
->last_fence
== NULL
)
370 fence
->pipe_fence
= NULL
;
372 screen
->fence_reference(screen
, &fence
->pipe_fence
, ctx
->last_fence
);
378 xa_fence_wait(struct xa_fence
*fence
, uint64_t timeout
)
383 if (fence
->pipe_fence
) {
384 struct pipe_screen
*screen
= fence
->xa
->screen
;
387 timed_out
= !screen
->fence_finish(screen
, NULL
, fence
->pipe_fence
, timeout
);
391 screen
->fence_reference(screen
, &fence
->pipe_fence
, NULL
);
397 xa_fence_destroy(struct xa_fence
*fence
)
402 if (fence
->pipe_fence
) {
403 struct pipe_screen
*screen
= fence
->xa
->screen
;
405 screen
->fence_reference(screen
, &fence
->pipe_fence
, NULL
);
412 xa_ctx_sampler_views_destroy(struct xa_context
*ctx
)
416 for (i
= 0; i
< ctx
->num_bound_samplers
; ++i
)
417 pipe_sampler_view_reference(&ctx
->bound_sampler_views
[i
], NULL
);
418 ctx
->num_bound_samplers
= 0;