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