Tizen 2.0 Release
[profile/ivi/osmesa.git] / src / mesa / drivers / x11 / glxapi.c
1 /*
2  * Mesa 3-D graphics library
3  * Version:  7.1
4  * 
5  * Copyright (C) 1999-2007  Brian Paul   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 "Software"),
9  * to deal in the Software without restriction, including without limitation
10  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
11  * and/or sell copies of the Software, and to permit persons to whom the
12  * Software is furnished to do so, subject to the following conditions:
13  * 
14  * The above copyright notice and this permission notice shall be included
15  * in all copies or substantial portions 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 MERCHANTABILITY,
19  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
20  * BRIAN PAUL BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
21  * AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
22  * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
23  */
24
25
26 /*
27  * This is the GLX API dispatcher.  Calls to the glX* functions are
28  * either routed to the real GLX encoders or to Mesa's pseudo-GLX functions.
29  * See the glxapi.h file for more details.
30  */
31
32
33 #include <assert.h>
34 #include <stdlib.h>
35 #include <stdio.h>
36 #include <string.h>
37 #include "main/glheader.h"
38 #include "main/compiler.h"
39 #include "glapi/glapi.h"
40 #include "glxapi.h"
41
42
43 extern struct _glxapi_table *_real_GetGLXDispatchTable(void);
44 extern struct _glxapi_table *_mesa_GetGLXDispatchTable(void);
45
46
47 struct display_dispatch {
48    Display *Dpy;
49    struct _glxapi_table *Table;
50    struct display_dispatch *Next;
51 };
52
53
54 /**
55  * When GLX_INDIRECT_RENDERING is defined, some symbols are missing in
56  * libglapi.a.  We need to define them here.
57  */
58 #ifdef GLX_INDIRECT_RENDERING
59
60 #include "glapi/glapitable.h"
61
62 #define KEYWORD1 PUBLIC
63
64 #if defined(USE_MGL_NAMESPACE)
65 #define NAME(func)  mgl##func
66 #else
67 #define NAME(func)  gl##func
68 #endif
69
70 #define DISPATCH(FUNC, ARGS, MESSAGE)           \
71    GET_DISPATCH()->FUNC ARGS
72
73 #define RETURN_DISPATCH(FUNC, ARGS, MESSAGE)    \
74    return GET_DISPATCH()->FUNC ARGS
75
76 /* skip normal ones */
77 #define _GLAPI_SKIP_NORMAL_ENTRY_POINTS
78 #include "glapi/glapitemp.h"
79
80 #endif /* GLX_INDIRECT_RENDERING */
81
82
83 static struct display_dispatch *DispatchList = NULL;
84
85
86 /* Display -> Dispatch caching */
87 static Display *prevDisplay = NULL;
88 static struct _glxapi_table *prevTable = NULL;
89
90
91 static struct _glxapi_table *
92 get_dispatch(Display *dpy)
93 {
94    if (!dpy)
95       return NULL;
96
97    /* search list of display/dispatch pairs for this display */
98    {
99       const struct display_dispatch *d = DispatchList;
100       while (d) {
101          if (d->Dpy == dpy) {
102             prevDisplay = dpy;
103             prevTable = d->Table;
104             return d->Table;  /* done! */
105          }
106          d = d->Next;
107       }
108    }
109
110    /* A new display, determine if we should use real GLX
111     * or Mesa's pseudo-GLX.
112     */
113    {
114       struct _glxapi_table *t = _mesa_GetGLXDispatchTable();
115
116       if (t) {
117          struct display_dispatch *d;
118          d = (struct display_dispatch *) malloc(sizeof(struct display_dispatch));
119          if (d) {
120             d->Dpy = dpy;
121             d->Table = t;
122             /* insert at head of list */
123             d->Next = DispatchList;
124             DispatchList = d;
125             /* update cache */
126             prevDisplay = dpy;
127             prevTable = t;
128             return t;
129          }
130       }
131    }
132
133    /* If we get here that means we can't use real GLX on this display
134     * and the Mesa pseudo-GLX software renderer wasn't compiled in.
135     * Or, we ran out of memory!
136     */
137    return NULL;
138 }
139
140
141 /* Don't use the GET_DISPATCH defined in glthread.h */
142 #undef GET_DISPATCH
143
144 #define GET_DISPATCH(DPY, TABLE)        \
145    if (DPY == prevDisplay) {            \
146       TABLE = prevTable;                \
147    }                                    \
148    else if (!DPY) {                     \
149       TABLE = NULL;                     \
150    }                                    \
151    else {                               \
152       TABLE = get_dispatch(DPY);        \
153    }
154
155    
156
157
158 /**
159  * GLX API current context.
160  */
161 #if defined(GLX_USE_TLS)
162 PUBLIC __thread void * CurrentContext
163     __attribute__((tls_model("initial-exec")));
164 #elif defined(THREADS)
165 static _glthread_TSD ContextTSD;         /**< Per-thread context pointer */
166 #else
167 static GLXContext CurrentContext = 0;
168 #endif
169
170
171 static void
172 SetCurrentContext(GLXContext c)
173 {
174 #if defined(GLX_USE_TLS)
175    CurrentContext = c;
176 #elif defined(THREADS)
177    _glthread_SetTSD(&ContextTSD, c);
178 #else
179    CurrentContext = c;
180 #endif
181 }
182
183
184 /*
185  * GLX API entrypoints
186  */
187
188 /*** GLX_VERSION_1_0 ***/
189
190 XVisualInfo PUBLIC *
191 glXChooseVisual(Display *dpy, int screen, int *list)
192 {
193    struct _glxapi_table *t;
194    GET_DISPATCH(dpy, t);
195    if (!t)
196       return NULL;
197    return (t->ChooseVisual)(dpy, screen, list);
198 }
199
200
201 void PUBLIC
202 glXCopyContext(Display *dpy, GLXContext src, GLXContext dst, unsigned long mask)
203 {
204    struct _glxapi_table *t;
205    GET_DISPATCH(dpy, t);
206    if (!t)
207       return;
208    (t->CopyContext)(dpy, src, dst, mask);
209 }
210
211
212 GLXContext PUBLIC
213 glXCreateContext(Display *dpy, XVisualInfo *visinfo, GLXContext shareList, Bool direct)
214 {
215    struct _glxapi_table *t;
216    GET_DISPATCH(dpy, t);
217    if (!t)
218       return 0;
219    return (t->CreateContext)(dpy, visinfo, shareList, direct);
220 }
221
222
223 GLXPixmap PUBLIC
224 glXCreateGLXPixmap(Display *dpy, XVisualInfo *visinfo, Pixmap pixmap)
225 {
226    struct _glxapi_table *t;
227    GET_DISPATCH(dpy, t);
228    if (!t)
229       return 0;
230    return (t->CreateGLXPixmap)(dpy, visinfo, pixmap);
231 }
232
233
234 void PUBLIC
235 glXDestroyContext(Display *dpy, GLXContext ctx)
236 {
237    struct _glxapi_table *t;
238    GET_DISPATCH(dpy, t);
239    if (!t)
240       return;
241    if (glXGetCurrentContext() == ctx)
242       SetCurrentContext(NULL);
243    (t->DestroyContext)(dpy, ctx);
244 }
245
246
247 void PUBLIC
248 glXDestroyGLXPixmap(Display *dpy, GLXPixmap pixmap)
249 {
250    struct _glxapi_table *t;
251    GET_DISPATCH(dpy, t);
252    if (!t)
253       return;
254    (t->DestroyGLXPixmap)(dpy, pixmap);
255 }
256
257
258 int PUBLIC
259 glXGetConfig(Display *dpy, XVisualInfo *visinfo, int attrib, int *value)
260 {
261    struct _glxapi_table *t;
262    GET_DISPATCH(dpy, t);
263    if (!t)
264       return GLX_NO_EXTENSION;
265    return (t->GetConfig)(dpy, visinfo, attrib, value);
266 }
267
268
269 GLXContext PUBLIC
270 glXGetCurrentContext(void)
271 {
272 #if defined(GLX_USE_TLS)
273    return CurrentContext;
274 #elif defined(THREADS)
275    return (GLXContext) _glthread_GetTSD(&ContextTSD);
276 #else
277    return CurrentContext;
278 #endif
279 }
280
281
282 GLXDrawable PUBLIC
283 glXGetCurrentDrawable(void)
284 {
285    __GLXcontext *gc = (__GLXcontext *) glXGetCurrentContext();
286    return gc ? gc->currentDrawable : 0;
287 }
288
289
290 Bool PUBLIC
291 glXIsDirect(Display *dpy, GLXContext ctx)
292 {
293    struct _glxapi_table *t;
294    GET_DISPATCH(dpy, t);
295    if (!t)
296       return False;
297    return (t->IsDirect)(dpy, ctx);
298 }
299
300
301 Bool PUBLIC
302 glXMakeCurrent(Display *dpy, GLXDrawable drawable, GLXContext ctx)
303 {
304    Bool b;
305    struct _glxapi_table *t;
306    GET_DISPATCH(dpy, t);
307    if (!t) {
308       return False;
309    }
310    b = (*t->MakeCurrent)(dpy, drawable, ctx);
311    if (b) {
312       SetCurrentContext(ctx);
313    }
314    return b;
315 }
316
317
318 Bool PUBLIC
319 glXQueryExtension(Display *dpy, int *errorb, int *event)
320 {
321    struct _glxapi_table *t;
322    GET_DISPATCH(dpy, t);
323    if (!t)
324       return False;
325    return (t->QueryExtension)(dpy, errorb, event);
326 }
327
328
329 Bool PUBLIC
330 glXQueryVersion(Display *dpy, int *maj, int *min)
331 {
332    struct _glxapi_table *t;
333    GET_DISPATCH(dpy, t);
334    if (!t)
335       return False;
336    return (t->QueryVersion)(dpy, maj, min);
337 }
338
339
340 void PUBLIC
341 glXSwapBuffers(Display *dpy, GLXDrawable drawable)
342 {
343    struct _glxapi_table *t;
344    GET_DISPATCH(dpy, t);
345    if (!t)
346       return;
347    (t->SwapBuffers)(dpy, drawable);
348 }
349
350
351 void PUBLIC
352 glXUseXFont(Font font, int first, int count, int listBase)
353 {
354    struct _glxapi_table *t;
355    Display *dpy = glXGetCurrentDisplay();
356    GET_DISPATCH(dpy, t);
357    if (!t)
358       return;
359    (t->UseXFont)(font, first, count, listBase);
360 }
361
362
363 void PUBLIC
364 glXWaitGL(void)
365 {
366    struct _glxapi_table *t;
367    Display *dpy = glXGetCurrentDisplay();
368    GET_DISPATCH(dpy, t);
369    if (!t)
370       return;
371    (t->WaitGL)();
372 }
373
374
375 void PUBLIC
376 glXWaitX(void)
377 {
378    struct _glxapi_table *t;
379    Display *dpy = glXGetCurrentDisplay();
380    GET_DISPATCH(dpy, t);
381    if (!t)
382       return;
383    (t->WaitX)();
384 }
385
386
387
388 /*** GLX_VERSION_1_1 ***/
389
390 const char PUBLIC *
391 glXGetClientString(Display *dpy, int name)
392 {
393    struct _glxapi_table *t;
394    GET_DISPATCH(dpy, t);
395    if (!t)
396       return NULL;
397    return (t->GetClientString)(dpy, name);
398 }
399
400
401 const char PUBLIC *
402 glXQueryExtensionsString(Display *dpy, int screen)
403 {
404    struct _glxapi_table *t;
405    GET_DISPATCH(dpy, t);
406    if (!t)
407       return NULL;
408    return (t->QueryExtensionsString)(dpy, screen);
409 }
410
411
412 const char PUBLIC *
413 glXQueryServerString(Display *dpy, int screen, int name)
414 {
415    struct _glxapi_table *t;
416    GET_DISPATCH(dpy, t);
417    if (!t)
418       return NULL;
419    return (t->QueryServerString)(dpy, screen, name);
420 }
421
422
423 /*** GLX_VERSION_1_2 ***/
424
425 Display PUBLIC *
426 glXGetCurrentDisplay(void)
427 {
428    /* Same code as in libGL's glxext.c */
429    __GLXcontext *gc = (__GLXcontext *) glXGetCurrentContext();
430    if (NULL == gc) return NULL;
431    return gc->currentDpy;
432 }
433
434
435
436 /*** GLX_VERSION_1_3 ***/
437
438 GLXFBConfig PUBLIC *
439 glXChooseFBConfig(Display *dpy, int screen, const int *attribList, int *nitems)
440 {
441    struct _glxapi_table *t;
442    GET_DISPATCH(dpy, t);
443    if (!t)
444       return 0;
445    return (t->ChooseFBConfig)(dpy, screen, attribList, nitems);
446 }
447
448
449 GLXContext PUBLIC
450 glXCreateNewContext(Display *dpy, GLXFBConfig config, int renderType, GLXContext shareList, Bool direct)
451 {
452    struct _glxapi_table *t;
453    GET_DISPATCH(dpy, t);
454    if (!t)
455       return 0;
456    return (t->CreateNewContext)(dpy, config, renderType, shareList, direct);
457 }
458
459
460 GLXPbuffer PUBLIC
461 glXCreatePbuffer(Display *dpy, GLXFBConfig config, const int *attribList)
462 {
463    struct _glxapi_table *t;
464    GET_DISPATCH(dpy, t);
465    if (!t)
466       return 0;
467    return (t->CreatePbuffer)(dpy, config, attribList);
468 }
469
470
471 GLXPixmap PUBLIC
472 glXCreatePixmap(Display *dpy, GLXFBConfig config, Pixmap pixmap, const int *attribList)
473 {
474    struct _glxapi_table *t;
475    GET_DISPATCH(dpy, t);
476    if (!t)
477       return 0;
478    return (t->CreatePixmap)(dpy, config, pixmap, attribList);
479 }
480
481
482 GLXWindow PUBLIC
483 glXCreateWindow(Display *dpy, GLXFBConfig config, Window win, const int *attribList)
484 {
485    struct _glxapi_table *t;
486    GET_DISPATCH(dpy, t);
487    if (!t)
488       return 0;
489    return (t->CreateWindow)(dpy, config, win, attribList);
490 }
491
492
493 void PUBLIC
494 glXDestroyPbuffer(Display *dpy, GLXPbuffer pbuf)
495 {
496    struct _glxapi_table *t;
497    GET_DISPATCH(dpy, t);
498    if (!t)
499       return;
500    (t->DestroyPbuffer)(dpy, pbuf);
501 }
502
503
504 void PUBLIC
505 glXDestroyPixmap(Display *dpy, GLXPixmap pixmap)
506 {
507    struct _glxapi_table *t;
508    GET_DISPATCH(dpy, t);
509    if (!t)
510       return;
511    (t->DestroyPixmap)(dpy, pixmap);
512 }
513
514
515 void PUBLIC
516 glXDestroyWindow(Display *dpy, GLXWindow window)
517 {
518    struct _glxapi_table *t;
519    GET_DISPATCH(dpy, t);
520    if (!t)
521       return;
522    (t->DestroyWindow)(dpy, window);
523 }
524
525
526 GLXDrawable PUBLIC
527 glXGetCurrentReadDrawable(void)
528 {
529    __GLXcontext *gc = (__GLXcontext *) glXGetCurrentContext();
530    return gc ? gc->currentReadable : 0;
531 }
532
533
534 int PUBLIC
535 glXGetFBConfigAttrib(Display *dpy, GLXFBConfig config, int attribute, int *value)
536 {
537    struct _glxapi_table *t;
538    GET_DISPATCH(dpy, t);
539    if (!t)
540       return GLX_NO_EXTENSION;
541    return (t->GetFBConfigAttrib)(dpy, config, attribute, value);
542 }
543
544
545 GLXFBConfig PUBLIC *
546 glXGetFBConfigs(Display *dpy, int screen, int *nelements)
547 {
548    struct _glxapi_table *t;
549    GET_DISPATCH(dpy, t);
550    if (!t)
551       return 0;
552    return (t->GetFBConfigs)(dpy, screen, nelements);
553 }
554
555 void PUBLIC
556 glXGetSelectedEvent(Display *dpy, GLXDrawable drawable, unsigned long *mask)
557 {
558    struct _glxapi_table *t;
559    GET_DISPATCH(dpy, t);
560    if (!t)
561       return;
562    (t->GetSelectedEvent)(dpy, drawable, mask);
563 }
564
565
566 XVisualInfo PUBLIC *
567 glXGetVisualFromFBConfig(Display *dpy, GLXFBConfig config)
568 {
569    struct _glxapi_table *t;
570    GET_DISPATCH(dpy, t);
571    if (!t)
572       return NULL;
573    return (t->GetVisualFromFBConfig)(dpy, config);
574 }
575
576
577 Bool PUBLIC
578 glXMakeContextCurrent(Display *dpy, GLXDrawable draw, GLXDrawable read, GLXContext ctx)
579 {
580    Bool b;
581    struct _glxapi_table *t;
582    GET_DISPATCH(dpy, t);
583    if (!t)
584       return False;
585    b = (t->MakeContextCurrent)(dpy, draw, read, ctx);
586    if (b) {
587       SetCurrentContext(ctx);
588    }
589    return b;
590 }
591
592
593 int PUBLIC
594 glXQueryContext(Display *dpy, GLXContext ctx, int attribute, int *value)
595 {
596    struct _glxapi_table *t;
597    GET_DISPATCH(dpy, t);
598    assert(t);
599    if (!t)
600       return 0; /* XXX correct? */
601    return (t->QueryContext)(dpy, ctx, attribute, value);
602 }
603
604
605 void PUBLIC
606 glXQueryDrawable(Display *dpy, GLXDrawable draw, int attribute, unsigned int *value)
607 {
608    struct _glxapi_table *t;
609    GET_DISPATCH(dpy, t);
610    if (!t)
611       return;
612    (t->QueryDrawable)(dpy, draw, attribute, value);
613 }
614
615
616 void PUBLIC
617 glXSelectEvent(Display *dpy, GLXDrawable drawable, unsigned long mask)
618 {
619    struct _glxapi_table *t;
620    GET_DISPATCH(dpy, t);
621    if (!t)
622       return;
623    (t->SelectEvent)(dpy, drawable, mask);
624 }
625
626
627
628 /*** GLX_SGI_swap_control ***/
629
630 int PUBLIC
631 glXSwapIntervalSGI(int interval)
632 {
633    struct _glxapi_table *t;
634    Display *dpy = glXGetCurrentDisplay();
635    GET_DISPATCH(dpy, t);
636    if (!t)
637       return 0;
638    return (t->SwapIntervalSGI)(interval);
639 }
640
641
642
643 /*** GLX_SGI_video_sync ***/
644
645 int PUBLIC
646 glXGetVideoSyncSGI(unsigned int *count)
647 {
648    struct _glxapi_table *t;
649    Display *dpy = glXGetCurrentDisplay();
650    GET_DISPATCH(dpy, t);
651    if (!t || !glXGetCurrentContext())
652       return GLX_BAD_CONTEXT;
653    return (t->GetVideoSyncSGI)(count);
654 }
655
656 int PUBLIC
657 glXWaitVideoSyncSGI(int divisor, int remainder, unsigned int *count)
658 {
659    struct _glxapi_table *t;
660    Display *dpy = glXGetCurrentDisplay();
661    GET_DISPATCH(dpy, t);
662    if (!t || !glXGetCurrentContext())
663       return GLX_BAD_CONTEXT;
664    return (t->WaitVideoSyncSGI)(divisor, remainder, count);
665 }
666
667
668
669 /*** GLX_SGI_make_current_read ***/
670
671 Bool PUBLIC
672 glXMakeCurrentReadSGI(Display *dpy, GLXDrawable draw, GLXDrawable read, GLXContext ctx)
673 {
674    struct _glxapi_table *t;
675    GET_DISPATCH(dpy, t);
676    if (!t)
677       return False;
678    return (t->MakeCurrentReadSGI)(dpy, draw, read, ctx);
679 }
680
681 GLXDrawable PUBLIC
682 glXGetCurrentReadDrawableSGI(void)
683 {
684    return glXGetCurrentReadDrawable();
685 }
686
687
688 #if defined(_VL_H)
689
690 GLXVideoSourceSGIX PUBLIC
691 glXCreateGLXVideoSourceSGIX(Display *dpy, int screen, VLServer server, VLPath path, int nodeClass, VLNode drainNode)
692 {
693    struct _glxapi_table *t;
694    GET_DISPATCH(dpy, t);
695    if (!t)
696       return 0;
697    return (t->CreateGLXVideoSourceSGIX)(dpy, screen, server, path, nodeClass, drainNode);
698 }
699
700 void PUBLIC
701 glXDestroyGLXVideoSourceSGIX(Display *dpy, GLXVideoSourceSGIX src)
702 {
703    struct _glxapi_table *t;
704    GET_DISPATCH(dpy, t);
705    if (!t)
706       return 0;
707    return (t->DestroyGLXVideoSourceSGIX)(dpy, src);
708 }
709
710 #endif
711
712
713 /*** GLX_EXT_import_context ***/
714
715 void PUBLIC
716 glXFreeContextEXT(Display *dpy, GLXContext context)
717 {
718    struct _glxapi_table *t;
719    GET_DISPATCH(dpy, t);
720    if (!t)
721       return;
722    (t->FreeContextEXT)(dpy, context);
723 }
724
725 GLXContextID PUBLIC
726 glXGetContextIDEXT(const GLXContext context)
727 {
728    return ((__GLXcontext *) context)->xid;
729 }
730
731 Display PUBLIC *
732 glXGetCurrentDisplayEXT(void)
733 {
734    return glXGetCurrentDisplay();
735 }
736
737 GLXContext PUBLIC
738 glXImportContextEXT(Display *dpy, GLXContextID contextID)
739 {
740    struct _glxapi_table *t;
741    GET_DISPATCH(dpy, t);
742    if (!t)
743       return 0;
744    return (t->ImportContextEXT)(dpy, contextID);
745 }
746
747 int PUBLIC
748 glXQueryContextInfoEXT(Display *dpy, GLXContext context, int attribute,int *value)
749 {
750    struct _glxapi_table *t;
751    GET_DISPATCH(dpy, t);
752    if (!t)
753       return 0;  /* XXX ok? */
754    return (t->QueryContextInfoEXT)(dpy, context, attribute, value);
755 }
756
757
758
759 /*** GLX_SGIX_fbconfig ***/
760
761 int PUBLIC
762 glXGetFBConfigAttribSGIX(Display *dpy, GLXFBConfigSGIX config, int attribute, int *value)
763 {
764    struct _glxapi_table *t;
765    GET_DISPATCH(dpy, t);
766    if (!t)
767       return 0;
768    return (t->GetFBConfigAttribSGIX)(dpy, config, attribute, value);
769 }
770
771 GLXFBConfigSGIX PUBLIC *
772 glXChooseFBConfigSGIX(Display *dpy, int screen, int *attrib_list, int *nelements)
773 {
774    struct _glxapi_table *t;
775    GET_DISPATCH(dpy, t);
776    if (!t)
777       return 0;
778    return (t->ChooseFBConfigSGIX)(dpy, screen, attrib_list, nelements);
779 }
780
781 GLXPixmap PUBLIC
782 glXCreateGLXPixmapWithConfigSGIX(Display *dpy, GLXFBConfigSGIX config, Pixmap pixmap)
783 {
784    struct _glxapi_table *t;
785    GET_DISPATCH(dpy, t);
786    if (!t)
787       return 0;
788    return (t->CreateGLXPixmapWithConfigSGIX)(dpy, config, pixmap);
789 }
790
791 GLXContext PUBLIC
792 glXCreateContextWithConfigSGIX(Display *dpy, GLXFBConfigSGIX config, int render_type, GLXContext share_list, Bool direct)
793 {
794    struct _glxapi_table *t;
795    GET_DISPATCH(dpy, t);
796    if (!t)
797       return 0;
798    return (t->CreateContextWithConfigSGIX)(dpy, config, render_type, share_list, direct);
799 }
800
801 XVisualInfo PUBLIC *
802 glXGetVisualFromFBConfigSGIX(Display *dpy, GLXFBConfigSGIX config)
803 {
804    struct _glxapi_table *t;
805    GET_DISPATCH(dpy, t);
806    if (!t)
807       return 0;
808    return (t->GetVisualFromFBConfigSGIX)(dpy, config);
809 }
810
811 GLXFBConfigSGIX PUBLIC
812 glXGetFBConfigFromVisualSGIX(Display *dpy, XVisualInfo *vis)
813 {
814    struct _glxapi_table *t;
815    GET_DISPATCH(dpy, t);
816    if (!t)
817       return 0;
818    return (t->GetFBConfigFromVisualSGIX)(dpy, vis);
819 }
820
821
822
823 /*** GLX_SGIX_pbuffer ***/
824
825 GLXPbufferSGIX PUBLIC
826 glXCreateGLXPbufferSGIX(Display *dpy, GLXFBConfigSGIX config, unsigned int width, unsigned int height, int *attrib_list)
827 {
828    struct _glxapi_table *t;
829    GET_DISPATCH(dpy, t);
830    if (!t)
831       return 0;
832    return (t->CreateGLXPbufferSGIX)(dpy, config, width, height, attrib_list);
833 }
834
835 void PUBLIC
836 glXDestroyGLXPbufferSGIX(Display *dpy, GLXPbufferSGIX pbuf)
837 {
838    struct _glxapi_table *t;
839    GET_DISPATCH(dpy, t);
840    if (!t)
841       return;
842    (t->DestroyGLXPbufferSGIX)(dpy, pbuf);
843 }
844
845 int PUBLIC
846 glXQueryGLXPbufferSGIX(Display *dpy, GLXPbufferSGIX pbuf, int attribute, unsigned int *value)
847 {
848    struct _glxapi_table *t;
849    GET_DISPATCH(dpy, t);
850    if (!t)
851       return 0;
852    return (t->QueryGLXPbufferSGIX)(dpy, pbuf, attribute, value);
853 }
854
855 void PUBLIC
856 glXSelectEventSGIX(Display *dpy, GLXDrawable drawable, unsigned long mask)
857 {
858    struct _glxapi_table *t;
859    GET_DISPATCH(dpy, t);
860    if (!t)
861       return;
862    (t->SelectEventSGIX)(dpy, drawable, mask);
863 }
864
865 void PUBLIC
866 glXGetSelectedEventSGIX(Display *dpy, GLXDrawable drawable, unsigned long *mask)
867 {
868    struct _glxapi_table *t;
869    GET_DISPATCH(dpy, t);
870    if (!t)
871       return;
872    (t->GetSelectedEventSGIX)(dpy, drawable, mask);
873 }
874
875
876
877 /*** GLX_SGI_cushion ***/
878
879 void PUBLIC
880 glXCushionSGI(Display *dpy, Window win, float cushion)
881 {
882    struct _glxapi_table *t;
883    GET_DISPATCH(dpy, t);
884    if (!t)
885       return;
886    (t->CushionSGI)(dpy, win, cushion);
887 }
888
889
890
891 /*** GLX_SGIX_video_resize ***/
892
893 int PUBLIC
894 glXBindChannelToWindowSGIX(Display *dpy, int screen, int channel , Window window)
895 {
896    struct _glxapi_table *t;
897    GET_DISPATCH(dpy, t);
898    if (!t)
899       return 0;
900    return (t->BindChannelToWindowSGIX)(dpy, screen, channel, window);
901 }
902
903 int PUBLIC
904 glXChannelRectSGIX(Display *dpy, int screen, int channel, int x, int y, int w, int h)
905 {
906    struct _glxapi_table *t;
907    GET_DISPATCH(dpy, t);
908    if (!t)
909       return 0;
910    return (t->ChannelRectSGIX)(dpy, screen, channel, x, y, w, h);
911 }
912
913 int PUBLIC
914 glXQueryChannelRectSGIX(Display *dpy, int screen, int channel, int *x, int *y, int *w, int *h)
915 {
916    struct _glxapi_table *t;
917    GET_DISPATCH(dpy, t);
918    if (!t)
919       return 0;
920    return (t->QueryChannelRectSGIX)(dpy, screen, channel, x, y, w, h);
921 }
922
923 int PUBLIC
924 glXQueryChannelDeltasSGIX(Display *dpy, int screen, int channel, int *dx, int *dy, int *dw, int *dh)
925 {
926    struct _glxapi_table *t;
927    GET_DISPATCH(dpy, t);
928    if (!t)
929       return 0;
930    return (t->QueryChannelDeltasSGIX)(dpy, screen, channel, dx, dy, dw, dh);
931 }
932
933 int PUBLIC
934 glXChannelRectSyncSGIX(Display *dpy, int screen, int channel, GLenum synctype)
935 {
936    struct _glxapi_table *t;
937    GET_DISPATCH(dpy, t);
938    if (!t)
939       return 0;
940    return (t->ChannelRectSyncSGIX)(dpy, screen, channel, synctype);
941 }
942
943
944
945 #if defined(_DM_BUFFER_H_)
946
947 Bool PUBLIC
948 glXAssociateDMPbufferSGIX(Display *dpy, GLXPbufferSGIX pbuffer, DMparams *params, DMbuffer dmbuffer)
949 {
950    struct _glxapi_table *t;
951    GET_DISPATCH(dpy, t);
952    if (!t)
953       return False;
954    return (t->AssociateDMPbufferSGIX)(dpy, pbuffer, params, dmbuffer);
955 }
956
957 #endif
958
959
960 /*** GLX_SGIX_swap_group ***/
961
962 void PUBLIC
963 glXJoinSwapGroupSGIX(Display *dpy, GLXDrawable drawable, GLXDrawable member)
964 {
965    struct _glxapi_table *t;
966    GET_DISPATCH(dpy, t);
967    if (!t)
968       return;
969    (*t->JoinSwapGroupSGIX)(dpy, drawable, member);
970 }
971
972
973 /*** GLX_SGIX_swap_barrier ***/
974
975 void PUBLIC
976 glXBindSwapBarrierSGIX(Display *dpy, GLXDrawable drawable, int barrier)
977 {
978    struct _glxapi_table *t;
979    GET_DISPATCH(dpy, t);
980    if (!t)
981       return;
982    (*t->BindSwapBarrierSGIX)(dpy, drawable, barrier);
983 }
984
985 Bool PUBLIC
986 glXQueryMaxSwapBarriersSGIX(Display *dpy, int screen, int *max)
987 {
988    struct _glxapi_table *t;
989    GET_DISPATCH(dpy, t);
990    if (!t)
991       return False;
992    return (*t->QueryMaxSwapBarriersSGIX)(dpy, screen, max);
993 }
994
995
996
997 /*** GLX_SUN_get_transparent_index ***/
998
999 Status PUBLIC
1000 glXGetTransparentIndexSUN(Display *dpy, Window overlay, Window underlay, long *pTransparent)
1001 {
1002    struct _glxapi_table *t;
1003    GET_DISPATCH(dpy, t);
1004    if (!t)
1005       return False;
1006    return (*t->GetTransparentIndexSUN)(dpy, overlay, underlay, pTransparent);
1007 }
1008
1009
1010
1011 /*** GLX_MESA_copy_sub_buffer ***/
1012
1013 void PUBLIC
1014 glXCopySubBufferMESA(Display *dpy, GLXDrawable drawable, int x, int y, int width, int height)
1015 {
1016    struct _glxapi_table *t;
1017    GET_DISPATCH(dpy, t);
1018    if (!t)
1019       return;
1020    (t->CopySubBufferMESA)(dpy, drawable, x, y, width, height);
1021 }
1022
1023
1024
1025 /*** GLX_MESA_release_buffers ***/
1026
1027 Bool PUBLIC
1028 glXReleaseBuffersMESA(Display *dpy, Window w)
1029 {
1030    struct _glxapi_table *t;
1031    GET_DISPATCH(dpy, t);
1032    if (!t)
1033       return False;
1034    return (t->ReleaseBuffersMESA)(dpy, w);
1035 }
1036
1037
1038
1039 /*** GLX_MESA_pixmap_colormap ***/
1040
1041 GLXPixmap PUBLIC
1042 glXCreateGLXPixmapMESA(Display *dpy, XVisualInfo *visinfo, Pixmap pixmap, Colormap cmap)
1043 {
1044    struct _glxapi_table *t;
1045    GET_DISPATCH(dpy, t);
1046    if (!t)
1047       return 0;
1048    return (t->CreateGLXPixmapMESA)(dpy, visinfo, pixmap, cmap);
1049 }
1050
1051
1052
1053 /*** GLX_MESA_set_3dfx_mode ***/
1054
1055 Bool PUBLIC
1056 glXSet3DfxModeMESA(int mode)
1057 {
1058    struct _glxapi_table *t;
1059    Display *dpy = glXGetCurrentDisplay();
1060    GET_DISPATCH(dpy, t);
1061    if (!t)
1062       return False;
1063    return (t->Set3DfxModeMESA)(mode);
1064 }
1065
1066
1067
1068 /*** GLX_NV_vertex_array_range ***/
1069
1070 void PUBLIC *
1071 glXAllocateMemoryNV( GLsizei size,
1072                      GLfloat readFrequency,
1073                      GLfloat writeFrequency,
1074                      GLfloat priority )
1075 {
1076    struct _glxapi_table *t;
1077    Display *dpy = glXGetCurrentDisplay();
1078    GET_DISPATCH(dpy, t);
1079    if (!t)
1080       return NULL;
1081    return (t->AllocateMemoryNV)(size, readFrequency, writeFrequency, priority);
1082 }
1083
1084
1085 void PUBLIC
1086 glXFreeMemoryNV( GLvoid *pointer )
1087 {
1088    struct _glxapi_table *t;
1089    Display *dpy = glXGetCurrentDisplay();
1090    GET_DISPATCH(dpy, t);
1091    if (!t)
1092       return;
1093    (t->FreeMemoryNV)(pointer);
1094 }
1095
1096
1097
1098
1099 /*** GLX_MESA_agp_offset */
1100
1101 GLuint PUBLIC
1102 glXGetAGPOffsetMESA( const GLvoid *pointer )
1103 {
1104    struct _glxapi_table *t;
1105    Display *dpy = glXGetCurrentDisplay();
1106    GET_DISPATCH(dpy, t);
1107    if (!t)
1108       return ~0;
1109    return (t->GetAGPOffsetMESA)(pointer);
1110 }
1111
1112
1113 /*** GLX_EXT_texture_from_pixmap */
1114
1115 void PUBLIC
1116 glXBindTexImageEXT(Display *dpy, GLXDrawable drawable, int buffer,
1117                    const int *attrib_list)
1118 {
1119    struct _glxapi_table *t;
1120    GET_DISPATCH(dpy, t);
1121    if (t)
1122       t->BindTexImageEXT(dpy, drawable, buffer, attrib_list);
1123 }
1124
1125 void PUBLIC
1126 glXReleaseTexImageEXT(Display *dpy, GLXDrawable drawable, int buffer)
1127 {
1128    struct _glxapi_table *t;
1129    GET_DISPATCH(dpy, t);
1130    if (t)
1131       t->ReleaseTexImageEXT(dpy, drawable, buffer);
1132 }
1133
1134
1135 /**********************************************************************/
1136 /* GLX API management functions                                       */
1137 /**********************************************************************/
1138
1139
1140 const char *
1141 _glxapi_get_version(void)
1142 {
1143    return "1.3";
1144 }
1145
1146
1147 /*
1148  * Return array of extension strings.
1149  */
1150 const char **
1151 _glxapi_get_extensions(void)
1152 {
1153    static const char *extensions[] = {
1154 #ifdef GLX_EXT_import_context
1155       "GLX_EXT_import_context",
1156 #endif
1157 #ifdef GLX_SGI_video_sync
1158       "GLX_SGI_video_sync",
1159 #endif
1160 #ifdef GLX_MESA_copy_sub_buffer
1161       "GLX_MESA_copy_sub_buffer",
1162 #endif
1163 #ifdef GLX_MESA_release_buffers
1164       "GLX_MESA_release_buffers",
1165 #endif
1166 #ifdef GLX_MESA_pixmap_colormap
1167       "GLX_MESA_pixmap_colormap",
1168 #endif
1169 #ifdef GLX_MESA_set_3dfx_mode
1170       "GLX_MESA_set_3dfx_mode",
1171 #endif
1172 #ifdef GLX_SGIX_fbconfig
1173       "GLX_SGIX_fbconfig",
1174 #endif
1175 #ifdef GLX_SGIX_pbuffer
1176       "GLX_SGIX_pbuffer",
1177 #endif
1178 #ifdef GLX_EXT_texture_from_pixmap
1179       "GLX_EXT_texture_from_pixmap",
1180 #endif
1181 #ifdef GLX_INTEL_swap_event
1182       "GLX_INTEL_swap_event",
1183 #endif
1184       NULL
1185    };
1186    return extensions;
1187 }
1188
1189
1190 /*
1191  * Return size of the GLX dispatch table, in entries, not bytes.
1192  */
1193 GLuint
1194 _glxapi_get_dispatch_table_size(void)
1195 {
1196    return sizeof(struct _glxapi_table) / sizeof(void *);
1197 }
1198
1199
1200 static int
1201 generic_no_op_func(void)
1202 {
1203    return 0;
1204 }
1205
1206
1207 /*
1208  * Initialize all functions in given dispatch table to be no-ops
1209  */
1210 void
1211 _glxapi_set_no_op_table(struct _glxapi_table *t)
1212 {
1213    typedef int (*nop_func)(void);
1214    nop_func *dispatch = (nop_func *) t;
1215    GLuint n = _glxapi_get_dispatch_table_size();
1216    GLuint i;
1217    for (i = 0; i < n; i++) {
1218       dispatch[i] = generic_no_op_func;
1219    }
1220 }
1221
1222
1223 struct name_address_pair {
1224    const char *Name;
1225    __GLXextFuncPtr Address;
1226 };
1227
1228 static struct name_address_pair GLX_functions[] = {
1229    /*** GLX_VERSION_1_0 ***/
1230    { "glXChooseVisual", (__GLXextFuncPtr) glXChooseVisual },
1231    { "glXCopyContext", (__GLXextFuncPtr) glXCopyContext },
1232    { "glXCreateContext", (__GLXextFuncPtr) glXCreateContext },
1233    { "glXCreateGLXPixmap", (__GLXextFuncPtr) glXCreateGLXPixmap },
1234    { "glXDestroyContext", (__GLXextFuncPtr) glXDestroyContext },
1235    { "glXDestroyGLXPixmap", (__GLXextFuncPtr) glXDestroyGLXPixmap },
1236    { "glXGetConfig", (__GLXextFuncPtr) glXGetConfig },
1237    { "glXGetCurrentContext", (__GLXextFuncPtr) glXGetCurrentContext },
1238    { "glXGetCurrentDrawable", (__GLXextFuncPtr) glXGetCurrentDrawable },
1239    { "glXIsDirect", (__GLXextFuncPtr) glXIsDirect },
1240    { "glXMakeCurrent", (__GLXextFuncPtr) glXMakeCurrent },
1241    { "glXQueryExtension", (__GLXextFuncPtr) glXQueryExtension },
1242    { "glXQueryVersion", (__GLXextFuncPtr) glXQueryVersion },
1243    { "glXSwapBuffers", (__GLXextFuncPtr) glXSwapBuffers },
1244    { "glXUseXFont", (__GLXextFuncPtr) glXUseXFont },
1245    { "glXWaitGL", (__GLXextFuncPtr) glXWaitGL },
1246    { "glXWaitX", (__GLXextFuncPtr) glXWaitX },
1247
1248    /*** GLX_VERSION_1_1 ***/
1249    { "glXGetClientString", (__GLXextFuncPtr) glXGetClientString },
1250    { "glXQueryExtensionsString", (__GLXextFuncPtr) glXQueryExtensionsString },
1251    { "glXQueryServerString", (__GLXextFuncPtr) glXQueryServerString },
1252
1253    /*** GLX_VERSION_1_2 ***/
1254    { "glXGetCurrentDisplay", (__GLXextFuncPtr) glXGetCurrentDisplay },
1255
1256    /*** GLX_VERSION_1_3 ***/
1257    { "glXChooseFBConfig", (__GLXextFuncPtr) glXChooseFBConfig },
1258    { "glXCreateNewContext", (__GLXextFuncPtr) glXCreateNewContext },
1259    { "glXCreatePbuffer", (__GLXextFuncPtr) glXCreatePbuffer },
1260    { "glXCreatePixmap", (__GLXextFuncPtr) glXCreatePixmap },
1261    { "glXCreateWindow", (__GLXextFuncPtr) glXCreateWindow },
1262    { "glXDestroyPbuffer", (__GLXextFuncPtr) glXDestroyPbuffer },
1263    { "glXDestroyPixmap", (__GLXextFuncPtr) glXDestroyPixmap },
1264    { "glXDestroyWindow", (__GLXextFuncPtr) glXDestroyWindow },
1265    { "glXGetCurrentReadDrawable", (__GLXextFuncPtr) glXGetCurrentReadDrawable },
1266    { "glXGetFBConfigAttrib", (__GLXextFuncPtr) glXGetFBConfigAttrib },
1267    { "glXGetFBConfigs", (__GLXextFuncPtr) glXGetFBConfigs },
1268    { "glXGetSelectedEvent", (__GLXextFuncPtr) glXGetSelectedEvent },
1269    { "glXGetVisualFromFBConfig", (__GLXextFuncPtr) glXGetVisualFromFBConfig },
1270    { "glXMakeContextCurrent", (__GLXextFuncPtr) glXMakeContextCurrent },
1271    { "glXQueryContext", (__GLXextFuncPtr) glXQueryContext },
1272    { "glXQueryDrawable", (__GLXextFuncPtr) glXQueryDrawable },
1273    { "glXSelectEvent", (__GLXextFuncPtr) glXSelectEvent },
1274
1275    /*** GLX_VERSION_1_4 ***/
1276    { "glXGetProcAddress", (__GLXextFuncPtr) glXGetProcAddress },
1277
1278    /*** GLX_SGI_swap_control ***/
1279    { "glXSwapIntervalSGI", (__GLXextFuncPtr) glXSwapIntervalSGI },
1280
1281    /*** GLX_SGI_video_sync ***/
1282    { "glXGetVideoSyncSGI", (__GLXextFuncPtr) glXGetVideoSyncSGI },
1283    { "glXWaitVideoSyncSGI", (__GLXextFuncPtr) glXWaitVideoSyncSGI },
1284
1285    /*** GLX_SGI_make_current_read ***/
1286    { "glXMakeCurrentReadSGI", (__GLXextFuncPtr) glXMakeCurrentReadSGI },
1287    { "glXGetCurrentReadDrawableSGI", (__GLXextFuncPtr) glXGetCurrentReadDrawableSGI },
1288
1289    /*** GLX_SGIX_video_source ***/
1290 #if defined(_VL_H)
1291    { "glXCreateGLXVideoSourceSGIX", (__GLXextFuncPtr) glXCreateGLXVideoSourceSGIX },
1292    { "glXDestroyGLXVideoSourceSGIX", (__GLXextFuncPtr) glXDestroyGLXVideoSourceSGIX },
1293 #endif
1294
1295    /*** GLX_EXT_import_context ***/
1296    { "glXFreeContextEXT", (__GLXextFuncPtr) glXFreeContextEXT },
1297    { "glXGetContextIDEXT", (__GLXextFuncPtr) glXGetContextIDEXT },
1298    { "glXGetCurrentDisplayEXT", (__GLXextFuncPtr) glXGetCurrentDisplayEXT },
1299    { "glXImportContextEXT", (__GLXextFuncPtr) glXImportContextEXT },
1300    { "glXQueryContextInfoEXT", (__GLXextFuncPtr) glXQueryContextInfoEXT },
1301
1302    /*** GLX_SGIX_fbconfig ***/
1303    { "glXGetFBConfigAttribSGIX", (__GLXextFuncPtr) glXGetFBConfigAttribSGIX },
1304    { "glXChooseFBConfigSGIX", (__GLXextFuncPtr) glXChooseFBConfigSGIX },
1305    { "glXCreateGLXPixmapWithConfigSGIX", (__GLXextFuncPtr) glXCreateGLXPixmapWithConfigSGIX },
1306    { "glXCreateContextWithConfigSGIX", (__GLXextFuncPtr) glXCreateContextWithConfigSGIX },
1307    { "glXGetVisualFromFBConfigSGIX", (__GLXextFuncPtr) glXGetVisualFromFBConfigSGIX },
1308    { "glXGetFBConfigFromVisualSGIX", (__GLXextFuncPtr) glXGetFBConfigFromVisualSGIX },
1309
1310    /*** GLX_SGIX_pbuffer ***/
1311    { "glXCreateGLXPbufferSGIX", (__GLXextFuncPtr) glXCreateGLXPbufferSGIX },
1312    { "glXDestroyGLXPbufferSGIX", (__GLXextFuncPtr) glXDestroyGLXPbufferSGIX },
1313    { "glXQueryGLXPbufferSGIX", (__GLXextFuncPtr) glXQueryGLXPbufferSGIX },
1314    { "glXSelectEventSGIX", (__GLXextFuncPtr) glXSelectEventSGIX },
1315    { "glXGetSelectedEventSGIX", (__GLXextFuncPtr) glXGetSelectedEventSGIX },
1316
1317    /*** GLX_SGI_cushion ***/
1318    { "glXCushionSGI", (__GLXextFuncPtr) glXCushionSGI },
1319
1320    /*** GLX_SGIX_video_resize ***/
1321    { "glXBindChannelToWindowSGIX", (__GLXextFuncPtr) glXBindChannelToWindowSGIX },
1322    { "glXChannelRectSGIX", (__GLXextFuncPtr) glXChannelRectSGIX },
1323    { "glXQueryChannelRectSGIX", (__GLXextFuncPtr) glXQueryChannelRectSGIX },
1324    { "glXQueryChannelDeltasSGIX", (__GLXextFuncPtr) glXQueryChannelDeltasSGIX },
1325    { "glXChannelRectSyncSGIX", (__GLXextFuncPtr) glXChannelRectSyncSGIX },
1326
1327    /*** GLX_SGIX_dmbuffer **/
1328 #if defined(_DM_BUFFER_H_)
1329    { "glXAssociateDMPbufferSGIX", (__GLXextFuncPtr) glXAssociateDMPbufferSGIX },
1330 #endif
1331
1332    /*** GLX_SGIX_swap_group ***/
1333    { "glXJoinSwapGroupSGIX", (__GLXextFuncPtr) glXJoinSwapGroupSGIX },
1334
1335    /*** GLX_SGIX_swap_barrier ***/
1336    { "glXBindSwapBarrierSGIX", (__GLXextFuncPtr) glXBindSwapBarrierSGIX },
1337    { "glXQueryMaxSwapBarriersSGIX", (__GLXextFuncPtr) glXQueryMaxSwapBarriersSGIX },
1338
1339    /*** GLX_SUN_get_transparent_index ***/
1340    { "glXGetTransparentIndexSUN", (__GLXextFuncPtr) glXGetTransparentIndexSUN },
1341
1342    /*** GLX_MESA_copy_sub_buffer ***/
1343    { "glXCopySubBufferMESA", (__GLXextFuncPtr) glXCopySubBufferMESA },
1344
1345    /*** GLX_MESA_pixmap_colormap ***/
1346    { "glXCreateGLXPixmapMESA", (__GLXextFuncPtr) glXCreateGLXPixmapMESA },
1347
1348    /*** GLX_MESA_release_buffers ***/
1349    { "glXReleaseBuffersMESA", (__GLXextFuncPtr) glXReleaseBuffersMESA },
1350
1351    /*** GLX_MESA_set_3dfx_mode ***/
1352    { "glXSet3DfxModeMESA", (__GLXextFuncPtr) glXSet3DfxModeMESA },
1353
1354    /*** GLX_ARB_get_proc_address ***/
1355    { "glXGetProcAddressARB", (__GLXextFuncPtr) glXGetProcAddressARB },
1356
1357    /*** GLX_NV_vertex_array_range ***/
1358    { "glXAllocateMemoryNV", (__GLXextFuncPtr) glXAllocateMemoryNV },
1359    { "glXFreeMemoryNV", (__GLXextFuncPtr) glXFreeMemoryNV },
1360
1361    /*** GLX_MESA_agp_offset ***/
1362    { "glXGetAGPOffsetMESA", (__GLXextFuncPtr) glXGetAGPOffsetMESA },
1363
1364    /*** GLX_EXT_texture_from_pixmap ***/
1365    { "glXBindTexImageEXT", (__GLXextFuncPtr) glXBindTexImageEXT },
1366    { "glXReleaseTexImageEXT", (__GLXextFuncPtr) glXReleaseTexImageEXT },
1367
1368    { NULL, NULL }   /* end of list */
1369 };
1370
1371
1372
1373 /*
1374  * Return address of named glX function, or NULL if not found.
1375  */
1376 __GLXextFuncPtr
1377 _glxapi_get_proc_address(const char *funcName)
1378 {
1379    GLuint i;
1380    for (i = 0; GLX_functions[i].Name; i++) {
1381 #ifdef MANGLE
1382       /* skip the "m" prefix on the name */
1383       if (strcmp(GLX_functions[i].Name, funcName+1) == 0)
1384 #else
1385       if (strcmp(GLX_functions[i].Name, funcName) == 0)
1386 #endif
1387          return GLX_functions[i].Address;
1388    }
1389    return NULL;
1390 }
1391
1392
1393
1394 /*
1395  * This function does not get dispatched through the dispatch table
1396  * since it's really a "meta" function.
1397  */
1398 __GLXextFuncPtr PUBLIC
1399 glXGetProcAddressARB(const GLubyte *procName)
1400 {
1401    __GLXextFuncPtr f;
1402
1403    f = _glxapi_get_proc_address((const char *) procName);
1404    if (f) {
1405       return f;
1406    }
1407
1408    f = (__GLXextFuncPtr) _glapi_get_proc_address((const char *) procName);
1409    return f;
1410 }
1411
1412
1413 /* GLX 1.4 */
1414 void PUBLIC
1415 (*glXGetProcAddress(const GLubyte *procName))()
1416 {
1417    return glXGetProcAddressARB(procName);
1418 }