xref: /OK3568_Linux_fs/external/xserver/glx/extension_string.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun  * (C) Copyright IBM Corporation 2002-2006
3*4882a593Smuzhiyun  * All Rights Reserved.
4*4882a593Smuzhiyun  *
5*4882a593Smuzhiyun  * Permission is hereby granted, free of charge, to any person obtaining a
6*4882a593Smuzhiyun  * copy of this software and associated documentation files (the "Software"),
7*4882a593Smuzhiyun  * to deal in the Software without restriction, including without limitation
8*4882a593Smuzhiyun  * on the rights to use, copy, modify, merge, publish, distribute, sub
9*4882a593Smuzhiyun  * license, and/or sell copies of the Software, and to permit persons to whom
10*4882a593Smuzhiyun  * the Software is furnished to do so, subject to the following conditions:
11*4882a593Smuzhiyun  *
12*4882a593Smuzhiyun  * The above copyright notice and this permission notice (including the next
13*4882a593Smuzhiyun  * paragraph) shall be included in all copies or substantial portions of the
14*4882a593Smuzhiyun  * Software.
15*4882a593Smuzhiyun  *
16*4882a593Smuzhiyun  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17*4882a593Smuzhiyun  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18*4882a593Smuzhiyun  * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.  IN NO EVENT SHALL
19*4882a593Smuzhiyun  * THE COPYRIGHT HOLDERS AND/OR THEIR SUPPLIERS BE LIABLE FOR ANY CLAIM,
20*4882a593Smuzhiyun  * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
21*4882a593Smuzhiyun  * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
22*4882a593Smuzhiyun  * USE OR OTHER DEALINGS IN THE SOFTWARE.
23*4882a593Smuzhiyun  */
24*4882a593Smuzhiyun 
25*4882a593Smuzhiyun /**
26*4882a593Smuzhiyun  * \file extension_string.c
27*4882a593Smuzhiyun  * Routines to manage the GLX extension string and GLX version for AIGLX
28*4882a593Smuzhiyun  * drivers.  This code is loosely based on src/glx/x11/glxextensions.c from
29*4882a593Smuzhiyun  * Mesa.
30*4882a593Smuzhiyun  *
31*4882a593Smuzhiyun  * \author Ian Romanick <idr@us.ibm.com>
32*4882a593Smuzhiyun  */
33*4882a593Smuzhiyun 
34*4882a593Smuzhiyun #include <string.h>
35*4882a593Smuzhiyun #include "extension_string.h"
36*4882a593Smuzhiyun 
37*4882a593Smuzhiyun #define SET_BIT(m,b)    (m[ (b) / 8 ] |=  (1U << ((b) % 8)))
38*4882a593Smuzhiyun #define CLR_BIT(m,b)    (m[ (b) / 8 ] &= ~(1U << ((b) % 8)))
39*4882a593Smuzhiyun #define IS_SET(m,b)    ((m[ (b) / 8 ] &   (1U << ((b) % 8))) != 0)
40*4882a593Smuzhiyun #define CONCAT(a,b) a ## b
41*4882a593Smuzhiyun #define GLX(n) "GLX_" # n, 4 + sizeof( # n ) - 1, CONCAT(n,_bit)
42*4882a593Smuzhiyun #define VER(a,b)  a, b
43*4882a593Smuzhiyun #define Y  1
44*4882a593Smuzhiyun #define N  0
45*4882a593Smuzhiyun #define EXT_ENABLED(bit,supported) (IS_SET(supported, bit))
46*4882a593Smuzhiyun 
47*4882a593Smuzhiyun struct extension_info {
48*4882a593Smuzhiyun     const char *const name;
49*4882a593Smuzhiyun     unsigned name_len;
50*4882a593Smuzhiyun 
51*4882a593Smuzhiyun     unsigned char bit;
52*4882a593Smuzhiyun 
53*4882a593Smuzhiyun     /**
54*4882a593Smuzhiyun      * This is the lowest version of GLX that "requires" this extension.
55*4882a593Smuzhiyun      * For example, GLX 1.3 requires SGIX_fbconfig, SGIX_pbuffer, and
56*4882a593Smuzhiyun      * SGI_make_current_read.  If the extension is not required by any known
57*4882a593Smuzhiyun      * version of GLX, use 0, 0.
58*4882a593Smuzhiyun      */
59*4882a593Smuzhiyun     unsigned char version_major;
60*4882a593Smuzhiyun     unsigned char version_minor;
61*4882a593Smuzhiyun 
62*4882a593Smuzhiyun     /**
63*4882a593Smuzhiyun      * Is driver support forced by the ABI?
64*4882a593Smuzhiyun      */
65*4882a593Smuzhiyun     unsigned char driver_support;
66*4882a593Smuzhiyun };
67*4882a593Smuzhiyun 
68*4882a593Smuzhiyun /**
69*4882a593Smuzhiyun  * List of known GLX Extensions.
70*4882a593Smuzhiyun  * The last Y/N switch informs whether the support of this extension is always enabled.
71*4882a593Smuzhiyun  */
72*4882a593Smuzhiyun static const struct extension_info known_glx_extensions[] = {
73*4882a593Smuzhiyun /*   GLX_ARB_get_proc_address is implemented on the client. */
74*4882a593Smuzhiyun     /* *INDENT-OFF* */
75*4882a593Smuzhiyun     { GLX(ARB_context_flush_control),   VER(0,0), N, },
76*4882a593Smuzhiyun     { GLX(ARB_create_context),          VER(0,0), N, },
77*4882a593Smuzhiyun     { GLX(ARB_create_context_no_error), VER(0,0), N, },
78*4882a593Smuzhiyun     { GLX(ARB_create_context_profile),  VER(0,0), N, },
79*4882a593Smuzhiyun     { GLX(ARB_create_context_robustness), VER(0,0), N, },
80*4882a593Smuzhiyun     { GLX(ARB_fbconfig_float),          VER(0,0), N, },
81*4882a593Smuzhiyun     { GLX(ARB_framebuffer_sRGB),        VER(0,0), N, },
82*4882a593Smuzhiyun     { GLX(ARB_multisample),             VER(1,4), Y, },
83*4882a593Smuzhiyun 
84*4882a593Smuzhiyun     { GLX(EXT_create_context_es_profile), VER(0,0), N, },
85*4882a593Smuzhiyun     { GLX(EXT_create_context_es2_profile), VER(0,0), N, },
86*4882a593Smuzhiyun     { GLX(EXT_fbconfig_packed_float),   VER(0,0), N, },
87*4882a593Smuzhiyun     { GLX(EXT_framebuffer_sRGB),        VER(0,0), N, },
88*4882a593Smuzhiyun     { GLX(EXT_import_context),          VER(0,0), Y, },
89*4882a593Smuzhiyun     { GLX(EXT_libglvnd),                VER(0,0), N, },
90*4882a593Smuzhiyun     { GLX(EXT_no_config_context),       VER(0,0), N, },
91*4882a593Smuzhiyun     { GLX(EXT_stereo_tree),             VER(0,0), N, },
92*4882a593Smuzhiyun     { GLX(EXT_texture_from_pixmap),     VER(0,0), N, },
93*4882a593Smuzhiyun     { GLX(EXT_visual_info),             VER(0,0), Y, },
94*4882a593Smuzhiyun     { GLX(EXT_visual_rating),           VER(0,0), Y, },
95*4882a593Smuzhiyun 
96*4882a593Smuzhiyun     { GLX(MESA_copy_sub_buffer),        VER(0,0), N, },
97*4882a593Smuzhiyun     { GLX(OML_swap_method),             VER(0,0), Y, },
98*4882a593Smuzhiyun     { GLX(SGI_make_current_read),       VER(1,3), Y, },
99*4882a593Smuzhiyun     { GLX(SGI_swap_control),            VER(0,0), N, },
100*4882a593Smuzhiyun     { GLX(SGIS_multisample),            VER(0,0), Y, },
101*4882a593Smuzhiyun     { GLX(SGIX_fbconfig),               VER(1,3), Y, },
102*4882a593Smuzhiyun     { GLX(SGIX_pbuffer),                VER(1,3), Y, },
103*4882a593Smuzhiyun     { GLX(SGIX_visual_select_group),    VER(0,0), Y, },
104*4882a593Smuzhiyun     { GLX(INTEL_swap_event),            VER(0,0), N, },
105*4882a593Smuzhiyun     { NULL }
106*4882a593Smuzhiyun     /* *INDENT-ON* */
107*4882a593Smuzhiyun };
108*4882a593Smuzhiyun 
109*4882a593Smuzhiyun /**
110*4882a593Smuzhiyun  * Create a GLX extension string for a set of enable bits.
111*4882a593Smuzhiyun  *
112*4882a593Smuzhiyun  * Creates a GLX extension string for the set of bit in \c enable_bits.  This
113*4882a593Smuzhiyun  * string is then stored in \c buffer if buffer is not \c NULL.  This allows
114*4882a593Smuzhiyun  * two-pass operation.  On the first pass the caller passes \c NULL for
115*4882a593Smuzhiyun  * \c buffer, and the function determines how much space is required to store
116*4882a593Smuzhiyun  * the extension string.  The caller allocates the buffer and calls the
117*4882a593Smuzhiyun  * function again.
118*4882a593Smuzhiyun  *
119*4882a593Smuzhiyun  * \param enable_bits  Bits representing the enabled extensions.
120*4882a593Smuzhiyun  * \param buffer       Buffer to store the extension string.  May be \c NULL.
121*4882a593Smuzhiyun  *
122*4882a593Smuzhiyun  * \return
123*4882a593Smuzhiyun  * The number of characters in \c buffer that were written to.  If \c buffer
124*4882a593Smuzhiyun  * is \c NULL, this is the size of buffer that must be allocated by the
125*4882a593Smuzhiyun  * caller.
126*4882a593Smuzhiyun  */
127*4882a593Smuzhiyun int
__glXGetExtensionString(const unsigned char * enable_bits,char * buffer)128*4882a593Smuzhiyun __glXGetExtensionString(const unsigned char *enable_bits, char *buffer)
129*4882a593Smuzhiyun {
130*4882a593Smuzhiyun     unsigned i;
131*4882a593Smuzhiyun     int length = 0;
132*4882a593Smuzhiyun 
133*4882a593Smuzhiyun     for (i = 0; known_glx_extensions[i].name != NULL; i++) {
134*4882a593Smuzhiyun         const unsigned bit = known_glx_extensions[i].bit;
135*4882a593Smuzhiyun         const size_t len = known_glx_extensions[i].name_len;
136*4882a593Smuzhiyun 
137*4882a593Smuzhiyun         if (EXT_ENABLED(bit, enable_bits)) {
138*4882a593Smuzhiyun             if (buffer != NULL) {
139*4882a593Smuzhiyun                 (void) memcpy(&buffer[length], known_glx_extensions[i].name,
140*4882a593Smuzhiyun                               len);
141*4882a593Smuzhiyun 
142*4882a593Smuzhiyun                 buffer[length + len + 0] = ' ';
143*4882a593Smuzhiyun                 buffer[length + len + 1] = '\0';
144*4882a593Smuzhiyun             }
145*4882a593Smuzhiyun 
146*4882a593Smuzhiyun             length += len + 1;
147*4882a593Smuzhiyun         }
148*4882a593Smuzhiyun     }
149*4882a593Smuzhiyun 
150*4882a593Smuzhiyun     return length + 1;
151*4882a593Smuzhiyun }
152*4882a593Smuzhiyun 
153*4882a593Smuzhiyun void
__glXEnableExtension(unsigned char * enable_bits,const char * ext)154*4882a593Smuzhiyun __glXEnableExtension(unsigned char *enable_bits, const char *ext)
155*4882a593Smuzhiyun {
156*4882a593Smuzhiyun     const size_t ext_name_len = strlen(ext);
157*4882a593Smuzhiyun     unsigned i;
158*4882a593Smuzhiyun 
159*4882a593Smuzhiyun     for (i = 0; known_glx_extensions[i].name != NULL; i++) {
160*4882a593Smuzhiyun         if ((ext_name_len == known_glx_extensions[i].name_len)
161*4882a593Smuzhiyun             && (memcmp(ext, known_glx_extensions[i].name, ext_name_len) == 0)) {
162*4882a593Smuzhiyun             SET_BIT(enable_bits, known_glx_extensions[i].bit);
163*4882a593Smuzhiyun             break;
164*4882a593Smuzhiyun         }
165*4882a593Smuzhiyun     }
166*4882a593Smuzhiyun }
167*4882a593Smuzhiyun 
168*4882a593Smuzhiyun void
__glXInitExtensionEnableBits(unsigned char * enable_bits)169*4882a593Smuzhiyun __glXInitExtensionEnableBits(unsigned char *enable_bits)
170*4882a593Smuzhiyun {
171*4882a593Smuzhiyun     unsigned i;
172*4882a593Smuzhiyun 
173*4882a593Smuzhiyun     (void) memset(enable_bits, 0, __GLX_EXT_BYTES);
174*4882a593Smuzhiyun 
175*4882a593Smuzhiyun     for (i = 0; known_glx_extensions[i].name != NULL; i++) {
176*4882a593Smuzhiyun         if (known_glx_extensions[i].driver_support) {
177*4882a593Smuzhiyun             SET_BIT(enable_bits, known_glx_extensions[i].bit);
178*4882a593Smuzhiyun         }
179*4882a593Smuzhiyun     }
180*4882a593Smuzhiyun }
181