a4eac11ae3451f5e8b85f73dac4cadaf2767b4f1
[mesa.git] / src / gallium / drivers / galahad / glhd_screen.c
1 /**************************************************************************
2 *
3 * Copyright 2009 VMware, Inc.
4 * 2010 Corbin Simpson <MostAwesomeDude@gmail.com>
5 * All Rights Reserved.
6 *
7 * Permission is hereby granted, free of charge, to any person obtaining a
8 * copy of this software and associated documentation files (the
9 * "Software"), to deal in the Software without restriction, including
10 * without limitation the rights to use, copy, modify, merge, publish,
11 * distribute, sub license, and/or sell copies of the Software, and to
12 * permit persons to whom the Software is furnished to do so, subject to
13 * the following conditions:
14 *
15 * The above copyright notice and this permission notice (including the
16 * next paragraph) shall be included in all copies or substantial portions
17 * of the Software.
18 *
19 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
20 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
21 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
22 * IN NO EVENT SHALL VMWARE AND/OR ITS SUPPLIERS BE LIABLE FOR
23 * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
24 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
25 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
26 *
27 **************************************************************************/
28
29
30 #include "pipe/p_screen.h"
31 #include "pipe/p_state.h"
32 #include "util/u_memory.h"
33 #include "util/u_math.h"
34
35 #include "glhd_public.h"
36 #include "glhd_screen.h"
37 #include "glhd_context.h"
38 #include "glhd_objects.h"
39
40 DEBUG_GET_ONCE_BOOL_OPTION(galahad, "GALLIUM_GALAHAD", FALSE)
41
42 static void
43 galahad_screen_destroy(struct pipe_screen *_screen)
44 {
45 struct galahad_screen *glhd_screen = galahad_screen(_screen);
46 struct pipe_screen *screen = glhd_screen->screen;
47
48 screen->destroy(screen);
49
50 FREE(glhd_screen);
51 }
52
53 static const char *
54 galahad_screen_get_name(struct pipe_screen *_screen)
55 {
56 struct galahad_screen *glhd_screen = galahad_screen(_screen);
57 struct pipe_screen *screen = glhd_screen->screen;
58
59 return screen->get_name(screen);
60 }
61
62 static const char *
63 galahad_screen_get_vendor(struct pipe_screen *_screen)
64 {
65 struct galahad_screen *glhd_screen = galahad_screen(_screen);
66 struct pipe_screen *screen = glhd_screen->screen;
67
68 return screen->get_vendor(screen);
69 }
70
71 static int
72 galahad_screen_get_param(struct pipe_screen *_screen,
73 enum pipe_cap param)
74 {
75 struct galahad_screen *glhd_screen = galahad_screen(_screen);
76 struct pipe_screen *screen = glhd_screen->screen;
77
78 return screen->get_param(screen,
79 param);
80 }
81
82 static float
83 galahad_screen_get_paramf(struct pipe_screen *_screen,
84 enum pipe_cap param)
85 {
86 struct galahad_screen *glhd_screen = galahad_screen(_screen);
87 struct pipe_screen *screen = glhd_screen->screen;
88
89 return screen->get_paramf(screen,
90 param);
91 }
92
93 static boolean
94 galahad_screen_is_format_supported(struct pipe_screen *_screen,
95 enum pipe_format format,
96 enum pipe_texture_target target,
97 unsigned sample_count,
98 unsigned tex_usage,
99 unsigned geom_flags)
100 {
101 struct galahad_screen *glhd_screen = galahad_screen(_screen);
102 struct pipe_screen *screen = glhd_screen->screen;
103
104 if (target >= PIPE_MAX_TEXTURE_TYPES) {
105 glhd_warn("Received bogus texture target %d", target);
106 }
107
108 return screen->is_format_supported(screen,
109 format,
110 target,
111 sample_count,
112 tex_usage,
113 geom_flags);
114 }
115
116 static struct pipe_context *
117 galahad_screen_context_create(struct pipe_screen *_screen,
118 void *priv)
119 {
120 struct galahad_screen *glhd_screen = galahad_screen(_screen);
121 struct pipe_screen *screen = glhd_screen->screen;
122 struct pipe_context *result;
123
124 result = screen->context_create(screen, priv);
125 if (result)
126 return galahad_context_create(_screen, result);
127 return NULL;
128 }
129
130 static struct pipe_resource *
131 galahad_screen_resource_create(struct pipe_screen *_screen,
132 const struct pipe_resource *templat)
133 {
134 struct galahad_screen *glhd_screen = galahad_screen(_screen);
135 struct pipe_screen *screen = glhd_screen->screen;
136 struct pipe_resource *result;
137
138 if (templat->target >= PIPE_MAX_TEXTURE_TYPES)
139 glhd_warn("Received bogus resource target %d", templat->target);
140
141 if(templat->target != PIPE_TEXTURE_RECT && templat->target != PIPE_BUFFER && !screen->get_param(screen, PIPE_CAP_NPOT_TEXTURES))
142 {
143 if(!util_is_pot(templat->width0) || !util_is_pot(templat->height0))
144 glhd_warn("Requested NPOT (%ux%u) non-rectangle texture without NPOT support", templat->width0, templat->height0);
145 }
146
147 if(templat->target == PIPE_TEXTURE_RECT && templat->last_level)
148 glhd_warn("Rectangle textures cannot have mipmaps, but last_level = %u", templat->last_level);
149
150 if(templat->target == PIPE_BUFFER && templat->last_level)
151 glhd_warn("Buffers cannot have mipmaps, but last_level = %u", templat->last_level);
152
153 if(templat->target != PIPE_TEXTURE_3D && templat->depth0 != 1)
154 glhd_warn("Only 3D textures can have depth != 1, but received target %u and depth %u", templat->target, templat->depth0);
155
156 if(templat->target == PIPE_TEXTURE_1D && templat->height0 != 1)
157 glhd_warn("1D textures must have height 1 but got asked for height %u", templat->height0);
158
159 if(templat->target == PIPE_BUFFER && templat->height0 != 1)
160 glhd_warn("Buffers must have height 1 but got asked for height %u", templat->height0);
161
162 if(templat->target == PIPE_TEXTURE_CUBE && templat->width0 != templat->height0)
163 glhd_warn("Cube maps must be square, but got asked for %ux%u", templat->width0, templat->height0);
164
165 result = screen->resource_create(screen,
166 templat);
167
168 if (result)
169 return galahad_resource_create(glhd_screen, result);
170 return NULL;
171 }
172
173 static struct pipe_resource *
174 galahad_screen_resource_from_handle(struct pipe_screen *_screen,
175 const struct pipe_resource *templ,
176 struct winsys_handle *handle)
177 {
178 struct galahad_screen *glhd_screen = galahad_screen(_screen);
179 struct pipe_screen *screen = glhd_screen->screen;
180 struct pipe_resource *result;
181
182 /* TODO trace call */
183
184 result = screen->resource_from_handle(screen, templ, handle);
185
186 result = galahad_resource_create(galahad_screen(_screen), result);
187
188 return result;
189 }
190
191 static boolean
192 galahad_screen_resource_get_handle(struct pipe_screen *_screen,
193 struct pipe_resource *_resource,
194 struct winsys_handle *handle)
195 {
196 struct galahad_screen *glhd_screen = galahad_screen(_screen);
197 struct galahad_resource *glhd_resource = galahad_resource(_resource);
198 struct pipe_screen *screen = glhd_screen->screen;
199 struct pipe_resource *resource = glhd_resource->resource;
200
201 /* TODO trace call */
202
203 return screen->resource_get_handle(screen, resource, handle);
204 }
205
206
207
208 static void
209 galahad_screen_resource_destroy(struct pipe_screen *screen,
210 struct pipe_resource *_resource)
211 {
212 galahad_resource_destroy(galahad_resource(_resource));
213 }
214
215 static struct pipe_surface *
216 galahad_screen_get_tex_surface(struct pipe_screen *_screen,
217 struct pipe_resource *_resource,
218 unsigned face,
219 unsigned level,
220 unsigned zslice,
221 unsigned usage)
222 {
223 struct galahad_screen *glhd_screen = galahad_screen(_screen);
224 struct galahad_resource *glhd_resource = galahad_resource(_resource);
225 struct pipe_screen *screen = glhd_screen->screen;
226 struct pipe_resource *resource = glhd_resource->resource;
227 struct pipe_surface *result;
228
229 result = screen->get_tex_surface(screen,
230 resource,
231 face,
232 level,
233 zslice,
234 usage);
235
236 if (result)
237 return galahad_surface_create(glhd_resource, result);
238 return NULL;
239 }
240
241 static void
242 galahad_screen_tex_surface_destroy(struct pipe_surface *_surface)
243 {
244 galahad_surface_destroy(galahad_surface(_surface));
245 }
246
247
248
249 static struct pipe_resource *
250 galahad_screen_user_buffer_create(struct pipe_screen *_screen,
251 void *ptr,
252 unsigned bytes,
253 unsigned usage)
254 {
255 struct galahad_screen *glhd_screen = galahad_screen(_screen);
256 struct pipe_screen *screen = glhd_screen->screen;
257 struct pipe_resource *result;
258
259 result = screen->user_buffer_create(screen,
260 ptr,
261 bytes,
262 usage);
263
264 if (result)
265 return galahad_resource_create(glhd_screen, result);
266 return NULL;
267 }
268
269
270
271 static void
272 galahad_screen_flush_frontbuffer(struct pipe_screen *_screen,
273 struct pipe_surface *_surface,
274 void *context_private)
275 {
276 struct galahad_screen *glhd_screen = galahad_screen(_screen);
277 struct galahad_surface *glhd_surface = galahad_surface(_surface);
278 struct pipe_screen *screen = glhd_screen->screen;
279 struct pipe_surface *surface = glhd_surface->surface;
280
281 screen->flush_frontbuffer(screen,
282 surface,
283 context_private);
284 }
285
286 static void
287 galahad_screen_fence_reference(struct pipe_screen *_screen,
288 struct pipe_fence_handle **ptr,
289 struct pipe_fence_handle *fence)
290 {
291 struct galahad_screen *glhd_screen = galahad_screen(_screen);
292 struct pipe_screen *screen = glhd_screen->screen;
293
294 screen->fence_reference(screen,
295 ptr,
296 fence);
297 }
298
299 static int
300 galahad_screen_fence_signalled(struct pipe_screen *_screen,
301 struct pipe_fence_handle *fence,
302 unsigned flags)
303 {
304 struct galahad_screen *glhd_screen = galahad_screen(_screen);
305 struct pipe_screen *screen = glhd_screen->screen;
306
307 return screen->fence_signalled(screen,
308 fence,
309 flags);
310 }
311
312 static int
313 galahad_screen_fence_finish(struct pipe_screen *_screen,
314 struct pipe_fence_handle *fence,
315 unsigned flags)
316 {
317 struct galahad_screen *glhd_screen = galahad_screen(_screen);
318 struct pipe_screen *screen = glhd_screen->screen;
319
320 return screen->fence_finish(screen,
321 fence,
322 flags);
323 }
324
325 struct pipe_screen *
326 galahad_screen_create(struct pipe_screen *screen)
327 {
328 struct galahad_screen *glhd_screen;
329
330 if (!debug_get_option_galahad())
331 return screen;
332
333 glhd_screen = CALLOC_STRUCT(galahad_screen);
334 if (!glhd_screen) {
335 return screen;
336 }
337
338 glhd_screen->base.winsys = NULL;
339
340 glhd_screen->base.destroy = galahad_screen_destroy;
341 glhd_screen->base.get_name = galahad_screen_get_name;
342 glhd_screen->base.get_vendor = galahad_screen_get_vendor;
343 glhd_screen->base.get_param = galahad_screen_get_param;
344 glhd_screen->base.get_paramf = galahad_screen_get_paramf;
345 glhd_screen->base.is_format_supported = galahad_screen_is_format_supported;
346 glhd_screen->base.context_create = galahad_screen_context_create;
347 glhd_screen->base.resource_create = galahad_screen_resource_create;
348 glhd_screen->base.resource_from_handle = galahad_screen_resource_from_handle;
349 glhd_screen->base.resource_get_handle = galahad_screen_resource_get_handle;
350 glhd_screen->base.resource_destroy = galahad_screen_resource_destroy;
351 glhd_screen->base.get_tex_surface = galahad_screen_get_tex_surface;
352 glhd_screen->base.tex_surface_destroy = galahad_screen_tex_surface_destroy;
353 glhd_screen->base.user_buffer_create = galahad_screen_user_buffer_create;
354 glhd_screen->base.flush_frontbuffer = galahad_screen_flush_frontbuffer;
355 glhd_screen->base.fence_reference = galahad_screen_fence_reference;
356 glhd_screen->base.fence_signalled = galahad_screen_fence_signalled;
357 glhd_screen->base.fence_finish = galahad_screen_fence_finish;
358
359 glhd_screen->screen = screen;
360
361 glhd_warn("Created screen %p", glhd_screen);
362
363 return &glhd_screen->base;
364 }