Merge branch '7.8'
[mesa.git] / src / gallium / auxiliary / util / u_surface.c
1 /**************************************************************************
2 *
3 * Copyright 2009 VMware, Inc. All Rights Reserved.
4 *
5 * Permission is hereby granted, free of charge, to any person obtaining a
6 * copy of this software and associated documentation files (the
7 * "Software"), to deal in the Software without restriction, including
8 * without limitation the rights to use, copy, modify, merge, publish,
9 * distribute, sub license, and/or sell copies of the Software, and to
10 * permit persons to whom the Software is furnished to do so, subject to
11 * the following conditions:
12 *
13 * The above copyright notice and this permission notice (including the
14 * next paragraph) shall be included in all copies or substantial portions
15 * of the Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
18 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
20 * IN NO EVENT SHALL VMWARE AND/OR ITS SUPPLIERS BE LIABLE FOR
21 * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
22 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
23 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
24 *
25 **************************************************************************/
26
27 /**
28 * @file
29 * Surface utility functions.
30 *
31 * @author Brian Paul
32 */
33
34
35 #include "pipe/p_screen.h"
36 #include "pipe/p_state.h"
37 #include "pipe/p_defines.h"
38 #include "util/u_inlines.h"
39
40 #include "util/u_memory.h"
41 #include "util/u_surface.h"
42
43
44 /**
45 * Helper to quickly create an RGBA rendering surface of a certain size.
46 * \param textureOut returns the new texture
47 * \param surfaceOut returns the new surface
48 * \return TRUE for success, FALSE if failure
49 */
50 boolean
51 util_create_rgba_surface(struct pipe_screen *screen,
52 uint width, uint height,
53 uint bind,
54 struct pipe_resource **textureOut,
55 struct pipe_surface **surfaceOut)
56 {
57 static const enum pipe_format rgbaFormats[] = {
58 PIPE_FORMAT_B8G8R8A8_UNORM,
59 PIPE_FORMAT_A8R8G8B8_UNORM,
60 PIPE_FORMAT_A8B8G8R8_UNORM,
61 PIPE_FORMAT_NONE
62 };
63 const uint target = PIPE_TEXTURE_2D;
64 enum pipe_format format = PIPE_FORMAT_NONE;
65 struct pipe_resource templ;
66 uint i;
67
68 /* Choose surface format */
69 for (i = 0; rgbaFormats[i]; i++) {
70 if (screen->is_format_supported(screen, rgbaFormats[i],
71 target, bind, 0)) {
72 format = rgbaFormats[i];
73 break;
74 }
75 }
76 if (format == PIPE_FORMAT_NONE)
77 return FALSE; /* unable to get an rgba format!?! */
78
79 /* create texture */
80 memset(&templ, 0, sizeof(templ));
81 templ.target = target;
82 templ.format = format;
83 templ.last_level = 0;
84 templ.width0 = width;
85 templ.height0 = height;
86 templ.depth0 = 1;
87 templ.bind = bind;
88
89 *textureOut = screen->resource_create(screen, &templ);
90 if (!*textureOut)
91 return FALSE;
92
93 /* create surface / view into texture */
94 *surfaceOut = screen->get_tex_surface(screen,
95 *textureOut,
96 0, 0, 0,
97 bind);
98 if (!*surfaceOut) {
99 pipe_resource_reference(textureOut, NULL);
100 return FALSE;
101 }
102
103 return TRUE;
104 }
105
106
107 /**
108 * Release the surface and texture from util_create_rgba_surface().
109 */
110 void
111 util_destroy_rgba_surface(struct pipe_resource *texture,
112 struct pipe_surface *surface)
113 {
114 pipe_surface_reference(&surface, NULL);
115 pipe_resource_reference(&texture, NULL);
116 }
117
118
119
120 /**
121 * Compare pipe_framebuffer_state objects.
122 * \return TRUE if same, FALSE if different
123 */
124 boolean
125 util_framebuffer_state_equal(const struct pipe_framebuffer_state *dst,
126 const struct pipe_framebuffer_state *src)
127 {
128 unsigned i;
129
130 if (dst->width != src->width ||
131 dst->height != src->height)
132 return FALSE;
133
134 for (i = 0; i < Elements(src->cbufs); i++) {
135 if (dst->cbufs[i] != src->cbufs[i]) {
136 return FALSE;
137 }
138 }
139
140 if (dst->nr_cbufs != src->nr_cbufs) {
141 return FALSE;
142 }
143
144 if (dst->zsbuf != src->zsbuf) {
145 return FALSE;
146 }
147
148 return TRUE;
149 }
150
151
152 /**
153 * Copy framebuffer state from src to dst, updating refcounts.
154 */
155 void
156 util_copy_framebuffer_state(struct pipe_framebuffer_state *dst,
157 const struct pipe_framebuffer_state *src)
158 {
159 unsigned i;
160
161 dst->width = src->width;
162 dst->height = src->height;
163
164 for (i = 0; i < Elements(src->cbufs); i++) {
165 pipe_surface_reference(&dst->cbufs[i], src->cbufs[i]);
166 }
167
168 dst->nr_cbufs = src->nr_cbufs;
169
170 pipe_surface_reference(&dst->zsbuf, src->zsbuf);
171 }
172
173
174 void
175 util_unreference_framebuffer_state(struct pipe_framebuffer_state *fb)
176 {
177 unsigned i;
178
179 for (i = 0; i < fb->nr_cbufs; i++) {
180 pipe_surface_reference(&fb->cbufs[i], NULL);
181 }
182
183 pipe_surface_reference(&fb->zsbuf, NULL);
184
185 fb->width = fb->height = 0;
186 fb->nr_cbufs = 0;
187 }