c9f020c40f83987f4dcd60edf3724f64c59ab8d7
[mesa.git] / src / mesa / state_tracker / st_format.c
1 /**************************************************************************
2 *
3 * Copyright 2007 Tungsten Graphics, Inc., Cedar Park, Texas.
4 * Copyright (c) 2008 VMware, Inc.
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 TUNGSTEN GRAPHICS 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 /**
31 * Texture Image-related functions.
32 * \author Brian Paul
33 */
34
35 #include "main/imports.h"
36 #include "main/context.h"
37 #include "main/texstore.h"
38 #include "main/texformat.h"
39 #include "main/enums.h"
40 #include "main/macros.h"
41
42 #include "pipe/p_context.h"
43 #include "pipe/p_defines.h"
44 #include "pipe/p_screen.h"
45 #include "util/u_format.h"
46 #include "st_context.h"
47 #include "st_format.h"
48
49
50 static GLuint
51 format_max_bits(
52 pipe_format_rgbazs_t info )
53 {
54 GLuint size = util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_RGB, 0);
55
56 size = MAX2(size, util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_RGB, 1));
57 size = MAX2(size, util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_RGB, 2));
58 size = MAX2(size, util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_RGB, 3));
59 size = MAX2(size, util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_ZS, 0));
60 size = MAX2(size, util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_ZS, 1));
61 return size;
62 }
63
64 static GLuint
65 format_size(
66 pipe_format_rgbazs_t info )
67 {
68 return
69 util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_RGB, 0) +
70 util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_RGB, 1) +
71 util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_RGB, 2) +
72 util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_RGB, 3) +
73 util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_ZS, 0) +
74 util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_ZS, 1);
75 }
76
77 /*
78 * XXX temporary here
79 */
80 GLboolean
81 st_get_format_info(enum pipe_format format, struct pipe_format_info *pinfo)
82 {
83 if (pf_layout(format) == PIPE_FORMAT_LAYOUT_RGBAZS) {
84 const struct util_format_description *desc = util_format_description(format);
85 pipe_format_rgbazs_t info;
86
87 assert(desc);
88
89 info = format;
90
91 #if 0
92 printf("%s\n", pf_name( format ) );
93 #endif
94
95 /* Data type */
96 if (format == PIPE_FORMAT_A1R5G5B5_UNORM || format == PIPE_FORMAT_R5G6B5_UNORM) {
97 pinfo->datatype = GL_UNSIGNED_SHORT;
98 }
99 else if (format == PIPE_FORMAT_S8Z24_UNORM) {
100 pinfo->datatype = GL_UNSIGNED_INT_24_8;
101 }
102 else {
103 const GLuint size = format_max_bits( info );
104 if (size == 8) {
105 if (pf_type(info) == PIPE_FORMAT_TYPE_UNORM)
106 pinfo->datatype = GL_UNSIGNED_BYTE;
107 else
108 pinfo->datatype = GL_BYTE;
109 }
110 else if (size == 16) {
111 if (pf_type(info) == PIPE_FORMAT_TYPE_UNORM)
112 pinfo->datatype = GL_UNSIGNED_SHORT;
113 else
114 pinfo->datatype = GL_SHORT;
115 }
116 else {
117 assert( size <= 32 );
118 if (pf_type(info) == PIPE_FORMAT_TYPE_UNORM)
119 pinfo->datatype = GL_UNSIGNED_INT;
120 else
121 pinfo->datatype = GL_INT;
122 }
123 }
124
125 /* Component bits */
126 pinfo->red_bits = util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_RGB, 0);
127 pinfo->green_bits = util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_RGB, 1);
128 pinfo->blue_bits = util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_RGB, 2);
129 pinfo->alpha_bits = util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_RGB, 3);
130 pinfo->depth_bits = util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_ZS, 0);
131 pinfo->stencil_bits = util_format_get_component_bits((enum pipe_format)info, UTIL_FORMAT_COLORSPACE_ZS, 1);
132 pinfo->luminance_bits = 0;
133 pinfo->intensity_bits = 0;
134
135 /* Format size */
136 pinfo->size = format_size( info ) / 8;
137
138 /* Luminance & Intensity bits */
139 if (desc->swizzle[0] == UTIL_FORMAT_SWIZZLE_X &&
140 desc->swizzle[1] == UTIL_FORMAT_SWIZZLE_X &&
141 desc->swizzle[2] == UTIL_FORMAT_SWIZZLE_X) {
142 if (desc->swizzle[3] == UTIL_FORMAT_SWIZZLE_X) {
143 pinfo->intensity_bits = pinfo->red_bits;
144 }
145 else {
146 pinfo->luminance_bits = pinfo->red_bits;
147 }
148 pinfo->red_bits = 0;
149 }
150
151 pinfo->mesa_format = st_pipe_format_to_mesa_format(format);
152 }
153 else if (pf_layout(format) == PIPE_FORMAT_LAYOUT_YCBCR) {
154 pinfo->mesa_format = MESA_FORMAT_YCBCR;
155 pinfo->datatype = GL_UNSIGNED_SHORT;
156 pinfo->size = 2; /* two bytes per "texel" */
157 }
158 else {
159 /* compressed format? */
160 assert(0);
161 }
162
163 #if 0
164 printf(
165 "ST_FORMAT: R(%u), G(%u), B(%u), A(%u), Z(%u), S(%u)\n",
166 pinfo->red_bits,
167 pinfo->green_bits,
168 pinfo->blue_bits,
169 pinfo->alpha_bits,
170 pinfo->depth_bits,
171 pinfo->stencil_bits );
172 #endif
173
174 pinfo->format = format;
175
176 return GL_TRUE;
177 }
178
179
180 /**
181 * Return bytes per pixel for the given format.
182 */
183 GLuint
184 st_sizeof_format(enum pipe_format format)
185 {
186 struct pipe_format_info info;
187 if (!st_get_format_info( format, &info )) {
188 assert( 0 );
189 return 0;
190 }
191 return info.size;
192 }
193
194
195 /**
196 * Return bytes per pixel for the given format.
197 */
198 GLenum
199 st_format_datatype(enum pipe_format format)
200 {
201 struct pipe_format_info info;
202 if (!st_get_format_info( format, &info )) {
203 assert( 0 );
204 return 0;
205 }
206 return info.datatype;
207 }
208
209
210 enum pipe_format
211 st_mesa_format_to_pipe_format(gl_format mesaFormat)
212 {
213 switch (mesaFormat) {
214 /* fix this */
215 case MESA_FORMAT_ARGB8888_REV:
216 case MESA_FORMAT_ARGB8888:
217 return PIPE_FORMAT_A8R8G8B8_UNORM;
218 case MESA_FORMAT_XRGB8888:
219 return PIPE_FORMAT_X8R8G8B8_UNORM;
220 case MESA_FORMAT_ARGB1555:
221 return PIPE_FORMAT_A1R5G5B5_UNORM;
222 case MESA_FORMAT_ARGB4444:
223 return PIPE_FORMAT_A4R4G4B4_UNORM;
224 case MESA_FORMAT_RGB565:
225 return PIPE_FORMAT_R5G6B5_UNORM;
226 case MESA_FORMAT_AL88:
227 return PIPE_FORMAT_A8L8_UNORM;
228 case MESA_FORMAT_A8:
229 return PIPE_FORMAT_A8_UNORM;
230 case MESA_FORMAT_L8:
231 return PIPE_FORMAT_L8_UNORM;
232 case MESA_FORMAT_I8:
233 return PIPE_FORMAT_I8_UNORM;
234 case MESA_FORMAT_Z16:
235 return PIPE_FORMAT_Z16_UNORM;
236 case MESA_FORMAT_Z32:
237 return PIPE_FORMAT_Z32_UNORM;
238 case MESA_FORMAT_Z24_S8:
239 return PIPE_FORMAT_Z24S8_UNORM;
240 case MESA_FORMAT_S8_Z24:
241 return PIPE_FORMAT_S8Z24_UNORM;
242 case MESA_FORMAT_YCBCR:
243 return PIPE_FORMAT_YCBCR;
244 #if FEATURE_texture_s3tc
245 case MESA_FORMAT_RGB_DXT1:
246 return PIPE_FORMAT_DXT1_RGB;
247 case MESA_FORMAT_RGBA_DXT1:
248 return PIPE_FORMAT_DXT1_RGBA;
249 case MESA_FORMAT_RGBA_DXT3:
250 return PIPE_FORMAT_DXT3_RGBA;
251 case MESA_FORMAT_RGBA_DXT5:
252 return PIPE_FORMAT_DXT5_RGBA;
253 #if FEATURE_EXT_texture_sRGB
254 case MESA_FORMAT_SRGB_DXT1:
255 return PIPE_FORMAT_DXT1_SRGB;
256 case MESA_FORMAT_SRGBA_DXT1:
257 return PIPE_FORMAT_DXT1_SRGBA;
258 case MESA_FORMAT_SRGBA_DXT3:
259 return PIPE_FORMAT_DXT3_SRGBA;
260 case MESA_FORMAT_SRGBA_DXT5:
261 return PIPE_FORMAT_DXT5_SRGBA;
262 #endif
263 #endif
264 #if FEATURE_EXT_texture_sRGB
265 case MESA_FORMAT_SLA8:
266 return PIPE_FORMAT_A8L8_SRGB;
267 case MESA_FORMAT_SL8:
268 return PIPE_FORMAT_L8_SRGB;
269 case MESA_FORMAT_SRGB8:
270 return PIPE_FORMAT_R8G8B8_SRGB;
271 case MESA_FORMAT_SRGBA8:
272 return PIPE_FORMAT_R8G8B8A8_SRGB;
273 case MESA_FORMAT_SARGB8:
274 return PIPE_FORMAT_A8R8G8B8_SRGB;
275 #endif
276 default:
277 assert(0);
278 return 0;
279 }
280 }
281
282
283 gl_format
284 st_pipe_format_to_mesa_format(enum pipe_format pipeFormat)
285 {
286 switch (pipeFormat) {
287 case PIPE_FORMAT_A8R8G8B8_UNORM:
288 return MESA_FORMAT_ARGB8888;
289 case PIPE_FORMAT_X8R8G8B8_UNORM:
290 return MESA_FORMAT_XRGB8888;
291 case PIPE_FORMAT_A1R5G5B5_UNORM:
292 return MESA_FORMAT_ARGB1555;
293 case PIPE_FORMAT_A4R4G4B4_UNORM:
294 return MESA_FORMAT_ARGB4444;
295 case PIPE_FORMAT_R5G6B5_UNORM:
296 return MESA_FORMAT_RGB565;
297 case PIPE_FORMAT_A8L8_UNORM:
298 return MESA_FORMAT_AL88;
299 case PIPE_FORMAT_A8_UNORM:
300 return MESA_FORMAT_A8;
301 case PIPE_FORMAT_L8_UNORM:
302 return MESA_FORMAT_L8;
303 case PIPE_FORMAT_I8_UNORM:
304 return MESA_FORMAT_I8;
305 case PIPE_FORMAT_Z16_UNORM:
306 return MESA_FORMAT_Z16;
307 case PIPE_FORMAT_Z32_UNORM:
308 return MESA_FORMAT_Z32;
309 case PIPE_FORMAT_Z24X8_UNORM:
310 return MESA_FORMAT_Z24_X8;
311 case PIPE_FORMAT_Z24S8_UNORM:
312 return MESA_FORMAT_Z24_S8;
313 case PIPE_FORMAT_X8Z24_UNORM:
314 return MESA_FORMAT_X8_Z24;
315 case PIPE_FORMAT_S8Z24_UNORM:
316 return MESA_FORMAT_S8_Z24;
317 case PIPE_FORMAT_S8_UNORM:
318 return MESA_FORMAT_S8;
319
320 case PIPE_FORMAT_YCBCR:
321 return MESA_FORMAT_YCBCR;
322 case PIPE_FORMAT_R16G16B16A16_SNORM:
323 return MESA_FORMAT_SIGNED_RGBA_16;
324
325 #if FEATURE_texture_s3tc
326 case PIPE_FORMAT_DXT1_RGB:
327 return MESA_FORMAT_RGB_DXT1;
328 case PIPE_FORMAT_DXT1_RGBA:
329 return MESA_FORMAT_RGBA_DXT1;
330 case PIPE_FORMAT_DXT3_RGBA:
331 return MESA_FORMAT_RGBA_DXT3;
332 case PIPE_FORMAT_DXT5_RGBA:
333 return MESA_FORMAT_RGBA_DXT5;
334 #if FEATURE_EXT_texture_sRGB
335 case PIPE_FORMAT_DXT1_SRGB:
336 return MESA_FORMAT_SRGB_DXT1;
337 case PIPE_FORMAT_DXT1_SRGBA:
338 return MESA_FORMAT_SRGBA_DXT1;
339 case PIPE_FORMAT_DXT3_SRGBA:
340 return MESA_FORMAT_SRGBA_DXT3;
341 case PIPE_FORMAT_DXT5_SRGBA:
342 return MESA_FORMAT_SRGBA_DXT5;
343 #endif
344 #endif
345 #if FEATURE_EXT_texture_sRGB
346 case PIPE_FORMAT_A8L8_SRGB:
347 return MESA_FORMAT_SLA8;
348 case PIPE_FORMAT_L8_SRGB:
349 return MESA_FORMAT_SL8;
350 case PIPE_FORMAT_R8G8B8_SRGB:
351 return MESA_FORMAT_SRGB8;
352 case PIPE_FORMAT_R8G8B8A8_SRGB:
353 return MESA_FORMAT_SRGBA8;
354 case PIPE_FORMAT_A8R8G8B8_SRGB:
355 return MESA_FORMAT_SARGB8;
356 #endif
357 default:
358 assert(0);
359 return 0;
360 }
361 }
362
363
364 /**
365 * Find an RGBA format supported by the context/winsys.
366 */
367 static enum pipe_format
368 default_rgba_format(struct pipe_screen *screen,
369 enum pipe_texture_target target,
370 unsigned tex_usage,
371 unsigned geom_flags)
372 {
373 static const enum pipe_format colorFormats[] = {
374 PIPE_FORMAT_A8R8G8B8_UNORM,
375 PIPE_FORMAT_B8G8R8A8_UNORM,
376 PIPE_FORMAT_R8G8B8A8_UNORM,
377 PIPE_FORMAT_R5G6B5_UNORM
378 };
379 uint i;
380 for (i = 0; i < Elements(colorFormats); i++) {
381 if (screen->is_format_supported( screen, colorFormats[i], target, tex_usage, geom_flags )) {
382 return colorFormats[i];
383 }
384 }
385 return PIPE_FORMAT_NONE;
386 }
387
388 /**
389 * Find an sRGBA format supported by the context/winsys.
390 */
391 static enum pipe_format
392 default_srgba_format(struct pipe_screen *screen,
393 enum pipe_texture_target target,
394 unsigned tex_usage,
395 unsigned geom_flags)
396 {
397 static const enum pipe_format colorFormats[] = {
398 PIPE_FORMAT_A8R8G8B8_SRGB,
399 PIPE_FORMAT_B8G8R8A8_SRGB,
400 PIPE_FORMAT_R8G8B8A8_SRGB,
401 };
402 uint i;
403 for (i = 0; i < Elements(colorFormats); i++) {
404 if (screen->is_format_supported( screen, colorFormats[i], target, tex_usage, geom_flags )) {
405 return colorFormats[i];
406 }
407 }
408 return PIPE_FORMAT_NONE;
409 }
410
411 /**
412 * Search list of formats for first RGBA format with >8 bits/channel.
413 */
414 static enum pipe_format
415 default_deep_rgba_format(struct pipe_screen *screen,
416 enum pipe_texture_target target,
417 unsigned tex_usage,
418 unsigned geom_flags)
419 {
420 if (screen->is_format_supported(screen, PIPE_FORMAT_R16G16B16A16_SNORM, target, tex_usage, geom_flags)) {
421 return PIPE_FORMAT_R16G16B16A16_SNORM;
422 }
423 if (tex_usage & PIPE_TEXTURE_USAGE_RENDER_TARGET)
424 return default_rgba_format(screen, target, tex_usage, geom_flags);
425 else
426 return PIPE_FORMAT_NONE;
427 }
428
429
430 /**
431 * Find an Z format supported by the context/winsys.
432 */
433 static enum pipe_format
434 default_depth_format(struct pipe_screen *screen,
435 enum pipe_texture_target target,
436 unsigned tex_usage,
437 unsigned geom_flags)
438 {
439 static const enum pipe_format zFormats[] = {
440 PIPE_FORMAT_Z16_UNORM,
441 PIPE_FORMAT_Z32_UNORM,
442 PIPE_FORMAT_S8Z24_UNORM,
443 PIPE_FORMAT_Z24S8_UNORM
444 };
445 uint i;
446 for (i = 0; i < Elements(zFormats); i++) {
447 if (screen->is_format_supported( screen, zFormats[i], target, tex_usage, geom_flags )) {
448 return zFormats[i];
449 }
450 }
451 return PIPE_FORMAT_NONE;
452 }
453
454
455 /**
456 * Given an OpenGL internalFormat value for a texture or surface, return
457 * the best matching PIPE_FORMAT_x, or PIPE_FORMAT_NONE if there's no match.
458 * \param target one of PIPE_TEXTURE_x
459 * \param tex_usage either PIPE_TEXTURE_USAGE_RENDER_TARGET
460 * or PIPE_TEXTURE_USAGE_SAMPLER
461 */
462 enum pipe_format
463 st_choose_format(struct pipe_screen *screen, GLenum internalFormat,
464 enum pipe_texture_target target, unsigned tex_usage)
465 {
466 unsigned geom_flags = 0;
467
468 switch (internalFormat) {
469 case 4:
470 case GL_RGBA:
471 case GL_COMPRESSED_RGBA:
472 case 3:
473 case GL_RGB:
474 case GL_COMPRESSED_RGB:
475 case GL_RGBA8:
476 case GL_RGB10_A2:
477 case GL_RGBA12:
478 return default_rgba_format( screen, target, tex_usage, geom_flags );
479 case GL_RGBA16:
480 if (tex_usage & PIPE_TEXTURE_USAGE_RENDER_TARGET)
481 return default_deep_rgba_format( screen, target, tex_usage, geom_flags );
482 else
483 return default_rgba_format( screen, target, tex_usage, geom_flags );
484
485 case GL_RGBA4:
486 case GL_RGBA2:
487 if (screen->is_format_supported( screen, PIPE_FORMAT_A4R4G4B4_UNORM, target, tex_usage, geom_flags ))
488 return PIPE_FORMAT_A4R4G4B4_UNORM;
489 return default_rgba_format( screen, target, tex_usage, geom_flags );
490
491 case GL_RGB5_A1:
492 if (screen->is_format_supported( screen, PIPE_FORMAT_A1R5G5B5_UNORM, target, tex_usage, geom_flags ))
493 return PIPE_FORMAT_A1R5G5B5_UNORM;
494 return default_rgba_format( screen, target, tex_usage, geom_flags );
495
496 case GL_RGB8:
497 case GL_RGB10:
498 case GL_RGB12:
499 case GL_RGB16:
500 return default_rgba_format( screen, target, tex_usage, geom_flags );
501
502 case GL_RGB5:
503 case GL_RGB4:
504 case GL_R3_G3_B2:
505 if (screen->is_format_supported( screen, PIPE_FORMAT_R5G6B5_UNORM, target, tex_usage, geom_flags ))
506 return PIPE_FORMAT_R5G6B5_UNORM;
507 if (screen->is_format_supported( screen, PIPE_FORMAT_A1R5G5B5_UNORM, target, tex_usage, geom_flags ))
508 return PIPE_FORMAT_A1R5G5B5_UNORM;
509 return default_rgba_format( screen, target, tex_usage, geom_flags );
510
511 case GL_ALPHA:
512 case GL_ALPHA4:
513 case GL_ALPHA8:
514 case GL_ALPHA12:
515 case GL_ALPHA16:
516 case GL_COMPRESSED_ALPHA:
517 if (screen->is_format_supported( screen, PIPE_FORMAT_A8_UNORM, target, tex_usage, geom_flags ))
518 return PIPE_FORMAT_A8_UNORM;
519 return default_rgba_format( screen, target, tex_usage, geom_flags );
520
521 case 1:
522 case GL_LUMINANCE:
523 case GL_LUMINANCE4:
524 case GL_LUMINANCE8:
525 case GL_LUMINANCE12:
526 case GL_LUMINANCE16:
527 case GL_COMPRESSED_LUMINANCE:
528 if (screen->is_format_supported( screen, PIPE_FORMAT_L8_UNORM, target, tex_usage, geom_flags ))
529 return PIPE_FORMAT_L8_UNORM;
530 return default_rgba_format( screen, target, tex_usage, geom_flags );
531
532 case 2:
533 case GL_LUMINANCE_ALPHA:
534 case GL_LUMINANCE4_ALPHA4:
535 case GL_LUMINANCE6_ALPHA2:
536 case GL_LUMINANCE8_ALPHA8:
537 case GL_LUMINANCE12_ALPHA4:
538 case GL_LUMINANCE12_ALPHA12:
539 case GL_LUMINANCE16_ALPHA16:
540 case GL_COMPRESSED_LUMINANCE_ALPHA:
541 if (screen->is_format_supported( screen, PIPE_FORMAT_A8L8_UNORM, target, tex_usage, geom_flags ))
542 return PIPE_FORMAT_A8L8_UNORM;
543 return default_rgba_format( screen, target, tex_usage, geom_flags );
544
545 case GL_INTENSITY:
546 case GL_INTENSITY4:
547 case GL_INTENSITY8:
548 case GL_INTENSITY12:
549 case GL_INTENSITY16:
550 case GL_COMPRESSED_INTENSITY:
551 if (screen->is_format_supported( screen, PIPE_FORMAT_I8_UNORM, target, tex_usage, geom_flags ))
552 return PIPE_FORMAT_I8_UNORM;
553 return default_rgba_format( screen, target, tex_usage, geom_flags );
554
555 case GL_YCBCR_MESA:
556 if (screen->is_format_supported(screen, PIPE_FORMAT_YCBCR,
557 target, tex_usage, geom_flags)) {
558 return PIPE_FORMAT_YCBCR;
559 }
560 if (screen->is_format_supported(screen, PIPE_FORMAT_YCBCR_REV,
561 target, tex_usage, geom_flags)) {
562 return PIPE_FORMAT_YCBCR_REV;
563 }
564 return PIPE_FORMAT_NONE;
565
566 case GL_RGB_S3TC:
567 case GL_RGB4_S3TC:
568 case GL_COMPRESSED_RGB_S3TC_DXT1_EXT:
569 return PIPE_FORMAT_DXT1_RGB;
570
571 case GL_COMPRESSED_RGBA_S3TC_DXT1_EXT:
572 return PIPE_FORMAT_DXT1_RGBA;
573
574 case GL_RGBA_S3TC:
575 case GL_RGBA4_S3TC:
576 case GL_COMPRESSED_RGBA_S3TC_DXT3_EXT:
577 return PIPE_FORMAT_DXT3_RGBA;
578
579 case GL_COMPRESSED_RGBA_S3TC_DXT5_EXT:
580 return PIPE_FORMAT_DXT5_RGBA;
581
582 #if 0
583 case GL_COMPRESSED_RGB_FXT1_3DFX:
584 return PIPE_FORMAT_RGB_FXT1;
585 case GL_COMPRESSED_RGBA_FXT1_3DFX:
586 return PIPE_FORMAT_RGB_FXT1;
587 #endif
588
589 case GL_DEPTH_COMPONENT16:
590 if (screen->is_format_supported( screen, PIPE_FORMAT_Z16_UNORM, target, tex_usage, geom_flags ))
591 return PIPE_FORMAT_Z16_UNORM;
592 /* fall-through */
593 case GL_DEPTH_COMPONENT24:
594 if (screen->is_format_supported( screen, PIPE_FORMAT_S8Z24_UNORM, target, tex_usage, geom_flags ))
595 return PIPE_FORMAT_S8Z24_UNORM;
596 if (screen->is_format_supported( screen, PIPE_FORMAT_Z24S8_UNORM, target, tex_usage, geom_flags ))
597 return PIPE_FORMAT_Z24S8_UNORM;
598 /* fall-through */
599 case GL_DEPTH_COMPONENT32:
600 if (screen->is_format_supported( screen, PIPE_FORMAT_Z32_UNORM, target, tex_usage, geom_flags ))
601 return PIPE_FORMAT_Z32_UNORM;
602 /* fall-through */
603 case GL_DEPTH_COMPONENT:
604 return default_depth_format( screen, target, tex_usage, geom_flags );
605
606 case GL_STENCIL_INDEX:
607 case GL_STENCIL_INDEX1_EXT:
608 case GL_STENCIL_INDEX4_EXT:
609 case GL_STENCIL_INDEX8_EXT:
610 case GL_STENCIL_INDEX16_EXT:
611 if (screen->is_format_supported( screen, PIPE_FORMAT_S8_UNORM, target, tex_usage, geom_flags ))
612 return PIPE_FORMAT_S8_UNORM;
613 if (screen->is_format_supported( screen, PIPE_FORMAT_S8Z24_UNORM, target, tex_usage, geom_flags ))
614 return PIPE_FORMAT_S8Z24_UNORM;
615 if (screen->is_format_supported( screen, PIPE_FORMAT_Z24S8_UNORM, target, tex_usage, geom_flags ))
616 return PIPE_FORMAT_Z24S8_UNORM;
617 return PIPE_FORMAT_NONE;
618
619 case GL_DEPTH_STENCIL_EXT:
620 case GL_DEPTH24_STENCIL8_EXT:
621 if (screen->is_format_supported( screen, PIPE_FORMAT_S8Z24_UNORM, target, tex_usage, geom_flags ))
622 return PIPE_FORMAT_S8Z24_UNORM;
623 if (screen->is_format_supported( screen, PIPE_FORMAT_Z24S8_UNORM, target, tex_usage, geom_flags ))
624 return PIPE_FORMAT_Z24S8_UNORM;
625 return PIPE_FORMAT_NONE;
626
627 case GL_SRGB_EXT:
628 case GL_SRGB8_EXT:
629 case GL_COMPRESSED_SRGB_EXT:
630 case GL_COMPRESSED_SRGB_ALPHA_EXT:
631 case GL_SRGB_ALPHA_EXT:
632 case GL_SRGB8_ALPHA8_EXT:
633 return default_srgba_format( screen, target, tex_usage, geom_flags );
634 case GL_COMPRESSED_SRGB_S3TC_DXT1_EXT:
635 return PIPE_FORMAT_DXT1_SRGB;
636 case GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT:
637 return PIPE_FORMAT_DXT1_SRGBA;
638 case GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT3_EXT:
639 return PIPE_FORMAT_DXT3_SRGBA;
640 case GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT:
641 return PIPE_FORMAT_DXT5_SRGBA;
642
643 case GL_SLUMINANCE_ALPHA_EXT:
644 case GL_SLUMINANCE8_ALPHA8_EXT:
645 case GL_COMPRESSED_SLUMINANCE_EXT:
646 case GL_COMPRESSED_SLUMINANCE_ALPHA_EXT:
647 if (screen->is_format_supported( screen, PIPE_FORMAT_A8L8_SRGB, target, tex_usage, geom_flags ))
648 return PIPE_FORMAT_A8L8_SRGB;
649 return default_srgba_format( screen, target, tex_usage, geom_flags );
650
651 case GL_SLUMINANCE_EXT:
652 case GL_SLUMINANCE8_EXT:
653 if (screen->is_format_supported( screen, PIPE_FORMAT_L8_SRGB, target, tex_usage, geom_flags ))
654 return PIPE_FORMAT_L8_SRGB;
655 return default_srgba_format( screen, target, tex_usage, geom_flags );
656
657 default:
658 return PIPE_FORMAT_NONE;
659 }
660 }
661
662
663 static GLboolean
664 is_depth_or_stencil_format(GLenum internalFormat)
665 {
666 switch (internalFormat) {
667 case GL_DEPTH_COMPONENT:
668 case GL_DEPTH_COMPONENT16:
669 case GL_DEPTH_COMPONENT24:
670 case GL_DEPTH_COMPONENT32:
671 case GL_STENCIL_INDEX:
672 case GL_STENCIL_INDEX1_EXT:
673 case GL_STENCIL_INDEX4_EXT:
674 case GL_STENCIL_INDEX8_EXT:
675 case GL_STENCIL_INDEX16_EXT:
676 case GL_DEPTH_STENCIL_EXT:
677 case GL_DEPTH24_STENCIL8_EXT:
678 return GL_TRUE;
679 default:
680 return GL_FALSE;
681 }
682 }
683
684 /**
685 * Called by FBO code to choose a PIPE_FORMAT_ for drawing surfaces.
686 */
687 enum pipe_format
688 st_choose_renderbuffer_format(struct pipe_screen *screen,
689 GLenum internalFormat)
690 {
691 uint usage;
692 if (is_depth_or_stencil_format(internalFormat))
693 usage = PIPE_TEXTURE_USAGE_DEPTH_STENCIL;
694 else
695 usage = PIPE_TEXTURE_USAGE_RENDER_TARGET;
696 return st_choose_format(screen, internalFormat, PIPE_TEXTURE_2D, usage);
697 }
698
699
700 static gl_format
701 translate_gallium_format_to_mesa_format(enum pipe_format format)
702 {
703 switch (format) {
704 case PIPE_FORMAT_A8R8G8B8_UNORM:
705 return MESA_FORMAT_ARGB8888;
706 case PIPE_FORMAT_X8R8G8B8_UNORM:
707 return MESA_FORMAT_XRGB8888;
708 case PIPE_FORMAT_A1R5G5B5_UNORM:
709 return MESA_FORMAT_ARGB1555;
710 case PIPE_FORMAT_A4R4G4B4_UNORM:
711 return MESA_FORMAT_ARGB4444;
712 case PIPE_FORMAT_R5G6B5_UNORM:
713 return MESA_FORMAT_RGB565;
714 case PIPE_FORMAT_A8L8_UNORM:
715 return MESA_FORMAT_AL88;
716 case PIPE_FORMAT_A8_UNORM:
717 return MESA_FORMAT_A8;
718 case PIPE_FORMAT_L8_UNORM:
719 return MESA_FORMAT_L8;
720 case PIPE_FORMAT_I8_UNORM:
721 return MESA_FORMAT_I8;
722 case PIPE_FORMAT_Z16_UNORM:
723 return MESA_FORMAT_Z16;
724 case PIPE_FORMAT_Z32_UNORM:
725 return MESA_FORMAT_Z32;
726 case PIPE_FORMAT_Z24S8_UNORM:
727 return MESA_FORMAT_Z24_S8;
728 case PIPE_FORMAT_X8Z24_UNORM:
729 return MESA_FORMAT_X8_Z24;
730 case PIPE_FORMAT_S8Z24_UNORM:
731 return MESA_FORMAT_S8_Z24;
732 case PIPE_FORMAT_YCBCR:
733 return MESA_FORMAT_YCBCR;
734 case PIPE_FORMAT_YCBCR_REV:
735 return MESA_FORMAT_YCBCR_REV;
736 #if FEATURE_texture_s3tc
737 case PIPE_FORMAT_DXT1_RGB:
738 return MESA_FORMAT_RGB_DXT1;
739 case PIPE_FORMAT_DXT1_RGBA:
740 return MESA_FORMAT_RGBA_DXT1;
741 case PIPE_FORMAT_DXT3_RGBA:
742 return MESA_FORMAT_RGBA_DXT3;
743 case PIPE_FORMAT_DXT5_RGBA:
744 return MESA_FORMAT_RGBA_DXT5;
745 #if FEATURE_EXT_texture_sRGB
746 case PIPE_FORMAT_DXT1_SRGB:
747 return MESA_FORMAT_SRGB_DXT1;
748 case PIPE_FORMAT_DXT1_SRGBA:
749 return MESA_FORMAT_SRGBA_DXT1;
750 case PIPE_FORMAT_DXT3_SRGBA:
751 return MESA_FORMAT_SRGBA_DXT3;
752 case PIPE_FORMAT_DXT5_SRGBA:
753 return MESA_FORMAT_SRGBA_DXT5;
754 #endif
755 #endif
756 #if FEATURE_EXT_texture_sRGB
757 case PIPE_FORMAT_A8L8_SRGB:
758 return MESA_FORMAT_SLA8;
759 case PIPE_FORMAT_L8_SRGB:
760 return MESA_FORMAT_SL8;
761 case PIPE_FORMAT_R8G8B8_SRGB:
762 return MESA_FORMAT_SRGB8;
763 case PIPE_FORMAT_R8G8B8A8_SRGB:
764 return MESA_FORMAT_SRGBA8;
765 case PIPE_FORMAT_A8R8G8B8_SRGB:
766 return MESA_FORMAT_SARGB8;
767 #endif
768 /* XXX add additional cases */
769 default:
770 assert(0);
771 return MESA_FORMAT_NONE;
772 }
773 }
774
775
776 /**
777 * Called via ctx->Driver.chooseTextureFormat().
778 */
779 gl_format
780 st_ChooseTextureFormat(GLcontext *ctx, GLint internalFormat,
781 GLenum format, GLenum type)
782 {
783 enum pipe_format pFormat;
784
785 (void) format;
786 (void) type;
787
788 pFormat = st_choose_format(ctx->st->pipe->screen, internalFormat,
789 PIPE_TEXTURE_2D, PIPE_TEXTURE_USAGE_SAMPLER);
790 if (pFormat == PIPE_FORMAT_NONE)
791 return MESA_FORMAT_NONE;
792
793 return translate_gallium_format_to_mesa_format(pFormat);
794 }
795
796
797 /**
798 * Test if a gallium format is equivalent to a GL format/type.
799 */
800 GLboolean
801 st_equal_formats(enum pipe_format pFormat, GLenum format, GLenum type)
802 {
803 switch (pFormat) {
804 case PIPE_FORMAT_R8G8B8A8_UNORM:
805 return format == GL_RGBA && type == GL_UNSIGNED_BYTE;
806 case PIPE_FORMAT_B8G8R8A8_UNORM:
807 return format == GL_BGRA && type == GL_UNSIGNED_BYTE;
808 case PIPE_FORMAT_R5G6B5_UNORM:
809 return format == GL_RGB && type == GL_UNSIGNED_SHORT_5_6_5;
810 /* XXX more combos... */
811 default:
812 return GL_FALSE;
813 }
814 }